https://github.com/justincady created https://github.com/llvm/llvm-project/pull/133463
In cases where a terminating statement exists within the try or catch block the coverage mapping can be incorrect. Coverage reports display lines following the last catch block as uncovered, when the lines have been executed. This change adjusts the mapping such that an extra region is only created after a try block if the try itself contains the terminating statement. The testing validates coverage more broadly than the above to ensure other mapping around exception handling does not regress with this change. >From 6f79ddba80fa391a8c1ddbd3b2217f2e5a9ce21b Mon Sep 17 00:00:00 2001 From: Justin Cady <d...@justincady.com> Date: Fri, 28 Mar 2025 12:07:44 -0400 Subject: [PATCH] [Coverage] Fix region creation after try statements In cases where a terminating statement exists within the try or catch block the coverage mapping can be incorrect. Coverage reports display lines following the last catch block as uncovered, when the lines have been executed. This change adjusts the mapping such that an extra region is only created after a try block if the try itself contains the terminating statement. The testing validates coverage more broadly than the above to ensure other mapping around exception handling does not regress with this change. --- clang/lib/CodeGen/CoverageMappingGen.cpp | 10 +- clang/test/CoverageMapping/trycatch.cpp | 4 +- .../test/profile/Linux/coverage-exception.cpp | 142 ++++++++++++++++++ 3 files changed, 152 insertions(+), 4 deletions(-) create mode 100644 compiler-rt/test/profile/Linux/coverage-exception.cpp diff --git a/clang/lib/CodeGen/CoverageMappingGen.cpp b/clang/lib/CodeGen/CoverageMappingGen.cpp index 73811d15979d5..f3b819bcf3b20 100644 --- a/clang/lib/CodeGen/CoverageMappingGen.cpp +++ b/clang/lib/CodeGen/CoverageMappingGen.cpp @@ -2123,11 +2123,17 @@ struct CounterCoverageMappingBuilder Counter ParentCount = getRegion().getCounter(); propagateCounts(ParentCount, S->getTryBlock()); + bool TryHasTerminateStmt = HasTerminateStmt; + for (unsigned I = 0, E = S->getNumHandlers(); I < E; ++I) Visit(S->getHandler(I)); - Counter ExitCount = getRegionCounter(S); - pushRegion(ExitCount); + if (TryHasTerminateStmt) { + Counter ExitCount = getRegionCounter(S); + pushRegion(ExitCount); + } + + HasTerminateStmt = TryHasTerminateStmt; } void VisitCXXCatchStmt(const CXXCatchStmt *S) { diff --git a/clang/test/CoverageMapping/trycatch.cpp b/clang/test/CoverageMapping/trycatch.cpp index 89fae8af9b720..e5d2482c6b151 100644 --- a/clang/test/CoverageMapping/trycatch.cpp +++ b/clang/test/CoverageMapping/trycatch.cpp @@ -33,5 +33,5 @@ int main() { // CHECK-NEXT: File 0, [[@LINE]]:12 -> [[@ catch(const Warning &w) { // CHECK-NEXT: File 0, [[@LINE]]:27 -> [[@LINE+2]]:4 = #4 j = 0; } - return 0; // CHECK-NEXT: File 0, [[@LINE]]:3 -> [[@LINE]]:11 = #1 -} + return 0; // CHECK-NOT: File 0 +} // CHECK-NOT: File 0 diff --git a/compiler-rt/test/profile/Linux/coverage-exception.cpp b/compiler-rt/test/profile/Linux/coverage-exception.cpp new file mode 100644 index 0000000000000..c12303363fbb6 --- /dev/null +++ b/compiler-rt/test/profile/Linux/coverage-exception.cpp @@ -0,0 +1,142 @@ +// REQUIRES: lld-available +// XFAIL: powerpc64-target-arch + +// RUN: %clangxx_profgen -std=c++17 -fuse-ld=lld -fcoverage-mapping -o %t %s +// RUN: env LLVM_PROFILE_FILE=%t.profraw %run %t +// RUN: llvm-profdata merge -o %t.profdata %t.profraw +// RUN: llvm-cov show %t -instr-profile=%t.profdata 2>&1 | FileCheck %s + +#include <stdio.h> +#include <stdlib.h> + +#define TRY_AND_CATCH_ALL(x) \ + try { \ + (x); \ + } catch (...) { \ + } + +#define TRY_MAYBE_CRASH(x) \ + try { \ + if ((x)) { \ + printf("no crash\n"); \ + } else { \ + abort(); \ + } \ + } catch (...) { \ + } + +#define TRY_AND_CATCH_CRASHES(x) \ + try { \ + (x); \ + } catch (...) { \ + abort(); \ + } + +// clang-format off +static +int test_no_exception() { // CHECK: [[@LINE]]| 1|int test_no_exception() + int i = 0; // CHECK: [[@LINE]]| 1| int i + try { // CHECK: [[@LINE]]| 1| try { + i = 1; // CHECK: [[@LINE]]| 1| i = + } catch (...) { // CHECK: [[@LINE]]| 1| } catch ( + abort(); // CHECK: [[@LINE]]| 0| abort(); + } // CHECK: [[@LINE]]| 0| } + printf("%s: %u\n", __func__, i); // CHECK: [[@LINE]]| 1| printf( + return 0; // CHECK: [[@LINE]]| 1| return +} // CHECK: [[@LINE]]| 1|} + +static +int test_no_exception_macro() { // CHECK: [[@LINE]]| 1|int test_no_exception_macro() + int i = 0; // CHECK: [[@LINE]]| 1| int i + TRY_AND_CATCH_ALL(i = 1); // CHECK: [[@LINE]]| 1| TRY_AND_CATCH_ALL( + printf("%s: %u\n", __func__, i); // CHECK: [[@LINE]]| 1| printf( + return 0; // CHECK: [[@LINE]]| 1| return +} // CHECK: [[@LINE]]| 1|} + +static +int test_exception() { // CHECK: [[@LINE]]| 1|int test_exception() + try { // CHECK: [[@LINE]]| 1| try { + throw 1; // CHECK: [[@LINE]]| 1| throw + } catch (...) { // CHECK: [[@LINE]]| 1| } catch ( + printf("%s\n", __func__); // CHECK: [[@LINE]]| 1| printf( + } // CHECK: [[@LINE]]| 1| } + return 0; // CHECK: [[@LINE]]| 1| return +} // CHECK: [[@LINE]]| 1|} + +static +int test_exception_macro() { // CHECK: [[@LINE]]| 1|int test_exception_macro() + TRY_AND_CATCH_ALL(throw 1); // CHECK: [[@LINE]]| 1| TRY_AND_CATCH_ALL( + printf("%s\n", __func__); // CHECK: [[@LINE]]| 1| printf( + return 0; // CHECK: [[@LINE]]| 1| return +} // CHECK: [[@LINE]]| 1|} + +static +int test_exception_macro_nested() { // CHECK: [[@LINE]]| 1|int test_exception_macro_nested() + try { // CHECK: [[@LINE]]| 1| try { + TRY_AND_CATCH_ALL(throw 1); // CHECK: [[@LINE]]| 1| TRY_AND_CATCH_ALL( + } catch (...) { // CHECK: [[@LINE]]| 1| } catch ( + abort(); // CHECK: [[@LINE]]| 0| abort(); + } // CHECK: [[@LINE]]| 0| } + printf("%s\n", __func__); // CHECK: [[@LINE]]| 1| printf( + return 0; // CHECK: [[@LINE]]| 1| return +} // CHECK: [[@LINE]]| 1|} + +static +int test_exception_try_crash() { // CHECK: [[@LINE]]| 1|int test_exception_try_crash() + int i = 1; // CHECK: [[@LINE]]| 1| int i + TRY_MAYBE_CRASH(i); // CHECK: [[@LINE]]| 1| TRY_MAYBE_CRASH( + printf("%s\n", __func__); // CHECK: [[@LINE]]| 1| printf( + return 0; // CHECK: [[@LINE]]| 1| return +} // CHECK: [[@LINE]]| 1|} + +static +int test_exception_crash() { // CHECK: [[@LINE]]| 1|int test_exception_crash() + int i = 0; // CHECK: [[@LINE]]| 1| int i + TRY_AND_CATCH_CRASHES(i = 1); // CHECK: [[@LINE]]| 1| TRY_AND_CATCH_CRASHES( + printf("%s\n", __func__); // CHECK: [[@LINE]]| 1| printf( + return 0; // CHECK: [[@LINE]]| 1| return +} // CHECK: [[@LINE]]| 1|} + +static +int test_conditional(int i) { // CHECK: [[@LINE]]| 1|int test_conditional(int i) + try { // CHECK: [[@LINE]]| 1| try { + if (i % 2 == 0) { // CHECK: [[@LINE]]| 1| if ( + printf("%s\n", __func__); // CHECK: [[@LINE]]| 1| printf( + } else { // CHECK: [[@LINE]]| 1| } else { + abort(); // CHECK: [[@LINE]]| 0| abort(); + } // CHECK: [[@LINE]]| 0| } + } catch (...) { // CHECK: [[@LINE]]| 1| } catch ( + abort(); // CHECK: [[@LINE]]| 0| abort(); + } // CHECK: [[@LINE]]| 0| } + return 0; // CHECK: [[@LINE]]| 1| return +} + +static +int test_multiple_catch() { // CHECK: [[@LINE]]| 1|int test_multiple_catch() + try { // CHECK: [[@LINE]]| 1| try { + throw 1; // CHECK: [[@LINE]]| 1| throw + } catch (double) { // CHECK: [[@LINE]]| 1| } catch (double) + abort(); // CHECK: [[@LINE]]| 0| abort(); + } catch (int) { // CHECK: [[@LINE]]| 1| } catch (int) + printf("int\n"); // CHECK: [[@LINE]]| 1| printf( + } catch (float) { // CHECK: [[@LINE]]| 1| } catch (float) + abort(); // CHECK: [[@LINE]]| 0| abort(); + } catch (...) { // CHECK: [[@LINE]]| 0| } catch ( + abort(); // CHECK: [[@LINE]]| 0| abort(); + } // CHECK: [[@LINE]]| 0| } + return 0; // CHECK: [[@LINE]]| 1| return +} // CHECK: [[@LINE]]| 1|} + +int main() { // CHECK: [[@LINE]]| 1|int main() + test_no_exception(); // CHECK: [[@LINE]]| 1| test_no_exception( + test_no_exception_macro(); // CHECK: [[@LINE]]| 1| test_no_exception_macro( + test_exception(); // CHECK: [[@LINE]]| 1| test_exception( + test_exception_macro(); // CHECK: [[@LINE]]| 1| test_exception_macro( + test_exception_macro_nested(); // CHECK: [[@LINE]]| 1| test_exception_macro_nested( + test_exception_try_crash(); // CHECK: [[@LINE]]| 1| test_exception_try_crash( + test_exception_crash(); // CHECK: [[@LINE]]| 1| test_exception_crash( + test_conditional(2); // CHECK: [[@LINE]]| 1| test_conditional( + test_multiple_catch(); // CHECK: [[@LINE]]| 1| test_multiple_catch( + return 0; // CHECK: [[@LINE]]| 1| return +} // CHECK: [[@LINE]]| 1|} +// clang-format on _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits