1*7c3d14c8STreehugger Robot // RUN: %clangxx_cfi -o %t1 %s 2*7c3d14c8STreehugger Robot // RUN: %expect_crash %t1 a 2>&1 | FileCheck --check-prefix=FAIL %s 3*7c3d14c8STreehugger Robot // RUN: %expect_crash %t1 b 2>&1 | FileCheck --check-prefix=FAIL %s 4*7c3d14c8STreehugger Robot // RUN: %expect_crash %t1 c 2>&1 | FileCheck --check-prefix=FAIL %s 5*7c3d14c8STreehugger Robot // RUN: %t1 d 2>&1 | FileCheck --check-prefix=PASS %s 6*7c3d14c8STreehugger Robot // RUN: %t1 e 2>&1 | FileCheck --check-prefix=PASS %s 7*7c3d14c8STreehugger Robot // RUN: %t1 f 2>&1 | FileCheck --check-prefix=PASS %s 8*7c3d14c8STreehugger Robot // RUN: %expect_crash %t1 g 2>&1 | FileCheck --check-prefix=FAIL %s 9*7c3d14c8STreehugger Robot // RUN: %t1 h 2>&1 | FileCheck --check-prefix=PASS %s 10*7c3d14c8STreehugger Robot 11*7c3d14c8STreehugger Robot // RUN: %clangxx_cfi -DB32 -o %t2 %s 12*7c3d14c8STreehugger Robot // RUN: %expect_crash %t2 a 2>&1 | FileCheck --check-prefix=FAIL %s 13*7c3d14c8STreehugger Robot // RUN: %expect_crash %t2 b 2>&1 | FileCheck --check-prefix=FAIL %s 14*7c3d14c8STreehugger Robot // RUN: %expect_crash %t2 c 2>&1 | FileCheck --check-prefix=FAIL %s 15*7c3d14c8STreehugger Robot // RUN: %t2 d 2>&1 | FileCheck --check-prefix=PASS %s 16*7c3d14c8STreehugger Robot // RUN: %t2 e 2>&1 | FileCheck --check-prefix=PASS %s 17*7c3d14c8STreehugger Robot // RUN: %t2 f 2>&1 | FileCheck --check-prefix=PASS %s 18*7c3d14c8STreehugger Robot // RUN: %expect_crash %t2 g 2>&1 | FileCheck --check-prefix=FAIL %s 19*7c3d14c8STreehugger Robot // RUN: %t2 h 2>&1 | FileCheck --check-prefix=PASS %s 20*7c3d14c8STreehugger Robot 21*7c3d14c8STreehugger Robot // RUN: %clangxx_cfi -DB64 -o %t3 %s 22*7c3d14c8STreehugger Robot // RUN: %expect_crash %t3 a 2>&1 | FileCheck --check-prefix=FAIL %s 23*7c3d14c8STreehugger Robot // RUN: %expect_crash %t3 b 2>&1 | FileCheck --check-prefix=FAIL %s 24*7c3d14c8STreehugger Robot // RUN: %expect_crash %t3 c 2>&1 | FileCheck --check-prefix=FAIL %s 25*7c3d14c8STreehugger Robot // RUN: %t3 d 2>&1 | FileCheck --check-prefix=PASS %s 26*7c3d14c8STreehugger Robot // RUN: %t3 e 2>&1 | FileCheck --check-prefix=PASS %s 27*7c3d14c8STreehugger Robot // RUN: %t3 f 2>&1 | FileCheck --check-prefix=PASS %s 28*7c3d14c8STreehugger Robot // RUN: %expect_crash %t3 g 2>&1 | FileCheck --check-prefix=FAIL %s 29*7c3d14c8STreehugger Robot // RUN: %t3 h 2>&1 | FileCheck --check-prefix=PASS %s 30*7c3d14c8STreehugger Robot 31*7c3d14c8STreehugger Robot // RUN: %clangxx_cfi -DBM -o %t4 %s 32*7c3d14c8STreehugger Robot // RUN: %expect_crash %t4 a 2>&1 | FileCheck --check-prefix=FAIL %s 33*7c3d14c8STreehugger Robot // RUN: %expect_crash %t4 b 2>&1 | FileCheck --check-prefix=FAIL %s 34*7c3d14c8STreehugger Robot // RUN: %expect_crash %t4 c 2>&1 | FileCheck --check-prefix=FAIL %s 35*7c3d14c8STreehugger Robot // RUN: %t4 d 2>&1 | FileCheck --check-prefix=PASS %s 36*7c3d14c8STreehugger Robot // RUN: %t4 e 2>&1 | FileCheck --check-prefix=PASS %s 37*7c3d14c8STreehugger Robot // RUN: %t4 f 2>&1 | FileCheck --check-prefix=PASS %s 38*7c3d14c8STreehugger Robot // RUN: %expect_crash %t4 g 2>&1 | FileCheck --check-prefix=FAIL %s 39*7c3d14c8STreehugger Robot // RUN: %t4 h 2>&1 | FileCheck --check-prefix=PASS %s 40*7c3d14c8STreehugger Robot 41*7c3d14c8STreehugger Robot // RUN: %clangxx_cfi -fsanitize=cfi-cast-strict -o %t5 %s 42*7c3d14c8STreehugger Robot // RUN: %expect_crash %t5 a 2>&1 | FileCheck --check-prefix=FAIL %s 43*7c3d14c8STreehugger Robot // RUN: %expect_crash %t5 b 2>&1 | FileCheck --check-prefix=FAIL %s 44*7c3d14c8STreehugger Robot // RUN: %expect_crash %t5 c 2>&1 | FileCheck --check-prefix=FAIL %s 45*7c3d14c8STreehugger Robot // RUN: %expect_crash %t5 d 2>&1 | FileCheck --check-prefix=FAIL %s 46*7c3d14c8STreehugger Robot // RUN: %expect_crash %t5 e 2>&1 | FileCheck --check-prefix=FAIL %s 47*7c3d14c8STreehugger Robot // RUN: %expect_crash %t5 f 2>&1 | FileCheck --check-prefix=FAIL %s 48*7c3d14c8STreehugger Robot // RUN: %expect_crash %t5 g 2>&1 | FileCheck --check-prefix=FAIL %s 49*7c3d14c8STreehugger Robot // RUN: %expect_crash %t5 h 2>&1 | FileCheck --check-prefix=FAIL %s 50*7c3d14c8STreehugger Robot 51*7c3d14c8STreehugger Robot // RUN: %clangxx -o %t6 %s 52*7c3d14c8STreehugger Robot // RUN: %t6 a 2>&1 | FileCheck --check-prefix=PASS %s 53*7c3d14c8STreehugger Robot // RUN: %t6 b 2>&1 | FileCheck --check-prefix=PASS %s 54*7c3d14c8STreehugger Robot // RUN: %t6 c 2>&1 | FileCheck --check-prefix=PASS %s 55*7c3d14c8STreehugger Robot // RUN: %t6 d 2>&1 | FileCheck --check-prefix=PASS %s 56*7c3d14c8STreehugger Robot // RUN: %t6 e 2>&1 | FileCheck --check-prefix=PASS %s 57*7c3d14c8STreehugger Robot // RUN: %t6 f 2>&1 | FileCheck --check-prefix=PASS %s 58*7c3d14c8STreehugger Robot // RUN: %t6 g 2>&1 | FileCheck --check-prefix=PASS %s 59*7c3d14c8STreehugger Robot // RUN: %t6 h 2>&1 | FileCheck --check-prefix=PASS %s 60*7c3d14c8STreehugger Robot 61*7c3d14c8STreehugger Robot // RUN: %clangxx_cfi_diag -o %t7 %s 62*7c3d14c8STreehugger Robot // RUN: %t7 a 2>&1 | FileCheck --check-prefix=CFI-DIAG-D %s 63*7c3d14c8STreehugger Robot // RUN: %t7 b 2>&1 | FileCheck --check-prefix=CFI-DIAG-D %s 64*7c3d14c8STreehugger Robot // RUN: %t7 c 2>&1 | FileCheck --check-prefix=CFI-DIAG-D %s 65*7c3d14c8STreehugger Robot // RUN: %t7 g 2>&1 | FileCheck --check-prefix=CFI-DIAG-U %s 66*7c3d14c8STreehugger Robot 67*7c3d14c8STreehugger Robot // Tests that the CFI enforcement detects bad casts. 68*7c3d14c8STreehugger Robot 69*7c3d14c8STreehugger Robot // REQUIRES: cxxabi 70*7c3d14c8STreehugger Robot 71*7c3d14c8STreehugger Robot #include <stdio.h> 72*7c3d14c8STreehugger Robot #include "utils.h" 73*7c3d14c8STreehugger Robot 74*7c3d14c8STreehugger Robot struct A { 75*7c3d14c8STreehugger Robot virtual void f(); 76*7c3d14c8STreehugger Robot }; 77*7c3d14c8STreehugger Robot f()78*7c3d14c8STreehugger Robotvoid A::f() {} 79*7c3d14c8STreehugger Robot 80*7c3d14c8STreehugger Robot struct B : A { 81*7c3d14c8STreehugger Robot virtual void f(); 82*7c3d14c8STreehugger Robot }; 83*7c3d14c8STreehugger Robot f()84*7c3d14c8STreehugger Robotvoid B::f() {} 85*7c3d14c8STreehugger Robot 86*7c3d14c8STreehugger Robot struct C : A { 87*7c3d14c8STreehugger Robot }; 88*7c3d14c8STreehugger Robot main(int argc,char ** argv)89*7c3d14c8STreehugger Robotint main(int argc, char **argv) { 90*7c3d14c8STreehugger Robot create_derivers<B>(); 91*7c3d14c8STreehugger Robot 92*7c3d14c8STreehugger Robot B *b = new B; 93*7c3d14c8STreehugger Robot break_optimization(b); 94*7c3d14c8STreehugger Robot 95*7c3d14c8STreehugger Robot // FAIL: 1 96*7c3d14c8STreehugger Robot // PASS: 1 97*7c3d14c8STreehugger Robot fprintf(stderr, "1\n"); 98*7c3d14c8STreehugger Robot 99*7c3d14c8STreehugger Robot A a; 100*7c3d14c8STreehugger Robot 101*7c3d14c8STreehugger Robot // CFI-DIAG-D: runtime error: control flow integrity check for type 'B' failed during base-to-derived cast 102*7c3d14c8STreehugger Robot // CFI-DIAG-D-NEXT: note: vtable is of type '{{(struct )?}}A' 103*7c3d14c8STreehugger Robot 104*7c3d14c8STreehugger Robot // CFI-DIAG-U: runtime error: control flow integrity check for type 'B' failed during cast to unrelated type 105*7c3d14c8STreehugger Robot // CFI-DIAG-U-NEXT: note: vtable is of type '{{(struct )?}}A' 106*7c3d14c8STreehugger Robot 107*7c3d14c8STreehugger Robot switch (argv[1][0]) { 108*7c3d14c8STreehugger Robot case 'a': 109*7c3d14c8STreehugger Robot static_cast<B *>(&a); // UB 110*7c3d14c8STreehugger Robot break; 111*7c3d14c8STreehugger Robot case 'b': 112*7c3d14c8STreehugger Robot static_cast<B &>(a); // UB 113*7c3d14c8STreehugger Robot break; 114*7c3d14c8STreehugger Robot case 'c': 115*7c3d14c8STreehugger Robot static_cast<B &&>(a); // UB 116*7c3d14c8STreehugger Robot break; 117*7c3d14c8STreehugger Robot case 'd': 118*7c3d14c8STreehugger Robot static_cast<C *>(&a); // UB, strict only 119*7c3d14c8STreehugger Robot break; 120*7c3d14c8STreehugger Robot case 'e': 121*7c3d14c8STreehugger Robot static_cast<C &>(a); // UB, strict only 122*7c3d14c8STreehugger Robot break; 123*7c3d14c8STreehugger Robot case 'f': 124*7c3d14c8STreehugger Robot static_cast<C &&>(a); // UB, strict only 125*7c3d14c8STreehugger Robot break; 126*7c3d14c8STreehugger Robot case 'g': 127*7c3d14c8STreehugger Robot static_cast<B *>(static_cast<void *>(&a)); // Non-UB bad cast 128*7c3d14c8STreehugger Robot break; 129*7c3d14c8STreehugger Robot case 'h': 130*7c3d14c8STreehugger Robot static_cast<C *>(static_cast<void *>(&a)); // Non-UB bad cast, strict only 131*7c3d14c8STreehugger Robot break; 132*7c3d14c8STreehugger Robot } 133*7c3d14c8STreehugger Robot 134*7c3d14c8STreehugger Robot // FAIL-NOT: {{^2$}} 135*7c3d14c8STreehugger Robot // PASS: {{^2$}} 136*7c3d14c8STreehugger Robot fprintf(stderr, "2\n"); 137*7c3d14c8STreehugger Robot } 138