C++: How to measure code coverage with lcov and GTest

Posted on

Why, You Ask?

Because you want to know which parts of your C++ code are actually being tested โ€“ not just assumed to be. Code coverage isn't about perfection, but about confidence. And with the right setup, you can easily visualize what's being tested and what's quietly collecting dust.

In this guide, we'll walk through:

  • Setting up lcov and GTest
  • Writing effective coverage-focused tests
  • Running the tests and generating reports
  • Interpreting results and achieving real insight
  • Automating it all via GitHub Actions

๐Ÿ“ฆ Try It Yourself

Check out the working example repository here:

๐Ÿ‘‰ github.com/svnscha/cpp-coverage-example

It contains:

  • A minimal MyQueue class
  • Unit tests using GTest
  • Full lcov and genhtml integration
  • A GitHub Actions workflow for automated code coverage with a threshold

1. Install Required Tools

Make sure your environment has:

  • lcov
  • A C++ compiler (GCC or Clang)
  • cmake, ninja
  • GTest (or install libgtest-dev on Ubuntu)

2. Build with Coverage Instrumentation

To generate coverage data, compile your code with:

  • --coverage: Coverage instrumentation
  • -g: Debug information
  • -O0: No optimization (to avoid inlining, etc.)

๐Ÿ›  Minimal CMake Snippet

option(ENABLE_COVERAGE "Enable code coverage reporting" OFF)

function(enable_coverage target)
    if(ENABLE_COVERAGE AND CMAKE_CXX_COMPILER_ID MATCHES "GNU|Clang")
        target_compile_options(${target} PRIVATE --coverage -O0 -g)
        target_link_options(${target} PRIVATE --coverage)
    endif()
endfunction()

enable_testing()

add_library(MyLibrary ...)
add_executable(MyTests ...)
target_link_libraries(MyTests PRIVATE MyLibrary GTest::gtest_main)

include(GTest)
gtest_discover_tests(MyTests)

enable_coverage(MyLibrary)
enable_coverage(MyTests)

3. Write Tests That Actually Trigger All Code Paths

Hereโ€™s a small utility class as an example:

class MyQueue
{
public:
    void Push(int val)
    { 
        _q.push(val); 
    }

    void Pop() 
    {
        if (_q.empty())
            return;
        _q.pop();
    }

    bool IsEmpty() const { return _q.empty(); }

private:
    std::queue<int> _q;
};

Now test both normal and edge cases:

TEST(MyQueueTest, PopWhenEmpty) {
    MyQueue q;
    q.Pop(); // Hit the early return
    EXPECT_TRUE(q.IsEmpty());
}

TEST(MyQueueTest, PushAndPop) {
    MyQueue q;
    q.Push(42);
    EXPECT_FALSE(q.IsEmpty());
    q.Pop();
    EXPECT_TRUE(q.IsEmpty());
}

4. Run Tests and Generate Coverage Report

cmake -B build -DENABLE_COVERAGE=ON -DCMAKE_BUILD_TYPE=Debug
cmake --build build
cd build
ctest --output-on-failure

Then generate the coverage report:

lcov --directory . --capture --output-file coverage.info --rc geninfo_auto_base=1 --ignore-errors mismatch
lcov --remove coverage.info '/usr/*' '*/tests/*' --output-file coverage.filtered.info
genhtml coverage.filtered.info --output-directory coverage-report

Open this in your browser:

build/coverage-report/index.html

5. Automation

Now, having the coverage report in place is a great start. But it's not enough to just have the coverage report. You need to automate it so that every time you run your tests, you get a detailed report of what was covered and what wasn't.

To get started, you can setup a GitHub workflow with threshold tests, upload the reports or do post-processing on the report.

Either way, the key is to have a way to run tests and generate reports. Here's an example workflow:

Summary

StepWhat You Did
1. InstallSet up lcov, gtest, gcc, and cmake
2. BuildCompiled with --coverage and no optimizations
3. TestHit both happy paths and edge cases
4. ReportGenerated detailed HTML coverage report

Conclusion

Code coverage doesn't guarantee perfect tests, but it gives you visibility. When paired with thoughtful test cases (especially edge cases), it helps ensure your code behaves correctly - even when things go sideways.

Even small utility classes deserve this level of care.

And with lcov, you can turn "I think I tested this" into "Yes, this line has been executed 12 times during CI."


Copyright ยฉ 2025 Sven Scharmentke. All rights reserved.