2.15 GDB断点后处理commands中finish/until/tb带来的问题
https://scz.617.cn/unix/200901071528.txt
Q:
我现在拦截recvmsg(),有几个目的:
1) 获取UDP报文的源IP、源PORT 2) 观察接收到的数据
由于recvmsg()被频繁调用,我需要设置条件断点,比如当源IP、源PORT、接收到的 数据满足一定条件时再命中断点。对于recvmsg()来说,这些信息都是函数返回之后 才能获取的,因此我设想了这样一种断点设置方式:
b recvmsg
commands
silent
set $x=(struct msghdr )((unsigned int )($esp+8))
finish
if ($x->msg_name!=0&&$x->msg_namelen==16)
set $y=(struct sockaddr_in )$x->msg_name
if ( ntohl( $y->sin_addr.s_addr )==<...>)
但这个断点不能达到预期效果,commands中finish之后的脚本未被执行,可能是BUG, 也可能是GDB就是这样实现的。将finish换成until ((unsigned int )$esp),情 况依旧。后来我试图将finish换成tb ((unsigned int )$esp),并针对tb设置后 处理commands,但GDB不允许这样做:
Can't use the "commands" command among a breakpoint's commands.
用define也无法绕过上述限制。
D: scz 2021-04-22
现在GDB允许在commands中tb并再次使用commands,后面的技巧过时了。
A: qfp 2009-01-07 14:20
用如下办法变通满足原始需求:
define finish_recvmsg
finish
if ($x->msg_name!=0&&$x->msg_namelen==16)
set $y=(struct sockaddr_in *)$x->msg_name
if ( ntohl( $y->sin_addr.s_addr )==<...>)
b recvmsg commands silent set $x=(struct msghdr )((unsigned int *)($esp+8)) finish_recvmsg end
A: emacsen@SMTH 2009-01-07 12:33:14
emacsen建议用hook试试,用Google搜"gdb hook"找到类似资料:
define hook-echo // 定义echo命令的"前处理" echo [ end
define hookpost-echo // 定义echo命令的"后处理" echo ]\n end
(gdb) echo Hello World [Hello World]
据此测试了hookpost-finish,也能满足原始需求:
define hookpost-finish
if ($x->msg_name!=0&&$x->msg_namelen==16)
set $y=(struct sockaddr_in *)$x->msg_name
if ( ntohl( $y->sin_addr.s_addr )==<...>)
b recvmsg commands silent set $x=(struct msghdr )((unsigned int *)($esp+8)) finish end