From cd59ab547816a62e84927b85baf9fe6450d1dc45 Mon Sep 17 00:00:00 2001 From: Jeremy <51220084+jeremy-rifkin@users.noreply.github.com> Date: Sun, 19 Nov 2023 23:46:49 -0600 Subject: [PATCH] Added signal tracing demo --- CMakeLists.txt | 4 +++ test/signal_demo.cpp | 81 ++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 85 insertions(+) create mode 100644 test/signal_demo.cpp diff --git a/CMakeLists.txt b/CMakeLists.txt index 7535ab3..7cdbaa7 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -494,6 +494,10 @@ if(CPPTRACE_BUILD_TESTING) endif() if(UNIX) + add_executable(signal_demo test/signal_demo.cpp) + target_compile_features(signal_demo PRIVATE cxx_std_11) + target_link_libraries(signal_demo PRIVATE ${target_name}) + add_executable(signal_tracer test/signal_tracer.cpp) target_compile_features(signal_tracer PRIVATE cxx_std_11) target_link_libraries(signal_tracer PRIVATE ${target_name}) diff --git a/test/signal_demo.cpp b/test/signal_demo.cpp new file mode 100644 index 0000000..8c6c082 --- /dev/null +++ b/test/signal_demo.cpp @@ -0,0 +1,81 @@ +#include + +#include +#include +#include +#include + +#include +#include +#include +#include +#include + +void trace() { + *(char*)0 = 2; +} + +void bar() { + trace(); +} + +void foo() { + bar(); +} + +struct pipe_t { + union { + struct { + int read_end; + int write_end; + }; + int data[2]; + }; +}; +static_assert(sizeof(pipe_t) == 2 * sizeof(int), "Unexpected struct packing"); + +void handler(int signo, siginfo_t* info, void* context) { + const char* message = "SIGSEGV ocurred:\n"; + write(STDERR_FILENO, message, strlen(message)); + cpptrace::frame_ptr buffer[100]; + std::size_t count = cpptrace::safe_generate_raw_trace(buffer, 100); + pipe_t input_pipe; + pipe(input_pipe.data); + const pid_t pid = fork(); + if(pid == -1) { return; } + if(pid == 0) { // child + dup2(input_pipe.read_end, STDIN_FILENO); + close(input_pipe.read_end); + close(input_pipe.write_end); + execl( + "signal_tracer", + "signal_tracer", + nullptr + ); + _exit(1); + } + for(std::size_t i = 0; i < count; i++) { + cpptrace::minimal_object_frame frame; + cpptrace::get_minimal_object_frame(buffer[i], &frame); + write(input_pipe.write_end, &frame, sizeof(frame)); + } + close(input_pipe.read_end); + close(input_pipe.write_end); + waitpid(pid, nullptr, 0); + _exit(1); +} + +int main() { + cpptrace::absorb_trace_exceptions(false); + cpptrace::register_terminate_handler(); + + struct sigaction action = { 0 }; + action.sa_flags = 0; + action.sa_sigaction = &handler; + if (sigaction(SIGSEGV, &action, NULL) == -1) { + perror("sigaction"); + exit(EXIT_FAILURE); + } + + foo(); +}