如何用gcov抑制模板的内联

How to suppress inlining of templates with gcov

本文关键字:何用 gcov      更新时间:2023-10-16

我使用GCC 4.9和GCOV来获得代码和分支覆盖。然而,分支覆盖率的结果对于我的c++代码来说是完全无用的。尽管使用了我所知道的所有-fno-*-inline标志,但似乎GCC内联模板。

下面是一个小的示例应用程序来说明这个问题:

#include <string>
#include <iostream>
using namespace std;
int main() {
  string foo;
  foo = "abc";
  cout << foo << endl;
}

我用g++ -O0 -fno-inline -fno-inline-small-functions -fno-default-inline --coverage -fprofile-arcs test.cpp -o test编译程序

运行test后,gcovr -r . -b打印:

------------------------------------------------------------------------------
                           GCC Code Coverage Report
Directory: .
------------------------------------------------------------------------------
File                                    Branches   Taken  Cover   Missing
------------------------------------------------------------------------------
test.cpp                                      14       7    50%   7,8,9,10
------------------------------------------------------------------------------
TOTAL                                         14       7    50%
------------------------------------------------------------------------------
在我们的main函数中没有一个分支。例如,第7行包含string foo;。似乎std::basic_string<...>的构造函数中有一些if语句,但是当查看main的覆盖率时,这不是有用的信息。

问题是,所有这些内联分支加起来,为我的实际单元测试计算的分支覆盖率大约是40%。我感兴趣的是我的代码的分支覆盖率,而不是我在c++标准库中碰到了多少分支。

有没有办法完全关闭编译器中的内联或告诉GCOV不考虑内联分支?我在GCOV主页或其他地方找不到任何关于这个主题的指南。

嗯,你应该总是仔细检查你的期望。非常感谢@Useless为我指出gcov输出本身。但是,您并不完全正确:分支不属于test.cpp文件。运行gcovr-k并查看所有中间文件,显示gcov正确地生成了诸如#usr#include#c++#4.9#bits#basic_string.h.gcov之类的文件,这些文件显示了c++标准库方面的覆盖范围。

然而,test.cpp中所有分支的原因都不是内联的。这是异常。由于潜在的异常(例如std::bad_alloc),每个对标准库的调用都是一个分支。将-fno-exceptions添加到编译器标志中会得到以下输出:

------------------------------------------------------------------------------
                           GCC Code Coverage Report
Directory: .
------------------------------------------------------------------------------
File                                    Branches   Taken  Cover   Missing
------------------------------------------------------------------------------
test.cpp                                       4       2    50%   10
------------------------------------------------------------------------------
TOTAL                                          4       2    50%
------------------------------------------------------------------------------

通过cat foo.cpp.gcov打印深入挖掘gcov输出:

        -:    0:Source:test.cpp
        -:    0:Graph:/home/neverlord/gcov/test.gcno
        -:    0:Data:/home/neverlord/gcov/test.gcda
        -:    0:Runs:1
        -:    0:Programs:1
        -:    1:#include <string>
        -:    2:#include <iostream>
        -:    3:
        -:    4:using namespace std;
        -:    5:
function main called 1 returned 100% blocks executed 100%
        1:    6:int main() {
        1:    7:  string foo;
call    0 returned 1
        1:    8:  foo = "abc";
call    0 returned 1
        1:    9:  cout << foo << endl;
call    0 returned 1
call    1 returned 1
call    2 returned 1
function _GLOBAL__sub_I_main called 1 returned 100% blocks executed 100%
function _Z41__static_initialization_and_destruction_0ii called 1 returned 100% blocks executed 100%
        4:   10:}
call    0 returned 1
branch  1 taken 1 (fallthrough)
branch  2 taken 0
branch  3 taken 1 (fallthrough)
branch  4 taken 0

不好意思。