0. 前言
gdb
是一个十分强大的调试器,而它又提供给python
扩展的gdb
模块。但是,该模块只有在gdb
中,才能被成功导入。
import gdb
运行上述代码,报错显示没有gdb
模块。因为只有在gdb
加载该python
文件时,才能运行成功:
(gdb) source xxx.py
为了解决这个问题,可以利用python
语言的强大,通过socket
套接字编写,在gdb
中开启一个服务器,从而到达外界与gdb
之间的交互。
1. 代码示例
import gdb
import socket
# TCP服务端
def main():
#创建一个socket对象,AF_INET指定使用IPv4协议(AF_INET6代表IPV6),SOCK_STREAM指定使用面向流的TCP协议
tcp_serve_socket = socket.socket(socket.AF_INET, socket.SOCK_STREAM)
tcp_serve_socket.bind(('localhost', 8001)) # 绑定本地ip和端口
tcp_serve_socket.listen(128) # 开始监听端口,数字表示等待连接的最大数量
# 循环目的:多次调用accept为多个客户服务
while True:
print('等待一个新请求到来....')
tcp_client_socket, client_addr = tcp_serve_socket.accept()
print('一个新请求到来,来自:%s' % str(client_addr))
recv_data = tcp_client_socket.recv(1024).decode('utf-8') # 从客户端接受消息,最多1024字节。recv_data为字节类型,.decode()将recv_data转化成字符型
print('请求发送的命令是%s' % recv_data)
retMsg = ''
if recv_data == 'start':
print('send start to gdb')
retMsg = gdb.execute('start', to_string=True)
elif recv_data == 'continue':
print('send continue to gdb')
retMsg = gdb.execute('c', to_string=True)
else:
print('no this commond')
print(retMsg)
tcp_client_socket.send(retMsg.encode('utf-8')) # 将字符串进行字节编码
# 关闭与客户端的连接
tcp_client_socket.close()
print('请求响应完毕')
tcp_serve_socket.close()
if __name__ == '__main__':
main()
通过socket
模块,开启一个tcp
服务器,然后监听一个端口,接收客户端的请求,并对请求做出响应。
过程中,通过gdb
模块中提供给python
的api
,进行对gdb
的调用,如:gdb.execute('start', to_string=True)
执行start
命令,并返回字符串。更多api
见官方文档。
2. 运行与结果
假如实现了gdb
服务器的代码文件为xxx.py
,所在路径为yyy
。
运行方法为:
gdb
进入gdb
命令行界面,然后:
(gdb) source xxx.py
或者:
echo "source yyy/xxx.py" >> ~/.gdbinit
gdb
最终结果为:

3. 问题
虽然通过python
编写socket
服务器,可以实现外界与gdb
的交互。但是,这样不优雅,也比较浪费。
同时,在gdb
中实现的这个服务端,只是运行当前gdb
中的一个扩展。而一个gdb
同一时刻只能调试一个文件,该服务器又不可能让外层gdb
多线程进行。
所以,任意一个用户访问,都只能得到当前gdb
调试的进度和状态。也就是说,它只能为一个用户服务。
而要为多用户服务,必须开启多个gdb
服务端,但如果只是主机间的进程交互的话,这样是十分低效的。
更高效的方法应该是,用进程间通讯代替tcp
通讯。
网友评论