xref: /aosp_15_r20/external/clang/test/SemaCXX/warn-reinterpret-base-class.cpp (revision 67e74705e28f6214e480b399dd47ea732279e315)
1*67e74705SXin Li // RUN: %clang_cc1 -std=c++11 -fsyntax-only -triple %itanium_abi_triple -verify -Wreinterpret-base-class -Wno-unused-volatile-lvalue %s
2*67e74705SXin Li // RUN: %clang_cc1 -std=c++11 -fsyntax-only -triple %ms_abi_triple -DMSABI -verify -Wreinterpret-base-class -Wno-unused-volatile-lvalue %s
3*67e74705SXin Li 
4*67e74705SXin Li // RUN: not %clang_cc1 -std=c++11 -fsyntax-only -triple %itanium_abi_triple -fdiagnostics-parseable-fixits -Wreinterpret-base-class -Wno-unused-volatile-lvalue %s 2>&1 | FileCheck %s
5*67e74705SXin Li // RUN: not %clang_cc1 -std=c++11 -fsyntax-only -triple %ms_abi_triple -fdiagnostics-parseable-fixits -Wreinterpret-base-class -Wno-unused-volatile-lvalue %s 2>&1 | FileCheck %s
6*67e74705SXin Li 
7*67e74705SXin Li // PR 13824
8*67e74705SXin Li class A {
9*67e74705SXin Li };
10*67e74705SXin Li class DA : public A {
11*67e74705SXin Li };
12*67e74705SXin Li class DDA : public DA {
13*67e74705SXin Li };
14*67e74705SXin Li class DAo : protected A {
15*67e74705SXin Li };
16*67e74705SXin Li class DAi : private A {
17*67e74705SXin Li };
18*67e74705SXin Li 
19*67e74705SXin Li class DVA : public virtual A {
20*67e74705SXin Li };
21*67e74705SXin Li class DDVA : public virtual DA {
22*67e74705SXin Li };
23*67e74705SXin Li class DMA : public virtual A, public virtual DA { //expected-warning{{direct base 'A' is inaccessible due to ambiguity:\n    class DMA -> class A\n    class DMA -> class DA -> class A}}
24*67e74705SXin Li };
25*67e74705SXin Li 
26*67e74705SXin Li class B;
27*67e74705SXin Li 
28*67e74705SXin Li struct C {
29*67e74705SXin Li   // Do not fail on incompletely-defined classes.
30*67e74705SXin Li   decltype(reinterpret_cast<C *>(0)) foo;
31*67e74705SXin Li   decltype(reinterpret_cast<A *>((C *) 0)) bar;
32*67e74705SXin Li   decltype(reinterpret_cast<C *>((A *) 0)) baz;
33*67e74705SXin Li };
34*67e74705SXin Li 
reinterpret_not_defined_class(B * b,C * c)35*67e74705SXin Li void reinterpret_not_defined_class(B *b, C *c) {
36*67e74705SXin Li   // Should not fail if class has no definition.
37*67e74705SXin Li   (void)*reinterpret_cast<C *>(b);
38*67e74705SXin Li   (void)*reinterpret_cast<B *>(c);
39*67e74705SXin Li 
40*67e74705SXin Li   (void)reinterpret_cast<C &>(*b);
41*67e74705SXin Li   (void)reinterpret_cast<B &>(*c);
42*67e74705SXin Li }
43*67e74705SXin Li 
44*67e74705SXin Li // Do not fail on erroneous classes with fields of incompletely-defined types.
45*67e74705SXin Li // Base class is malformed.
46*67e74705SXin Li namespace BaseMalformed {
47*67e74705SXin Li   struct A; // expected-note {{forward declaration of 'BaseMalformed::A'}}
48*67e74705SXin Li   struct B {
49*67e74705SXin Li     A a; // expected-error {{field has incomplete type 'BaseMalformed::A'}}
50*67e74705SXin Li   };
51*67e74705SXin Li   struct C : public B {} c;
52*67e74705SXin Li   B *b = reinterpret_cast<B *>(&c);
53*67e74705SXin Li } // end anonymous namespace
54*67e74705SXin Li 
55*67e74705SXin Li // Child class is malformed.
56*67e74705SXin Li namespace ChildMalformed {
57*67e74705SXin Li   struct A; // expected-note {{forward declaration of 'ChildMalformed::A'}}
58*67e74705SXin Li   struct B {};
59*67e74705SXin Li   struct C : public B {
60*67e74705SXin Li     A a; // expected-error {{field has incomplete type 'ChildMalformed::A'}}
61*67e74705SXin Li   } c;
62*67e74705SXin Li   B *b = reinterpret_cast<B *>(&c);
63*67e74705SXin Li } // end anonymous namespace
64*67e74705SXin Li 
65*67e74705SXin Li // Base class outside upcast base-chain is malformed.
66*67e74705SXin Li namespace BaseBaseMalformed {
67*67e74705SXin Li   struct A; // expected-note {{forward declaration of 'BaseBaseMalformed::A'}}
68*67e74705SXin Li   struct Y {};
69*67e74705SXin Li   struct X { A a; }; // expected-error {{field has incomplete type 'BaseBaseMalformed::A'}}
70*67e74705SXin Li   struct B : Y, X {};
71*67e74705SXin Li   struct C : B {} c;
72*67e74705SXin Li   B *p = reinterpret_cast<B*>(&c);
73*67e74705SXin Li }
74*67e74705SXin Li 
75*67e74705SXin Li namespace InheritanceMalformed {
76*67e74705SXin Li   struct A; // expected-note {{forward declaration of 'InheritanceMalformed::A'}}
77*67e74705SXin Li   struct B : A {}; // expected-error {{base class has incomplete type}}
78*67e74705SXin Li   struct C : B {} c;
79*67e74705SXin Li   B *p = reinterpret_cast<B*>(&c);
80*67e74705SXin Li }
81*67e74705SXin Li 
82*67e74705SXin Li // Virtual base class outside upcast base-chain is malformed.
83*67e74705SXin Li namespace VBaseMalformed{
84*67e74705SXin Li   struct A; // expected-note {{forward declaration of 'VBaseMalformed::A'}}
85*67e74705SXin Li   struct X { A a; };  // expected-error {{field has incomplete type 'VBaseMalformed::A'}}
86*67e74705SXin Li   struct B : public virtual X {};
87*67e74705SXin Li   struct C : B {} c;
88*67e74705SXin Li   B *p = reinterpret_cast<B*>(&c);
89*67e74705SXin Li }
90*67e74705SXin Li 
reinterpret_not_updowncast(A * pa,const A * pca,A & a,const A & ca)91*67e74705SXin Li void reinterpret_not_updowncast(A *pa, const A *pca, A &a, const A &ca) {
92*67e74705SXin Li   (void)*reinterpret_cast<C *>(pa);
93*67e74705SXin Li   (void)*reinterpret_cast<const C *>(pa);
94*67e74705SXin Li   (void)*reinterpret_cast<volatile C *>(pa);
95*67e74705SXin Li   (void)*reinterpret_cast<const volatile C *>(pa);
96*67e74705SXin Li 
97*67e74705SXin Li   (void)*reinterpret_cast<const C *>(pca);
98*67e74705SXin Li   (void)*reinterpret_cast<const volatile C *>(pca);
99*67e74705SXin Li 
100*67e74705SXin Li   (void)reinterpret_cast<C &>(a);
101*67e74705SXin Li   (void)reinterpret_cast<const C &>(a);
102*67e74705SXin Li   (void)reinterpret_cast<volatile C &>(a);
103*67e74705SXin Li   (void)reinterpret_cast<const volatile C &>(a);
104*67e74705SXin Li 
105*67e74705SXin Li   (void)reinterpret_cast<const C &>(ca);
106*67e74705SXin Li   (void)reinterpret_cast<const volatile C &>(ca);
107*67e74705SXin Li }
108*67e74705SXin Li 
reinterpret_pointer_downcast(A * a,const A * ca)109*67e74705SXin Li void reinterpret_pointer_downcast(A *a, const A *ca) {
110*67e74705SXin Li   (void)*reinterpret_cast<DA *>(a);
111*67e74705SXin Li   (void)*reinterpret_cast<const DA *>(a);
112*67e74705SXin Li   (void)*reinterpret_cast<volatile DA *>(a);
113*67e74705SXin Li   (void)*reinterpret_cast<const volatile DA *>(a);
114*67e74705SXin Li 
115*67e74705SXin Li   (void)*reinterpret_cast<const DA *>(ca);
116*67e74705SXin Li   (void)*reinterpret_cast<const volatile DA *>(ca);
117*67e74705SXin Li 
118*67e74705SXin Li   (void)*reinterpret_cast<DDA *>(a);
119*67e74705SXin Li   (void)*reinterpret_cast<DAo *>(a);
120*67e74705SXin Li   (void)*reinterpret_cast<DAi *>(a);
121*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'DVA *' from its virtual base 'A *' behaves differently from 'static_cast'}}
122*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
123*67e74705SXin Li   (void)*reinterpret_cast<DVA *>(a);
124*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast"
125*67e74705SXin Li 
126*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'DDVA *' from its virtual base 'A *' behaves differently from 'static_cast'}}
127*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
128*67e74705SXin Li   (void)*reinterpret_cast<DDVA *>(a);
129*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast"
130*67e74705SXin Li 
131*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'DMA *' from its virtual base 'A *' behaves differently from 'static_cast'}}
132*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
133*67e74705SXin Li   (void)*reinterpret_cast<DMA *>(a);
134*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast"
135*67e74705SXin Li }
136*67e74705SXin Li 
reinterpret_reference_downcast(A a,A & ra,const A & cra)137*67e74705SXin Li void reinterpret_reference_downcast(A a, A &ra, const A &cra) {
138*67e74705SXin Li   (void)reinterpret_cast<DA &>(a);
139*67e74705SXin Li   (void)reinterpret_cast<const DA &>(a);
140*67e74705SXin Li   (void)reinterpret_cast<volatile DA &>(a);
141*67e74705SXin Li   (void)reinterpret_cast<const volatile DA &>(a);
142*67e74705SXin Li 
143*67e74705SXin Li   (void)reinterpret_cast<DA &>(ra);
144*67e74705SXin Li   (void)reinterpret_cast<const DA &>(ra);
145*67e74705SXin Li   (void)reinterpret_cast<volatile DA &>(ra);
146*67e74705SXin Li   (void)reinterpret_cast<const volatile DA &>(ra);
147*67e74705SXin Li 
148*67e74705SXin Li   (void)reinterpret_cast<const DA &>(cra);
149*67e74705SXin Li   (void)reinterpret_cast<const volatile DA &>(cra);
150*67e74705SXin Li 
151*67e74705SXin Li   (void)reinterpret_cast<DDA &>(a);
152*67e74705SXin Li   (void)reinterpret_cast<DAo &>(a);
153*67e74705SXin Li   (void)reinterpret_cast<DAi &>(a);
154*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'DVA &' from its virtual base 'A' behaves differently from 'static_cast'}}
155*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
156*67e74705SXin Li   (void)reinterpret_cast<DVA &>(a);
157*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
158*67e74705SXin Li 
159*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'DDVA &' from its virtual base 'A' behaves differently from 'static_cast'}}
160*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
161*67e74705SXin Li   (void)reinterpret_cast<DDVA &>(a);
162*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
163*67e74705SXin Li 
164*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'DMA &' from its virtual base 'A' behaves differently from 'static_cast'}}
165*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
166*67e74705SXin Li   (void)reinterpret_cast<DMA &>(a);
167*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
168*67e74705SXin Li }
169*67e74705SXin Li 
reinterpret_pointer_upcast(DA * da,const DA * cda,DDA * dda,DAo * dao,DAi * dai,DVA * dva,DDVA * ddva,DMA * dma)170*67e74705SXin Li void reinterpret_pointer_upcast(DA *da, const DA *cda, DDA *dda, DAo *dao,
171*67e74705SXin Li                                 DAi *dai, DVA *dva, DDVA *ddva, DMA *dma) {
172*67e74705SXin Li   (void)*reinterpret_cast<A *>(da);
173*67e74705SXin Li   (void)*reinterpret_cast<const A *>(da);
174*67e74705SXin Li   (void)*reinterpret_cast<volatile A *>(da);
175*67e74705SXin Li   (void)*reinterpret_cast<const volatile A *>(da);
176*67e74705SXin Li 
177*67e74705SXin Li   (void)*reinterpret_cast<const A *>(cda);
178*67e74705SXin Li   (void)*reinterpret_cast<const volatile A *>(cda);
179*67e74705SXin Li 
180*67e74705SXin Li   (void)*reinterpret_cast<A *>(dda);
181*67e74705SXin Li   (void)*reinterpret_cast<DA *>(dda);
182*67e74705SXin Li   (void)*reinterpret_cast<A *>(dao);
183*67e74705SXin Li   (void)*reinterpret_cast<A *>(dai);
184*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DVA *' to its virtual base 'A *' behaves differently from 'static_cast'}}
185*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
186*67e74705SXin Li   (void)*reinterpret_cast<A *>(dva);
187*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast"
188*67e74705SXin Li 
189*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DDVA *' to its virtual base 'A *' behaves differently from 'static_cast'}}
190*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
191*67e74705SXin Li   (void)*reinterpret_cast<A *>(ddva);
192*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast"
193*67e74705SXin Li 
194*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DDVA *' to its virtual base 'DA *' behaves differently from 'static_cast'}}
195*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
196*67e74705SXin Li   (void)*reinterpret_cast<DA *>(ddva);
197*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast"
198*67e74705SXin Li 
199*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DMA *' to its virtual base 'A *' behaves differently from 'static_cast'}}
200*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
201*67e74705SXin Li   (void)*reinterpret_cast<A *>(dma);
202*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast"
203*67e74705SXin Li 
204*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DMA *' to its virtual base 'DA *' behaves differently from 'static_cast'}}
205*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
206*67e74705SXin Li   (void)*reinterpret_cast<DA *>(dma);
207*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast"
208*67e74705SXin Li }
209*67e74705SXin Li 
reinterpret_reference_upcast(DA & da,const DA & cda,DDA & dda,DAo & dao,DAi & dai,DVA & dva,DDVA & ddva,DMA & dma)210*67e74705SXin Li void reinterpret_reference_upcast(DA &da, const DA &cda, DDA &dda, DAo &dao,
211*67e74705SXin Li                                   DAi &dai, DVA &dva, DDVA &ddva, DMA &dma) {
212*67e74705SXin Li   (void)reinterpret_cast<A &>(da);
213*67e74705SXin Li   (void)reinterpret_cast<const A &>(da);
214*67e74705SXin Li   (void)reinterpret_cast<volatile A &>(da);
215*67e74705SXin Li   (void)reinterpret_cast<const volatile A &>(da);
216*67e74705SXin Li 
217*67e74705SXin Li   (void)reinterpret_cast<const A &>(cda);
218*67e74705SXin Li   (void)reinterpret_cast<const volatile A &>(cda);
219*67e74705SXin Li 
220*67e74705SXin Li   (void)reinterpret_cast<A &>(dda);
221*67e74705SXin Li   (void)reinterpret_cast<DA &>(dda);
222*67e74705SXin Li   (void)reinterpret_cast<A &>(dao);
223*67e74705SXin Li   (void)reinterpret_cast<A &>(dai);
224*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DVA' to its virtual base 'A &' behaves differently from 'static_cast'}}
225*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
226*67e74705SXin Li   (void)reinterpret_cast<A &>(dva);
227*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
228*67e74705SXin Li 
229*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DDVA' to its virtual base 'A &' behaves differently from 'static_cast'}}
230*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
231*67e74705SXin Li   (void)reinterpret_cast<A &>(ddva);
232*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
233*67e74705SXin Li 
234*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DDVA' to its virtual base 'DA &' behaves differently from 'static_cast'}}
235*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
236*67e74705SXin Li   (void)reinterpret_cast<DA &>(ddva);
237*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
238*67e74705SXin Li 
239*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DMA' to its virtual base 'A &' behaves differently from 'static_cast'}}
240*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
241*67e74705SXin Li   (void)reinterpret_cast<A &>(dma);
242*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
243*67e74705SXin Li 
244*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'DMA' to its virtual base 'DA &' behaves differently from 'static_cast'}}
245*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
246*67e74705SXin Li   (void)reinterpret_cast<DA &>(dma);
247*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
248*67e74705SXin Li }
249*67e74705SXin Li 
250*67e74705SXin Li struct E {
251*67e74705SXin Li   int x;
252*67e74705SXin Li };
253*67e74705SXin Li 
254*67e74705SXin Li class F : public E {
foo()255*67e74705SXin Li   virtual int foo() { return x; }
256*67e74705SXin Li };
257*67e74705SXin Li 
258*67e74705SXin Li class G : public F {
259*67e74705SXin Li };
260*67e74705SXin Li 
261*67e74705SXin Li class H : public E, public A {
262*67e74705SXin Li };
263*67e74705SXin Li 
264*67e74705SXin Li class I : virtual public F {
265*67e74705SXin Li };
266*67e74705SXin Li 
267*67e74705SXin Li typedef const F * K;
268*67e74705SXin Li typedef volatile K L;
269*67e74705SXin Li 
different_subobject_downcast(E * e,F * f,A * a)270*67e74705SXin Li void different_subobject_downcast(E *e, F *f, A *a) {
271*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'F *' from its base at non-zero offset 'E *' behaves differently from 'static_cast'}}
272*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
273*67e74705SXin Li   (void)reinterpret_cast<F *>(e);
274*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
275*67e74705SXin Li 
276*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'G *' from its base at non-zero offset 'E *' behaves differently from 'static_cast'}}
277*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
278*67e74705SXin Li   (void)reinterpret_cast<G *>(e);
279*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
280*67e74705SXin Li 
281*67e74705SXin Li   (void)reinterpret_cast<H *>(e);
282*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'I *' from its virtual base 'E *' behaves differently from 'static_cast'}}
283*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
284*67e74705SXin Li   (void)reinterpret_cast<I *>(e);
285*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
286*67e74705SXin Li 
287*67e74705SXin Li 
288*67e74705SXin Li   (void)reinterpret_cast<G *>(f);
289*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'I *' from its virtual base 'F *' behaves differently from 'static_cast'}}
290*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
291*67e74705SXin Li   (void)reinterpret_cast<I *>(f);
292*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
293*67e74705SXin Li 
294*67e74705SXin Li #ifdef MSABI
295*67e74705SXin Li   // In MS ABI mode, A is at non-zero offset in H.
296*67e74705SXin Li   // expected-warning@+3 {{'reinterpret_cast' to class 'H *' from its base at non-zero offset 'A *' behaves differently from 'static_cast'}}
297*67e74705SXin Li   // expected-note@+2 {{use 'static_cast'}}
298*67e74705SXin Li #endif
299*67e74705SXin Li   (void)reinterpret_cast<H *>(a);
300*67e74705SXin Li 
301*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' to class 'L' (aka 'const F *volatile') from its base at non-zero offset 'E *' behaves differently from 'static_cast'}}
302*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}}
303*67e74705SXin Li   (void)reinterpret_cast<L>(e);
304*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
305*67e74705SXin Li }
306*67e74705SXin Li 
different_subobject_upcast(F * f,G * g,H * h,I * i)307*67e74705SXin Li void different_subobject_upcast(F *f, G *g, H *h, I *i) {
308*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'F *' to its base at non-zero offset 'E *' behaves differently from 'static_cast'}}
309*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
310*67e74705SXin Li   (void)reinterpret_cast<E *>(f);
311*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
312*67e74705SXin Li 
313*67e74705SXin Li   (void)reinterpret_cast<F *>(g);
314*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'G *' to its base at non-zero offset 'E *' behaves differently from 'static_cast'}}
315*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
316*67e74705SXin Li   (void)reinterpret_cast<E *>(g);
317*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
318*67e74705SXin Li 
319*67e74705SXin Li   (void)reinterpret_cast<E *>(h);
320*67e74705SXin Li 
321*67e74705SXin Li #ifdef MSABI
322*67e74705SXin Li   // In MS ABI mode, A is at non-zero offset in H.
323*67e74705SXin Li   // expected-warning@+3 {{'reinterpret_cast' from class 'H *' to its base at non-zero offset 'A *' behaves differently from 'static_cast'}}
324*67e74705SXin Li   // expected-note@+2 {{use 'static_cast'}}
325*67e74705SXin Li #endif
326*67e74705SXin Li   (void)reinterpret_cast<A *>(h);
327*67e74705SXin Li 
328*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'I *' to its virtual base 'F *' behaves differently from 'static_cast'}}
329*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
330*67e74705SXin Li   (void)reinterpret_cast<F *>(i);
331*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
332*67e74705SXin Li 
333*67e74705SXin Li   // expected-warning@+2 {{'reinterpret_cast' from class 'I *' to its virtual base 'E *' behaves differently from 'static_cast'}}
334*67e74705SXin Li   // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}}
335*67e74705SXin Li   (void)reinterpret_cast<E *>(i);
336*67e74705SXin Li   // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast"
337*67e74705SXin Li }
338