Ubuntu 14.04 LTS:
peter@peterubuntu0:$ insmod mymodule
peter@peterubuntu0:/sys/module/mymodule/sections$ cat .text .data .bss
0x0000000000000000
0x0000000000000000
0x0000000000000000
那么,当我进行远程调试时,如何告诉 gdb 我的模块在目标机器上的加载位置?我是否只使用偏移量(那么 .text 的偏移量为 0x64 或十进制 64?)?
peter@peterubuntu0:~$ objdump mymodule.ko --section-headers
mymodule.ko: file format elf64-x86-64
Sections:
Idx Name Size VMA LMA File off Algn
0 .note.gnu.build-id 00000024 0000000000000000 0000000000000000 00000040 2**2
CONTENTS, ALLOC, LOAD, READONLY, DATA
1 .text 0000b345 0000000000000000 0000000000000000 00000064 2**0
CONTENTS, ALLOC, LOAD, RELOC, READONLY, CODE
12 .data 000001e4 0000000000000000 0000000000000000 00012620 2**5
CONTENTS, ALLOC, LOAD, RELOC, DATA, LINK_ONCE_DISCARD
15 .bss 00000014 0000000000000000 0000000000000000 00012a80 2**3
ALLOC
答案1
我发现,当目标是另一个xubuntu时,也可以通过以下方式获取文本地址:
(gdb) monitor lsmod
Module Size modstruct Used by
iptable_mangle 16384 0xffffffffc0f57040 1 (Live) 0xffffffffc0f55000 [ ]
...
然后你可以使用其他地方记录的命令来告诉 gdb 偏移量
(gdb) add-symbol-file mymodule 0xmymoduleaddress
答案2
将 .gdbinit 文件添加到您的主目录。Gdb 在启动时会获取此文件。在此文件中,您可以为 gdb 定义宏,也可以执行普通的 shell 命令。
#gdb implementation of the linux lsmod
define gdblsmod
set $current = modules.next
set $offset = ((int)&((struct module *)0).list)
printf "Module\tAddress\n"
while($current.next != modules.next)
printf "%s\t%p\n", \
((struct module *) (((void *) ($current)) - $offset ) )->name ,\
((struct module *) (((void *) ($current)) - $offset ) )->module_core
set $current = $current.next
end
end
此宏打印所有当前加载模块的文本段地址。打开 gdb 控制台并键入gdblsmod
以使用它。您还可以在其中输入 gdb 命令,例如:
set serial baud 115200
如果您通过串行端口进行远程调试。