如何在Xcode中方法调用的最开始设置断点,以便检查$rdi$rsi等



我想在方法调用的最开始设置断点,这样我就可以检查它是$rdi$rsi等。

在Xcode中,当我在方法调用(比如-HelperClass doThingWithBlock:])处设置符号断点时,它会停在方法体的第一行,这已经是方法调用开始后的几条指令了,如下面对方法调用的反汇编所示。

DebugBlock`-[HelperClass doThingWithBlock:]:
        0x109844aa0 <+0>:  pushq  %rbp
        0x109844aa1 <+1>:  movq   %rsp, %rbp
        0x109844aa4 <+4>:  subq   $0x30, %rsp
        0x109844aa8 <+8>:  leaq   -0x18(%rbp), %rax
        0x109844aac <+12>: movq   %rdi, -0x8(%rbp)
        0x109844ab0 <+16>: movq   %rsi, -0x10(%rbp)
        0x109844ab4 <+20>: movq   $0x0, -0x18(%rbp)
        0x109844abc <+28>: movq   %rax, %rdi
        0x109844abf <+31>: movq   %rdx, %rsi
        0x109844ac2 <+34>: callq  0x109844c74               ; symbol stub for: objc_storeStrong
        0x109844ac7 <+39>: leaq   0x15a2(%rip), %rax        ; @"hi"
        0x109844ace <+46>: movl   $0x16, %ecx
        0x109844ad3 <+51>: movl   %ecx, %edx
    ->  0x109844ad5 <+53>: movq   -0x18(%rbp), %rsi
        0x109844ad9 <+57>: movq   %rsi, %rdi
        0x109844adc <+60>: movq   %rsi, -0x20(%rbp)
        0x109844ae0 <+64>: movq   %rax, %rsi
        0x109844ae3 <+67>: movq   -0x20(%rbp), %rax
        0x109844ae7 <+71>: callq  *0x10(%rax)
        0x109844aea <+74>: xorl   %ecx, %ecx
        0x109844aec <+76>: movl   %ecx, %esi
        0x109844aee <+78>: leaq   -0x18(%rbp), %rdx
        0x109844af2 <+82>: movq   %rdx, %rdi
        0x109844af5 <+85>: movb   %al, -0x21(%rbp)
        0x109844af8 <+88>: callq  0x109844c74               ; symbol stub for: objc_storeStrong
        0x109844afd <+93>: addq   $0x30, %rsp
        0x109844b01 <+97>: popq   %rbp
        0x109844b02 <+98>: retq

Jason的建议适用于更复杂的问题,但这是一个非常常见的要求,因此break set有一个专门用于控制将断点推过序言的选项:

(lldb)break set-n main--跳过序言0

当您有调试信息时,

lldb会将断点位置提前到函数中的第一个源行。这个想法是,大多数拥有源级别信息的人更感兴趣的是用他们的名字打印自变量,而不是查看用于传递它们的寄存器

如果你使用的是一个简单的C函数,你可以设置一个地址断点来评估函数名到一个地址,例如

(lldb) br s -n main
Breakpoint 1: where = a.out`main + 11 at a.c:3, address = 0x0000000100000f8b
(lldb) br s -a `main`
Breakpoint 2: address = 0x0000000100000f80
(lldb) 

倒勾符号"将倒勾中的表达式计算为地址/值。breakpoint set --address的工作方式正如您所期望的那样。

甚至还有一些特殊的魔术,你不需要对期望地址的东西进行回溯,比如br s -a。你可以做br s -a main,它会起作用——作为一个非常特殊的奖励,你可以在这里为函数添加偏移量,比如br s -a main+5,它通常不是一个有效的C表达式。

不幸的是,我们使用的是objc方法-[HelperClass doThingWithBlock:],不能像我使用main那样将其作为表达式。我认为在这种情况下,您需要自己找到它的地址,例如,您可以在开始时反汇编一条指令,如dis -c 1 -n '-[ViewController setRepresentedObject:]',然后将该地址输入到br s -a中。

最新更新