https://github.com/Jlalond updated https://github.com/llvm/llvm-project/pull/140150
>From 86ec6c076b9cf8e7afeb7d6bb0e334434f6e0d9e Mon Sep 17 00:00:00 2001 From: Jacob Lalonde <jalalo...@fb.com> Date: Thu, 15 May 2025 13:57:11 -0700 Subject: [PATCH 1/7] Update ThreadElfCore --- lldb/include/lldb/Target/UnixSignals.h | 6 ++++-- .../Plugins/Process/Utility/LinuxSignals.cpp | 17 ++++++++++++++--- .../Plugins/Process/elf-core/ThreadElfCore.cpp | 10 +++++++--- .../Plugins/Process/elf-core/ThreadElfCore.h | 6 ++++++ lldb/source/Target/UnixSignals.cpp | 9 +++++++-- 5 files changed, 38 insertions(+), 10 deletions(-) diff --git a/lldb/include/lldb/Target/UnixSignals.h b/lldb/include/lldb/Target/UnixSignals.h index b3605ccefddbe..a1807d69f329b 100644 --- a/lldb/include/lldb/Target/UnixSignals.h +++ b/lldb/include/lldb/Target/UnixSignals.h @@ -36,7 +36,9 @@ class UnixSignals { std::optional<int32_t> code = std::nullopt, std::optional<lldb::addr_t> addr = std::nullopt, std::optional<lldb::addr_t> lower = std::nullopt, - std::optional<lldb::addr_t> upper = std::nullopt) const; + std::optional<lldb::addr_t> upper = std::nullopt, + std::optional<uint32_t> pid = std::nullopt, + std::optional<uint32_t> uid = std::nullopt) const; bool SignalIsValid(int32_t signo) const; @@ -105,7 +107,7 @@ class UnixSignals { llvm::StringRef description, llvm::StringRef alias = llvm::StringRef()); - enum SignalCodePrintOption { None, Address, Bounds }; + enum SignalCodePrintOption { None, Address, Bounds, Sender }; // Instead of calling this directly, use a ADD_SIGCODE macro to get compile // time checks when on the native platform. diff --git a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp index 9c4fe55147a28..25d4e4609bbb8 100644 --- a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp +++ b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp @@ -38,6 +38,17 @@ #define ADD_SIGCODE(signal_name, signal_value, code_name, code_value, ...) \ AddSignalCode(signal_value, code_value, __VA_ARGS__) #endif /* if defined(__linux__) && !defined(__mips__) */ +// See siginfo.h in the Linux Kernel, these codes can be sent for any signal. +#define ADD_LINUX_SIGNAL(signo, name, ...) \ + AddSignal(signo, name, __VA_ARGS__); \ + ADD_SIGCODE(signo, signo, SI_QUEUE, -1, "sent by sigqueue"); \ + ADD_SIGCODE(signo, signo, SI_TIMER, -2, "sent by timer expiration"); \ + ADD_SIGCODE(signo, signo, SI_MESGQ, -3, "sent by real time mesq state change"); \ + ADD_SIGCODE(signo, signo, SI_ASYNCIO, -4, "sent by AIO completion"); \ + ADD_SIGCODE(signo, signo, SI_SIGIO, -5, "sent by queued SIGIO"); \ + ADD_SIGCODE(signo, signo, SI_TKILL, -6, "sent by tkill system call"); \ + ADD_SIGCODE(signo, signo, SI_DETHREAD, -7, "sent by execve() killing subsidiary threads"); \ + ADD_SIGCODE(signo, signo, SI_ASYNCNL, -60, "sent by glibc async name lookup completion"); using namespace lldb_private; @@ -46,9 +57,9 @@ LinuxSignals::LinuxSignals() : UnixSignals() { Reset(); } void LinuxSignals::Reset() { m_signals.clear(); // clang-format off - // SIGNO NAME SUPPRESS STOP NOTIFY DESCRIPTION - // ====== ============== ======== ====== ====== =================================================== - AddSignal(1, "SIGHUP", false, true, true, "hangup"); + // SIGNO NAME SUPPRESS STOP NOTIFY DESCRIPTION + // ====== ============== ======== ====== ====== =================================================== + ADD_LINUX_SIGNAL(1, "SIGHUP", false, true, true, "hangup"); AddSignal(2, "SIGINT", true, true, true, "interrupt"); AddSignal(3, "SIGQUIT", false, true, true, "quit"); diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp index a0cd0ee5025bd..267879a473463 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp @@ -584,9 +584,13 @@ Status ELFLinuxSigInfo::Parse(const DataExtractor &data, const ArchSpec &arch, // 64b ELF have a 4 byte pad. if (data.GetAddressByteSize() == 8) offset += 4; - // Not every stop signal has a valid address, but that will get resolved in - // the unix_signals.GetSignalDescription() call below. - if (unix_signals.GetShouldStop(si_signo)) { + + if (si_code < 0) { + sigfault.kill._pid = data.GetU32(&offset); + sigfault.kill._uid = data.GetU32(&offset); + } else if (unix_signals.GetShouldStop(si_signo)) { + // Not every stop signal has a valid address, but that will get resolved in + // the unix_signals.GetSignalDescription() call below. // Instead of memcpy we call all these individually as the extractor will // handle endianness for us. sigfault.si_addr = data.GetAddress(&offset); diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h index 6f8d41351a6bf..2cbf794c2b5b1 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h @@ -96,6 +96,12 @@ struct ELFLinuxSigInfo { /* used when si_code=SEGV_PKUERR */ uint32_t _pkey; } bounds; + + // We need this for all the generic signals. + struct { + uint32_t _pid; /* sender's pid */ + uint32_t _uid; /* sender's uid */ + } _kill; } sigfault; enum SigInfoNoteType : uint8_t { eUnspecified, eNT_SIGINFO }; diff --git a/lldb/source/Target/UnixSignals.cpp b/lldb/source/Target/UnixSignals.cpp index da661003925c7..a5dbfd029410a 100644 --- a/lldb/source/Target/UnixSignals.cpp +++ b/lldb/source/Target/UnixSignals.cpp @@ -141,7 +141,9 @@ std::string UnixSignals::GetSignalDescription(int32_t signo, std::optional<int32_t> code, std::optional<lldb::addr_t> addr, std::optional<lldb::addr_t> lower, - std::optional<lldb::addr_t> upper) const { + std::optional<lldb::addr_t> upper, + std::optional<uint32_t> pid, + std::optional<uint32_t> uid) const { std::string str; collection::const_iterator pos = m_signals.find(signo); @@ -180,6 +182,10 @@ UnixSignals::GetSignalDescription(int32_t signo, std::optional<int32_t> code, strm << sc.m_description.str(); break; + case SignalCodePrintOption::Sender: + if (pid && uid) + strm << " (sender pid=" << *pid << ", uid=" << *uid << ")"; + break; } str += strm.str(); } @@ -397,4 +403,3 @@ bool UnixSignals::ResetSignal(int32_t signo, bool reset_stop, (*elem).second.Reset(reset_stop, reset_notify, reset_suppress); return true; } - >From 6d8a30cc38816f661cd3126613987ccbde39ab05 Mon Sep 17 00:00:00 2001 From: Jacob Lalonde <jalalo...@fb.com> Date: Thu, 15 May 2025 14:33:15 -0700 Subject: [PATCH 2/7] Add sender option to LinuxSignals.cpp --- .../Plugins/Process/Utility/LinuxSignals.cpp | 142 +++++++++--------- .../Process/elf-core/ThreadElfCore.cpp | 32 ++-- .../Plugins/Process/elf-core/ThreadElfCore.h | 44 +++--- 3 files changed, 111 insertions(+), 107 deletions(-) diff --git a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp index 25d4e4609bbb8..da0abf5a1f471 100644 --- a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp +++ b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp @@ -41,14 +41,14 @@ // See siginfo.h in the Linux Kernel, these codes can be sent for any signal. #define ADD_LINUX_SIGNAL(signo, name, ...) \ AddSignal(signo, name, __VA_ARGS__); \ - ADD_SIGCODE(signo, signo, SI_QUEUE, -1, "sent by sigqueue"); \ - ADD_SIGCODE(signo, signo, SI_TIMER, -2, "sent by timer expiration"); \ - ADD_SIGCODE(signo, signo, SI_MESGQ, -3, "sent by real time mesq state change"); \ - ADD_SIGCODE(signo, signo, SI_ASYNCIO, -4, "sent by AIO completion"); \ - ADD_SIGCODE(signo, signo, SI_SIGIO, -5, "sent by queued SIGIO"); \ - ADD_SIGCODE(signo, signo, SI_TKILL, -6, "sent by tkill system call"); \ - ADD_SIGCODE(signo, signo, SI_DETHREAD, -7, "sent by execve() killing subsidiary threads"); \ - ADD_SIGCODE(signo, signo, SI_ASYNCNL, -60, "sent by glibc async name lookup completion"); + ADD_SIGCODE(signo, signo, SI_QUEUE, -1, "sent by sigqueue", SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_TIMER, -2, "sent by timer expiration", SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_MESGQ, -3, "sent by real time mesq state change", SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_ASYNCIO, -4, "sent by AIO completion", SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_SIGIO, -5, "sent by queued SIGIO", SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_TKILL, -6, "sent by tkill system call", SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_DETHREAD, -7, "sent by execve() killing subsidiary threads", SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_ASYNCNL, -60, "sent by glibc async name lookup completion", SignalCodePrintOption::Sender); using namespace lldb_private; @@ -60,10 +60,10 @@ void LinuxSignals::Reset() { // SIGNO NAME SUPPRESS STOP NOTIFY DESCRIPTION // ====== ============== ======== ====== ====== =================================================== ADD_LINUX_SIGNAL(1, "SIGHUP", false, true, true, "hangup"); - AddSignal(2, "SIGINT", true, true, true, "interrupt"); - AddSignal(3, "SIGQUIT", false, true, true, "quit"); + ADD_LINUX_SIGNAL(2, "SIGINT", true, true, true, "interrupt"); + ADD_LINUX_SIGNAL(3, "SIGQUIT", false, true, true, "quit"); - AddSignal(4, "SIGILL", false, true, true, "illegal instruction"); + ADD_LINUX_SIGNAL(4, "SIGILL", false, true, true, "illegal instruction"); ADD_SIGCODE(SIGILL, 4, ILL_ILLOPC, 1, "illegal opcode"); ADD_SIGCODE(SIGILL, 4, ILL_ILLOPN, 2, "illegal operand"); ADD_SIGCODE(SIGILL, 4, ILL_ILLADR, 3, "illegal addressing mode"); @@ -73,15 +73,15 @@ void LinuxSignals::Reset() { ADD_SIGCODE(SIGILL, 4, ILL_COPROC, 7, "coprocessor error"); ADD_SIGCODE(SIGILL, 4, ILL_BADSTK, 8, "internal stack error"); - AddSignal(5, "SIGTRAP", true, true, true, "trace trap (not reset when caught)"); - AddSignal(6, "SIGABRT", false, true, true, "abort()/IOT trap", "SIGIOT"); + ADD_LINUX_SIGNAL(5, "SIGTRAP", true, true, true, "trace trap (not reset when caught)"); + ADD_LINUX_SIGNAL(6, "SIGABRT", false, true, true, "abort()/IOT trap", "SIGIOT"); - AddSignal(7, "SIGBUS", false, true, true, "bus error"); + ADD_LINUX_SIGNAL(7, "SIGBUS", false, true, true, "bus error"); ADD_SIGCODE(SIGBUS, 7, BUS_ADRALN, 1, "illegal alignment"); ADD_SIGCODE(SIGBUS, 7, BUS_ADRERR, 2, "illegal address"); ADD_SIGCODE(SIGBUS, 7, BUS_OBJERR, 3, "hardware error"); - AddSignal(8, "SIGFPE", false, true, true, "floating point exception"); + ADD_LINUX_SIGNAL(8, "SIGFPE", false, true, true, "floating point exception"); ADD_SIGCODE(SIGFPE, 8, FPE_INTDIV, 1, "integer divide by zero"); ADD_SIGCODE(SIGFPE, 8, FPE_INTOVF, 2, "integer overflow"); ADD_SIGCODE(SIGFPE, 8, FPE_FLTDIV, 3, "floating point divide by zero"); @@ -91,10 +91,10 @@ void LinuxSignals::Reset() { ADD_SIGCODE(SIGFPE, 8, FPE_FLTINV, 7, "floating point invalid operation"); ADD_SIGCODE(SIGFPE, 8, FPE_FLTSUB, 8, "subscript out of range"); - AddSignal(9, "SIGKILL", false, true, true, "kill"); - AddSignal(10, "SIGUSR1", false, true, true, "user defined signal 1"); + ADD_LINUX_SIGNAL(9, "SIGKILL", false, true, true, "kill"); + ADD_LINUX_SIGNAL(10, "SIGUSR1", false, true, true, "user defined signal 1"); - AddSignal(11, "SIGSEGV", false, true, true, "segmentation violation"); + ADD_LINUX_SIGNAL(11, "SIGSEGV", false, true, true, "segmentation violation"); ADD_SIGCODE(SIGSEGV, 11, SEGV_MAPERR, 1, "address not mapped to object", SignalCodePrintOption::Address); ADD_SIGCODE(SIGSEGV, 11, SEGV_ACCERR, 2, "invalid permissions for mapped object", SignalCodePrintOption::Address); ADD_SIGCODE(SIGSEGV, 11, SEGV_BNDERR, 3, "failed address bounds checks", SignalCodePrintOption::Bounds); @@ -105,58 +105,58 @@ void LinuxSignals::Reset() { // codes. One way to get this is via unaligned SIMD loads. Treat it as invalid address. ADD_SIGCODE(SIGSEGV, 11, SI_KERNEL, 0x80, "invalid address", SignalCodePrintOption::Address); - AddSignal(12, "SIGUSR2", false, true, true, "user defined signal 2"); - AddSignal(13, "SIGPIPE", false, true, true, "write to pipe with reading end closed"); - AddSignal(14, "SIGALRM", false, false, false, "alarm"); - AddSignal(15, "SIGTERM", false, true, true, "termination requested"); - AddSignal(16, "SIGSTKFLT", false, true, true, "stack fault"); - AddSignal(17, "SIGCHLD", false, false, true, "child status has changed", "SIGCLD"); - AddSignal(18, "SIGCONT", false, false, true, "process continue"); - AddSignal(19, "SIGSTOP", true, true, true, "process stop"); - AddSignal(20, "SIGTSTP", false, true, true, "tty stop"); - AddSignal(21, "SIGTTIN", false, true, true, "background tty read"); - AddSignal(22, "SIGTTOU", false, true, true, "background tty write"); - AddSignal(23, "SIGURG", false, true, true, "urgent data on socket"); - AddSignal(24, "SIGXCPU", false, true, true, "CPU resource exceeded"); - AddSignal(25, "SIGXFSZ", false, true, true, "file size limit exceeded"); - AddSignal(26, "SIGVTALRM", false, true, true, "virtual time alarm"); - AddSignal(27, "SIGPROF", false, false, false, "profiling time alarm"); - AddSignal(28, "SIGWINCH", false, true, true, "window size changes"); - AddSignal(29, "SIGIO", false, true, true, "input/output ready/Pollable event", "SIGPOLL"); - AddSignal(30, "SIGPWR", false, true, true, "power failure"); - AddSignal(31, "SIGSYS", false, true, true, "invalid system call"); - AddSignal(32, "SIG32", false, false, false, "threading library internal signal 1"); - AddSignal(33, "SIG33", false, false, false, "threading library internal signal 2"); - AddSignal(34, "SIGRTMIN", false, false, false, "real time signal 0"); - AddSignal(35, "SIGRTMIN+1", false, false, false, "real time signal 1"); - AddSignal(36, "SIGRTMIN+2", false, false, false, "real time signal 2"); - AddSignal(37, "SIGRTMIN+3", false, false, false, "real time signal 3"); - AddSignal(38, "SIGRTMIN+4", false, false, false, "real time signal 4"); - AddSignal(39, "SIGRTMIN+5", false, false, false, "real time signal 5"); - AddSignal(40, "SIGRTMIN+6", false, false, false, "real time signal 6"); - AddSignal(41, "SIGRTMIN+7", false, false, false, "real time signal 7"); - AddSignal(42, "SIGRTMIN+8", false, false, false, "real time signal 8"); - AddSignal(43, "SIGRTMIN+9", false, false, false, "real time signal 9"); - AddSignal(44, "SIGRTMIN+10", false, false, false, "real time signal 10"); - AddSignal(45, "SIGRTMIN+11", false, false, false, "real time signal 11"); - AddSignal(46, "SIGRTMIN+12", false, false, false, "real time signal 12"); - AddSignal(47, "SIGRTMIN+13", false, false, false, "real time signal 13"); - AddSignal(48, "SIGRTMIN+14", false, false, false, "real time signal 14"); - AddSignal(49, "SIGRTMIN+15", false, false, false, "real time signal 15"); - AddSignal(50, "SIGRTMAX-14", false, false, false, "real time signal 16"); // switching to SIGRTMAX-xxx to match "kill -l" output - AddSignal(51, "SIGRTMAX-13", false, false, false, "real time signal 17"); - AddSignal(52, "SIGRTMAX-12", false, false, false, "real time signal 18"); - AddSignal(53, "SIGRTMAX-11", false, false, false, "real time signal 19"); - AddSignal(54, "SIGRTMAX-10", false, false, false, "real time signal 20"); - AddSignal(55, "SIGRTMAX-9", false, false, false, "real time signal 21"); - AddSignal(56, "SIGRTMAX-8", false, false, false, "real time signal 22"); - AddSignal(57, "SIGRTMAX-7", false, false, false, "real time signal 23"); - AddSignal(58, "SIGRTMAX-6", false, false, false, "real time signal 24"); - AddSignal(59, "SIGRTMAX-5", false, false, false, "real time signal 25"); - AddSignal(60, "SIGRTMAX-4", false, false, false, "real time signal 26"); - AddSignal(61, "SIGRTMAX-3", false, false, false, "real time signal 27"); - AddSignal(62, "SIGRTMAX-2", false, false, false, "real time signal 28"); - AddSignal(63, "SIGRTMAX-1", false, false, false, "real time signal 29"); - AddSignal(64, "SIGRTMAX", false, false, false, "real time signal 30"); + ADD_LINUX_SIGNAL(12, "SIGUSR2", false, true, true, "user defined signal 2"); + ADD_LINUX_SIGNAL(13, "SIGPIPE", false, true, true, "write to pipe with reading end closed"); + ADD_LINUX_SIGNAL(14, "SIGALRM", false, false, false, "alarm"); + ADD_LINUX_SIGNAL(15, "SIGTERM", false, true, true, "termination requested"); + ADD_LINUX_SIGNAL(16, "SIGSTKFLT", false, true, true, "stack fault"); + ADD_LINUX_SIGNAL(17, "SIGCHLD", false, false, true, "child status has changed", "SIGCLD"); + ADD_LINUX_SIGNAL(18, "SIGCONT", false, false, true, "process continue"); + ADD_LINUX_SIGNAL(19, "SIGSTOP", true, true, true, "process stop"); + ADD_LINUX_SIGNAL(20, "SIGTSTP", false, true, true, "tty stop"); + ADD_LINUX_SIGNAL(21, "SIGTTIN", false, true, true, "background tty read"); + ADD_LINUX_SIGNAL(22, "SIGTTOU", false, true, true, "background tty write"); + ADD_LINUX_SIGNAL(23, "SIGURG", false, true, true, "urgent data on socket"); + ADD_LINUX_SIGNAL(24, "SIGXCPU", false, true, true, "CPU resource exceeded"); + ADD_LINUX_SIGNAL(25, "SIGXFSZ", false, true, true, "file size limit exceeded"); + ADD_LINUX_SIGNAL(26, "SIGVTALRM", false, true, true, "virtual time alarm"); + ADD_LINUX_SIGNAL(27, "SIGPROF", false, false, false, "profiling time alarm"); + ADD_LINUX_SIGNAL(28, "SIGWINCH", false, true, true, "window size changes"); + ADD_LINUX_SIGNAL(29, "SIGIO", false, true, true, "input/output ready/Pollable event", "SIGPOLL"); + ADD_LINUX_SIGNAL(30, "SIGPWR", false, true, true, "power failure"); + ADD_LINUX_SIGNAL(31, "SIGSYS", false, true, true, "invalid system call"); + ADD_LINUX_SIGNAL(32, "SIG32", false, false, false, "threading library internal signal 1"); + ADD_LINUX_SIGNAL(33, "SIG33", false, false, false, "threading library internal signal 2"); + ADD_LINUX_SIGNAL(34, "SIGRTMIN", false, false, false, "real time signal 0"); + ADD_LINUX_SIGNAL(35, "SIGRTMIN+1", false, false, false, "real time signal 1"); + ADD_LINUX_SIGNAL(36, "SIGRTMIN+2", false, false, false, "real time signal 2"); + ADD_LINUX_SIGNAL(37, "SIGRTMIN+3", false, false, false, "real time signal 3"); + ADD_LINUX_SIGNAL(38, "SIGRTMIN+4", false, false, false, "real time signal 4"); + ADD_LINUX_SIGNAL(39, "SIGRTMIN+5", false, false, false, "real time signal 5"); + ADD_LINUX_SIGNAL(40, "SIGRTMIN+6", false, false, false, "real time signal 6"); + ADD_LINUX_SIGNAL(41, "SIGRTMIN+7", false, false, false, "real time signal 7"); + ADD_LINUX_SIGNAL(42, "SIGRTMIN+8", false, false, false, "real time signal 8"); + ADD_LINUX_SIGNAL(43, "SIGRTMIN+9", false, false, false, "real time signal 9"); + ADD_LINUX_SIGNAL(44, "SIGRTMIN+10", false, false, false, "real time signal 10"); + ADD_LINUX_SIGNAL(45, "SIGRTMIN+11", false, false, false, "real time signal 11"); + ADD_LINUX_SIGNAL(46, "SIGRTMIN+12", false, false, false, "real time signal 12"); + ADD_LINUX_SIGNAL(47, "SIGRTMIN+13", false, false, false, "real time signal 13"); + ADD_LINUX_SIGNAL(48, "SIGRTMIN+14", false, false, false, "real time signal 14"); + ADD_LINUX_SIGNAL(49, "SIGRTMIN+15", false, false, false, "real time signal 15"); + ADD_LINUX_SIGNAL(50, "SIGRTMAX-14", false, false, false, "real time signal 16"); // switching to SIGRTMAX-xxx to match "kill -l" output + ADD_LINUX_SIGNAL(51, "SIGRTMAX-13", false, false, false, "real time signal 17"); + ADD_LINUX_SIGNAL(52, "SIGRTMAX-12", false, false, false, "real time signal 18"); + ADD_LINUX_SIGNAL(53, "SIGRTMAX-11", false, false, false, "real time signal 19"); + ADD_LINUX_SIGNAL(54, "SIGRTMAX-10", false, false, false, "real time signal 20"); + ADD_LINUX_SIGNAL(55, "SIGRTMAX-9", false, false, false, "real time signal 21"); + ADD_LINUX_SIGNAL(56, "SIGRTMAX-8", false, false, false, "real time signal 22"); + ADD_LINUX_SIGNAL(57, "SIGRTMAX-7", false, false, false, "real time signal 23"); + ADD_LINUX_SIGNAL(58, "SIGRTMAX-6", false, false, false, "real time signal 24"); + ADD_LINUX_SIGNAL(59, "SIGRTMAX-5", false, false, false, "real time signal 25"); + ADD_LINUX_SIGNAL(60, "SIGRTMAX-4", false, false, false, "real time signal 26"); + ADD_LINUX_SIGNAL(61, "SIGRTMAX-3", false, false, false, "real time signal 27"); + ADD_LINUX_SIGNAL(62, "SIGRTMAX-2", false, false, false, "real time signal 28"); + ADD_LINUX_SIGNAL(63, "SIGRTMAX-1", false, false, false, "real time signal 29"); + ADD_LINUX_SIGNAL(64, "SIGRTMAX", false, false, false, "real time signal 30"); // clang-format on } diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp index 267879a473463..ce0f65cd9f14c 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp @@ -586,24 +586,24 @@ Status ELFLinuxSigInfo::Parse(const DataExtractor &data, const ArchSpec &arch, offset += 4; if (si_code < 0) { - sigfault.kill._pid = data.GetU32(&offset); - sigfault.kill._uid = data.GetU32(&offset); + sifields.kill.pid = data.GetU32(&offset); + sifields.kill.uid = data.GetU32(&offset); } else if (unix_signals.GetShouldStop(si_signo)) { // Not every stop signal has a valid address, but that will get resolved in // the unix_signals.GetSignalDescription() call below. // Instead of memcpy we call all these individually as the extractor will // handle endianness for us. - sigfault.si_addr = data.GetAddress(&offset); - sigfault.si_addr_lsb = data.GetU16(&offset); - if (data.GetByteSize() - offset >= sizeof(sigfault.bounds)) { - sigfault.bounds._addr_bnd._lower = data.GetAddress(&offset); - sigfault.bounds._addr_bnd._upper = data.GetAddress(&offset); - sigfault.bounds._pkey = data.GetU32(&offset); + sifields.sigfault.si_addr = data.GetAddress(&offset); + sifields.sigfault.si_addr_lsb = data.GetU16(&offset); + if (data.GetByteSize() - offset >= sizeof(sifields.sigfault.bounds)) { + sifields.sigfault.bounds._addr_bnd._lower = data.GetAddress(&offset); + sifields.sigfault.bounds._addr_bnd._upper = data.GetAddress(&offset); + sifields.sigfault.bounds._pkey = data.GetU32(&offset); } else { // Set these to 0 so we don't use bogus data for the description. - sigfault.bounds._addr_bnd._lower = 0; - sigfault.bounds._addr_bnd._upper = 0; - sigfault.bounds._pkey = 0; + sifields.sigfault.bounds._addr_bnd._lower = 0; + sifields.sigfault.bounds._addr_bnd._upper = 0; + sifields.sigfault.bounds._pkey = 0; } } @@ -613,13 +613,15 @@ Status ELFLinuxSigInfo::Parse(const DataExtractor &data, const ArchSpec &arch, std::string ELFLinuxSigInfo::GetDescription( const lldb_private::UnixSignals &unix_signals) const { if (unix_signals.GetShouldStop(si_signo) && note_type == eNT_SIGINFO) { - if (sigfault.bounds._addr_bnd._upper != 0) + if (si_code < 0) + return unix_signals.GetSignalDescription(si_signo, si_code, std::nullopt, std::nullopt, std::nullopt, sifields.kill.pid, sifields.kill.uid); + else if (sifields.sigfault.bounds._addr_bnd._upper != 0) return unix_signals.GetSignalDescription( - si_signo, si_code, sigfault.si_addr, sigfault.bounds._addr_bnd._lower, - sigfault.bounds._addr_bnd._upper); + si_signo, si_code, sifields.sigfault.si_addr, sifields.sigfault.bounds._addr_bnd._lower, + sifields.sigfault.bounds._addr_bnd._upper); else return unix_signals.GetSignalDescription(si_signo, si_code, - sigfault.si_addr); + sifields.sigfault.si_addr); } // This looks weird, but there is an existing pattern where we don't pass a diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h index 2cbf794c2b5b1..2c254b4b522e9 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h @@ -82,27 +82,29 @@ struct ELFLinuxSigInfo { int32_t si_signo; // Order matters for the first 3. int32_t si_errno; int32_t si_code; - // Copied from siginfo_t so we don't have to include signal.h on non 'Nix - // builds. Slight modifications to ensure no 32b vs 64b differences. - struct alignas(8) { - lldb::addr_t si_addr; /* faulting insn/memory ref. */ - int16_t si_addr_lsb; /* Valid LSB of the reported address. */ - union { - /* used when si_code=SEGV_BNDERR */ - struct { - lldb::addr_t _lower; - lldb::addr_t _upper; - } _addr_bnd; - /* used when si_code=SEGV_PKUERR */ - uint32_t _pkey; - } bounds; - - // We need this for all the generic signals. - struct { - uint32_t _pid; /* sender's pid */ - uint32_t _uid; /* sender's uid */ - } _kill; - } sigfault; + union alignas(8) { + struct alignas(8) { + uint32_t pid; /* sender's pid */ + uint32_t uid; /* sender's uid */ + } kill; + // Copied from siginfo_t so we don't have to include signal.h on non 'Nix + // builds. Slight modifications to ensure no 32b vs 64b differences. + struct alignas(8) { + lldb::addr_t si_addr; /* faulting insn/memory ref. */ + int16_t si_addr_lsb; /* Valid LSB of the reported address. */ + union { + /* used when si_code=SEGV_BNDERR */ + struct { + lldb::addr_t _lower; + lldb::addr_t _upper; + } _addr_bnd; + /* used when si_code=SEGV_PKUERR */ + uint32_t _pkey; + } bounds; + + // We need this for all the generic signals. + } sigfault; + } sifields; enum SigInfoNoteType : uint8_t { eUnspecified, eNT_SIGINFO }; SigInfoNoteType note_type; >From 9e352e71666628c7e81f520e8619779cc7a6e150 Mon Sep 17 00:00:00 2001 From: Jacob Lalonde <jalalo...@fb.com> Date: Thu, 15 May 2025 14:46:40 -0700 Subject: [PATCH 3/7] Add test for sender case --- lldb/unittests/Signals/UnixSignalsTest.cpp | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/lldb/unittests/Signals/UnixSignalsTest.cpp b/lldb/unittests/Signals/UnixSignalsTest.cpp index 9a7d9afc2b185..825cc5ea6a782 100644 --- a/lldb/unittests/Signals/UnixSignalsTest.cpp +++ b/lldb/unittests/Signals/UnixSignalsTest.cpp @@ -27,6 +27,7 @@ class TestSignals : public UnixSignals { AddSignalCode(16, 2, "SIG16 with a fault address", SignalCodePrintOption::Address); AddSignalCode(16, 3, "bounds violation", SignalCodePrintOption::Bounds); + AddSignalCode(16, -6, "sent by tkill system call", SignalCodePrintOption::Sender); } }; @@ -124,6 +125,10 @@ TEST(UnixSignalsTest, GetAsString) { // No address given just print the code description. ASSERT_EQ("SIG16: SIG16 with a fault address", signals.GetSignalDescription(16, 2)); + // TKill, but with no sender + ASSERT_EQ("SIG16: sent by tkill system call", signals.GetSignalDescription(16, -6, 0xCAFEF00D)); + // TKill, but with no sender + ASSERT_EQ("SIG16: sent by tkill system call (sender pid=912, uid=99)", signals.GetSignalDescription(16, -6, 0xCAFEF00D, std::nullopt, std::nullopt, 912, 99)); const char *expected = "SIG16: bounds violation"; // Must pass all needed info to get full output. >From 1a1b9fa48bead59a15291a58814e363cd4412f55 Mon Sep 17 00:00:00 2001 From: Jacob Lalonde <jalalo...@fb.com> Date: Thu, 15 May 2025 15:03:32 -0700 Subject: [PATCH 4/7] run GCF --- .../Plugins/Process/Utility/LinuxSignals.cpp | 31 +++++++++++++------ .../Process/elf-core/ThreadElfCore.cpp | 17 +++++----- .../Plugins/Process/elf-core/ThreadElfCore.h | 4 +-- lldb/source/Target/UnixSignals.cpp | 12 +++---- lldb/unittests/Signals/UnixSignalsTest.cpp | 10 ++++-- 5 files changed, 45 insertions(+), 29 deletions(-) diff --git a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp index da0abf5a1f471..76c32e376eb4b 100644 --- a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp +++ b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp @@ -39,16 +39,27 @@ AddSignalCode(signal_value, code_value, __VA_ARGS__) #endif /* if defined(__linux__) && !defined(__mips__) */ // See siginfo.h in the Linux Kernel, these codes can be sent for any signal. -#define ADD_LINUX_SIGNAL(signo, name, ...) \ - AddSignal(signo, name, __VA_ARGS__); \ - ADD_SIGCODE(signo, signo, SI_QUEUE, -1, "sent by sigqueue", SignalCodePrintOption::Sender); \ - ADD_SIGCODE(signo, signo, SI_TIMER, -2, "sent by timer expiration", SignalCodePrintOption::Sender); \ - ADD_SIGCODE(signo, signo, SI_MESGQ, -3, "sent by real time mesq state change", SignalCodePrintOption::Sender); \ - ADD_SIGCODE(signo, signo, SI_ASYNCIO, -4, "sent by AIO completion", SignalCodePrintOption::Sender); \ - ADD_SIGCODE(signo, signo, SI_SIGIO, -5, "sent by queued SIGIO", SignalCodePrintOption::Sender); \ - ADD_SIGCODE(signo, signo, SI_TKILL, -6, "sent by tkill system call", SignalCodePrintOption::Sender); \ - ADD_SIGCODE(signo, signo, SI_DETHREAD, -7, "sent by execve() killing subsidiary threads", SignalCodePrintOption::Sender); \ - ADD_SIGCODE(signo, signo, SI_ASYNCNL, -60, "sent by glibc async name lookup completion", SignalCodePrintOption::Sender); +#define ADD_LINUX_SIGNAL(signo, name, ...) \ + AddSignal(signo, name, __VA_ARGS__); \ + ADD_SIGCODE(signo, signo, SI_QUEUE, -1, "sent by sigqueue", \ + SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_TIMER, -2, "sent by timer expiration", \ + SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_MESGQ, -3, \ + "sent by real time mesq state change", \ + SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_ASYNCIO, -4, "sent by AIO completion", \ + SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_SIGIO, -5, "sent by queued SIGIO", \ + SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_TKILL, -6, "sent by tkill system call", \ + SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_DETHREAD, -7, \ + "sent by execve() killing subsidiary threads", \ + SignalCodePrintOption::Sender); \ + ADD_SIGCODE(signo, signo, SI_ASYNCNL, -60, \ + "sent by glibc async name lookup completion", \ + SignalCodePrintOption::Sender); using namespace lldb_private; diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp index ce0f65cd9f14c..907e009bc7b80 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp @@ -585,10 +585,10 @@ Status ELFLinuxSigInfo::Parse(const DataExtractor &data, const ArchSpec &arch, if (data.GetAddressByteSize() == 8) offset += 4; - if (si_code < 0) { - sifields.kill.pid = data.GetU32(&offset); - sifields.kill.uid = data.GetU32(&offset); - } else if (unix_signals.GetShouldStop(si_signo)) { + if (si_code < 0) { + sifields.kill.pid = data.GetU32(&offset); + sifields.kill.uid = data.GetU32(&offset); + } else if (unix_signals.GetShouldStop(si_signo)) { // Not every stop signal has a valid address, but that will get resolved in // the unix_signals.GetSignalDescription() call below. // Instead of memcpy we call all these individually as the extractor will @@ -614,14 +614,17 @@ std::string ELFLinuxSigInfo::GetDescription( const lldb_private::UnixSignals &unix_signals) const { if (unix_signals.GetShouldStop(si_signo) && note_type == eNT_SIGINFO) { if (si_code < 0) - return unix_signals.GetSignalDescription(si_signo, si_code, std::nullopt, std::nullopt, std::nullopt, sifields.kill.pid, sifields.kill.uid); + return unix_signals.GetSignalDescription( + si_signo, si_code, std::nullopt, std::nullopt, std::nullopt, + sifields.kill.pid, sifields.kill.uid); else if (sifields.sigfault.bounds._addr_bnd._upper != 0) return unix_signals.GetSignalDescription( - si_signo, si_code, sifields.sigfault.si_addr, sifields.sigfault.bounds._addr_bnd._lower, + si_signo, si_code, sifields.sigfault.si_addr, + sifields.sigfault.bounds._addr_bnd._lower, sifields.sigfault.bounds._addr_bnd._upper); else return unix_signals.GetSignalDescription(si_signo, si_code, - sifields.sigfault.si_addr); + sifields.sigfault.si_addr); } // This looks weird, but there is an existing pattern where we don't pass a diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h index 2c254b4b522e9..40434543b7bb2 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h @@ -84,8 +84,8 @@ struct ELFLinuxSigInfo { int32_t si_code; union alignas(8) { struct alignas(8) { - uint32_t pid; /* sender's pid */ - uint32_t uid; /* sender's uid */ + uint32_t pid; /* sender's pid */ + uint32_t uid; /* sender's uid */ } kill; // Copied from siginfo_t so we don't have to include signal.h on non 'Nix // builds. Slight modifications to ensure no 32b vs 64b differences. diff --git a/lldb/source/Target/UnixSignals.cpp b/lldb/source/Target/UnixSignals.cpp index a5dbfd029410a..6113c6648817c 100644 --- a/lldb/source/Target/UnixSignals.cpp +++ b/lldb/source/Target/UnixSignals.cpp @@ -137,13 +137,11 @@ llvm::StringRef UnixSignals::GetSignalAsStringRef(int32_t signo) const { return pos->second.m_name; } -std::string -UnixSignals::GetSignalDescription(int32_t signo, std::optional<int32_t> code, - std::optional<lldb::addr_t> addr, - std::optional<lldb::addr_t> lower, - std::optional<lldb::addr_t> upper, - std::optional<uint32_t> pid, - std::optional<uint32_t> uid) const { +std::string UnixSignals::GetSignalDescription( + int32_t signo, std::optional<int32_t> code, + std::optional<lldb::addr_t> addr, std::optional<lldb::addr_t> lower, + std::optional<lldb::addr_t> upper, std::optional<uint32_t> pid, + std::optional<uint32_t> uid) const { std::string str; collection::const_iterator pos = m_signals.find(signo); diff --git a/lldb/unittests/Signals/UnixSignalsTest.cpp b/lldb/unittests/Signals/UnixSignalsTest.cpp index 825cc5ea6a782..582e441556067 100644 --- a/lldb/unittests/Signals/UnixSignalsTest.cpp +++ b/lldb/unittests/Signals/UnixSignalsTest.cpp @@ -27,7 +27,8 @@ class TestSignals : public UnixSignals { AddSignalCode(16, 2, "SIG16 with a fault address", SignalCodePrintOption::Address); AddSignalCode(16, 3, "bounds violation", SignalCodePrintOption::Bounds); - AddSignalCode(16, -6, "sent by tkill system call", SignalCodePrintOption::Sender); + AddSignalCode(16, -6, "sent by tkill system call", + SignalCodePrintOption::Sender); } }; @@ -126,9 +127,12 @@ TEST(UnixSignalsTest, GetAsString) { ASSERT_EQ("SIG16: SIG16 with a fault address", signals.GetSignalDescription(16, 2)); // TKill, but with no sender - ASSERT_EQ("SIG16: sent by tkill system call", signals.GetSignalDescription(16, -6, 0xCAFEF00D)); + ASSERT_EQ("SIG16: sent by tkill system call", + signals.GetSignalDescription(16, -6, 0xCAFEF00D)); // TKill, but with no sender - ASSERT_EQ("SIG16: sent by tkill system call (sender pid=912, uid=99)", signals.GetSignalDescription(16, -6, 0xCAFEF00D, std::nullopt, std::nullopt, 912, 99)); + ASSERT_EQ("SIG16: sent by tkill system call (sender pid=912, uid=99)", + signals.GetSignalDescription(16, -6, 0xCAFEF00D, std::nullopt, + std::nullopt, 912, 99)); const char *expected = "SIG16: bounds violation"; // Must pass all needed info to get full output. >From d8c2a04859e7c10617c15eacc02a303d66b83121 Mon Sep 17 00:00:00 2001 From: Jacob Lalonde <jalalo...@fb.com> Date: Fri, 16 May 2025 14:56:06 -0700 Subject: [PATCH 5/7] Reconfigure the parsing to get the bytes and convert to value object, then parse that in thlinux signals --- lldb/include/lldb/Target/UnixSignals.h | 3 + .../Plugins/Process/Utility/LinuxSignals.cpp | 42 ++++++ .../Plugins/Process/Utility/LinuxSignals.h | 2 + .../Process/elf-core/ProcessElfCore.cpp | 19 ++- .../Process/elf-core/ThreadElfCore.cpp | 129 +++++------------- .../Plugins/Process/elf-core/ThreadElfCore.h | 69 ++-------- 6 files changed, 105 insertions(+), 159 deletions(-) diff --git a/lldb/include/lldb/Target/UnixSignals.h b/lldb/include/lldb/Target/UnixSignals.h index a1807d69f329b..9ae4048ed683d 100644 --- a/lldb/include/lldb/Target/UnixSignals.h +++ b/lldb/include/lldb/Target/UnixSignals.h @@ -15,6 +15,7 @@ #include <vector> #include "lldb/lldb-private.h" +#include "lldb/ValueObject/ValueObject.h" #include "llvm/Support/JSON.h" namespace lldb_private { @@ -31,6 +32,8 @@ class UnixSignals { llvm::StringRef GetSignalAsStringRef(int32_t signo) const; + virtual std::string GetSignalDescriptionFromSiginfo(lldb::ValueObjectSP siginfo_sp) const { return ""; }; + std::string GetSignalDescription(int32_t signo, std::optional<int32_t> code = std::nullopt, diff --git a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp index 76c32e376eb4b..392675a265f5d 100644 --- a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp +++ b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp @@ -171,3 +171,45 @@ void LinuxSignals::Reset() { ADD_LINUX_SIGNAL(64, "SIGRTMAX", false, false, false, "real time signal 30"); // clang-format on } + +std::string LinuxSignals::GetSignalDescriptionFromSiginfo(lldb::ValueObjectSP siginfo_sp) const { + if (!siginfo_sp) + return ""; + + int code = siginfo_sp->GetChildMemberWithName("si_code")->GetValueAsSigned(0); + int signo = siginfo_sp->GetChildMemberWithName("si_signo")->GetValueAsSigned(-1); + // si_code = 0 is SI_NOINFO, we just want the description with nothing important + if (code == 0) + return GetSignalDescription(signo, code); + + lldb::ValueObjectSP sifields = siginfo_sp->GetChildMemberWithName("_sifields"); + // The negative si_codes are special and mean this signal was sent from user space + // not the kernel. These take precedence because they break some of the invariants + // around kernel sent signals. Such as SIGSEGV won't have an address. + if (code < 0) { + lldb::ValueObjectSP sikill = sifields->GetChildMemberWithName("_kill"); + uint32_t pid = sikill->GetChildMemberWithName("_pid")->GetValueAsUnsigned(-1); + uint32_t uid = sikill->GetChildMemberWithName("_uid")->GetValueAsUnsigned(-1); + return GetSignalDescription(signo, code, std::nullopt, std::nullopt, std::nullopt, pid, uid); + } + + switch (signo) { + case SIGILL: + case SIGFPE: + case SIGBUS: { + lldb::ValueObjectSP sigfault = sifields->GetChildMemberWithName("_sigfault"); + lldb::addr_t addr = sigfault->GetChildMemberWithName("_addr")->GetValueAsUnsigned(-1); + return GetSignalDescription(signo, code, addr); + } + case SIGSEGV: { + lldb::ValueObjectSP sigfault = sifields->GetChildMemberWithName("_sigfault"); + lldb::addr_t addr = sigfault->GetChildMemberWithName("_addr")->GetValueAsUnsigned(-1); + lldb::ValueObjectSP bounds = sigfault->GetChildMemberWithName("_addr_bnd"); + lldb::addr_t lower = bounds->GetChildMemberWithName("_lower")->GetValueAsUnsigned(-1); + lldb::addr_t upper = bounds->GetChildMemberWithName("_upper")->GetValueAsUnsigned(-1); + return GetSignalDescription(signo, code, addr, lower, upper); + } + default: + return GetSignalDescription(signo, code); + } +} diff --git a/lldb/source/Plugins/Process/Utility/LinuxSignals.h b/lldb/source/Plugins/Process/Utility/LinuxSignals.h index 32c4744a96d04..414cfd531388f 100644 --- a/lldb/source/Plugins/Process/Utility/LinuxSignals.h +++ b/lldb/source/Plugins/Process/Utility/LinuxSignals.h @@ -18,6 +18,8 @@ class LinuxSignals : public UnixSignals { public: LinuxSignals(); + std::string GetSignalDescriptionFromSiginfo(lldb::ValueObjectSP siginfo_sp) const override; + private: void Reset() override; }; diff --git a/lldb/source/Plugins/Process/elf-core/ProcessElfCore.cpp b/lldb/source/Plugins/Process/elf-core/ProcessElfCore.cpp index 6635b15b669f1..dc3e9616dc9c0 100644 --- a/lldb/source/Plugins/Process/elf-core/ProcessElfCore.cpp +++ b/lldb/source/Plugins/Process/elf-core/ProcessElfCore.cpp @@ -232,7 +232,7 @@ Status ProcessElfCore::DoLoadCore() { bool prstatus_signal_found = false; // Check we found a signal in a SIGINFO note. for (const auto &thread_data : m_thread_data) { - if (thread_data.siginfo.si_signo != 0) + if (!thread_data.siginfo_bytes.empty()) siginfo_signal_found = true; if (thread_data.prstatus_sig != 0) prstatus_signal_found = true; @@ -242,10 +242,10 @@ Status ProcessElfCore::DoLoadCore() { // PRSTATUS note. if (prstatus_signal_found) { for (auto &thread_data : m_thread_data) - thread_data.siginfo.si_signo = thread_data.prstatus_sig; + thread_data.signo = thread_data.prstatus_sig; } else if (m_thread_data.size() > 0) { // If all else fails force the first thread to be SIGSTOP - m_thread_data.begin()->siginfo.si_signo = + m_thread_data.begin()->signo = GetUnixSignals()->GetSignalNumberFromName("SIGSTOP"); } } @@ -506,7 +506,7 @@ static void ParseFreeBSDPrStatus(ThreadData &thread_data, else offset += 16; - thread_data.siginfo.si_signo = data.GetU32(&offset); // pr_cursig + thread_data.signo = data.GetU32(&offset); // pr_cursig thread_data.tid = data.GetU32(&offset); // pr_pid if (lp64) offset += 4; @@ -589,7 +589,7 @@ static void ParseOpenBSDProcInfo(ThreadData &thread_data, return; offset += 4; - thread_data.siginfo.si_signo = data.GetU32(&offset); + thread_data.signo = data.GetU32(&offset); } llvm::Expected<std::vector<CoreNote>> @@ -827,7 +827,7 @@ llvm::Error ProcessElfCore::parseNetBSDNotes(llvm::ArrayRef<CoreNote> notes) { // Signal targeted at the whole process. if (siglwp == 0) { for (auto &data : m_thread_data) - data.siginfo.si_signo = signo; + data.signo = signo; } // Signal destined for a particular LWP. else { @@ -835,7 +835,7 @@ llvm::Error ProcessElfCore::parseNetBSDNotes(llvm::ArrayRef<CoreNote> notes) { for (auto &data : m_thread_data) { if (data.tid == siglwp) { - data.siginfo.si_signo = signo; + data.signo = signo; passed = true; break; } @@ -938,12 +938,9 @@ llvm::Error ProcessElfCore::parseLinuxNotes(llvm::ArrayRef<CoreNote> notes) { break; } case ELF::NT_SIGINFO: { - const lldb_private::UnixSignals &unix_signals = *GetUnixSignals(); - ELFLinuxSigInfo siginfo; - Status status = siginfo.Parse(note.data, arch, unix_signals); + Status status = ELFLinuxSigInfo::Parse(note.data, arch, GetTarget().GetPlatform(), thread_data); if (status.Fail()) return status.ToError(); - thread_data.siginfo = siginfo; break; } case ELF::NT_FILE: { diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp index 907e009bc7b80..766b9b6d331fb 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp @@ -52,7 +52,7 @@ using namespace lldb_private; ThreadElfCore::ThreadElfCore(Process &process, const ThreadData &td) : Thread(process, td.tid), m_thread_name(td.name), m_thread_reg_ctx_sp(), m_gpregset_data(td.gpregset), m_notes(td.notes), - m_siginfo(std::move(td.siginfo)) {} + m_siginfo_bytes(std::move(td.siginfo_bytes)), m_signo(td.signo) {} ThreadElfCore::~ThreadElfCore() { DestroyThread(); } @@ -243,6 +243,14 @@ ThreadElfCore::CreateRegisterContextForFrame(StackFrame *frame) { return reg_ctx_sp; } +llvm::Expected<std::unique_ptr<llvm::MemoryBuffer>> ThreadElfCore::GetSiginfo(size_t max_size) const { + if (m_siginfo_bytes.empty()) + return llvm::createStringError(llvm::inconvertibleErrorCode(), + "no siginfo note"); + + return llvm::MemoryBuffer::getMemBufferCopy(m_siginfo_bytes, "siginfo note bytes"); +} + bool ThreadElfCore::CalculateStopInfo() { ProcessSP process_sp(GetProcess()); if (!process_sp) @@ -252,15 +260,17 @@ bool ThreadElfCore::CalculateStopInfo() { if (!unix_signals_sp) return false; - const char *sig_description; - std::string description = m_siginfo.GetDescription(*unix_signals_sp); - if (description.empty()) - sig_description = nullptr; - else - sig_description = description.c_str(); - - SetStopInfo(StopInfo::CreateStopReasonWithSignal( - *this, m_siginfo.si_signo, sig_description, m_siginfo.si_code)); + lldb::ValueObjectSP siginfo = GetSiginfoValue(); + if (!siginfo || !siginfo->GetValueIsValid()) { + std::string description = unix_signals_sp->GetSignalDescription(m_signo, 0); + SetStopInfo(StopInfo::CreateStopReasonWithSignal(*this, m_signo, description.c_str(), 0)); + } else { + std::string description = unix_signals_sp->GetSignalDescriptionFromSiginfo(siginfo); + uint32_t signo = siginfo->GetChildMemberWithName("si_signo")->GetValueAsUnsigned(-1); + uint32_t code = siginfo->GetChildMemberWithName("si_code")->GetValueAsUnsigned(0); + SetStopInfo(StopInfo::CreateStopReasonWithSignal( + *this, signo, description.c_str(), code)); + } SetStopInfo(m_stop_info_sp); return true; @@ -544,91 +554,22 @@ ELFLinuxPrPsInfo::Populate(const lldb_private::ProcessInstanceInfo &info, return prpsinfo; } -// Parse SIGINFO from NOTE entry -ELFLinuxSigInfo::ELFLinuxSigInfo() { memset(this, 0, sizeof(ELFLinuxSigInfo)); } +Status ELFLinuxSigInfo::Parse(const DataExtractor &data, const ArchSpec &arch, const lldb::PlatformSP platform_sp, ThreadData &thread_data) { + if (!platform_sp) + return Status::FromErrorString("No platform for arch."); + CompilerType type = platform_sp->GetSiginfoType(arch.GetTriple()); + if (!type.IsValid()) + return Status::FromErrorString("no siginfo_t for platform."); -size_t ELFLinuxSigInfo::GetSize(const lldb_private::ArchSpec &arch) { - if (arch.IsMIPS()) - return sizeof(ELFLinuxSigInfo); - switch (arch.GetCore()) { - case lldb_private::ArchSpec::eCore_x86_64_x86_64: - return sizeof(ELFLinuxSigInfo); - case lldb_private::ArchSpec::eCore_s390x_generic: - case lldb_private::ArchSpec::eCore_x86_32_i386: - case lldb_private::ArchSpec::eCore_x86_32_i486: - return 12; - default: - return 0; - } -} + auto type_size_or_err = type.GetByteSize(nullptr); + if (!type_size_or_err) + return Status::FromError(type_size_or_err.takeError()); -Status ELFLinuxSigInfo::Parse(const DataExtractor &data, const ArchSpec &arch, - const lldb_private::UnixSignals &unix_signals) { - Status error; - uint64_t size = GetSize(arch); - if (size > data.GetByteSize()) { - error = Status::FromErrorStringWithFormat( - "NT_SIGINFO size should be %zu, but the remaining bytes are: %" PRIu64, - GetSize(arch), data.GetByteSize()); - return error; - } - - // Set that we've parsed the siginfo from a SIGINFO note. - note_type = eNT_SIGINFO; - // Parsing from a 32 bit ELF core file, and populating/reusing the structure - // properly, because the struct is for the 64 bit version - offset_t offset = 0; - si_signo = data.GetU32(&offset); - si_errno = data.GetU32(&offset); - si_code = data.GetU32(&offset); - // 64b ELF have a 4 byte pad. - if (data.GetAddressByteSize() == 8) - offset += 4; - - if (si_code < 0) { - sifields.kill.pid = data.GetU32(&offset); - sifields.kill.uid = data.GetU32(&offset); - } else if (unix_signals.GetShouldStop(si_signo)) { - // Not every stop signal has a valid address, but that will get resolved in - // the unix_signals.GetSignalDescription() call below. - // Instead of memcpy we call all these individually as the extractor will - // handle endianness for us. - sifields.sigfault.si_addr = data.GetAddress(&offset); - sifields.sigfault.si_addr_lsb = data.GetU16(&offset); - if (data.GetByteSize() - offset >= sizeof(sifields.sigfault.bounds)) { - sifields.sigfault.bounds._addr_bnd._lower = data.GetAddress(&offset); - sifields.sigfault.bounds._addr_bnd._upper = data.GetAddress(&offset); - sifields.sigfault.bounds._pkey = data.GetU32(&offset); - } else { - // Set these to 0 so we don't use bogus data for the description. - sifields.sigfault.bounds._addr_bnd._lower = 0; - sifields.sigfault.bounds._addr_bnd._upper = 0; - sifields.sigfault.bounds._pkey = 0; - } - } - - return error; -} - -std::string ELFLinuxSigInfo::GetDescription( - const lldb_private::UnixSignals &unix_signals) const { - if (unix_signals.GetShouldStop(si_signo) && note_type == eNT_SIGINFO) { - if (si_code < 0) - return unix_signals.GetSignalDescription( - si_signo, si_code, std::nullopt, std::nullopt, std::nullopt, - sifields.kill.pid, sifields.kill.uid); - else if (sifields.sigfault.bounds._addr_bnd._upper != 0) - return unix_signals.GetSignalDescription( - si_signo, si_code, sifields.sigfault.si_addr, - sifields.sigfault.bounds._addr_bnd._lower, - sifields.sigfault.bounds._addr_bnd._upper); - else - return unix_signals.GetSignalDescription(si_signo, si_code, - sifields.sigfault.si_addr); - } + if (data.GetByteSize() < *type_size_or_err) + return Status::FromErrorString("siginfo note byte size smaller than siginfo_t for platform."); - // This looks weird, but there is an existing pattern where we don't pass a - // description to keep up with that, we return empty here, and then the above - // function will set the description whether or not this is empty. - return std::string(); + lldb::offset_t offset = 0; + const char *bytes = static_cast<const char*>(data.GetData(&offset, *type_size_or_err)); + thread_data.siginfo_bytes = llvm::StringRef(bytes, *type_size_or_err); + return Status(); } diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h index 40434543b7bb2..413719a1a26f4 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h @@ -12,6 +12,8 @@ #include "Plugins/Process/elf-core/RegisterUtilities.h" #include "lldb/Target/Thread.h" #include "lldb/Utility/DataExtractor.h" +#include "lldb/ValueObject/ValueObject.h" +#include "lldb/Target/Platform.h" #include "llvm/ADT/DenseMap.h" #include <optional> #include <string> @@ -77,57 +79,14 @@ struct ELFLinuxPrStatus { static_assert(sizeof(ELFLinuxPrStatus) == 112, "sizeof ELFLinuxPrStatus is not correct!"); -struct ELFLinuxSigInfo { - - int32_t si_signo; // Order matters for the first 3. - int32_t si_errno; - int32_t si_code; - union alignas(8) { - struct alignas(8) { - uint32_t pid; /* sender's pid */ - uint32_t uid; /* sender's uid */ - } kill; - // Copied from siginfo_t so we don't have to include signal.h on non 'Nix - // builds. Slight modifications to ensure no 32b vs 64b differences. - struct alignas(8) { - lldb::addr_t si_addr; /* faulting insn/memory ref. */ - int16_t si_addr_lsb; /* Valid LSB of the reported address. */ - union { - /* used when si_code=SEGV_BNDERR */ - struct { - lldb::addr_t _lower; - lldb::addr_t _upper; - } _addr_bnd; - /* used when si_code=SEGV_PKUERR */ - uint32_t _pkey; - } bounds; - - // We need this for all the generic signals. - } sigfault; - } sifields; - - enum SigInfoNoteType : uint8_t { eUnspecified, eNT_SIGINFO }; - SigInfoNoteType note_type; - - ELFLinuxSigInfo(); - - lldb_private::Status Parse(const lldb_private::DataExtractor &data, - const lldb_private::ArchSpec &arch, - const lldb_private::UnixSignals &unix_signals); - - std::string - GetDescription(const lldb_private::UnixSignals &unix_signals) const; - - // Return the bytesize of the structure - // 64 bit - just sizeof - // 32 bit - hardcoded because we are reusing the struct, but some of the - // members are smaller - - // so the layout is not the same - static size_t GetSize(const lldb_private::ArchSpec &arch); +class ELFLinuxSigInfo { +public: + static lldb_private::Status Parse(const lldb_private::DataExtractor &data, + const lldb_private::ArchSpec &arch, + const lldb::PlatformSP platform_sp, + ThreadData &thread_data); }; -static_assert(sizeof(ELFLinuxSigInfo) == 56, - "sizeof ELFLinuxSigInfo is not correct!"); // PRPSINFO structure's size differs based on architecture. // This is the layout in the x86-64 arch case. @@ -176,8 +135,9 @@ struct ThreadData { std::vector<lldb_private::CoreNote> notes; lldb::tid_t tid; std::string name; - ELFLinuxSigInfo siginfo; - int prstatus_sig = 0; + llvm::StringRef siginfo_bytes; + int prstatus_sig; + int signo; }; class ThreadElfCore : public lldb_private::Thread { @@ -208,8 +168,7 @@ class ThreadElfCore : public lldb_private::Thread { m_thread_name.clear(); } - void CreateStopFromSigInfo(const ELFLinuxSigInfo &siginfo, - const lldb_private::UnixSignals &unix_signals); + llvm::Expected<std::unique_ptr<llvm::MemoryBuffer>> GetSiginfo(size_t max_size) const override; protected: // Member variables. @@ -218,7 +177,9 @@ class ThreadElfCore : public lldb_private::Thread { lldb_private::DataExtractor m_gpregset_data; std::vector<lldb_private::CoreNote> m_notes; - ELFLinuxSigInfo m_siginfo; + llvm::StringRef m_siginfo_bytes; + // Only used if no siginfo note. + int m_signo; bool CalculateStopInfo() override; }; >From f2b2f8d37029b8172eac844dc2f812d46556d2c8 Mon Sep 17 00:00:00 2001 From: Jacob Lalonde <jalalo...@fb.com> Date: Fri, 16 May 2025 15:52:19 -0700 Subject: [PATCH 6/7] Refactor from testing, discovered that the compiler type generated signal differs from the linux definition --- .../Plugins/Process/Utility/LinuxSignals.cpp | 11 ++++++----- .../Plugins/Process/elf-core/ThreadElfCore.h | 19 ++++++++++--------- 2 files changed, 16 insertions(+), 14 deletions(-) diff --git a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp index 392675a265f5d..aef1b3f1d0a12 100644 --- a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp +++ b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp @@ -188,8 +188,8 @@ std::string LinuxSignals::GetSignalDescriptionFromSiginfo(lldb::ValueObjectSP si // around kernel sent signals. Such as SIGSEGV won't have an address. if (code < 0) { lldb::ValueObjectSP sikill = sifields->GetChildMemberWithName("_kill"); - uint32_t pid = sikill->GetChildMemberWithName("_pid")->GetValueAsUnsigned(-1); - uint32_t uid = sikill->GetChildMemberWithName("_uid")->GetValueAsUnsigned(-1); + uint32_t pid = sikill->GetChildMemberWithName("si_pid")->GetValueAsUnsigned(-1); + uint32_t uid = sikill->GetChildMemberWithName("si_uid")->GetValueAsUnsigned(-1); return GetSignalDescription(signo, code, std::nullopt, std::nullopt, std::nullopt, pid, uid); } @@ -198,13 +198,14 @@ std::string LinuxSignals::GetSignalDescriptionFromSiginfo(lldb::ValueObjectSP si case SIGFPE: case SIGBUS: { lldb::ValueObjectSP sigfault = sifields->GetChildMemberWithName("_sigfault"); - lldb::addr_t addr = sigfault->GetChildMemberWithName("_addr")->GetValueAsUnsigned(-1); + lldb::addr_t addr = sigfault->GetChildMemberWithName("si_addr")->GetValueAsUnsigned(-1); return GetSignalDescription(signo, code, addr); } case SIGSEGV: { lldb::ValueObjectSP sigfault = sifields->GetChildMemberWithName("_sigfault"); - lldb::addr_t addr = sigfault->GetChildMemberWithName("_addr")->GetValueAsUnsigned(-1); - lldb::ValueObjectSP bounds = sigfault->GetChildMemberWithName("_addr_bnd"); + lldb::addr_t addr = sigfault->GetChildMemberWithName("si_addr")->GetValueAsUnsigned(-1); + + lldb::ValueObjectSP bounds = sigfault->GetChildMemberWithName("_bounds")->GetChildMemberWithName("_addr_bnd"); lldb::addr_t lower = bounds->GetChildMemberWithName("_lower")->GetValueAsUnsigned(-1); lldb::addr_t upper = bounds->GetChildMemberWithName("_upper")->GetValueAsUnsigned(-1); return GetSignalDescription(signo, code, addr, lower, upper); diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h index 413719a1a26f4..8b5e593528125 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h @@ -79,6 +79,16 @@ struct ELFLinuxPrStatus { static_assert(sizeof(ELFLinuxPrStatus) == 112, "sizeof ELFLinuxPrStatus is not correct!"); +struct ThreadData { + lldb_private::DataExtractor gpregset; + std::vector<lldb_private::CoreNote> notes; + lldb::tid_t tid; + std::string name; + llvm::StringRef siginfo_bytes; + int prstatus_sig; + int signo; +}; + class ELFLinuxSigInfo { public: static lldb_private::Status Parse(const lldb_private::DataExtractor &data, @@ -130,15 +140,6 @@ struct ELFLinuxPrPsInfo { static_assert(sizeof(ELFLinuxPrPsInfo) == 136, "sizeof ELFLinuxPrPsInfo is not correct!"); -struct ThreadData { - lldb_private::DataExtractor gpregset; - std::vector<lldb_private::CoreNote> notes; - lldb::tid_t tid; - std::string name; - llvm::StringRef siginfo_bytes; - int prstatus_sig; - int signo; -}; class ThreadElfCore : public lldb_private::Thread { public: >From 12e725417075b09739da786417dac3898e7d4a35 Mon Sep 17 00:00:00 2001 From: Jacob Lalonde <jalalo...@fb.com> Date: Fri, 16 May 2025 15:52:41 -0700 Subject: [PATCH 7/7] Run GCF --- lldb/include/lldb/Target/UnixSignals.h | 7 +- .../Plugins/Process/Utility/LinuxSignals.cpp | 72 +++++++++++-------- .../Plugins/Process/Utility/LinuxSignals.h | 3 +- .../Process/elf-core/ProcessElfCore.cpp | 3 +- .../Process/elf-core/ThreadElfCore.cpp | 32 ++++++--- .../Plugins/Process/elf-core/ThreadElfCore.h | 7 +- 6 files changed, 77 insertions(+), 47 deletions(-) diff --git a/lldb/include/lldb/Target/UnixSignals.h b/lldb/include/lldb/Target/UnixSignals.h index 9ae4048ed683d..53b718e917d25 100644 --- a/lldb/include/lldb/Target/UnixSignals.h +++ b/lldb/include/lldb/Target/UnixSignals.h @@ -14,8 +14,8 @@ #include <string> #include <vector> -#include "lldb/lldb-private.h" #include "lldb/ValueObject/ValueObject.h" +#include "lldb/lldb-private.h" #include "llvm/Support/JSON.h" namespace lldb_private { @@ -32,7 +32,10 @@ class UnixSignals { llvm::StringRef GetSignalAsStringRef(int32_t signo) const; - virtual std::string GetSignalDescriptionFromSiginfo(lldb::ValueObjectSP siginfo_sp) const { return ""; }; + virtual std::string + GetSignalDescriptionFromSiginfo(lldb::ValueObjectSP siginfo_sp) const { + return ""; + }; std::string GetSignalDescription(int32_t signo, diff --git a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp index aef1b3f1d0a12..06b8423000026 100644 --- a/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp +++ b/lldb/source/Plugins/Process/Utility/LinuxSignals.cpp @@ -172,45 +172,61 @@ void LinuxSignals::Reset() { // clang-format on } -std::string LinuxSignals::GetSignalDescriptionFromSiginfo(lldb::ValueObjectSP siginfo_sp) const { +std::string LinuxSignals::GetSignalDescriptionFromSiginfo( + lldb::ValueObjectSP siginfo_sp) const { if (!siginfo_sp) return ""; int code = siginfo_sp->GetChildMemberWithName("si_code")->GetValueAsSigned(0); - int signo = siginfo_sp->GetChildMemberWithName("si_signo")->GetValueAsSigned(-1); - // si_code = 0 is SI_NOINFO, we just want the description with nothing important + int signo = + siginfo_sp->GetChildMemberWithName("si_signo")->GetValueAsSigned(-1); + // si_code = 0 is SI_NOINFO, we just want the description with nothing + // important if (code == 0) return GetSignalDescription(signo, code); - lldb::ValueObjectSP sifields = siginfo_sp->GetChildMemberWithName("_sifields"); - // The negative si_codes are special and mean this signal was sent from user space - // not the kernel. These take precedence because they break some of the invariants - // around kernel sent signals. Such as SIGSEGV won't have an address. + lldb::ValueObjectSP sifields = + siginfo_sp->GetChildMemberWithName("_sifields"); + // The negative si_codes are special and mean this signal was sent from user + // space not the kernel. These take precedence because they break some of the + // invariants around kernel sent signals. Such as SIGSEGV won't have an + // address. if (code < 0) { lldb::ValueObjectSP sikill = sifields->GetChildMemberWithName("_kill"); - uint32_t pid = sikill->GetChildMemberWithName("si_pid")->GetValueAsUnsigned(-1); - uint32_t uid = sikill->GetChildMemberWithName("si_uid")->GetValueAsUnsigned(-1); - return GetSignalDescription(signo, code, std::nullopt, std::nullopt, std::nullopt, pid, uid); + uint32_t pid = + sikill->GetChildMemberWithName("si_pid")->GetValueAsUnsigned(-1); + uint32_t uid = + sikill->GetChildMemberWithName("si_uid")->GetValueAsUnsigned(-1); + return GetSignalDescription(signo, code, std::nullopt, std::nullopt, + std::nullopt, pid, uid); } switch (signo) { - case SIGILL: - case SIGFPE: - case SIGBUS: { - lldb::ValueObjectSP sigfault = sifields->GetChildMemberWithName("_sigfault"); - lldb::addr_t addr = sigfault->GetChildMemberWithName("si_addr")->GetValueAsUnsigned(-1); - return GetSignalDescription(signo, code, addr); - } - case SIGSEGV: { - lldb::ValueObjectSP sigfault = sifields->GetChildMemberWithName("_sigfault"); - lldb::addr_t addr = sigfault->GetChildMemberWithName("si_addr")->GetValueAsUnsigned(-1); - - lldb::ValueObjectSP bounds = sigfault->GetChildMemberWithName("_bounds")->GetChildMemberWithName("_addr_bnd"); - lldb::addr_t lower = bounds->GetChildMemberWithName("_lower")->GetValueAsUnsigned(-1); - lldb::addr_t upper = bounds->GetChildMemberWithName("_upper")->GetValueAsUnsigned(-1); - return GetSignalDescription(signo, code, addr, lower, upper); - } - default: - return GetSignalDescription(signo, code); + case SIGILL: + case SIGFPE: + case SIGBUS: { + lldb::ValueObjectSP sigfault = + sifields->GetChildMemberWithName("_sigfault"); + lldb::addr_t addr = + sigfault->GetChildMemberWithName("si_addr")->GetValueAsUnsigned(-1); + return GetSignalDescription(signo, code, addr); + } + case SIGSEGV: { + lldb::ValueObjectSP sigfault = + sifields->GetChildMemberWithName("_sigfault"); + lldb::addr_t addr = + sigfault->GetChildMemberWithName("si_addr")->GetValueAsUnsigned(-1); + + lldb::ValueObjectSP bounds = + sigfault->GetChildMemberWithName("_bounds")->GetChildMemberWithName( + "_addr_bnd"); + lldb::addr_t lower = + bounds->GetChildMemberWithName("_lower")->GetValueAsUnsigned(-1); + lldb::addr_t upper = + bounds->GetChildMemberWithName("_upper")->GetValueAsUnsigned(-1); + return GetSignalDescription(signo, code, addr, lower, upper); + } + default: + return GetSignalDescription(signo, code); } } diff --git a/lldb/source/Plugins/Process/Utility/LinuxSignals.h b/lldb/source/Plugins/Process/Utility/LinuxSignals.h index 414cfd531388f..fab8e4d0526a4 100644 --- a/lldb/source/Plugins/Process/Utility/LinuxSignals.h +++ b/lldb/source/Plugins/Process/Utility/LinuxSignals.h @@ -18,7 +18,8 @@ class LinuxSignals : public UnixSignals { public: LinuxSignals(); - std::string GetSignalDescriptionFromSiginfo(lldb::ValueObjectSP siginfo_sp) const override; + std::string GetSignalDescriptionFromSiginfo( + lldb::ValueObjectSP siginfo_sp) const override; private: void Reset() override; diff --git a/lldb/source/Plugins/Process/elf-core/ProcessElfCore.cpp b/lldb/source/Plugins/Process/elf-core/ProcessElfCore.cpp index dc3e9616dc9c0..eda30c9c6f82b 100644 --- a/lldb/source/Plugins/Process/elf-core/ProcessElfCore.cpp +++ b/lldb/source/Plugins/Process/elf-core/ProcessElfCore.cpp @@ -938,7 +938,8 @@ llvm::Error ProcessElfCore::parseLinuxNotes(llvm::ArrayRef<CoreNote> notes) { break; } case ELF::NT_SIGINFO: { - Status status = ELFLinuxSigInfo::Parse(note.data, arch, GetTarget().GetPlatform(), thread_data); + Status status = ELFLinuxSigInfo::Parse( + note.data, arch, GetTarget().GetPlatform(), thread_data); if (status.Fail()) return status.ToError(); break; diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp index 766b9b6d331fb..32948d6b0de9e 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.cpp @@ -243,12 +243,14 @@ ThreadElfCore::CreateRegisterContextForFrame(StackFrame *frame) { return reg_ctx_sp; } -llvm::Expected<std::unique_ptr<llvm::MemoryBuffer>> ThreadElfCore::GetSiginfo(size_t max_size) const { +llvm::Expected<std::unique_ptr<llvm::MemoryBuffer>> +ThreadElfCore::GetSiginfo(size_t max_size) const { if (m_siginfo_bytes.empty()) return llvm::createStringError(llvm::inconvertibleErrorCode(), - "no siginfo note"); + "no siginfo note"); - return llvm::MemoryBuffer::getMemBufferCopy(m_siginfo_bytes, "siginfo note bytes"); + return llvm::MemoryBuffer::getMemBufferCopy(m_siginfo_bytes, + "siginfo note bytes"); } bool ThreadElfCore::CalculateStopInfo() { @@ -263,13 +265,17 @@ bool ThreadElfCore::CalculateStopInfo() { lldb::ValueObjectSP siginfo = GetSiginfoValue(); if (!siginfo || !siginfo->GetValueIsValid()) { std::string description = unix_signals_sp->GetSignalDescription(m_signo, 0); - SetStopInfo(StopInfo::CreateStopReasonWithSignal(*this, m_signo, description.c_str(), 0)); + SetStopInfo(StopInfo::CreateStopReasonWithSignal(*this, m_signo, + description.c_str(), 0)); } else { - std::string description = unix_signals_sp->GetSignalDescriptionFromSiginfo(siginfo); - uint32_t signo = siginfo->GetChildMemberWithName("si_signo")->GetValueAsUnsigned(-1); - uint32_t code = siginfo->GetChildMemberWithName("si_code")->GetValueAsUnsigned(0); + std::string description = + unix_signals_sp->GetSignalDescriptionFromSiginfo(siginfo); + uint32_t signo = + siginfo->GetChildMemberWithName("si_signo")->GetValueAsUnsigned(-1); + uint32_t code = + siginfo->GetChildMemberWithName("si_code")->GetValueAsUnsigned(0); SetStopInfo(StopInfo::CreateStopReasonWithSignal( - *this, signo, description.c_str(), code)); + *this, signo, description.c_str(), code)); } SetStopInfo(m_stop_info_sp); @@ -554,7 +560,9 @@ ELFLinuxPrPsInfo::Populate(const lldb_private::ProcessInstanceInfo &info, return prpsinfo; } -Status ELFLinuxSigInfo::Parse(const DataExtractor &data, const ArchSpec &arch, const lldb::PlatformSP platform_sp, ThreadData &thread_data) { +Status ELFLinuxSigInfo::Parse(const DataExtractor &data, const ArchSpec &arch, + const lldb::PlatformSP platform_sp, + ThreadData &thread_data) { if (!platform_sp) return Status::FromErrorString("No platform for arch."); CompilerType type = platform_sp->GetSiginfoType(arch.GetTriple()); @@ -566,10 +574,12 @@ Status ELFLinuxSigInfo::Parse(const DataExtractor &data, const ArchSpec &arch, c return Status::FromError(type_size_or_err.takeError()); if (data.GetByteSize() < *type_size_or_err) - return Status::FromErrorString("siginfo note byte size smaller than siginfo_t for platform."); + return Status::FromErrorString( + "siginfo note byte size smaller than siginfo_t for platform."); lldb::offset_t offset = 0; - const char *bytes = static_cast<const char*>(data.GetData(&offset, *type_size_or_err)); + const char *bytes = + static_cast<const char *>(data.GetData(&offset, *type_size_or_err)); thread_data.siginfo_bytes = llvm::StringRef(bytes, *type_size_or_err); return Status(); } diff --git a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h index 8b5e593528125..2ab9b90f9cc17 100644 --- a/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h +++ b/lldb/source/Plugins/Process/elf-core/ThreadElfCore.h @@ -10,10 +10,10 @@ #define LLDB_SOURCE_PLUGINS_PROCESS_ELF_CORE_THREADELFCORE_H #include "Plugins/Process/elf-core/RegisterUtilities.h" +#include "lldb/Target/Platform.h" #include "lldb/Target/Thread.h" #include "lldb/Utility/DataExtractor.h" #include "lldb/ValueObject/ValueObject.h" -#include "lldb/Target/Platform.h" #include "llvm/ADT/DenseMap.h" #include <optional> #include <string> @@ -97,7 +97,6 @@ class ELFLinuxSigInfo { ThreadData &thread_data); }; - // PRPSINFO structure's size differs based on architecture. // This is the layout in the x86-64 arch case. // In the i386 case we parse it manually and fill it again @@ -140,7 +139,6 @@ struct ELFLinuxPrPsInfo { static_assert(sizeof(ELFLinuxPrPsInfo) == 136, "sizeof ELFLinuxPrPsInfo is not correct!"); - class ThreadElfCore : public lldb_private::Thread { public: ThreadElfCore(lldb_private::Process &process, const ThreadData &td); @@ -169,7 +167,8 @@ class ThreadElfCore : public lldb_private::Thread { m_thread_name.clear(); } - llvm::Expected<std::unique_ptr<llvm::MemoryBuffer>> GetSiginfo(size_t max_size) const override; + llvm::Expected<std::unique_ptr<llvm::MemoryBuffer>> + GetSiginfo(size_t max_size) const override; protected: // Member variables. _______________________________________________ lldb-commits mailing list lldb-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/lldb-commits