xref: /aosp_15_r20/external/compiler-rt/test/cfi/bad-cast.cpp (revision 7c3d14c8b49c529e04be81a3ce6f5cc23712e4c6)
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 Robot void 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 Robot void B::f() {}
85*7c3d14c8STreehugger Robot 
86*7c3d14c8STreehugger Robot struct C : A {
87*7c3d14c8STreehugger Robot };
88*7c3d14c8STreehugger Robot 
main(int argc,char ** argv)89*7c3d14c8STreehugger Robot int 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