summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorDmitry Mikulin <dmitry.mikulin@sony.com>2019-10-15 16:32:50 +0000
committerDmitry Mikulin <dmitry.mikulin@sony.com>2019-10-15 16:32:50 +0000
commit034badb312bedfee9e81a933ad3ef7f0a1d8853d (patch)
tree0ae251c29f1de5f4615bcd4e1d82d6f0390b072f
parent6e8599d939791c3751509c5ceebc29e78b61ac57 (diff)
downloadbcm5719-llvm-034badb312bedfee9e81a933ad3ef7f0a1d8853d.tar.gz
bcm5719-llvm-034badb312bedfee9e81a933ad3ef7f0a1d8853d.zip
CFI: wrong type passed to llvm.type.test with multiple inheritance devirtualization.
Differential Revision: https://reviews.llvm.org/D67985 llvm-svn: 374909
-rw-r--r--clang/lib/CodeGen/CGExprCXX.cpp2
-rw-r--r--clang/test/CodeGenCXX/cfi-multiple-inheritance.cpp31
-rw-r--r--compiler-rt/test/cfi/multiple-inheritance2.cpp38
3 files changed, 70 insertions, 1 deletions
diff --git a/clang/lib/CodeGen/CGExprCXX.cpp b/clang/lib/CodeGen/CGExprCXX.cpp
index 49527e9691a..114d806d454 100644
--- a/clang/lib/CodeGen/CGExprCXX.cpp
+++ b/clang/lib/CodeGen/CGExprCXX.cpp
@@ -382,7 +382,7 @@ RValue CodeGenFunction::EmitCXXMemberOrOperatorMemberCallExpr(
const CXXRecordDecl *RD;
std::tie(VTable, RD) =
CGM.getCXXABI().LoadVTablePtr(*this, This.getAddress(),
- MD->getParent());
+ CalleeDecl->getParent());
EmitVTablePtrCheckForCall(RD, VTable, CFITCK_NVCall, CE->getBeginLoc());
}
diff --git a/clang/test/CodeGenCXX/cfi-multiple-inheritance.cpp b/clang/test/CodeGenCXX/cfi-multiple-inheritance.cpp
new file mode 100644
index 00000000000..bebf30d22aa
--- /dev/null
+++ b/clang/test/CodeGenCXX/cfi-multiple-inheritance.cpp
@@ -0,0 +1,31 @@
+// Test that correct vtable ptr and type metadata are passed to llvm.type.test
+// Related to Bugzilla 43390.
+
+// RUN: %clang_cc1 -triple x86_64-unknown-linux -fvisibility hidden -std=c++11 -fsanitize=cfi-nvcall -emit-llvm -o - %s | FileCheck %s
+
+class A1 {
+public:
+ virtual int f1() = 0;
+};
+
+class A2 {
+public:
+ virtual int f2() = 0;
+};
+
+
+class B : public A1, public A2 {
+public:
+ int f2() final { return 1; }
+ int f1() final { return 2; }
+};
+
+// CHECK-LABEL: define hidden i32 @_Z3foov
+int foo() {
+ B b;
+ return static_cast<A2*>(&b)->f2();
+ // CHECK: [[P:%[^ ]*]] = bitcast %class.B* %b to i8**
+ // CHECK: [[V:%[^ ]*]] = load i8*, i8** [[P]], align 8
+ // CHECK: call i1 @llvm.type.test(i8* [[V]], metadata !"_ZTS1B")
+ // CHECK: call i1 @llvm.type.test(i8* [[V]], metadata !"all-vtables")
+}
diff --git a/compiler-rt/test/cfi/multiple-inheritance2.cpp b/compiler-rt/test/cfi/multiple-inheritance2.cpp
new file mode 100644
index 00000000000..f6b38452ee7
--- /dev/null
+++ b/compiler-rt/test/cfi/multiple-inheritance2.cpp
@@ -0,0 +1,38 @@
+// Test that virtual functions of the derived class can be called through
+// pointers of both base classes without CFI errors.
+// Related to Bugzilla 43390.
+
+// RUN: %clangxx_cfi -o %t1 %s
+// RUN: %run %t1 2>&1 | FileCheck --check-prefix=CFI %s
+
+// CFI: In f1
+// CFI: In f2
+// CFI-NOT: control flow integrity check
+
+// REQUIRES: cxxabi
+
+#include <stdio.h>
+
+class A1 {
+public:
+ virtual void f1() = 0;
+};
+
+class A2 {
+public:
+ virtual void f2() = 0;
+};
+
+
+class B : public A1, public A2 {
+public:
+ void f2() final { fprintf(stderr, "In f2\n"); }
+ void f1() final { fprintf(stderr, "In f1\n"); }
+};
+
+int main() {
+ B b;
+
+ static_cast<A1*>(&b)->f1();
+ static_cast<A2*>(&b)->f2();
+}
OpenPOWER on IntegriCloud