summaryrefslogtreecommitdiff
path: root/Kernel/Tasks/CrashHandler.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'Kernel/Tasks/CrashHandler.cpp')
-rw-r--r--Kernel/Tasks/CrashHandler.cpp45
1 files changed, 45 insertions, 0 deletions
diff --git a/Kernel/Tasks/CrashHandler.cpp b/Kernel/Tasks/CrashHandler.cpp
new file mode 100644
index 0000000000..3715bbe236
--- /dev/null
+++ b/Kernel/Tasks/CrashHandler.cpp
@@ -0,0 +1,45 @@
+/*
+ * Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
+ *
+ * SPDX-License-Identifier: BSD-2-Clause
+ */
+
+#include <Kernel/Arch/CPU.h>
+#include <Kernel/Arch/RegisterState.h>
+
+#include <Kernel/Panic.h>
+#include <Kernel/Tasks/Process.h>
+#include <Kernel/Tasks/Thread.h>
+
+namespace Kernel {
+
+void handle_crash(Kernel::RegisterState const& regs, char const* description, int signal, bool out_of_memory)
+{
+ auto* current_thread = Thread::current();
+ if (!current_thread)
+ PANIC("{} with !Thread::current()", description);
+
+ auto crashed_in_kernel = regs.previous_mode() == ExecutionMode::Kernel;
+ if (!crashed_in_kernel && current_thread->has_signal_handler(signal) && !current_thread->should_ignore_signal(signal) && !current_thread->is_signal_masked(signal)) {
+ current_thread->send_urgent_signal_to_self(signal);
+ return;
+ }
+
+ auto& process = current_thread->process();
+
+ // If a process crashed while inspecting another process,
+ // make sure we switch back to the right page tables.
+ Memory::MemoryManager::enter_process_address_space(process);
+
+ dmesgln("CRASH: CPU #{} {} in {}", Processor::current_id(), description, regs.previous_mode() == ExecutionMode::Kernel ? "kernel"sv : "userspace"sv);
+ dump_registers(regs);
+
+ if (crashed_in_kernel) {
+ process.address_space().with([&](auto& space) { space->dump_regions(); });
+ PANIC("Crash in kernel");
+ }
+
+ process.crash(signal, { regs }, out_of_memory);
+}
+
+}