+
+ /* <rdar://problem/7397932> abort() should call pthread_kill to deliver a signal to the aborting thread
+ * This helps gdb focus on the thread calling abort()
+ */
+
+ /* Block all signals on all other threads */
+ sigset_t fullmask;
+ sigfillset(&fullmask);
+ (void)_sigprocmask(SIG_SETMASK, &fullmask, NULL);
+
+ /* <rdar://problem/8400096> Set the workqueue killable */
+ __pthread_workqueue_setkill(1);
+
+ (void)pthread_sigmask(SIG_SETMASK, &act.sa_mask, NULL);
+ (void)pthread_kill(pthread_self(), SIGABRT);
+
+ usleep(TIMEOUT); /* give time for signal to happen */
+
+ /* If for some reason SIGABRT was not delivered, we exit using __builtin_trap
+ * which generates an illegal instruction on i386: <rdar://problem/8400958>
+ * and SIGTRAP on arm.
+ */
+ sigfillset(&act.sa_mask);
+ sigdelset(&act.sa_mask, SIGILL);
+ sigdelset(&act.sa_mask, SIGTRAP);
+ (void)_sigprocmask(SIG_SETMASK, &act.sa_mask, NULL);
+ __builtin_trap();
+}
+
+void
+abort_report_np(const char *fmt, ...)
+{
+ _SIMPLE_STRING s;
+ va_list ap;
+
+ if ((s = _simple_salloc()) != NULL) {
+ va_start(ap, fmt);
+ _simple_vsprintf(s, fmt, ap);
+ va_end(ap);
+ CRSetCrashLogMessage(_simple_string(s));
+ } else
+ CRSetCrashLogMessage(fmt); /* the format string is better than nothing */
+ abort();