2

Suppose I have the following C++:

template <int I> int bar(int i) {
  ++i;
label1:
  return I * i;
}
int main(int argc, char **) { return bar<2>(argc); }

Is it possible to set a gdb breakpoint on label1 for all instantiations of bar? In the above, it is easy -- there is just one instantiation. In my actual use case there are a large number spread all over the code base.

Said another way, in my gdb commandfile, is there a way to avoid the need to know a priori about every template instantiation of bar? In my actual use case, my aim is to emit some information about program state at the point of the labels (worry not, no programs were harmed by goto).

I know I can set a breakpoint on the label for a specific instantiation of bar as follows:

break -function bar<2> -label label1

I also know about rbreak which can be used to break on the entry of all template functions, but apparently does not have the -label option. I don't want to break on the entry point -- just the the label(s). I also looked to see if I could combine rbreak with until label1 but that didn't work.

Two other approaches I've considered are:

  1. grep for the labels (or could even just be specially formed comments), emit line numbers and use this information to populate/generate my gdb command file with source file and line number break points. This will definitely work, but was sort of hoping to avoid the need to "generate" my command file.
  2. Since I intend to implement some gdb Python pretty printers, perhaps I can parse the source from within python to get the line numbers and subsequently set the breakpoints. I'm just learning the Python API and it wasn't immediately obvious how to get the equivalent of list from within the gdb python package.

Any suggestions?

4

2 回答 2

4

这个答案显示了如何迭代程序中的所有全局符号。

对于您的测试程序,它产生:

bar<2>(int) True
main(int, char**) True

从这里,您可以轻松找到 的所有实例bar<>(),并使用Python 断点 API在它们上设置断点。

(gdb) py
>gdb.Breakpoint(function='bar<2>(int)', label='label1')
Breakpoint 1 at 0x1158: file t.cc, line 3.

使用function='bar<2>'也有效。

于 2021-05-01T06:54:18.467 回答
0

根据Employed Russian 的建议,我想出了以下 gdb python 脚本:

import re


class Custombreakpoint(gdb.Breakpoint):
    def __init__(self, function, label):
        super().__init__(function=function, label=label, internal=True)

    def stop(self):
        I = gdb.parse_and_eval("I")
        i = gdb.parse_and_eval("i")
        print(f"I = {I}, i = {i}")
        return False

pattern = re.compile('.*foo.*')
for sym in gdb.lookup_global_symbol('main').symtab.global_block():
    if sym.is_function:
        match = pattern.search(sym.name)
        if match:
            Custombreakpoint(function=match.string, label='label1')

gdb.execute('run')
gdb.execute('quit')

产生以下输出:

$ g++ -g ./test.cc && gdb --command=test.py ./a.out                                                                                                                                      
Reading symbols from ./a.out...                                                                                                                                                                                      
I = 2, i = 2                                                                                                                                                                                                         
I = 3, i = 2                                                                                                                                                                                                         
[Inferior 1 (process 17224) exited with code 012]

这将使我能够在我的实际用例中准确地做我想做的事情。谢谢您的帮助!

于 2021-05-02T05:11:51.587 回答