123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106 |
- #import "PLCrashReporter.h"
- #import "CrashReporter.h"
- extern NSString* GetCrashReportsPath();
- void CrashedCheckBelowForHintsWhy();
- static NSUncaughtExceptionHandler* gsCrashReporterUEHandler = NULL;
- static void SavePendingCrashReport()
- {
- if (![[UnityPLCrashReporter sharedReporter] hasPendingCrashReport])
- return;
- NSFileManager *fm = [NSFileManager defaultManager];
- NSError *error;
- if (![fm createDirectoryAtPath: GetCrashReportsPath() withIntermediateDirectories: YES attributes: nil error: &error])
- {
- ::printf("CrashReporter: could not create crash report directory: %s\n", [[error localizedDescription] UTF8String]);
- return;
- }
- NSData *data = [[UnityPLCrashReporter sharedReporter] loadPendingCrashReportDataAndReturnError: &error];
- if (data == nil)
- {
- ::printf("CrashReporter: failed to load crash report data: %s\n", [[error localizedDescription] UTF8String]);
- return;
- }
- NSString* file = [GetCrashReportsPath() stringByAppendingPathComponent: @"crash-"];
- unsigned long long seconds = (unsigned long long)[[NSDate date] timeIntervalSince1970];
- file = [file stringByAppendingString: [NSString stringWithFormat: @"%llu", seconds]];
- file = [file stringByAppendingString: @".plcrash"];
- if ([data writeToFile: file atomically: YES])
- {
- ::printf("CrashReporter: saved pending crash report.\n");
- if (![[UnityPLCrashReporter sharedReporter] purgePendingCrashReportAndReturnError: &error])
- {
- ::printf("CrashReporter: couldn't remove pending report: %s\n", [[error localizedDescription] UTF8String]);
- }
- }
- else
- {
- ::printf("CrashReporter: couldn't save crash report.\n");
- }
- }
- static void InitCrashReporter()
- {
- NSError *error;
- UnityInstallPostCrashCallback();
- if ([[UnityPLCrashReporter sharedReporter] enableCrashReporterAndReturnError: &error])
- ::printf("CrashReporter: initialized\n");
- else
- NSLog(@"CrashReporter: could not enable crash reporter: %@", error);
- SavePendingCrashReport();
- }
- static void UncaughtExceptionHandler(NSException *exception)
- {
- NSLog(@"Uncaught exception: %@: %@\n%@", [exception name], [exception reason], [exception callStackSymbols]);
- if (gsCrashReporterUEHandler)
- gsCrashReporterUEHandler(exception);
- }
- static void InitObjCUEHandler()
- {
- // Crash reporter sets its own handler, so we have to save it and call it manually
- gsCrashReporterUEHandler = NSGetUncaughtExceptionHandler();
- NSSetUncaughtExceptionHandler(&UncaughtExceptionHandler);
- }
- void InitCrashHandling()
- {
- #if ENABLE_CUSTOM_CRASH_REPORTER
- InitCrashReporter();
- #endif
- #if ENABLE_OBJC_UNCAUGHT_EXCEPTION_HANDLER
- InitObjCUEHandler();
- #endif
- }
- // This function will be called when AppDomain.CurrentDomain.UnhandledException event is triggered.
- // When running on device the app will do a hard crash and it will generate a crash log.
- void CrashedCheckBelowForHintsWhy()
- {
- #if ENABLE_CRASH_REPORT_SUBMISSION
- // Wait if app has crashed before we were able to submit an older pending crash report. This
- // could happen if app crashes at startup.
- WaitWhileCrashReportsAreSent();
- #endif
- #if ENABLE_IOS_CRASH_REPORTING || ENABLE_CUSTOM_CRASH_REPORTER
- // Make app crash hard here
- __builtin_trap();
- // Just in case above doesn't work
- abort();
- #endif
- }
|