llvmbot wrote:
<!--LLVM PR SUMMARY COMMENT--> @llvm/pr-subscribers-clang-codegen <details> <summary>Changes</summary> When emitting non-virtual base initializers for the constructor prologue, we would potentially use a re-laundered this pointer value from a previous block, which subsequently would not dominate this use. With this fix, we always launder the original CXXThisValue. This fixes https://github.com/llvm/llvm-project/issues/67937 Note: First commit just adds the test, which will be fixed by the second commit. I will land the test separately after the first round. --- Full diff: https://github.com/llvm/llvm-project/pull/68169.diff 2 Files Affected: - (modified) clang/lib/CodeGen/CGClass.cpp (+3-4) - (added) clang/test/CodeGenCXX/strict-vtable-pointers-GH67937.cpp (+22) ``````````diff diff --git a/clang/lib/CodeGen/CGClass.cpp b/clang/lib/CodeGen/CGClass.cpp index 57a424c6f176c47..d18f186ce5b4157 100644 --- a/clang/lib/CodeGen/CGClass.cpp +++ b/clang/lib/CodeGen/CGClass.cpp @@ -28,6 +28,7 @@ #include "clang/CodeGen/CGFunctionInfo.h" #include "llvm/IR/Intrinsics.h" #include "llvm/IR/Metadata.h" +#include "llvm/Support/SaveAndRestore.h" #include "llvm/Transforms/Utils/SanitizerStats.h" #include <optional> @@ -1291,10 +1292,10 @@ void CodeGenFunction::EmitCtorPrologue(const CXXConstructorDecl *CD, assert(BaseCtorContinueBB); } - llvm::Value *const OldThis = CXXThisValue; for (; B != E && (*B)->isBaseInitializer() && (*B)->isBaseVirtual(); B++) { if (!ConstructVBases) continue; + SaveAndRestore ThisRAII(CXXThisValue); if (CGM.getCodeGenOpts().StrictVTablePointers && CGM.getCodeGenOpts().OptimizationLevel > 0 && isInitializerOfDynamicClass(*B)) @@ -1311,7 +1312,7 @@ void CodeGenFunction::EmitCtorPrologue(const CXXConstructorDecl *CD, // Then, non-virtual base initializers. for (; B != E && (*B)->isBaseInitializer(); B++) { assert(!(*B)->isBaseVirtual()); - + SaveAndRestore ThisRAII(CXXThisValue); if (CGM.getCodeGenOpts().StrictVTablePointers && CGM.getCodeGenOpts().OptimizationLevel > 0 && isInitializerOfDynamicClass(*B)) @@ -1319,8 +1320,6 @@ void CodeGenFunction::EmitCtorPrologue(const CXXConstructorDecl *CD, EmitBaseInitializer(*this, ClassDecl, *B); } - CXXThisValue = OldThis; - InitializeVTablePointers(ClassDecl); // And finally, initialize class members. diff --git a/clang/test/CodeGenCXX/strict-vtable-pointers-GH67937.cpp b/clang/test/CodeGenCXX/strict-vtable-pointers-GH67937.cpp new file mode 100644 index 000000000000000..692ca23a69abe6a --- /dev/null +++ b/clang/test/CodeGenCXX/strict-vtable-pointers-GH67937.cpp @@ -0,0 +1,22 @@ +// RUN: %clang_cc1 %s -I%S -triple=x86_64-pc-windows-msvc -fstrict-vtable-pointers -disable-llvm-passes -O1 -emit-llvm -o %t.ll +// RUN: FileCheck %s < %t.ll + +struct A { + virtual ~A(); +}; +struct B : virtual A {}; +class C : B {}; +C foo; + +// CHECK-LABEL: define {{.*}} @"??0C@@QEAA@XZ"(ptr {{.*}} %this, i32 {{.*}} %is_most_derived) +// CHECK: ctor.init_vbases: +// CHECK-NEXT: %0 = getelementptr inbounds i8, ptr %this1, i64 0 +// CHECK-NEXT: store ptr @"??_8C@@7B@", ptr %0 +// CHECK-NEXT: %1 = call ptr @llvm.launder.invariant.group.p0(ptr %this1) +// CHECK-NEXT: %2 = getelementptr inbounds i8, ptr %1, i64 8 +// CHECK-NEXT: %call = call noundef ptr @"??0A@@QEAA@XZ"(ptr {{.*}} %2) #2 +// CHECK-NEXT: br label %ctor.skip_vbases +// CHECK-EMPTY: +// CHECK-NEXT: ctor.skip_vbases: +// CHECK-NEXT: %3 = call ptr @llvm.launder.invariant.group.p0(ptr %this1) +// CHECK-NEXT: %call3 = call noundef ptr @"??0B@@QEAA@XZ"(ptr {{.*}} %3, i32 noundef 0) #2 `````````` </details> https://github.com/llvm/llvm-project/pull/68169 _______________________________________________ cfe-commits mailing list cfe-commits@lists.llvm.org https://lists.llvm.org/cgi-bin/mailman/listinfo/cfe-commits