/* * mini-darwin.c: Darwin/MacOS support for Mono. * * Authors: * Mono Team (mono-list@lists.ximian.com) * * Copyright 2001-2003 Ximian, Inc. * Copyright 2003-2011 Novell, Inc (http://www.novell.com) * Copyright 2011 Xamarin, Inc (http://www.xamarin.com) * * Licensed under the MIT license. See LICENSE file in the project root for full license information. */ #include #include #ifdef HAVE_ALLOCA_H #include #endif #ifdef HAVE_UNISTD_H #include #endif #include #ifdef HAVE_SYS_TIME_H #include #endif #include #include #include #include #include #include #include #include #include #include #include #include "mono/metadata/profiler.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "mini.h" #include #include #include "trace.h" #include "version.h" #include "jit-icalls.h" /* MacOS includes */ #include #include #include #include #include #include #include /* This is #define'd by Boehm GC to _GC_dlopen. */ #undef dlopen void* dlopen(const char* path, int mode); void mono_runtime_install_handlers (void) { mono_runtime_posix_install_handlers (); /* Snow Leopard has a horrible bug: http://openradar.appspot.com/7209349 * This causes obscure SIGTRAP's for any application that comes across this built on * Snow Leopard. This is a horrible hack to ensure that the private __CFInitialize * is run on the main thread, so that we don't get SIGTRAPs later */ #if defined (__APPLE__) && (defined (__i386__) || defined (__x86_64__)) { void *handle = dlopen ("/System/Library/Frameworks/CoreFoundation.framework/CoreFoundation", RTLD_LAZY); if (handle == NULL) return; dlclose (handle); } #endif } pid_t mono_runtime_syscall_fork () { #ifdef HAVE_FORK return (pid_t) fork (); #else g_assert_not_reached (); #endif } void mono_gdb_render_native_backtraces (pid_t crashed_pid) { #ifdef HAVE_EXECV const char *argv [5]; char template [] = "/tmp/mono-gdb-commands.XXXXXX"; FILE *commands; gboolean using_lldb = FALSE; using_lldb = TRUE; argv [0] = g_find_program_in_path ("gdb"); if (argv [0]) using_lldb = FALSE; if (using_lldb) argv [0] = g_find_program_in_path ("lldb"); if (argv [0] == NULL) return; if (mkstemp (template) == -1) return; commands = fopen (template, "w"); if (using_lldb) { fprintf (commands, "process attach --pid %ld\n", (long) crashed_pid); fprintf (commands, "thread list\n"); fprintf (commands, "thread backtrace all\n"); fprintf (commands, "detach\n"); fprintf (commands, "quit\n"); argv [1] = "--source"; argv [2] = template; argv [3] = 0; } else { fprintf (commands, "attach %ld\n", (long) crashed_pid); fprintf (commands, "info threads\n"); fprintf (commands, " t a a info thread\n"); fprintf (commands, "thread apply all bt\n"); argv [1] = "-batch"; argv [2] = "-x"; argv [3] = template; argv [4] = 0; } fflush (commands); fclose (commands); fclose (stdin); execv (argv [0], (char**)argv); unlink (template); #else fprintf (stderr, "mono_gdb_render_native_backtraces not supported on this platform\n"); #endif // HAVE_EXECV } gboolean mono_thread_state_init_from_handle (MonoThreadUnwindState *tctx, MonoThreadInfo *info) { kern_return_t ret; mach_msg_type_number_t num_state; thread_state_t state; ucontext_t ctx; mcontext_t mctx; MonoJitTlsData *jit_tls; void *domain; MonoLMF *lmf = NULL; gpointer *addr; g_assert (info); /*Zero enough state to make sure the caller doesn't confuse itself*/ tctx->valid = FALSE; tctx->unwind_data [MONO_UNWIND_DATA_DOMAIN] = NULL; tctx->unwind_data [MONO_UNWIND_DATA_LMF] = NULL; tctx->unwind_data [MONO_UNWIND_DATA_JIT_TLS] = NULL; state = (thread_state_t) alloca (mono_mach_arch_get_thread_state_size ()); mctx = (mcontext_t) alloca (mono_mach_arch_get_mcontext_size ()); do { ret = mono_mach_arch_get_thread_state (info->native_handle, state, &num_state); } while (ret == KERN_ABORTED); if (ret != KERN_SUCCESS) return FALSE; mono_mach_arch_thread_state_to_mcontext (state, mctx); ctx.uc_mcontext = mctx; mono_sigctx_to_monoctx (&ctx, &tctx->ctx); /* mono_set_jit_tls () sets this */ jit_tls = mono_thread_info_tls_get (info, TLS_KEY_JIT_TLS); /* SET_APPDOMAIN () sets this */ domain = mono_thread_info_tls_get (info, TLS_KEY_DOMAIN); /*Thread already started to cleanup, can no longer capture unwind state*/ if (!jit_tls || !domain) return FALSE; /* * The current LMF address is kept in a separate TLS variable, and its hard to read its value without * arch-specific code. But the address of the TLS variable is stored in another TLS variable which * can be accessed through MonoThreadInfo. */ /* mono_set_lmf_addr () sets this */ addr = mono_thread_info_tls_get (info, TLS_KEY_LMF_ADDR); if (addr) lmf = *addr; tctx->unwind_data [MONO_UNWIND_DATA_DOMAIN] = domain; tctx->unwind_data [MONO_UNWIND_DATA_JIT_TLS] = jit_tls; tctx->unwind_data [MONO_UNWIND_DATA_LMF] = lmf; tctx->valid = TRUE; return TRUE; }