diff --git a/README.md b/README.md index 4ce94e4..8d48585 100644 --- a/README.md +++ b/README.md @@ -88,6 +88,7 @@ Of course, most use-cases will want to support additional debugging features as - Can be used to automatically read the remote executable on attach (using `ExecFile`) - Read auxiliary vector (`info auxv`) - Extra thread info (`info threads`) +- Extra library information (`info sharedlibraries`) _Note:_ GDB features are implemented on an as-needed basis by `gdbstub`'s contributors. If there's a missing GDB feature that you'd like `gdbstub` to implement, please file an issue and/or open a PR! diff --git a/examples/armv4t/gdb/libraries.rs b/examples/armv4t/gdb/libraries.rs new file mode 100644 index 0000000..0c64dd7 --- /dev/null +++ b/examples/armv4t/gdb/libraries.rs @@ -0,0 +1,28 @@ +use super::copy_range_to_buf; +use crate::emu::Emu; +use gdbstub::target; +use gdbstub::target::TargetResult; + +impl target::ext::libraries::LibrariesSvr4 for Emu { + fn get_libraries_svr4( + &self, + offset: u64, + length: usize, + buf: &mut [u8], + ) -> TargetResult { + // `l_ld` is the address of the `PT_DYNAMIC` ELF segment, so fake an + // address here. + // + // The `main-lm`, `lm`, and `lmid` seem to refer to in-memory structures + // which gdb may read, but gdb also seems to work well enough if they're + // null-ish or otherwise pointing to non-present things. + let xml = r#" + + + +"# + .trim() + .as_bytes(); + Ok(copy_range_to_buf(xml, offset, length, buf)) + } +} diff --git a/examples/armv4t/gdb/mod.rs b/examples/armv4t/gdb/mod.rs index 5837ddd..b56ce9a 100644 --- a/examples/armv4t/gdb/mod.rs +++ b/examples/armv4t/gdb/mod.rs @@ -20,6 +20,7 @@ mod catch_syscalls; mod exec_file; mod extended_mode; mod host_io; +mod libraries; mod lldb_register_info_override; mod memory_map; mod monitor_cmd; @@ -153,6 +154,13 @@ impl Target for Emu { fn support_auxv(&mut self) -> Option> { Some(self) } + + #[inline(always)] + fn support_libraries_svr4( + &mut self, + ) -> Option> { + Some(self) + } } impl SingleThreadBase for Emu { diff --git a/src/protocol/commands.rs b/src/protocol/commands.rs index 1229899..115e76b 100644 --- a/src/protocol/commands.rs +++ b/src/protocol/commands.rs @@ -330,4 +330,8 @@ commands! { lldb_register_info { "qRegisterInfo" => _qRegisterInfo::qRegisterInfo, } + + libraries_svr4 use 'a { + "qXfer:libraries-svr4:read" => _qXfer_libraries_svr4_read::qXferLibrariesSvr4Read<'a>, + } } diff --git a/src/protocol/commands/_qXfer_libraries_svr4_read.rs b/src/protocol/commands/_qXfer_libraries_svr4_read.rs new file mode 100644 index 0000000..be050f4 --- /dev/null +++ b/src/protocol/commands/_qXfer_libraries_svr4_read.rs @@ -0,0 +1,18 @@ +use crate::protocol::common::qxfer::ParseAnnex; +use crate::protocol::common::qxfer::QXferReadBase; + +pub type qXferLibrariesSvr4Read<'a> = QXferReadBase<'a, LibrariesSvr4Annex>; + +#[derive(Debug)] +pub struct LibrariesSvr4Annex; + +impl<'a> ParseAnnex<'a> for LibrariesSvr4Annex { + #[inline(always)] + fn from_buf(buf: &[u8]) -> Option { + if buf != b"" { + return None; + } + + Some(LibrariesSvr4Annex) + } +} diff --git a/src/stub/core_impl.rs b/src/stub/core_impl.rs index 07774d4..2ea3bc8 100644 --- a/src/stub/core_impl.rs +++ b/src/stub/core_impl.rs @@ -31,6 +31,7 @@ mod catch_syscalls; mod exec_file; mod extended_mode; mod host_io; +mod libraries; mod lldb_register_info; mod memory_map; mod monitor_cmd; @@ -216,6 +217,7 @@ impl GdbStubImpl { Command::Auxv(cmd) => self.handle_auxv(res, target, cmd), Command::ThreadExtraInfo(cmd) => self.handle_thread_extra_info(res, target, cmd), Command::LldbRegisterInfo(cmd) => self.handle_lldb_register_info(res, target, cmd), + Command::LibrariesSvr4(cmd) => self.handle_libraries_svr4(res, target, cmd), // in the worst case, the command could not be parsed... Command::Unknown(cmd) => { // HACK: if the user accidentally sends a resume command to a diff --git a/src/stub/core_impl/base.rs b/src/stub/core_impl/base.rs index 9110297..eebb715 100644 --- a/src/stub/core_impl/base.rs +++ b/src/stub/core_impl/base.rs @@ -195,6 +195,10 @@ impl GdbStubImpl { res.write_str(";qXfer:auxv:read+")?; } + if target.support_libraries_svr4().is_some() { + res.write_str(";qXfer:libraries-svr4:read+")?; + } + HandlerStatus::Handled } diff --git a/src/stub/core_impl/libraries.rs b/src/stub/core_impl/libraries.rs new file mode 100644 index 0000000..dd18cb7 --- /dev/null +++ b/src/stub/core_impl/libraries.rs @@ -0,0 +1,36 @@ +use super::prelude::*; +use crate::protocol::commands::ext::LibrariesSvr4; + +impl GdbStubImpl { + pub(crate) fn handle_libraries_svr4( + &mut self, + res: &mut ResponseWriter<'_, C>, + target: &mut T, + command: LibrariesSvr4<'_>, + ) -> Result> { + let ops = match target.support_libraries_svr4() { + Some(ops) => ops, + None => return Ok(HandlerStatus::Handled), + }; + + crate::__dead_code_marker!("libraries-svr4", "impl"); + + let handler_status = match command { + LibrariesSvr4::qXferLibrariesSvr4Read(cmd) => { + let ret = ops + .get_libraries_svr4(cmd.offset, cmd.length, cmd.buf) + .handle_error()?; + if ret == 0 { + res.write_str("l")?; + } else { + res.write_str("m")?; + // TODO: add more specific error variant? + res.write_binary(cmd.buf.get(..ret).ok_or(Error::PacketBufferOverflow)?)?; + } + HandlerStatus::Handled + } + }; + + Ok(handler_status) + } +} diff --git a/src/target/ext/libraries.rs b/src/target/ext/libraries.rs new file mode 100644 index 0000000..5feea82 --- /dev/null +++ b/src/target/ext/libraries.rs @@ -0,0 +1,29 @@ +//! Report information about the loaded shared libraries for targets where there +//! are possibly multiple files to be debugged mapped into the same address +//! space. + +use crate::target::Target; +use crate::target::TargetResult; + +/// Target Extension - List an SVR4 (System-V/Unix) target's libraries. +pub trait LibrariesSvr4: Target { + /// Get library list XML for this target. + /// + /// See the [GDB Documentation] for a description of the format. + /// + /// [GDB Documentation]: https://sourceware.org/gdb/current/onlinedocs/gdb.html/Library-List-Format-for-SVR4-Targets.html + /// + /// Return the number of bytes written into `buf` (which may be less than + /// `length`). + /// + /// If `offset` is greater than the length of the underlying data, return + /// `Ok(0)`. + fn get_libraries_svr4( + &self, + offset: u64, + length: usize, + buf: &mut [u8], + ) -> TargetResult; +} + +define_ext!(LibrariesSvr4Ops, LibrariesSvr4); diff --git a/src/target/ext/mod.rs b/src/target/ext/mod.rs index b7e2d8f..a6f3d95 100644 --- a/src/target/ext/mod.rs +++ b/src/target/ext/mod.rs @@ -265,6 +265,7 @@ pub mod catch_syscalls; pub mod exec_file; pub mod extended_mode; pub mod host_io; +pub mod libraries; pub mod lldb_register_info_override; pub mod memory_map; pub mod monitor_cmd; diff --git a/src/target/ext/section_offsets.rs b/src/target/ext/section_offsets.rs index eb80578..e778018 100644 --- a/src/target/ext/section_offsets.rs +++ b/src/target/ext/section_offsets.rs @@ -8,6 +8,13 @@ //! limited to reporting the offsets for code, data and bss, and is //! generally considered a legacy feature. //! +//! For System-V architectures GDB may use the `qXfer:libraries-svr4:read` +//! command to try to learn about loaded libraries and this can be implemented +//! with the [`LibrariesSvr4` +//! trait](crate::target::ext::libraries::LibrariesSvr4). Note that not all +//! targets may query this and it may not be applicable in all situations +//! either. +//! //! For targets where library offsets are maintained externally (e.g. Windows) //! you should consider implementing the more flexible `qXfer:library:read`. //! See issue [#20](https://github.com/daniel5151/gdbstub/issues/20) for more diff --git a/src/target/mod.rs b/src/target/mod.rs index 2b00d0b..da9541c 100644 --- a/src/target/mod.rs +++ b/src/target/mod.rs @@ -676,6 +676,13 @@ pub trait Target { fn support_auxv(&mut self) -> Option> { None } + + /// Support for reading a list of libraries for SVR4 (System-V/Unix) + /// platforms. + #[inline(always)] + fn support_libraries_svr4(&mut self) -> Option> { + None + } } macro_rules! __delegate {