]> git.cworth.org Git - apitrace/blob - common/os_posix.cpp
os: Be more concise with the semantics of os::abort.
[apitrace] / common / os_posix.cpp
1 /**************************************************************************
2  *
3  * Copyright 2010-2011 VMware, Inc.
4  * All Rights Reserved.
5  *
6  * Permission is hereby granted, free of charge, to any person obtaining a copy
7  * of this software and associated documentation files (the "Software"), to deal
8  * in the Software without restriction, including without limitation the rights
9  * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10  * copies of the Software, and to permit persons to whom the Software is
11  * furnished to do so, subject to the following conditions:
12  *
13  * The above copyright notice and this permission notice shall be included in
14  * all copies or substantial portions of the Software.
15  *
16  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17  * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19  * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20  * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22  * THE SOFTWARE.
23  *
24  **************************************************************************/
25
26
27 #include <assert.h>
28 #include <string.h>
29 #include <stdio.h>
30 #include <stdlib.h>
31
32 #include <unistd.h>
33 #include <sys/wait.h>
34 #include <sys/stat.h>
35 #include <fcntl.h>
36 #include <signal.h>
37
38 #if defined(__linux__)
39 #include <linux/limits.h> // PATH_MAX
40 #endif
41
42 #ifdef __APPLE__
43 #include <sys/syslimits.h> // PATH_MAX
44 #include <mach-o/dyld.h>
45 #endif
46
47 #ifdef ANDROID
48 #include <android/log.h>
49 #endif
50
51 #ifndef PATH_MAX
52 #warning PATH_MAX undefined
53 #define PATH_MAX 4096
54 #endif
55
56 #include "os.hpp"
57 #include "os_string.hpp"
58
59
60 namespace os {
61
62
63 String
64 getProcessName(void)
65 {
66     String path;
67     size_t size = PATH_MAX;
68     char *buf = path.buf(size);
69
70     // http://stackoverflow.com/questions/1023306/finding-current-executables-path-without-proc-self-exe
71 #ifdef __APPLE__
72     uint32_t len = size;
73     if (_NSGetExecutablePath(buf, &len) != 0) {
74         *buf = 0;
75         return path;
76     }
77     len = strlen(buf);
78 #else
79     ssize_t len;
80     len = readlink("/proc/self/exe", buf, size - 1);
81     if (len == -1) {
82         // /proc/self/exe is not available on setuid processes, so fallback to
83         // /proc/self/cmdline.
84         int fd = open("/proc/self/cmdline", O_RDONLY);
85         if (fd >= 0) {
86             len = read(fd, buf, size - 1);
87             close(fd);
88         }
89     }
90     if (len <= 0) {
91         snprintf(buf, size, "%i", (int)getpid());
92         return path;
93     }
94 #endif
95     path.truncate(len);
96
97     return path;
98 }
99
100 String
101 getCurrentDir(void)
102 {
103     String path;
104     size_t size = PATH_MAX;
105     char *buf = path.buf(size);
106
107     getcwd(buf, size);
108     buf[size - 1] = 0;
109     
110     path.truncate();
111     return path;
112 }
113
114 bool
115 String::exists(void) const
116 {
117     struct stat st;
118     int err;
119
120     err = stat(str(), &st);
121     if (err) {
122         return false;
123     }
124
125     if (!S_ISREG(st.st_mode))
126         return false;
127
128     return true;
129 }
130
131 int execute(char * const * args)
132 {
133     pid_t pid = fork();
134     if (pid == 0) {
135         // child
136         execvp(args[0], args);
137         fprintf(stderr, "error: failed to execute:");
138         for (unsigned i = 0; args[i]; ++i) {
139             fprintf(stderr, " %s", args[i]);
140         }
141         fprintf(stderr, "\n");
142         exit(-1);
143     } else {
144         // parent
145         if (pid == -1) {
146             fprintf(stderr, "error: failed to fork\n");
147             return -1;
148         }
149         int status = -1;
150         int ret;
151         waitpid(pid, &status, 0);
152         if (WIFEXITED(status)) {
153             ret = WEXITSTATUS(status);
154         } else if (WIFSIGNALED(status)) {
155             // match shell return code
156             ret = WTERMSIG(status) + 128;
157         } else {
158             ret = 128;
159         }
160         return ret;
161     }
162 }
163
164 static volatile bool logging = false;
165
166 void
167 log(const char *format, ...)
168 {
169     logging = true;
170     va_list ap;
171     va_start(ap, format);
172     fflush(stdout);
173 #ifdef ANDROID
174     __android_log_vprint(ANDROID_LOG_DEBUG, "apitrace", format, ap);
175 #else
176     vfprintf(stderr, format, ap);
177     fflush(stderr);
178 #endif
179     va_end(ap);
180     logging = false;
181 }
182
183 #if defined(__APPLE__)
184 long long timeFrequency = 0LL;
185 #endif
186
187 void
188 abort(void)
189 {
190     _exit(1);
191 }
192
193
194 static void (*gCallback)(void) = NULL;
195
196 #define NUM_SIGNALS 16
197
198 struct sigaction old_actions[NUM_SIGNALS];
199
200
201 /*
202  * See also:
203  * - http://sourceware.org/git/?p=glibc.git;a=blob;f=debug/segfault.c
204  * - http://ggi.cvs.sourceforge.net/viewvc/ggi/ggi-core/libgg/gg/cleanup.c?view=markup
205  */
206 static void
207 signalHandler(int sig, siginfo_t *info, void *context)
208 {
209     /*
210      * There are several signals that can happen when logging to stdout/stderr.
211      * For example, SIGPIPE will be emitted if stderr is a pipe with no
212      * readers.  Therefore ignore any signal while logging by returning
213      * immediately, to prevent deadlocks.
214      */
215     if (logging) {
216         return;
217     }
218
219     static int recursion_count = 0;
220
221     log("apitrace: warning: caught signal %i\n", sig);
222
223     if (recursion_count) {
224         log("apitrace: warning: recursion handling signal %i\n", sig);
225     } else {
226         if (gCallback) {
227             ++recursion_count;
228             gCallback();
229             --recursion_count;
230         }
231     }
232
233     struct sigaction *old_action;
234     if (sig >= NUM_SIGNALS) {
235         /* This should never happen */
236         log("error: unexpected signal %i\n", sig);
237         raise(SIGKILL);
238     }
239     old_action = &old_actions[sig];
240
241     if (old_action->sa_flags & SA_SIGINFO) {
242         // Handler is in sa_sigaction
243         old_action->sa_sigaction(sig, info, context);
244     } else {
245         if (old_action->sa_handler == SIG_DFL) {
246             log("apitrace: info: taking default action for signal %i\n", sig);
247
248 #if 1
249             struct sigaction dfl_action;
250             dfl_action.sa_handler = SIG_DFL;
251             sigemptyset (&dfl_action.sa_mask);
252             dfl_action.sa_flags = 0;
253             sigaction(sig, &dfl_action, NULL);
254
255             raise(sig);
256 #else
257             raise(SIGKILL);
258 #endif
259         } else if (old_action->sa_handler == SIG_IGN) {
260             /* ignore */
261         } else {
262             /* dispatch to handler */
263             old_action->sa_handler(sig);
264         }
265     }
266 }
267
268 void
269 setExceptionCallback(void (*callback)(void))
270 {
271     assert(!gCallback);
272     if (!gCallback) {
273         gCallback = callback;
274
275         struct sigaction new_action;
276         new_action.sa_sigaction = signalHandler;
277         sigemptyset(&new_action.sa_mask);
278         new_action.sa_flags = SA_SIGINFO | SA_RESTART;
279
280
281         for (int sig = 1; sig < NUM_SIGNALS; ++sig) {
282             // SIGKILL and SIGSTOP can't be handled.
283             if (sig == SIGKILL || sig == SIGSTOP) {
284                 continue;
285             }
286
287             /*
288              * SIGPIPE can be emitted when writing to stderr that is redirected
289              * to a pipe without readers.  It is also very unlikely to ocurr
290              * inside graphics APIs, and most applications where it can occur
291              * normally already ignore it.  In summary, it is unlikely that a
292              * SIGPIPE will cause abnormal termination, which it is likely that
293              * intercepting here will cause problems, so simple don't intercept
294              * it here.
295              */
296             if (sig == SIGPIPE) {
297                 continue;
298             }
299
300             if (sigaction(sig,  NULL, &old_actions[sig]) >= 0) {
301                 sigaction(sig,  &new_action, NULL);
302             }
303         }
304     }
305 }
306
307 void
308 resetExceptionCallback(void)
309 {
310     gCallback = NULL;
311 }
312
313 } /* namespace os */
314