2

我有以下要求:

  • 在任何函数的入口出口处添加文本。
  • 改变源代码,除了从上面插入(所以没有预处理器或任何东西)

例如:

void fn(param-list)
{
    ENTRY_TEXT (param-list)
    //some code
    EXIT_TEXT
}

但不仅在这种简单的情况下,它还可以使用预处理器指令运行!

例子:

void fn(param-list)
#ifdef __WIN__
{
  ENTRY_TEXT (param-list)
  //some windows code
  EXIT_TEXT
}
#else
{
    ENTRY_TEXT (param-list)
    //some any-os code

    if (condition)
    {
        return; //should become EXIT_TEXT
    }

    EXIT_TEXT
}

所以我的问题是:有正确的方法吗?

我已经尝试过使用编译器使用的解析器进行一些工作,但是由于它们都依赖于在解析之前运行预处理器,因此它们对我来说毫无用处。

此外,一些不需要预处理器的令牌生成解析器有些无用,因为它们生成令牌的内存映射,然后导致完整的新源代码,而不仅仅是插入文本。

我正在做的一件事是尝试使用 FLEX(或 JFlex),如果这是一个有效的选项,我将不胜感激。;-)

编辑:澄清一点:目的是允许堆栈跟踪之类的东西。我想跟踪每个函数调用,为了遵循调用层次结构,我需要在函数的入口点和函数的出口点放置一个宏。这构建了一个函数调用跟踪。:-)

EDIT2:特定于编译器的选项不太合适,因为我们有许多不同的编译器可供使用,而且许多编译器可能没有得到任何工具的良好支持。

4

3 回答 3

3

不幸的是,您的想法不仅不切实际(C++ 解析起来很复杂),而且注定要失败。

您遇到的主要问题是异常将EXIT_TEXT完全绕过您的宏。


你有几个解决方案。

如前所述,第一个解决方案是使用依赖于平台的方式来计算堆栈跟踪。它可能有些不精确,特别是因为内联:即,在调用者中内联的小函数,它们不会出现在堆栈跟踪中,因为在程序集级别没有生成函数调用。另一方面,它广泛可用,不需要对代码进行任何手术,也不会影响性能

第二种解决方案是只在进入时引入一些东西,并使用 RAII 来完成退出工作。比您的方案要好得多,因为它自动处理多个返回和异常,但它遇到了同样的问题:如何自动执行插入。为此,您可能希望在 AST 级别进行操作,并修改 AST 以引入您的小宝石。您可以使用 Clang(查找 c++11 迁移工具以获取大量重写示例)或 gcc(使用插件)来完成。

最后,您还有手动注释。虽然它可能看起来动力不足(而且工作量很大),但我要强调的是,您不会将日志记录留给工具...我看到手动执行此操作的 3 个优点:您可以避免在性能敏感部分引入这种开销,您可以仅保留大参数的“摘要”,您可以根据对当前函数感兴趣的内容自定义摘要。

于 2013-05-17T07:14:18.657 回答
1

我建议使用LLVM 库和 Clang开始。

您还可以利用 C++ 语言来简化您的流程。如果您只是将一个小对象插入到在函数范围入口处构造的代码中,并依赖于它将在退出时被销毁的事实。这应该会大大简化记录函数的“退出”。

于 2013-05-17T07:04:16.383 回答
0

这并不能真正回答您的问题,但是,对于您的初始需求,您可以使用 backtrace() 函数 execinfo.h(如果您使用的是 GCC)。

当我的 gcc C++ 应用程序崩溃时如何生成堆栈跟踪

于 2013-05-17T06:52:05.470 回答