summaryrefslogtreecommitdiffhomepage
path: root/include/DebuggerKit/Mach.h
diff options
context:
space:
mode:
Diffstat (limited to 'include/DebuggerKit/Mach.h')
-rw-r--r--include/DebuggerKit/Mach.h153
1 files changed, 153 insertions, 0 deletions
diff --git a/include/DebuggerKit/Mach.h b/include/DebuggerKit/Mach.h
new file mode 100644
index 0000000..05dc37f
--- /dev/null
+++ b/include/DebuggerKit/Mach.h
@@ -0,0 +1,153 @@
+// Copyright 2024-2025, Amlal El Mahrouss (amlal@nekernel.org)
+// Licensed under the Apache License, Version 2.0 (See accompanying
+// file LICENSE or copy at http://www.apache.org/licenses/LICENSE-2.0)
+// Official repository: https://github.com/ne-foss-org/nectar
+
+#ifndef NECTAR_DEBUGGERKIT_MACHCONTRACT_H
+#define NECTAR_DEBUGGERKIT_MACHCONTRACT_H
+
+#ifdef DK_MACH_DEBUGGER
+
+/// @file MachDebugger.h
+/// @brief POSIX Mach debugger.
+
+#include <DebuggerKit/IDebugger.h>
+#include <filesystem>
+#include <vector>
+
+CK_IMPORT_C kern_return_t mach_vm_write(vm_map_t target_task, mach_vm_address_t address,
+ vm_offset_t data, mach_msg_type_number_t dataCnt);
+
+CK_IMPORT_C kern_return_t mach_vm_protect(vm_map_t target_task, mach_vm_address_t address,
+ mach_vm_size_t size, boolean_t set_maximum,
+ vm_prot_t new_protection);
+
+#define PTRACE_ATTACH PT_ATTACHEXC
+#define PTRACE_DETACH PT_DETACH
+#define PTRACE_POKETEXT PT_WRITE_I
+#define PTRACE_CONT PT_CONTINUE
+#define PTRACE_PEEKTEXT PT_READ_I
+
+namespace DebuggerKit::POSIX {
+/// =========================================================== ///
+/// \brief MachDebugger engine class in C++
+/// \author Amlal El Mahrouss
+/// =========================================================== ///
+class MachDebugger DK_DEBUGGER_CONTRACT {
+ public:
+ explicit MachDebugger() = default;
+ ~MachDebugger() override = default;
+
+ public:
+ MachDebugger& operator=(const MachDebugger&) = default;
+ MachDebugger(const MachDebugger&) = default;
+
+ public:
+ bool Attach(const CompilerKit::STLString& path, const CompilerKit::STLString& argv,
+ ProcessID& pid) noexcept override {
+ pid = fork();
+
+ if (pid == 0) {
+ if (argv.empty()) {
+ ptrace(PT_TRACE_ME, 0, nullptr, 0);
+ kill(getpid(), SIGSTOP);
+ }
+
+ std::vector<char*> argv_arr;
+
+ argv_arr.push_back(const_cast<char*>(path.c_str()));
+ argv_arr.push_back(const_cast<char*>(argv.c_str()));
+ argv_arr.push_back(nullptr);
+
+ execv(path.c_str(), argv_arr.data());
+
+ _exit(1);
+ }
+
+ m_path = path;
+ mPid = pid;
+
+ pid = this->mPid;
+
+ return true;
+ }
+
+ void SetPath(const CompilerKit::STLString& path) noexcept {
+ if (path.empty()) {
+ return;
+ }
+
+ m_path = path;
+ }
+
+ bool BreakAt(const CompilerKit::STLString& symbol) noexcept override {
+ if (!m_path.empty() && std::filesystem::exists(m_path) &&
+ std::filesystem::is_regular_file(m_path)) {
+ auto handle = dlopen(m_path.c_str(), RTLD_LAZY);
+
+ if (handle == nullptr) {
+ return false;
+ }
+
+ auto addr = dlsym(handle, symbol.c_str());
+
+ if (addr == nullptr) {
+ return false;
+ }
+
+ task_read_t task;
+ task_for_pid(mach_task_self(), mPid, &task);
+
+#ifdef __x86_64__
+ uint32_t brk_inst = 0xD43E0000;
+
+ mach_vm_protect(task, (mach_vm_address_t) addr, sizeof(uint32_t), false,
+ VM_PROT_READ | VM_PROT_WRITE | VM_PROT_EXECUTE);
+
+ mach_vm_write(task, (mach_vm_address_t) addr, (vm_offset_t) &brk_inst, sizeof(addr));
+#endif
+
+ return true;
+ }
+
+ return false;
+ }
+
+ bool Break() noexcept override {
+ task_read_t task;
+ task_for_pid(mach_task_self(), mPid, &task);
+
+ kern_return_t ret = task_suspend(task);
+
+ return ret == KERN_SUCCESS;
+ }
+
+ bool Continue() noexcept override {
+ task_read_t task;
+ task_for_pid(mach_task_self(), mPid, &task);
+
+ kern_return_t ret = task_resume(task);
+
+ return ret == KERN_SUCCESS;
+ }
+
+ bool Detach() noexcept override {
+ this->Continue();
+
+ task_read_t task;
+ task_for_pid(mach_task_self(), mPid, &task);
+
+ kern_return_t kr = mach_port_deallocate(mach_task_self(), task);
+
+ return kr = KERN_SUCCESS;
+ }
+
+ private:
+ ProcessID mPid{0};
+ CompilerKit::STLString m_path;
+};
+} // namespace DebuggerKit::POSIX
+
+#endif // DK_MACH_DEBUGGER
+
+#endif // NECTAR_DEBUGGERKIT_MACHCONTRACT_H