Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Tracepoint extension support #160

Merged
merged 54 commits into from
Mar 2, 2025

Conversation

cczetier
Copy link
Contributor

@cczetier cczetier commented Dec 11, 2024

Description

This PR adds basic tracepoint extension support to GDB stub. Closes #157.

API Stability

  • This PR does not require a breaking API change

Checklist

  • Documentation
    • Ensured any public-facing rustdoc formatting looks good (via cargo doc)
    • (if appropriate) Added feature to "Debugging Features" in README.md
  • Validation
    • Included output of running examples/armv4t with RUST_LOG=trace + any relevant GDB output under the "Validation" section below
    • Included output of running ./example_no_std/check_size.sh before/after changes under the "Validation" section below
  • If implementing a new protocol extension IDET
    • Included a basic sample implementation in examples/armv4t
    • IDET can be optimized out (confirmed via ./example_no_std/check_size.sh)
    • OR implementation requires introducing non-optional binary bloat (please elaborate under "Description")
  • If upstreaming an Arch implementation
    • I have tested this code in my project, and to the best of my knowledge, it is working as intended.

Validation

GDB output
(gdb) trace test.c:10
Tracepoint 1 at 0x55550040: file test.c, line 10.
(gdb) break *0x55550044
Breakpoint 2 at 0x55550044: file test.c, line 10.
(gdb) break main
Breakpoint 3 at 0x5555000c: file test.c, line 2.
(gdb) r
The program being debugged has been started already.
Start it from the beginning? (y or n) y
Starting program: target:/test.elf 
Reading /test.elf from remote target...

Breakpoint 3, main () at test.c:2
2	in test.c
(gdb) tstart
warning: Target does not support trace user/notes, info ignored
(gdb) c
Continuing.

Breakpoint 2, 0x55550044 in main () at test.c:10
10	in test.c
(gdb) tstop
warning: Target does not support trace notes, note ignored
(gdb) tstatus
Trace stopped for an unknown reason.
Collected 1 trace frames.
Trace will stop if GDB disconnects.
Not looking at any trace frame.
(gdb) info tracepoints
Num     Type           Disp Enb Address    What
1       tracepoint     keep y   0x55550040 in main at test.c:10
	tracepoint already hit 1 time
	installed on target
(gdb) tfind
Found trace frame 0, tracepoint 1
#0  main () at test.c:10
10	in test.c
(gdb) i r
r0             0x0                 0
r1             0x0                 0
r2             0x0                 0
r3             0x0                 0
r4             0x0                 0
r5             0x0                 0
r6             0x0                 0
r7             0x0                 0
r8             0x0                 0
r9             0x0                 0
r10            0x0                 0
r11            0xffffffc           268435452
r12            0x0                 0
sp             0xfffffe8           0xfffffe8
lr             0x12345678          305419896
pc             0x55550040          0x55550040 <main+64>
cpsr           0x80000010          -2147483632
custom         0x12345678          305419896
time           0x3a87325d          981938781
unavailable    <unavailable>
loading section ".text" into memory from [0x55550000..0x55550078] Setting PC to 0x55550000 Waiting for a GDB connection on "127.0.0.1:9001"...

(The start of the cargo run output is corrupted by binary data that's printed, so I cut out the portion of the output relevant for tracepoint packets)

TRACE gdbstub::protocol::recv_packet     > <-- $qTStatus#49
 TRACE gdbstub::protocol::response_writer > --> $T0;tframes:00#4b
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $08301be5#f8
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,4#69
 TRACE gdbstub::protocol::response_writer > --> $013083e2#c6
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,4#94
 TRACE gdbstub::protocol::response_writer > --> $0430a0e3#f0
 TRACE gdbstub::protocol::recv_packet     > <-- $qTStatus#49
 TRACE gdbstub::protocol::response_writer > --> $T0;tframes:00#4b
 TRACE gdbstub::protocol::recv_packet     > <-- $Z0,5555000c,4#dd
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $Z0,55550044,4#b2
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $qTStatus#49
 TRACE gdbstub::protocol::response_writer > --> $T0;tframes:00#4b
 TRACE gdbstub::protocol::recv_packet     > <-- $vCont;c:p1.-1#0f
 TRACE gdbstub::stub::state_machine       > transition: "Idle<armv4t::emu::Emu>" --> "Running"
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550000, inst: 0xe52db004, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550004, inst: 0xe28db000, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: DataProc2
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550008, inst: 0xe24dd014, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: DataProc2
 TRACE gdbstub::protocol::response_writer > --> $T05thread:p01.01;swbreak:;#6a
 TRACE gdbstub::stub::state_machine       > transition: "Running" --> "Idle<armv4t::emu::Emu>"
 TRACE gdbstub::protocol::recv_packet     > <-- $g#67
 TRACE gdbstub::protocol::response_writer > --> $0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000fcffff0f00000000e8ffff0f785634120c005555xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx1000000078563412#06
 TRACE gdbstub::protocol::recv_packet     > <-- $qfThreadInfo#bb
 TRACE gdbstub::protocol::response_writer > --> $mp01.01#cd
 TRACE gdbstub::protocol::recv_packet     > <-- $qsThreadInfo#c8
 TRACE gdbstub::protocol::response_writer > --> $l#6c
 TRACE gdbstub::protocol::recv_packet     > <-- $z0,5555000c,4#fd
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $z0,55550044,4#d2
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,4#94
 TRACE gdbstub::protocol::response_writer > --> $0430a0e3#f0
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550008,4#69
 TRACE gdbstub::protocol::response_writer > --> $14d04de2#28
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,4#94
 TRACE gdbstub::protocol::response_writer > --> $0430a0e3#f0
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550008,4#69
 TRACE gdbstub::protocol::response_writer > --> $14d04de2#28
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,2#92
 TRACE gdbstub::protocol::response_writer > --> $0430#c7
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000a,2#90
 TRACE gdbstub::protocol::response_writer > --> $4de2#2f
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550008,2#67
 TRACE gdbstub::protocol::response_writer > --> $14d0#f9
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,2#92
 TRACE gdbstub::protocol::response_writer > --> $0430#c7
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000a,2#90
 TRACE gdbstub::protocol::response_writer > --> $4de2#2f
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550008,2#67
 TRACE gdbstub::protocol::response_writer > --> $14d0#f9
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,4#94
 TRACE gdbstub::protocol::response_writer > --> $0430a0e3#f0
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550008,4#69
 TRACE gdbstub::protocol::response_writer > --> $14d04de2#28
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,4#94
 TRACE gdbstub::protocol::response_writer > --> $0430a0e3#f0
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550008,4#69
 TRACE gdbstub::protocol::response_writer > --> $14d04de2#28
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,4#94
 TRACE gdbstub::protocol::response_writer > --> $0430a0e3#f0
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,4#94
 TRACE gdbstub::protocol::response_writer > --> $0430a0e3#f0
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555000c,4#94
 TRACE gdbstub::protocol::response_writer > --> $0430a0e3#f0
 TRACE gdbstub::protocol::recv_packet     > <-- $QTinit#59
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $QTDP:1:0000000055550040:E:0:0#49
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $QTDPsrc:1:55550040:at:0:9:746573742e633a3130#ea
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $QTro:55550000,55550078#23
 INFO  gdbstub::stub::core_impl           > Unknown command: Ok("QTro:55550000,55550078")
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $QTBuffer:circular:0#f8
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $QTBuffer:size:-1#8c
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $QTNotes:user:;notes:;#ba
 INFO  gdbstub::stub::core_impl           > Unknown command: Ok("QTNotes:user:;notes:;")
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $QTStart#b3
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $Z0,55550044,4#b2
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $qTStatus#49
 TRACE gdbstub::protocol::response_writer > --> $T1;tframes:00#4c
 TRACE gdbstub::protocol::recv_packet     > <-- $vCont;s:p1.1#f2
 TRACE gdbstub::stub::state_machine       > transition: "Idle<armv4t::emu::Emu>" --> "Running"
 TRACE armv4t_emu::arm                    > ARM: pc: 0x5555000c, inst: 0xe3a03004, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: DataProc2
 TRACE gdbstub::protocol::response_writer > --> $S05#b8
 TRACE gdbstub::stub::state_machine       > transition: "Running" --> "Idle<armv4t::emu::Emu>"
 TRACE gdbstub::protocol::recv_packet     > <-- $g#67
 TRACE gdbstub::protocol::response_writer > --> $0000000000000000000000000400000000000000000000000000000000000000000000000000000000000000fcffff0f00000000e8ffff0f7856341210005555xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx1000000078563412#7d
 TRACE gdbstub::protocol::recv_packet     > <-- $Z0,5555000c,4#dd
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $qTStatus#49
 TRACE gdbstub::protocol::response_writer > --> $T1;tframes:00#4c
 TRACE gdbstub::protocol::recv_packet     > <-- $vCont;c:p1.-1#0f
 TRACE gdbstub::stub::state_machine       > transition: "Idle<armv4t::emu::Emu>" --> "Running"
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550010, inst: 0xe50b3008, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550014, inst: 0xe3a03003, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: DataProc2
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550018, inst: 0xe50b3010, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE armv4t_emu::arm                    > ARM: pc: 0x5555001c, inst: 0xe51b3008, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550020, inst: 0xe2833001, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: DataProc2
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550024, inst: 0xe50b3008, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550028, inst: 0xe51b3010, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE armv4t_emu::arm                    > ARM: pc: 0x5555002c, inst: 0xe2833003, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: DataProc2
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550030, inst: 0xe50b3010, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550034, inst: 0xe3a03000, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: DataProc2
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550038, inst: 0xe50b300c, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE armv4t_emu::arm                    > ARM: pc: 0x5555003c, inst: 0xea000005, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: Branch
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550058, inst: 0xe51b300c, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE armv4t_emu::arm                    > ARM: pc: 0x5555005c, inst: 0xe3530902, cond: 0xe, cflags: 0000
 TRACE armv4t_emu::arm                    > Instruction: DataProc2
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550060, inst: 0xbafffff6, cond: 0xb, cflags: 1000
 TRACE armv4t_emu::arm                    > Instruction: Branch
 TRACE armv4t_emu::arm                    > ARM: pc: 0x55550040, inst: 0xe51b3008, cond: 0xe, cflags: 1000
 TRACE armv4t_emu::arm                    > Instruction: SingleXferI
 TRACE gdbstub::protocol::response_writer > --> $T05thread:p01.01;swbreak:;#6a
 TRACE gdbstub::stub::state_machine       > transition: "Running" --> "Idle<armv4t::emu::Emu>"
 TRACE gdbstub::protocol::recv_packet     > <-- $g#67
 TRACE gdbstub::protocol::response_writer > --> $0000000000000000000000000500000000000000000000000000000000000000000000000000000000000000fcffff0f00000000e8ffff0f7856341244005555xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx1000008078563412#8d
 TRACE gdbstub::protocol::recv_packet     > <-- $qfThreadInfo#bb
 TRACE gdbstub::protocol::response_writer > --> $mp01.01#cd
 TRACE gdbstub::protocol::recv_packet     > <-- $qsThreadInfo#c8
 TRACE gdbstub::protocol::response_writer > --> $l#6c
 TRACE gdbstub::protocol::recv_packet     > <-- $z0,5555000c,4#fd
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $z0,55550044,4#d2
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,4#69
 TRACE gdbstub::protocol::response_writer > --> $013083e2#c6
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $08301be5#f8
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,4#69
 TRACE gdbstub::protocol::response_writer > --> $013083e2#c6
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $08301be5#f8
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,2#67
 TRACE gdbstub::protocol::response_writer > --> $0130#c4
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550042,2#65
 TRACE gdbstub::protocol::response_writer > --> $1be5#2d
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,2#63
 TRACE gdbstub::protocol::response_writer > --> $0830#cb
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,2#67
 TRACE gdbstub::protocol::response_writer > --> $0130#c4
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550042,2#65
 TRACE gdbstub::protocol::response_writer > --> $1be5#2d
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,2#63
 TRACE gdbstub::protocol::response_writer > --> $0830#cb
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,4#69
 TRACE gdbstub::protocol::response_writer > --> $013083e2#c6
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $08301be5#f8
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,4#69
 TRACE gdbstub::protocol::response_writer > --> $013083e2#c6
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $08301be5#f8
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,4#69
 TRACE gdbstub::protocol::response_writer > --> $013083e2#c6
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,4#69
 TRACE gdbstub::protocol::response_writer > --> $013083e2#c6
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550044,4#69
 TRACE gdbstub::protocol::response_writer > --> $013083e2#c6
 TRACE gdbstub::protocol::recv_packet     > <-- $QTStop#4b
 TRACE gdbstub::protocol::response_writer > --> $OK#9a
 TRACE gdbstub::protocol::recv_packet     > <-- $QTNotes:tstop:;#97
 INFO  gdbstub::stub::core_impl           > Unknown command: Ok("QTNotes:tstop:;")
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $qTStatus#49
 TRACE gdbstub::protocol::response_writer > --> $T0;tframes:01#4c
 TRACE gdbstub::protocol::recv_packet     > <-- $qTP:1:55550040#52
 TRACE gdbstub::protocol::response_writer > --> $V01:00#51
 TRACE gdbstub::protocol::recv_packet     > <-- $QTFrame:0#fa
 TRACE gdbstub::protocol::response_writer > --> $F00T01#5b
 TRACE gdbstub::protocol::recv_packet     > <-- $QTFrame:0#fa
 TRACE gdbstub::protocol::response_writer > --> $F00T01#5b
 TRACE gdbstub::protocol::recv_packet     > <-- $g#67
 TRACE gdbstub::protocol::response_writer > --> $0000000000000000000000000000000000000000000000000000000000000000000000000000000000000000fcffff0f00000000e8ffff0f7856341240005555xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx1000008078563412#df
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555003c,4#97
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555003c,4#97
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,2#63
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555003e,2#97
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555003c,2#95
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,2#63
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555003e,2#97
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555003c,2#95
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555003c,4#97
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m5555003c,4#97
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $m55550040,4#65
 TRACE gdbstub::protocol::response_writer > --> $#00
 TRACE gdbstub::protocol::recv_packet     > <-- $p1b#03
 TRACE gdbstub::protocol::response_writer > --> $5d32873a#01
 TRACE gdbstub::protocol::recv_packet     > <-- $p1c#04
 TRACE gdbstub::protocol::response_writer > --> $xxxxxxxx#c6

Before/After `./example_no_std/check_size.sh` output

Before

Analyzing target/release/gdbstub-nostd

File  .text    Size          Crate Name
2.4%  71.2% 11.9KiB      [Unknown] main
0.2%   5.7%    969B        gdbstub gdbstub::stub::state_machine::GdbStubStateMachineInner<gdbstub::stub::state_machine::state::Running,T,C>::report_stop
0.1%   2.2%    374B        gdbstub gdbstub::protocol::commands::breakpoint::BasicBreakpoint::from_slice
0.1%   1.7%    295B        gdbstub <gdbstub::protocol::common::thread_id::ThreadId as core::convert::TryFrom<&[u8]>>::try_from
0.1%   1.7%    295B        gdbstub gdbstub::stub::core_impl::resume::<impl gdbstub::stub::core_impl::GdbStubImpl<T,C>>::write_stop_common
0.1%   1.6%    278B        gdbstub gdbstub::protocol::packet::PacketBuf::new
0.1%   1.6%    277B        gdbstub gdbstub::protocol::common::hex::decode_hex_buf
0.1%   1.6%    273B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write_specific_thread_id
0.0%   1.2%    213B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write
0.0%   0.8%    132B        gdbstub gdbstub::protocol::common::hex::decode_hex
0.0%   0.7%    122B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::inner_write
0.0%   0.6%    111B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::flush
0.0%   0.6%    107B        gdbstub gdbstub::protocol::common::hex::decode_hex
0.0%   0.6%    106B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadBase>::read_addrs
0.0%   0.6%    104B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write_num
0.0%   0.6%    101B      [Unknown] __libc_csu_init
0.0%   0.6%     97B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write_num
0.0%   0.6%     95B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write_hex
0.0%   0.5%     92B        gdbstub <gdbstub::protocol::common::thread_id::IdKind as core::convert::TryFrom<&[u8]>>::try_from
0.0%   0.5%     87B        gdbstub <usize as gdbstub::internal::be_bytes::BeBytes>::from_be_bytes
0.0%   0.5%     85B        gdbstub <u32 as gdbstub::internal::be_bytes::BeBytes>::from_be_bytes
0.0%   0.4%     71B   gdbstub_arch <gdbstub_arch::arm::reg::arm_core::ArmCoreRegs as gdbstub::arch::Registers>::gdb_deserialize
0.0%   0.4%     65B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadBase>::write_addrs
0.0%   0.4%     65B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadBase>::write_registers
0.0%   0.4%     65B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadBase>::read_registers
0.0%   0.3%     50B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadResume>::set_resume_action_cont...
0.0%   0.3%     50B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadResume>::clear_resume_actions
0.0%   0.3%     50B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadResume>::resume
0.0%   0.3%     43B      [Unknown] _start
0.0%   0.0%      6B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::breakpoints::SwBreakpoint>::add_sw_breakpoint
0.0%   0.0%      2B      [Unknown] __libc_csu_fini
3.4% 100.0% 16.7KiB                .text section size, the file size is 488.4KiB
target/release/gdbstub-nostd  :
section               size      addr
.interp                 28       568
.note.ABI-tag           32       596
.note.gnu.build-id      36       628
.gnu.hash               28       664
.dynsym                360       696
.dynstr                193      1056
.gnu.version            30      1250
.gnu.version_r          48      1280
.rela.dyn              408      1328
.init                   23      1736
.plt                    16      1760
.plt.got                 8      1776
.text                17106      1792
.fini                    9     18900
.rodata                920     18912
.eh_frame_hdr          276     19832
.eh_frame             1488     20112
.init_array              8   2121128
.fini_array              8   2121136
.dynamic               448   2121144
.got                   136   2121592
.data                    8   2121728
.bss                     8   2121736
.comment                93         0
Total                21718

After

Analyzing target/release/gdbstub-nostd

File  .text    Size          Crate Name
2.3%  71.2% 11.9KiB      [Unknown] main
0.2%   5.7%    969B        gdbstub gdbstub::stub::state_machine::GdbStubStateMachineInner<gdbstub::stub::state_machine::state::Running,T,C>::report_stop
0.1%   2.2%    374B        gdbstub gdbstub::protocol::commands::breakpoint::BasicBreakpoint::from_slice
0.1%   1.7%    295B        gdbstub <gdbstub::protocol::common::thread_id::ThreadId as core::convert::TryFrom<&[u8]>>::try_from
0.1%   1.7%    295B        gdbstub gdbstub::stub::core_impl::resume::<impl gdbstub::stub::core_impl::GdbStubImpl<T,C>>::write_stop_common
0.1%   1.6%    278B        gdbstub gdbstub::protocol::packet::PacketBuf::new
0.1%   1.6%    277B        gdbstub gdbstub::protocol::common::hex::decode_hex_buf
0.1%   1.6%    273B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write_specific_thread_id
0.0%   1.2%    213B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write
0.0%   0.8%    132B        gdbstub gdbstub::protocol::common::hex::decode_hex
0.0%   0.7%    122B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::inner_write
0.0%   0.6%    111B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::flush
0.0%   0.6%    107B        gdbstub gdbstub::protocol::common::hex::decode_hex
0.0%   0.6%    106B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadBase>::read_addrs
0.0%   0.6%    104B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write_num
0.0%   0.6%    101B      [Unknown] __libc_csu_init
0.0%   0.6%     97B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write_num
0.0%   0.6%     95B        gdbstub gdbstub::protocol::response_writer::ResponseWriter<C>::write_hex
0.0%   0.5%     92B        gdbstub <gdbstub::protocol::common::thread_id::IdKind as core::convert::TryFrom<&[u8]>>::try_from
0.0%   0.5%     87B        gdbstub <usize as gdbstub::internal::be_bytes::BeBytes>::from_be_bytes
0.0%   0.5%     85B        gdbstub <u32 as gdbstub::internal::be_bytes::BeBytes>::from_be_bytes
0.0%   0.4%     71B   gdbstub_arch <gdbstub_arch::arm::reg::arm_core::ArmCoreRegs as gdbstub::arch::Registers>::gdb_deserialize
0.0%   0.4%     65B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadBase>::write_addrs
0.0%   0.4%     65B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadBase>::write_registers
0.0%   0.4%     65B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadBase>::read_registers
0.0%   0.3%     50B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadResume>::set_resume_action_cont...
0.0%   0.3%     50B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadResume>::clear_resume_actions
0.0%   0.3%     50B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::base::multithread::MultiThreadResume>::resume
0.0%   0.3%     43B      [Unknown] _start
0.0%   0.0%      6B gdbstub_nostd? <gdbstub_nostd::gdb::DummyTarget as gdbstub::target::ext::breakpoints::SwBreakpoint>::add_sw_breakpoint
0.0%   0.0%      2B      [Unknown] __libc_csu_fini
3.3% 100.0% 16.7KiB                .text section size, the file size is 506.0KiB
target/release/gdbstub-nostd  :
section               size      addr
.interp                 28       568
.note.ABI-tag           32       596
.note.gnu.build-id      36       628
.gnu.hash               28       664
.dynsym                360       696
.dynstr                193      1056
.gnu.version            30      1250
.gnu.version_r          48      1280
.rela.dyn              408      1328
.init                   23      1736
.plt                    16      1760
.plt.got                 8      1776
.text                17090      1792
.fini                    9     18884
.rodata                920     18896
.eh_frame_hdr          276     19816
.eh_frame             1488     20096
.init_array              8   2121128
.fini_array              8   2121136
.dynamic               448   2121144
.got                   136   2121592
.data                    8   2121728
.bss                     8   2121736
.comment                93         0
Total                21702

@cczetier
Copy link
Contributor Author

I wasn't sure how the zero panic verification guarantee can be checked - there isn't a how to section in the README about it. I tried to write code as panic-free as I could, but I may have missed some pieces.

@daniel5151
Copy link
Owner

Thanks for sending in this PR - I'm excited to dig in here!

Unfortunately, I'm just about to leave for a vacation where I'll be completely AFK, so I'll only be able to take a look sometime in the range of ~Dec 20th to ~Dec 23rd.

Just wanted to give you a heads up, so that you're not worried about the lack of movement here 😅

Copy link
Owner

@daniel5151 daniel5151 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hello again, and happy holidays!

I've finally found some time to sit down and give this PR a review.
Please find a plethora of comments attached.


Broadly speaking, I think this is an impressive chunk of work, implementing what appears to be a very annoying and non-trivial part of the GDB RSP. From an organizational and syntactical POV, there's nothing that a few review comments can't polish up, and the overall vision here appears to be consistent and well put together. Kudos!

That said, I do have some concerns about the amount of API surface area we're taking on here, and the feasibility of testing all of it. Its great that the you've got some things working in the armv4t example, but from looking at the code (notably: obvious errors such as handlers which send responses with spaces delimiting various part of the packet), it seems that you've got a lot of code here that theoretically works, but hasn't been directly validated.

That's not to say we shouldn't try to land this work!

I think we should certainly try to get this PR landed... but to temper expectations for end-users, I might suggest we land this code under a mod experimental, with some documentation mentioning that this code covers a lot of surface area, and may not be fully tested.

Alternatively, if you're so inclined, I'd be happy to see more investment into the armv4t example code, with some corresponding logs that show all these codepaths having been smoke-tested. Or, of course, logs from whatever project you're implementing this feature for (and ideally, a link to the implementation itself - assuming you're working on something open-source).


I wasn't sure how the zero panic verification guarantee can be checked - there isn't a how to section in the README about it. I tried to write code as panic-free as I could, but I may have missed some pieces.

CI has a check for this, but it seems that the check doesn't run if clippy fails. Oops.

I should probably re-jig CI a bit so that clippy failing still gives no_panic feedback... my apologies.

Comment on lines 41 to 43
// Our response has to be a hex encoded buffer that fits within
// our packet size, which means we actually have half as much space
// as our slice would indicate.
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

so, interestingly enough, this is not the approach gdbstub has taken thus-far when it comes to responses.

While we certainly have a fixed buffer for incoming packets (i.e: the buffer we are parsing from here), we assume that the GDB client is capable of accepting any amount of data we stream out as part of our responses. This aligns with my particular reading of the GDB spec, which discusses the size of packets the stub can accept... but doesn't make a judgement on the size of response packets the client can receive.

as such - I would suggest skipping this buffer-slicing step entirely, and instead offering the handler a callback function they can write an arbirarily sized &[u8] into, which gdbstub can then simply stream out. If you poke around some of the other target APIs, you'll see examples of this callback-based / streaming-based pattern being employed to great effect.

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

alternatively, if you think that from a purely ergonomic POV, its nicer to provide end-users with a buffer to write data into... lets make sure to pass along the entire buffer we have access to, and let downstream response-writer code stream out the corresponding hex bytes.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The tracepoint packet documentation specifically calling out

The reply consists as many hex-encoded bytes as the target can deliver in a packet

makes me think that limiting our response to one packet is important and we can't just stream. Some other places like e.g.

Any number of actions may be packed together in a single ‘QTDP’ packet, as long as the packet does not exceed the maximum packet length (400 bytes, for many stubs).

I could see it not being important for, however, if we're the stub and because the QTDP packets are bidirectional...but QTBuffer responses are only ever being returned by the stub.

The packet buffer size in general is kind of confusing to me. Like you said, in most other places gdbstub just ignores the concept of a packet size and assumes it can stream data and it works fine, but then we have documentation like these occasionally...?

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think its fair to say that both the GDB RSP and gdbstub aren't always consistent about how these sorts of variable-length packets are implemented...

There are many instances where gdbstub has opted to recycle the trailing packet buffer in order to provide the Target a &mut [u8] it can write data into... while other times, it has opted to hand a callback-based object to the Target for it to write data into. But even in the former case, as I alluded to earlier / elsewhere, the data written in that buffer is nonetheless streamed out via a sequence of what are essentially putc calls, without any bound on output packet size.

As I mull over this further, I'm realizing that I should really spin up a tracking issue to document and discuss potential solutions for these sorts of project-wide inconsistencies. It may even tie into #159, and possibly extend #88, in the sense that gdbstub may need to care more about dealing with backpressure on outbound write operations...

But in any case, for this PR and packet specifically - I'm not sure I'll push hard in either direction. Feel free to do what you think is best, and I'll add this API to the growing list of APIs that aught to be more consistent.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Removed.

@cczetier
Copy link
Contributor Author

Thanks for the feedback! I'll look over and resolve them. For some background, we're using gdbstub in order to build out debugging introspection for a project we have. Our current tooling is via Python, and we're using gdbstub via some PyO3 bindings I hacked together (which I do intend to open source eventually, once I find the time), but that also isn't very "interesting" code. I'm working on this tracepoint support in tandem with building out the rest of the debugging stack, and so the API surface here was the minimal amount I needed in order to get gdb to not error out with "not supported" and implement the tracepoint functionality. We do have functionality using this, however, so none of it should be untested code (although some parts, like trace_buffer_request, we're also stubbing out in our closed source implementation similarly to the armv4t example).

@cczetier
Copy link
Contributor Author

cczetier commented Jan 6, 2025

I added additional error handling to DefineTracepoint::actions: it was returning an Option before instead of a Result, so it was possible for target implementations to ignore the fact that it returned None indicating error. Bubbling the PacketParseError is fairly ugly, however, and requires the user to write a map_err case for the result - is there a more idiomatic method it should look like instead?

@daniel5151
Copy link
Owner

Hey! Just wanted to drop in and mention that I see all the commits flying here, and that I'll definitely try to find some time soon to look into them, hopefully at some point in the next couple days before the weekend. Things have gotten unexpectedly busy on my end now that the new year is back in full swing... but I'll try to give timely feedback here to keep the ball rolling.

Thanks for all the work an iteration you're doing here!

@daniel5151
Copy link
Owner

Hey, really sorry for the lack of activity here. A ton of personal and work stuff has come up over the past few weeks, and I've yet to find a moment of focus time to give feedback here... My apologies :(

I'm going to try and set aside some time this weekend to give this a review.

Copy link
Owner

@daniel5151 daniel5151 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've spent some time looking at this PR, as well as the "Tracepoint Packets" part of the GDB RSP spec, and I think I've got a good idea / guess behind the how-and-why of the Create/Define/Query packet spec.

My take on the matter is that the GDB RSP didn't want to have a single "mega-packet" to define tracepoints, as it would've more likely than not blown past the packet limit set by various stubs. Instead, it opts to "drip-feed" the full context behind a particular tracepoint over the span of several QTDP packets, relying on the stub to gradually build up a full picture of the tracepoint its expected to create.

This is backed up by the careful wording in the QTDP:- packet:

This packet may only be sent immediately after another ‘QTDP’ packet that ended with a ‘-’

...implying that what we're really dealing with here is one big composite "create a new tracepoint" packet, as opposed to two distinct "create" and "define" tracepoint packets!

As such, I'm not convinced tracepoint_create and tracepoint_define expose the right level of abstraction we want in the API. Instead - I think we want to have methods like tracepoint_create_begin, tracepoint_create_continue, and then one final (synthetic) tracepoint_create_complete, and then later, tracepoint_enumerate_start and tracepoint_enumerate_step should operate on NewTracepoint and DefineTracepoint, instead of this "synthetic" TracepointItem type (which should go away).

Fundamentally, I don't believe we want end-user implementations to directly store the bundles of info contained within NewTracepoint and DefineTracepoint. i.e: storing a Vec<TracepointItem> directly, and then operating on that representation in other parts of the tracepoint API, seems like the wrong approach for end-users.

Instead, I believe the API methods / types (and in turn, the arvm4t example that shows off how they are used) should reflect the true intent of this API: that user-defined stubs are responsible for converting between the drip-fed "wire protocol" data representation of tracepoints (as encapsulated by the NewTracepoint and DefineTracepoint types), into a stub-specific Tracepoint type.

Similarly, when implementing tracepoint_enumerate_start and tracepoint_enumerate_step, users shouldn't just report some long-lived stashed reprs of NewTracepoint and DefineTracepoint - they should instead create them "on the fly" using whatever tracepoint representation it "committed" to as part of tracepoint_create_complete.

The key ethos behind gdbstub is that we aren't just in the business of munging the raw GDB RSP and presenting it to the end user. Rather - we interpret the intent behind the operations the GDB RSP is modeling, and present a new, sane API on-top of the base protocol semantics, making it easy for end users to interface with GDB without having to deal with much of the underlying "insanity".


Does this reasoning make sense?

That's not a rhetorical question or anything - I'd like to get your honest thoughts on the matter here. I'm just making an educated assumption based on my experience with the rest of the GDB RSP, but you're the one who's actually using this in a real project.


I didn't finish reviewing the whole PR, but given the cascading refactors that might fall out of this observation, it might make sense for me to hold off until we iterate some more here.

Apologies again for the delay in getting this feedback, and for not pointing this out sooner. I wish I could've found some time earlier to sit down and page in all the context here, but its been quite a busy January...

Let me know what you think, and where we want to go from here.

@cczetier
Copy link
Contributor Author

Fundamentally, I don't believe we want end-user implementations to directly store the bundles of info contained within NewTracepoint and DefineTracepoint. i.e: storing a Vec directly, and then operating on that representation in other parts of the tracepoint API, seems like the wrong approach for end-users.

I think this is in reality a fairly small difference in perspectives. In our project we aren't storing a Vec<TracepointItem>, or the stream of NewTracepoint/DefineTracepoint we get to echo back: we are creating an internal target specific tracepoint type with a target specific actions list, and use that to reconstruct the stream for enumeration like you say. The armv4t example saves them as full TracepointItems because that was the most minimal way of adding support without adding an internal tracepoint datatype and de/reserialization logic like we do internally - if it's more illustrative I can change that so that the internal representation is enough to reconstruct a TracepointItem without saving off the entire thing.

You're right in that tracepoint_create and tracepoint_define are "synthetic" in that in reality we are declaratively given an entire tracepoint at once and it is split across multiple packets - but the tracepoint_create_begin/_continue/_complete API is splitting that up into multiple synthetic steps in the same way that tracepoint_create/tracepoint_define does. We could refactor it so that if the define items don't have more then we additionally call a "complete" method to signal the tracepoint item definition is finished, but that is to be a difference of inches and not miles from the current API.

tracepoint_enumerate_start and tracepoint_enumerate_step should operate on NewTracepoint and DefineTracepoint, instead of this "synthetic" TracepointItem type (which should go away).

Switching to the alternative API surface doesn't get rid of TracepointItem. Stepping the enumeration state machine needs for the target to be able to give us both NewTracepoint or DefineTracepoint like you say, and the synthetic TracepointItem enum is the way that I went about doing that. To be clear, there's only one tracepoint_enumerate_start for the entire state machine, not one for each NewTracepoint: we can't have start return a NewTracepoint and step return a DefineTracepoint.

@daniel5151
Copy link
Owner

Note that the arvm4t example is actually quite important to landing features in gdbstub, as it serves as the "baseline" for how a particular API should be used, which is not only useful after a feature lands (to help others ramp-up on some particular APIs), but also during development, as a way for us to have a common baseline of how this part of the protocol is supposed to work. There have been other times when I was able to do that sort of collaboration in the context of the contributor's open-source project, but since I can't actually see your private integration, I'm having to riff off the proposed API, and their use in the armv4t example 😅

All that is to say - we can (and should!) cut corners on the armvt4 example implementation, but I want to make sure that some of the fundamental patterns of how the API should be used are presented clearly - both in the code, and in the inline code comments, pointing out when corners are being cut.


Regarding the create path - you're correct that the necessary code-level tweaks aren't that significant (inches, not miles). The key thing I want to refine here is the clarity of the resulting API, both via its documentation, but also via method naming, and the types that are used in the API.

On the enumerate path, indeed, I think its pretty fair to say that the specific thing concerning me about the current API is the TracepointItem enum. I now understand why you've structured it this way - it really comes down to the fact that there's no great way for gdbstub to maintain an internal state machine representation to handle the qTfP and qTsP packet responses.

But what if there was a way?

Here's the idea I just had:

  • lets add a new Target API tracepoint_enumerate_state() -> &mut TracepointEnumerateState, where struct TracepointEnumerateState { current_tracepoint: Option<Tracepoint> } (with a public new method)
    • i.e: the Target is responsible for allocating this gdbstub-defined state somewhere on themselves, and managing its memory
  • lets update tracepoint_enumerate_start is updated to take a Option<Tracepoint> as a parameter (pulled ad-hoc, via tracepoint_enumerate_state in the handler impl), and have it use NewTracepoint
    • If None, this means "enumerate from the start"
    • If Some, this means "report the specified tracepoint"
  • lets update tracepoint_enumerate_step to take a Tracepoint parameter (pulled ad-hoc, via tracepoint_enumerate_state in the handler impl), and have it use DefineTracepoint, and then have it return a enum TracepointEnumerateStep { More, Next(Tracepoint), Done }

I believe this is a reasonable approach that avoids having GdbStub itself store extra Tracepoint-specific state-machine state in the general-case, while nonetheless still letting gdbstub maintain a state machine on the user's behalf (albeit with a bit of extra plumbing on the user-side).

The reason I think this approach can work is that Option<Tracepoint> is a simple, Copy-able object, that the gdbstub handler code can easily maintain a copy of as part of the packet-handler's local stack-frame. This approach likely wouldn't work for more complex state (e.g: something that would require gdbstub to take a long-lived reference to part of TracepointEnumerateState), but should be viable here?

What do you think?

@cczetier
Copy link
Contributor Author

I implemented the target provided state machine code, which has the benefit of the target not needing to keep the state for enumeration at all, although the target code is a bit more gross for calculating the TracepointEnumerationStep logic. However, when testing it turns out that our internal repo never actually returned any tracepoints when enumerating...which reveals that the "FIXME add detailed syntax" line in the documentation is actually fairly important: they don't actually use QTDP or need the - for more actions so it was wrong, although it does probably means we can nix that field as part of the API.

Unfortunately, trying to enumerate uploaded tracepoints with any actions (including just collect $regs!) errors "warning: Uploaded tracepoint 2 actions have no source form, ignoring them". This is, as far as I can tell, is because we didn't support the (optional!) feature TracepointSource and so don't handle QTDPsrc packets, and the throw away line in its description about qTfP/qTsP echoing them back is load bearing: without the source description GDB doesn't know how to process the actions when enumerating.

I added QTDPsrc support and enumerating those source items as part of the refactor. This complicates the state machine logic a bit, but is still workable. I need to do some more work gating these bits behind a supports_tracepoint_source IDET trait, since not all targets probably will care about supporting source so GDB can download actions, however.

The documentation for the entire enumeration machinery is kind of unwieldy, but I'm also not entirely sure how to make it better...

Copy link
Owner

@daniel5151 daniel5151 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I really like this latest iteration! I think we're getting super close to landing this PR. Thanks again for all the iterations and back and forth here - I really appreciate it.

I've got a few more questions / comments about the current API... but the broad shape and approach taken here looks to be quite solid.

// Report our tracepoint
(f)(&self.tracepoints[&tp].0);

let ret = if !self.tracepoints[&tp].1.is_empty() {
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

consider the following simplification to the enumeration API:

what if, instead of having each of enumerate_{start,action,source} use the exact same TracepointEnumerateStep (which requires target authors to carefully avoid accidentally "looping" between enumerating actions/sources), we instead simplified the API such that each method could only transition to the "next" category.

i.e: instead of the following state machine:

start -> source
start -> action
start -> done
source -> source
source -> action
source -> done
action -> source
action -> action
action -> done

we instead had

start -> source
source -> source
source -> action
action -> action
action -> done

...which guarantees forward-progress through the various states, and simplifies the target implementation substantially (no more multi-armed if statements).

the downside, of course, is that each tracepoint will always result in each of the 3 enumerate methods being invoked on it, without any ability to "short circuit".

I'll let you tell me if you think this sort of inneficiency will result in tangible slowdowns in the real world, but if the overhead is likely to be minimal... I'd err on the side of having an API that's less likely to be misused.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The current API is a bit misuse prone, in that it's possible to trap yourself in a loop around action/source enumeration, but I think any other API would be equally misuse prone in other ways:; you can always trap yourself in a loop over your tracepoints, for example, and if you had types of TracepointEnumerateStep { More, MoveTo(Tracepoint), Done } where More means you continue pumping the current type of item and Done means move to the next type, you still need to correctly implement Done vs. MoveTo logic or else you could MoveTo from actions and skip all your source items.

One approach would be if you have struct Start; struct Actions; struct Source; struct Done(Option<MoveTo>); and TracepointEnumerateStep<Current, Next> so that you can enforce that enumerate_source is the only transition that can MoveTo a new tracepoint or stop enumeration via parameterization, but that runs into issues around moving source enumeration to a separate IDET: we'd always need to have the enumerate_source function available and implemented, since we can't conditionally have the Next type parameter be either Source or Done depending on if the trait is implement or not (or, well, maybe with an associated generic on the trait, but that's gross).

I'm not worry about slowdown, tracepoint enumeration is fairly rare and only(?) happens on attach: on program we're using target extended-remote mode for multiple PIDs and so need the download functionality to install tracepoints in multiple PIDs at the same time, but switching between them and triggering the tracepoint download happens interactively for the user and isn't performance critical (at least for our usecase!).

Copy link
Contributor Author

@cczetier cczetier Feb 10, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Also, to be clear, the current API is like:

start -> source
source -> source
source -> action
action -> action
action -> next(1)
start -> source
source -> action
action -> action
action -> next(2)
start -> source
source -> action
action -> done

You don't return done multiple times, it signals that you have no more information at all, and it pumps each information type in a row before moving to the next type

Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fair enough! Happy to leave it here, and maybe noodle around with it myself whenever I finally find the time to start tackling the big 0.8 backlog

@cczetier
Copy link
Contributor Author

There was an issue where we were returning an ExperimentStatus with the required lifetime bound before, which hit the same issue as ExperimentExplanation previously: I moved it to a callback so that you can report a borrowed error string.

Copy link
Owner

@daniel5151 daniel5151 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm happy with this implementation! It was a long road to get here, and I really appreciate you sticking with me here - I think we landed on something quite solid here, especially given all the wackiness in the GDB RSP, as well as some of the architectural limitations currently in gdbstub. 🎉

The only things left are:

  1. A quick polish-pass through the docs (see my comments)
  2. Making sure the PR description is up-to-date. i.e: re-running the various manual validations for dead-code-elimination, updating the armv4t trace logs, etc...
  3. Making rustfmt and clippy happy :)

I'm super excited to get this in!

@cczetier
Copy link
Contributor Author

It's great this is getting close to being merged! I'm going to be working on another project for a while, and so if there are more final comments I may be slow to respond - I'm still interested in working to get this merged, however.

Copy link
Owner

@daniel5151 daniel5151 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Latest round of edits is looking good! Not much feedback here, aide from that one comment wrt. making a failure mode a loud error, instead of silently accepting it.

As a reminder, you'll still need to:

  • Make sure the PR description is up-to-date. i.e: re-running the various manual validations for dead-code-elimination, updating the armv4t trace logs, etc...
  • Make rustfmt and clippy happy, so the PR checks are all green

.handle_error()?,
)
} else {
// If the target doesn't support tracepoint sources but told
Copy link
Owner

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This feels like it should result in a Error, no? i.e: the only way this code path could be hit is due to a faulty Target implementation?

If so, lets upgrade this to a loud error that notifies the user of their mistake

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Added new error variant, and now throwing it here.

@cczetier
Copy link
Contributor Author

Latest round of edits is looking good! Not much feedback here, aide from that one comment wrt. making a failure mode a loud error, instead of silently accepting it.

As a reminder, you'll still need to:

* Make sure the PR description is up-to-date. i.e: re-running the various manual validations for dead-code-elimination, updating the `armv4t` trace logs, etc...

* Make `rustfmt` and `clippy` happy, so the PR checks are all green

I fixed up the clippy errors (or, well, suppressed them, which is almost the same thing...), along with updated the OP command with up-to-date logs.

@daniel5151
Copy link
Owner

Have you been running cargo clippy locally? I cloned your branch locally, and I can def repro the same lifetime elision issues that CI is reporting.

@cczetier
Copy link
Contributor Author

cczetier commented Mar 2, 2025

I am running clippy locally - I'm on clippy 0.1.75 (9d83ac2 2023-10-31), however, which might be the difference? I'll try to fix the remaining errors, sorry.

@daniel5151
Copy link
Owner

Ah, that's do it.

gdbstub's CI tracks the current stable rustc, which is on Rust version 1.85.0.

@cczetier
Copy link
Contributor Author

cczetier commented Mar 2, 2025

Running an up-to-date cargo clippy gives me the errors in the failed CI, but also a bunch of extra instances of the unused lifetime lint in existing code (e.g. impl<'a> ParseAnnex<'a> for LibrariesSvr4Annex in src/protocol/commands/_qXfer_libraries_svr4_read.rs) that don't show up. I should just ignore those and only fix the ones that show up in CI, but not sure if there's still some mismatch between the clippy I'm running locally and what's in the pipeline that will cause another error...

@cczetier cczetier force-pushed the tracepoint_public branch from 03698fc to fc191a2 Compare March 2, 2025 22:57
@cczetier
Copy link
Contributor Author

cczetier commented Mar 2, 2025

Yup, that was it. This gives me a clean cargo clippy now.

@cczetier
Copy link
Contributor Author

cczetier commented Mar 2, 2025

...And now also gives a clean cargo clippy --example=armv4t as well 😓

@daniel5151
Copy link
Owner

CI is all green!
Wooo!!! 🚀

Thanks again for seeing this though!

I know we've still got a way to go until gdbstub can support all the bells-and-whistles that tracepoints have to offer... but now that we have this baseline implementation working, it should be a lot easier to make incremental progress in this space.

I'll go ahead and merge this PR, and then cut a shiny new 0.7.4 release ASAP.

@daniel5151 daniel5151 merged commit ed43ae1 into daniel5151:master Mar 2, 2025
2 checks passed
@daniel5151
Copy link
Owner

Release 0.7.4 is out 🎉

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Tracepoints support
2 participants