}
#endif
-static void ggml_print_backtrace(void) {
+void ggml_print_backtrace(void) {
const char * GGML_NO_BACKTRACE = getenv("GGML_NO_BACKTRACE");
if (GGML_NO_BACKTRACE) {
return;
const int parent_pid = getpid();
const int child_pid = fork();
if (child_pid < 0) { // error
+#if defined(__linux__)
+ close(lock[1]);
+ close(lock[0]);
+#endif
return;
} else if (child_pid == 0) { // child
char attach[32];
#if defined(__linux__)
close(lock[1]);
(void) !read(lock[0], lock, 1);
+ close(lock[0]);
#endif
// try gdb
execlp("gdb", "gdb", "--batch",
abort();
}
+// ggml_print_backtrace is registered with std::set_terminate by ggml.cpp
+
//
// logging
//
--- /dev/null
+#include "ggml-impl.h"
+
+#include <cstdlib>
+#include <exception>
+
+static std::terminate_handler previous_terminate_handler;
+
+GGML_NORETURN static void ggml_uncaught_exception() {
+ ggml_print_backtrace();
+ if (previous_terminate_handler) {
+ previous_terminate_handler();
+ }
+ abort(); // unreachable unless previous_terminate_handler was nullptr
+}
+
+static bool ggml_uncaught_exception_init = []{
+ const char * GGML_NO_BACKTRACE = getenv("GGML_NO_BACKTRACE");
+ if (GGML_NO_BACKTRACE) {
+ return false;
+ }
+ const auto prev{std::get_terminate()};
+ GGML_ASSERT(prev != ggml_uncaught_exception);
+ previous_terminate_handler = prev;
+ std::set_terminate(ggml_uncaught_exception);
+ return true;
+}();