有人有用于Linux上fd_set的gdb美化打印代码吗?
现代版本的gdb(调试工具)可以结合Python代码来“美化打印”复杂的数据结构。对于C++的标准模板库(STL)类和一些常见的boost.org
类型,有一些很棒的美化打印实现。
在网络编程中,常常会遇到select
和poll
的调用。poll()
使用的是一个数据结构数组,而select()
使用的是fd_set
。
有没有人见过fd_set
的美化打印实现?最好是可以在不同平台上使用的,如果只能在特定平台上用也没关系。理想情况下,我希望是linux/x86的版本,但任何版本我都愿意尝试,并希望能进行适当的调整。
1 个回答
4
好的,这是我写的一个在Linux上似乎能用的东西。如果你试过后也能用,记得告诉我哦:
anonprint.py
import gdb
class fd_set_printer:
"""
Prints an fd_set, which is normally an opaque
array of ints, each bit representing one file descriptor
"""
def __init__(self, val, val_array):
self.val = val
self.val_array = val_array
@staticmethod
def find_set_bits(bit_array):
"""
Finds set bits in a long bit list.
Expects a gdb.Value which contains a C array,
such as int[10], and treats it as a bitlist
of int_size * 10 bits long. Returns an array of
bit positions, starting with 0, for which the bits
are on.
e.g. for int foo[] = [1, 6], it will return [ 0, 33, 34 ]
The array should be given as a gdb.Value
"""
set_bits = []
bits_length = bit_array[0].type.sizeof * 8
current_bit = 0
# Can not use 'for current_byte in gdb.Value:' even if
# gdb.Value.type.code == gdb.TYPE_CODE_ARRAY
# So iteration happens this ugly C-style way
for current_byte_pos in xrange(*bit_array.type.range()):
current_byte = bit_array[current_byte_pos]
for bit in xrange(0, bits_length):
bit_mask = 1 << bit
if bit_mask & current_byte == bit_mask:
set_bits.append(current_bit)
current_bit += 1
return set_bits
def to_string(self):
fd_list = self.find_set_bits(self.val_array)
if len(fd_list) == 0:
output = "Empty file descriptor set."
else:
output = "File descriptor set: "
output += ', '.join(map(str, fd_list))
return output
def anon_struct_lookup_function(val):
"""
Checks if the given value looks like an fd_set.
If it does, delegates printing to the printer
"""
lookup_tag = val.type.tag
if lookup_tag == None:
return None
if lookup_tag != "<anonymous struct>":
return None
fields = val.type.fields()
val_array = None
if len(fields) == 1 and fields[0].name == 'fds_bits':
val_array = val['fds_bits']
elif len(fields) == 1 and fields[0].name == '__fds_bits':
val_array = val['__fds_bits']
if not val_array is None:
return fd_set_printer(val, val_array)
return None
def add_fd_set_printer(obj = gdb):
"Adds the fd_set pretty printer to the given object"
obj.pretty_printers.append(anon_struct_lookup_function)
然后你需要创建你的 ~/.gdbinit
文件:
python
import sys
sys.path.insert(0, '/home/user/anonprint_py_directory_here')
from anonprint import add_fd_set_printer
add_fd_set_printer()
end
这是我第一次尝试通过Python与gdb内部进行交互,所以欢迎大家给我评论和建议。