1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53
|
// RUN: %clang_cc1 -triple x86_64-unknown-linux-gnu -emit-llvm -fsanitize=cfi-mfcall -o - %s -fvisibility=hidden | FileCheck %s
#define CFI_UNCHECKED_CALLEE __attribute__((cfi_unchecked_callee))
class A {};
// CHECK-LABEL: _Z14MemberFuncCallP1AMS_FvvE
void MemberFuncCall(A *s, void (A::*p)()) {
// CHECK: memptr.virtual:
// CHECK-NEXT: [[VTABLE:%.*]] = load ptr, ptr {{.*}}, align 8
// CHECK-NEXT: [[OFFSET:%.*]] = sub i64 %memptr.ptr, 1
// CHECK-NEXT: [[FUNC:%.*]] = getelementptr i8, ptr [[VTABLE]], i64 [[OFFSET]]
// CHECK-NEXT: [[VALID:%.*]] = call i1 @llvm.type.test(ptr [[FUNC]], metadata !"_ZTSM1AFvvE.virtual")
// CHECK-NEXT: [[FUNC:%.*]] = getelementptr i8, ptr [[VTABLE]], i64 [[OFFSET]]
// CHECK-NEXT: %memptr.virtualfn = load ptr, ptr [[FUNC]], align 8
// CHECK-NEXT: {{.*}}= call i1 @llvm.type.test(ptr [[VTABLE]], metadata !"all-vtables")
// CHECK-NEXT: br i1 [[VALID]], label %[[CONT:.*]], label %handler.cfi_check_fail{{.*}}
// CHECK: [[CONT]]:
// CHECK-NEXT: br label %memptr.end
// CHECK: memptr.nonvirtual:
// CHECK-NEXT: %memptr.nonvirtualfn = inttoptr i64 %memptr.ptr to ptr
// CHECK-NEXT: [[VALID:%.*]] = call i1 @llvm.type.test(ptr %memptr.nonvirtualfn, metadata !"_ZTSM1AFvvE")
// CHECK-NEXT: [[VALID2:%.*]] = or i1 false, [[VALID]]
// CHECK-NEXT: br i1 [[VALID2]], label %[[CONT2:.*]], label %handler.cfi_check_fail{{.*}}
// CHECK: [[CONT2]]:
// CHECK-NEXT: br label %memptr.end
// CHECK: memptr.end:
// CHECK-NEXT: {{.*}} = phi ptr [ %memptr.virtualfn, %[[CONT]] ], [ %memptr.nonvirtualfn, %[[CONT2]] ]
(s->*p)();
}
// CHECK-LABEL: _Z19MemberFuncCallNoCFIP1AMS_FvvE
// CHECK-NOT: llvm.type.test
void MemberFuncCallNoCFI(A *s, void (CFI_UNCHECKED_CALLEE A::*p)()) {
// CHECK: memptr.virtual:
// CHECK-NEXT: [[VTABLE:%.*]] = load ptr, ptr {{.*}}, align 8
// CHECK-NEXT: [[OFFSET:%.*]] = sub i64 %memptr.ptr, 1
// CHECK-NEXT: [[FUNC:%.*]] = getelementptr i8, ptr [[VTABLE]], i64 [[OFFSET]]
// CHECK-NEXT: %memptr.virtualfn = load ptr, ptr [[FUNC]], align 8
// CHECK-NEXT: br label %memptr.end
// CHECK: memptr.nonvirtual:
// CHECK-NEXT: %memptr.nonvirtualfn = inttoptr i64 %memptr.ptr to ptr
// CHECK-NEXT: br label %memptr.end
// CHECK: memptr.end:
// CHECK-NEXT: {{.*}} = phi ptr [ %memptr.virtualfn, %memptr.virtual ], [ %memptr.nonvirtualfn, %memptr.nonvirtual ]
(s->*p)();
}
|