在 lldb 中设置大型数据结构的观察点

cod*_*kid 5 c++ vector watchpoint lldb

我正在学习 lldb,我很好奇你如何为更大的数据结构(例如向量)设置观察点。我知道我可以使用打印并且可以工作,但我收到一条消息说不支持大小为“x”的观察点。有没有解决的办法?谢谢您的帮助!

(lldb) s
Process 36110 stopped
* thread #1, queue = 'com.apple.main-thread', stop reason = step in
    frame #0: 0x0000000100001600 a.out`main at test.cpp:10
   7        vector<int> arr;
   8        arr.push_back(1);
   9        arr.push_back(2);
-> 10       arr.push_back(3);
   11       arr.push_back(4);
   12       arr.push_back(5);
   13
Target 0: (a.out) stopped.
(lldb) print arr
(std::__1::vector<int, std::__1::allocator<int> >) $2 = size=2 {
  [0] = 1
  [1] = 2
}
(lldb) w s v arr
error: Watchpoint creation failed (addr=0x7ffeefbff458, size=24, variable expression='arr').
error: watch size of 24 is not supported
Run Code Online (Sandbox Code Playgroud)

Jim*_*ham 5

如果您使用的是 Mac,x86_64 架构允许 4 个独立的监视区域,每个区域最多 8 个字节。目前,lldb 对于每个监视请求仅使用一个区域。它可以将多个监视区域组合在一起来处理适合此结构的更大请求。请随时通过http://bugs.llvm.org提交此功能的增强请求。但观察点的资源确实有限,因此您通常必须非常有针对性地关注您想要观看的内容 - 这可能就是为什么没有人抽出时间支持 > 8 字节的原因。

如果您想在向量中添加或删除元素时停止,那么观察向量中的结束指针(即__end_)就足够了。--raw您可以使用“frame var”参数查看向量的实际内容:

(lldb) fr v --raw arr
(std::__1::vector<int, std::__1::allocator<int> >) arr = {
  std::__1::__vector_base<int, std::__1::allocator<int> > = {
    __begin_ = 0x0000000100400000
    __end_ = 0x000000010040001c
    __end_cap_ = {
      std::__1::__compressed_pair_elem<int *, 0, false> = {
        __value_ = 0x0000000100400038
      }
    }
  }
}
Run Code Online (Sandbox Code Playgroud)

每当向量增大或缩小时,结束标记都会进行调整,因此观察点设置为:

(lldb) watch set v arr.__end_
Watchpoint created: Watchpoint 1: addr = 0x7ffeefbff1c8 size = 8 state = enabled type = w
    declare @ '/tmp/vectors.cpp:6'
    watchpoint spec = 'arr.__end_'
    new value: 0x000000010030020c
Run Code Online (Sandbox Code Playgroud)

会捕获push_back、erase等。

如果您想在向量值发生变化时停止,则必须观察各个值;如果只提供 32 个字节,您将无法查看有意义大小的向量中的所有数据。当然,当向量调整大小时,旧数据的观察点现在将指向已释放的内存......