3 * Support for interop with the Microsoft Error Reporting tool
6 * Alexander Kyte (alkyte@microsoft.com)
8 * (C) 2018 Microsoft, Inc.
15 #if defined(TARGET_OSX) && !defined(DISABLE_CRASH_REPORTING)
16 #include "mono-merp.h"
21 // OSX OS stuff now, for merpGUI interop
22 #include <mach/mach.h>
23 #include <mach/task_info.h>
24 #include <mach/mach_types.h>
25 #include <mach/mach_traps.h>
26 #include <servers/bootstrap.h>
28 #include <metadata/locales.h>
31 #if defined(HAVE_SYS_UTSNAME_H)
32 #include <sys/utsname.h>
35 // To get the apple machine model
36 #include <sys/param.h>
37 #include <sys/sysctl.h>
40 #include <mono/utils/json.h>
41 #include <mono/utils/mono-state.h>
42 #include <utils/mono-threads-debug.h>
45 kernel_version_string (void)
47 #ifdef HAVE_SYS_UTSNAME_H
48 static struct utsname name
;
49 static const char *version_string
;
51 if (!version_string
) {
52 // WSL does not always nul terminate. WSL was fixed February 2018.
53 // Restore memset if variable made non-static.
54 //memset (&name, 0, sizeof (name));
56 if (uname (&name
) >= 0)
57 version_string
= name
.release
;
62 return version_string
;
68 starts_with (const char *pre
, size_t pre_sz
, const char *str
)
70 return strncmp (pre
, str
, pre_sz
) == 0;
74 macos_version_string (void)
76 /* Can't allocate in here, could be called from a signal handler in a
79 static const char *version_string
;
81 static const size_t buf_size
= sizeof (buf
);
84 return version_string
;
86 /* macOS 10.13.6 or later */
87 if (!version_string
) {
89 if (sysctlbyname ("kern.osproductversion", NULL
, &size
, NULL
, 0) < 0 || size
>= buf_size
) {
90 /* if we couldn't get the size or if it needs more space that we have in buf, leave it empty */
92 return version_string
;
95 if (sysctlbyname ("kern.osproductversion", (char*)buf
, &size
, NULL
, 0) >= 0)
96 version_string
= &buf
[0];
98 /* macOS 10.13.5 or older */
99 if (!version_string
) {
100 const char *kv_string
= kernel_version_string ();
101 if (starts_with (G_STRING_CONSTANT_AND_LENGTH ("17"), kv_string
))
102 version_string
= "10.13"; // High Sierra
103 else if (starts_with (G_STRING_CONSTANT_AND_LENGTH ("16"), kv_string
))
104 version_string
= "10.12"; // Sierra
105 else if (starts_with (G_STRING_CONSTANT_AND_LENGTH ("15"), kv_string
))
106 version_string
= "10.11"; // El Capitan
107 else if (starts_with (G_STRING_CONSTANT_AND_LENGTH ("14"), kv_string
))
108 version_string
= "10.10"; // Yosemite
109 else if (starts_with (G_STRING_CONSTANT_AND_LENGTH ("13"), kv_string
))
110 version_string
= "10.9"; // Mavericks
111 else if (starts_with (G_STRING_CONSTANT_AND_LENGTH ("12"), kv_string
))
112 version_string
= "10.8"; // Mountain Lion
113 else if (starts_with (G_STRING_CONSTANT_AND_LENGTH ("11"), kv_string
))
114 version_string
= "10.7"; // Lion
119 return version_string
;
122 // To get the path of the running process
138 MERP_EXC_FORCE_QUIT
= 1,
139 MERP_EXC_SIGSEGV
= 2,
140 MERP_EXC_SIGABRT
= 3,
144 MERP_EXC_SIGFPE
= 7 ,
145 MERP_EXC_SIGTRAP
= 8,
146 MERP_EXC_SIGKILL
= 9,
151 const char *merpFilePath
;
152 const char *crashLogPath
;
153 const char *werXmlPath
;
155 const char *bundleIDArg
; // App Bundle ID (required for bucketization)
156 const char *versionArg
; // App Version (required for bucketization)
158 MerpArch archArg
; // Arch, MacOS only, bails out if not found also required for bucketization. (required)
159 MERPExcType exceptionArg
; // Exception type (refer to merpcommon.h and mach/exception_types.h for more info (optional)
161 const char *serviceNameArg
; // This is the Bootstrap service name that MERP GUI will create to receive mach_task_self on a port created. Bails out if MERP GUI fails to receive mach_task_self from the crashed app. (Required for crash log generation)
162 const char *servicePathArg
; // The path to the executable, used to relaunch the crashed app.
164 const char *moduleName
;
165 const char *moduleVersion
;
168 const char *osVersion
;
169 int uiLidArg
; // MONO_LOCALE_INVARIANT 0x007F
171 char systemModel
[100];
172 const char *systemManufacturer
;
174 const char *eventType
;
176 MonoStackHash hashes
;
181 gboolean enable_merp
;
183 const char *appBundleID
;
185 const char *appSignature
;
186 const char *appVersion
;
187 const char *merpGUIPath
;
188 const char *eventType
;
189 const char *merpFilePath
;
190 const char *crashLogPath
;
191 const char *werXmlPath
;
192 const char *moduleVersion
;
198 static MerpOptions config
;
203 } MonoMerpAnnotationEntry
;
206 get_merp_bitness (MerpArch arch
)
214 g_assert_not_reached ();
223 #elif defined(TARGET_AMD64)
224 return MerpArchx86_64
;
225 #elif defined(TARGET_POWERPC)
227 #elif defined(TARGET_POWERPC64)
228 return MerpArchPPC64
;
230 g_assert_not_reached ();
235 get_merp_exctype (MERPExcType exc
)
238 case MERP_EXC_FORCE_QUIT
:
240 case MERP_EXC_SIGSEGV
:
242 case MERP_EXC_SIGABRT
:
244 case MERP_EXC_SIGSYS
:
246 case MERP_EXC_SIGILL
:
248 case MERP_EXC_SIGBUS
:
250 case MERP_EXC_SIGFPE
:
252 case MERP_EXC_SIGTRAP
:
254 case MERP_EXC_SIGKILL
:
259 // Exception type documented as optional, not optional
260 g_assert_not_reached ();
262 g_assert_not_reached ();
267 parse_exception_type (const char *signal
)
269 if (!strcmp (signal
, "SIGSEGV"))
270 return MERP_EXC_SIGSEGV
;
272 if (!strcmp (signal
, "SIGFPE"))
273 return MERP_EXC_SIGFPE
;
275 if (!strcmp (signal
, "SIGILL"))
276 return MERP_EXC_SIGILL
;
278 if (!strcmp (signal
, "SIGABRT"))
279 return MERP_EXC_SIGABRT
;
281 // Force quit == hang?
282 // We need a default for this
283 if (!strcmp (signal
, "SIGTERM"))
284 return MERP_EXC_HANG
;
286 // FIXME: There are no other such signal
287 // strings passed to mono_handle_native_crash at the
288 // time of writing this
289 g_error ("Merp doesn't know how to handle %s\n", signal
);
292 static int merp_file_permissions
= S_IWUSR
| S_IRUSR
| S_IRGRP
| S_IROTH
;
295 mono_merp_write_params (MERPStruct
*merp
)
297 int handle
= g_open (merp
->merpFilePath
, O_TRUNC
| O_WRONLY
| O_CREAT
, merp_file_permissions
);
298 g_assertf (handle
!= -1, "Could not open MERP file at %s", merp
->merpFilePath
);
300 g_async_safe_fprintf(handle
, "ApplicationBundleId: %s\n", merp
->bundleIDArg
);
301 g_async_safe_fprintf(handle
, "ApplicationVersion: %s\n", merp
->versionArg
);
302 g_async_safe_fprintf(handle
, "ApplicationBitness: %s\n", get_merp_bitness (merp
->archArg
));
304 g_async_safe_fprintf(handle
, "ApplicationName: %s\n", merp
->serviceNameArg
);
305 g_async_safe_fprintf(handle
, "ApplicationPath: %s\n", merp
->servicePathArg
);
306 g_async_safe_fprintf(handle
, "BlameModuleName: %s\n", merp
->moduleName
);
307 g_async_safe_fprintf(handle
, "BlameModuleVersion: %s\n", merp
->moduleVersion
);
308 g_async_safe_fprintf(handle
, "BlameModuleOffset: 0x%llx\n", (unsigned long long)merp
->moduleOffset
);
309 g_async_safe_fprintf(handle
, "ExceptionType: %s\n", get_merp_exctype (merp
->exceptionArg
));
310 g_async_safe_fprintf(handle
, "StackChecksum: 0x%llx\n", merp
->hashes
.offset_free_hash
);
311 g_async_safe_fprintf(handle
, "StackHash: 0x%llx\n", merp
->hashes
.offset_rich_hash
);
314 g_async_safe_fprintf(handle
, "OSVersion: %s\n", merp
->osVersion
);
315 g_async_safe_fprintf(handle
, "LanguageID: 0x%x\n", merp
->uiLidArg
);
316 g_async_safe_fprintf(handle
, "SystemManufacturer: %s\n", merp
->systemManufacturer
);
317 g_async_safe_fprintf(handle
, "SystemModel: %s\n", merp
->systemModel
);
318 g_async_safe_fprintf(handle
, "EventType: %s\n", merp
->eventType
);
325 mono_merp_send (MERPStruct
*merp
)
327 gboolean invoke_success
= FALSE
;
329 #if defined(HAVE_EXECV) && defined(HAVE_FORK)
330 pid_t pid
= (pid_t
) fork ();
332 // Only one we define on OSX
334 const char *open_path
= "/usr/bin/open";
335 const char *argvOpen
[] = {open_path
, "-a", config
.merpGUIPath
, NULL
};
336 execv (open_path
, (char**)argvOpen
);
340 waitpid (pid
, &status
, 0);
341 gboolean exit_success
= FALSE
;
342 int exit_status
= FALSE
;
345 if (waitpid(pid
, &status
, WUNTRACED
| WCONTINUED
) == -1)
348 if (WIFEXITED(status
)) {
349 exit_status
= WEXITSTATUS(status
);
351 invoke_success
= exit_status
== TRUE
;
353 } else if (WIFSIGNALED(status
)) {
359 // // Create process to launch merp gui application
362 return invoke_success
;
366 get_apple_model (char *buffer
, size_t max_length
)
370 // Get the number of bytes to copy
371 sysctlbyname("hw.model", NULL
, &sz
, NULL
, 0);
373 if (sz
> max_length
) {
378 sysctlbyname("hw.model", buffer
, &sz
, NULL
, 0);
382 mono_init_merp (const intptr_t crashed_pid
, const char *signal
, MonoStackHash
*hashes
, MERPStruct
*merp
)
384 mono_memory_barrier ();
385 g_assert (mono_merp_enabled ());
387 merp
->merpFilePath
= config
.merpFilePath
;
388 merp
->crashLogPath
= config
.crashLogPath
;
389 merp
->werXmlPath
= config
.werXmlPath
;
391 // If these aren't set, icall wasn't made
392 // don't do merp? / don't set the variable to use merp;
393 g_assert (config
.appBundleID
);
394 g_assert (config
.appVersion
);
395 merp
->bundleIDArg
= config
.appSignature
;
396 merp
->versionArg
= config
.appVersion
;
398 merp
->archArg
= get_merp_arch ();
399 merp
->exceptionArg
= parse_exception_type (signal
);
401 merp
->serviceNameArg
= config
.appBundleID
;
402 merp
->servicePathArg
= config
.appPath
;
404 merp
->moduleName
= "Mono Exception";
405 merp
->moduleVersion
= config
.moduleVersion
;
407 merp
->moduleOffset
= 0;
409 merp
->uiLidArg
= MONO_LOCALE_INVARIANT
;
410 #if defined (TARGET_OSX)
411 merp
->osVersion
= macos_version_string ();
413 merp
->osVersion
= kernel_version_string ();
416 // FIXME: THis is apple-only for now
417 merp
->systemManufacturer
= "apple";
418 get_apple_model ((char *) merp
->systemModel
, sizeof (merp
->systemModel
));
420 merp
->eventType
= config
.eventType
;
422 merp
->hashes
= *hashes
;
424 merp
->annotations
= config
.annotations
;
428 mono_merp_write_fingerprint_payload (const char *non_param_data
, const MERPStruct
*merp
)
430 int handle
= g_open (merp
->crashLogPath
, O_TRUNC
| O_WRONLY
| O_CREAT
, merp_file_permissions
);
431 g_assertf (handle
!= -1, "Could not open crash log file at %s", merp
->crashLogPath
);
433 g_async_safe_fprintf(handle
, "{\n");
434 g_async_safe_fprintf(handle
, "\t\"payload\" : \n");
435 g_write (handle
, non_param_data
, (guint32
)strlen (non_param_data
));
436 g_async_safe_fprintf(handle
, ",\n");
438 g_async_safe_fprintf(handle
, "\t\"parameters\" : \n{\n");
439 g_async_safe_fprintf(handle
, "\t\t\"ApplicationBundleId\" : \"%s\",\n", merp
->bundleIDArg
);
440 g_async_safe_fprintf(handle
, "\t\t\"ApplicationVersion\" : \"%s\",\n", merp
->versionArg
);
441 g_async_safe_fprintf(handle
, "\t\t\"ApplicationBitness\" : \"%s\",\n", get_merp_bitness (merp
->archArg
));
442 g_async_safe_fprintf(handle
, "\t\t\"ApplicationName\" : \"%s\",\n", merp
->serviceNameArg
);
443 g_async_safe_fprintf(handle
, "\t\t\"BlameModuleName\" : \"%s\",\n", merp
->moduleName
);
444 g_async_safe_fprintf(handle
, "\t\t\"BlameModuleVersion\" : \"%s\",\n", merp
->moduleVersion
);
445 g_async_safe_fprintf(handle
, "\t\t\"BlameModuleOffset\" : \"0x%lx\",\n", merp
->moduleOffset
);
446 g_async_safe_fprintf(handle
, "\t\t\"ExceptionType\" : \"%s\",\n", get_merp_exctype (merp
->exceptionArg
));
447 g_async_safe_fprintf(handle
, "\t\t\"StackChecksum\" : \"0x%llx\",\n", merp
->hashes
.offset_free_hash
);
448 g_async_safe_fprintf(handle
, "\t\t\"StackHash\" : \"0x%llx\",\n", merp
->hashes
.offset_rich_hash
);
449 g_async_safe_fprintf(handle
, "\t\t\"Extra\" : \n\t\t{\n");
451 for (GSList
*cursor
= merp
->annotations
; cursor
; cursor
= cursor
->next
) {
452 MonoMerpAnnotationEntry
*iter
= (MonoMerpAnnotationEntry
*) cursor
->data
;
453 g_async_safe_fprintf(handle
, "\t\t\t\"%s\" : \"%s\"\n", iter
->key
, iter
->value
);
456 g_async_safe_fprintf(handle
, "\t\t},\n");
459 g_async_safe_fprintf(handle
, "\t\t\"OSVersion\" : \"%s\",\n", merp
->osVersion
);
460 g_async_safe_fprintf(handle
, "\t\t\"LanguageID\" : \"0x%x\",\n", merp
->uiLidArg
);
461 g_async_safe_fprintf(handle
, "\t\t\"SystemManufacturer\" : \"%s\",\n", merp
->systemManufacturer
);
462 g_async_safe_fprintf(handle
, "\t\t\"SystemModel\" : \"%s\",\n", merp
->systemModel
);
463 g_async_safe_fprintf(handle
, "\t\t\"EventType\" : \"%s\"\n", merp
->eventType
);
466 g_async_safe_fprintf(handle
, "\t}\n");
467 g_async_safe_fprintf(handle
, "}\n");
476 mono_write_wer_template (MERPStruct
*merp
)
478 // Note about missing ProcessInformation block: we have no PID that makes sense
479 // and when mono is embedded and used to run functions without an entry point,
480 // there is no image that would make any semantic sense to send either.
481 // It's a nuanced problem, each way we can run mono would need a separate fix.
483 int handle
= g_open (merp
->werXmlPath
, O_WRONLY
| O_CREAT
| O_TRUNC
, merp_file_permissions
);
484 g_assertf (handle
!= -1, "Could not open WER XML file at %s", merp
->werXmlPath
);
487 g_async_safe_fprintf(handle
, "<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n");
488 g_async_safe_fprintf(handle
, "<WERReportMetadata>\n");
489 g_async_safe_fprintf(handle
, "<ProblemSignatures>\n");
490 g_async_safe_fprintf(handle
, "<EventType>%s</EventType>\n", merp
->eventType
);
493 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, merp
->bundleIDArg
, i
);
495 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, merp
->versionArg
, i
);
497 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, get_merp_bitness (merp
->archArg
), i
);
499 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, merp
->serviceNameArg
, i
);
501 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, merp
->moduleName
, i
);
503 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, merp
->moduleVersion
, i
);
505 g_async_safe_fprintf(handle
, "<Parameter%d>0x%zx</Parameter%d>\n", i
, merp
->moduleOffset
, i
);
507 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, get_merp_exctype (merp
->exceptionArg
), i
);
509 g_async_safe_fprintf(handle
, "<Parameter%d>0x%llx</Parameter%d>\n", i
, merp
->hashes
.offset_free_hash
, i
);
511 g_async_safe_fprintf(handle
, "<Parameter%d>0x%llx</Parameter%d>\n", i
, merp
->hashes
.offset_rich_hash
, i
);
513 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, merp
->osVersion
, i
);
515 g_async_safe_fprintf(handle
, "<Parameter%d>0x%x</Parameter%d>\n", i
, merp
->uiLidArg
, i
);
517 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, merp
->systemManufacturer
, i
);
519 g_async_safe_fprintf(handle
, "<Parameter%d>%s</Parameter%d>\n", i
, merp
->systemModel
, i
);
522 g_async_safe_fprintf(handle
, "</ProblemSignatures>\n");
523 g_async_safe_fprintf(handle
, "</WERReportMetadata>\n");
532 mono_merp_invoke (const intptr_t crashed_pid
, const char *signal
, const char *non_param_data
, MonoStackHash
*hashes
)
535 int merp_tmp_file_tag
= 2;
536 gboolean alloc_success
= mono_state_alloc_mem (&mem
, merp_tmp_file_tag
, sizeof (MERPStruct
));
540 MERPStruct
*merp
= (MERPStruct
*) mem
.mem
;
541 memset (merp
, 0, sizeof (*merp
));
543 mono_summarize_timeline_phase_log (MonoSummaryMerpWriter
);
545 mono_init_merp (crashed_pid
, signal
, hashes
, merp
);
547 if (!mono_merp_write_params (merp
))
550 if (!mono_merp_write_fingerprint_payload (non_param_data
, merp
))
553 if (!mono_write_wer_template (merp
))
557 mono_summarize_timeline_phase_log (MonoSummaryMerpInvoke
);
558 gboolean success
= mono_merp_send (merp
);
561 mono_summarize_timeline_phase_log (MonoSummaryCleanup
);
563 mono_state_free_mem (&mem
);
569 mono_merp_add_annotation (const char *key
, const char *value
)
571 MonoMerpAnnotationEntry
*entry
= g_new0 (MonoMerpAnnotationEntry
, 1);
572 entry
->key
= g_strdup (key
);
573 entry
->value
= g_strdup (value
);
574 config
.annotations
= g_slist_prepend (config
.annotations
, entry
);
578 mono_merp_disable (void)
580 mono_memory_barrier ();
582 if (!config
.enable_merp
)
585 g_free ((char*)config
.appBundleID
); // cast away const
586 g_free ((char*)config
.appSignature
);
587 g_free ((char*)config
.appVersion
);
588 g_free ((char*)config
.merpGUIPath
);
589 g_free ((char*)config
.eventType
);
590 g_free ((char*)config
.appPath
);
591 g_free ((char*)config
.moduleVersion
);
592 g_slist_free (config
.annotations
);
593 memset (&config
, 0, sizeof (config
));
595 mono_memory_barrier ();
599 mono_merp_enable (const char *appBundleID
, const char *appSignature
, const char *appVersion
, const char *merpGUIPath
, const char *eventType
, const char *appPath
, const char *configDir
)
601 mono_memory_barrier ();
603 g_assert (!config
.enable_merp
);
608 const char *home
= g_get_home_dir ();
609 prefix
= g_strdup_printf ("%s/Library/Group Containers/UBF8T346G9.ms/", home
);
611 prefix
= g_strdup (configDir
);
613 config
.merpFilePath
= g_strdup_printf ("%s%s", prefix
, "MERP.uploadparams.txt");
614 config
.crashLogPath
= g_strdup_printf ("%s%s", prefix
, "lastcrashlog.txt");
615 config
.werXmlPath
= g_strdup_printf ("%s%s", prefix
, "CustomLogsMetadata.xml");
618 config
.moduleVersion
= mono_get_runtime_callbacks ()->get_runtime_build_info ();
620 config
.appBundleID
= g_strdup (appBundleID
);
621 config
.appSignature
= g_strdup (appSignature
);
622 config
.appVersion
= g_strdup (appVersion
);
623 config
.merpGUIPath
= g_strdup (merpGUIPath
);
624 config
.eventType
= g_strdup (eventType
);
625 config
.appPath
= g_strdup (appPath
);
627 config
.log
= g_getenv ("MONO_MERP_VERBOSE") != NULL
;
629 config
.enable_merp
= TRUE
;
631 mono_memory_barrier ();
635 mono_merp_enabled (void)
637 return config
.enable_merp
;