// RUN: %clang_cc1 -std=c++11 -fsyntax-only -triple %itanium_abi_triple -verify -Wreinterpret-base-class -Wno-unused-volatile-lvalue %s // RUN: %clang_cc1 -std=c++11 -fsyntax-only -triple %ms_abi_triple -DMSABI -verify -Wreinterpret-base-class -Wno-unused-volatile-lvalue %s // 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 // 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 // PR 13824 class A { }; class DA : public A { }; class DDA : public DA { }; class DAo : protected A { }; class DAi : private A { }; class DVA : public virtual A { }; class DDVA : public virtual DA { }; 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}} }; class B; struct C { // Do not fail on incompletely-defined classes. decltype(reinterpret_cast(0)) foo; decltype(reinterpret_cast((C *) 0)) bar; decltype(reinterpret_cast((A *) 0)) baz; }; void reinterpret_not_defined_class(B *b, C *c) { // Should not fail if class has no definition. (void)*reinterpret_cast(b); (void)*reinterpret_cast(c); (void)reinterpret_cast(*b); (void)reinterpret_cast(*c); } // Do not fail on erroneous classes with fields of incompletely-defined types. // Base class is malformed. namespace BaseMalformed { struct A; // expected-note {{forward declaration of 'BaseMalformed::A'}} struct B { A a; // expected-error {{field has incomplete type 'BaseMalformed::A'}} }; struct C : public B {} c; B *b = reinterpret_cast(&c); } // end anonymous namespace // Child class is malformed. namespace ChildMalformed { struct A; // expected-note {{forward declaration of 'ChildMalformed::A'}} struct B {}; struct C : public B { A a; // expected-error {{field has incomplete type 'ChildMalformed::A'}} } c; B *b = reinterpret_cast(&c); } // end anonymous namespace // Base class outside upcast base-chain is malformed. namespace BaseBaseMalformed { struct A; // expected-note {{forward declaration of 'BaseBaseMalformed::A'}} struct Y {}; struct X { A a; }; // expected-error {{field has incomplete type 'BaseBaseMalformed::A'}} struct B : Y, X {}; struct C : B {} c; B *p = reinterpret_cast(&c); } namespace InheritanceMalformed { struct A; // expected-note {{forward declaration of 'InheritanceMalformed::A'}} struct B : A {}; // expected-error {{base class has incomplete type}} struct C : B {} c; B *p = reinterpret_cast(&c); } // Virtual base class outside upcast base-chain is malformed. namespace VBaseMalformed{ struct A; // expected-note {{forward declaration of 'VBaseMalformed::A'}} struct X { A a; }; // expected-error {{field has incomplete type 'VBaseMalformed::A'}} struct B : public virtual X {}; struct C : B {} c; B *p = reinterpret_cast(&c); } void reinterpret_not_updowncast(A *pa, const A *pca, A &a, const A &ca) { (void)*reinterpret_cast(pa); (void)*reinterpret_cast(pa); (void)*reinterpret_cast(pa); (void)*reinterpret_cast(pa); (void)*reinterpret_cast(pca); (void)*reinterpret_cast(pca); (void)reinterpret_cast(a); (void)reinterpret_cast(a); (void)reinterpret_cast(a); (void)reinterpret_cast(a); (void)reinterpret_cast(ca); (void)reinterpret_cast(ca); } void reinterpret_pointer_downcast(A *a, const A *ca) { (void)*reinterpret_cast(a); (void)*reinterpret_cast(a); (void)*reinterpret_cast(a); (void)*reinterpret_cast(a); (void)*reinterpret_cast(ca); (void)*reinterpret_cast(ca); (void)*reinterpret_cast(a); (void)*reinterpret_cast(a); (void)*reinterpret_cast(a); // expected-warning@+2 {{'reinterpret_cast' to class 'DVA *' from its virtual base 'A *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)*reinterpret_cast(a); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' to class 'DDVA *' from its virtual base 'A *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)*reinterpret_cast(a); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' to class 'DMA *' from its virtual base 'A *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)*reinterpret_cast(a); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast" } void reinterpret_reference_downcast(A a, A &ra, const A &cra) { (void)reinterpret_cast(a); (void)reinterpret_cast(a); (void)reinterpret_cast(a); (void)reinterpret_cast(a); (void)reinterpret_cast(ra); (void)reinterpret_cast(ra); (void)reinterpret_cast(ra); (void)reinterpret_cast(ra); (void)reinterpret_cast(cra); (void)reinterpret_cast(cra); (void)reinterpret_cast(a); (void)reinterpret_cast(a); (void)reinterpret_cast(a); // expected-warning@+2 {{'reinterpret_cast' to class 'DVA &' from its virtual base 'A' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)reinterpret_cast(a); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' to class 'DDVA &' from its virtual base 'A' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)reinterpret_cast(a); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' to class 'DMA &' from its virtual base 'A' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)reinterpret_cast(a); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" } void reinterpret_pointer_upcast(DA *da, const DA *cda, DDA *dda, DAo *dao, DAi *dai, DVA *dva, DDVA *ddva, DMA *dma) { (void)*reinterpret_cast(da); (void)*reinterpret_cast(da); (void)*reinterpret_cast(da); (void)*reinterpret_cast(da); (void)*reinterpret_cast(cda); (void)*reinterpret_cast(cda); (void)*reinterpret_cast(dda); (void)*reinterpret_cast(dda); (void)*reinterpret_cast(dao); (void)*reinterpret_cast(dai); // expected-warning@+2 {{'reinterpret_cast' from class 'DVA *' to its virtual base 'A *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)*reinterpret_cast(dva); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' from class 'DDVA *' to its virtual base 'A *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)*reinterpret_cast(ddva); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' from class 'DDVA *' to its virtual base 'DA *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)*reinterpret_cast(ddva); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' from class 'DMA *' to its virtual base 'A *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)*reinterpret_cast(dma); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' from class 'DMA *' to its virtual base 'DA *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)*reinterpret_cast(dma); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:10-[[@LINE-1]]:26}:"static_cast" } void reinterpret_reference_upcast(DA &da, const DA &cda, DDA &dda, DAo &dao, DAi &dai, DVA &dva, DDVA &ddva, DMA &dma) { (void)reinterpret_cast(da); (void)reinterpret_cast(da); (void)reinterpret_cast(da); (void)reinterpret_cast(da); (void)reinterpret_cast(cda); (void)reinterpret_cast(cda); (void)reinterpret_cast(dda); (void)reinterpret_cast(dda); (void)reinterpret_cast(dao); (void)reinterpret_cast(dai); // expected-warning@+2 {{'reinterpret_cast' from class 'DVA' to its virtual base 'A &' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)reinterpret_cast(dva); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' from class 'DDVA' to its virtual base 'A &' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)reinterpret_cast(ddva); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' from class 'DDVA' to its virtual base 'DA &' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)reinterpret_cast(ddva); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' from class 'DMA' to its virtual base 'A &' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)reinterpret_cast(dma); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' from class 'DMA' to its virtual base 'DA &' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)reinterpret_cast(dma); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" } struct E { int x; }; class F : public E { virtual int foo() { return x; } }; class G : public F { }; class H : public E, public A { }; class I : virtual public F { }; typedef const F * K; typedef volatile K L; void different_subobject_downcast(E *e, F *f, A *a) { // expected-warning@+2 {{'reinterpret_cast' to class 'F *' from its base at non-zero offset 'E *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)reinterpret_cast(e); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' to class 'G *' from its base at non-zero offset 'E *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)reinterpret_cast(e); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" (void)reinterpret_cast(e); // expected-warning@+2 {{'reinterpret_cast' to class 'I *' from its virtual base 'E *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)reinterpret_cast(e); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" (void)reinterpret_cast(f); // expected-warning@+2 {{'reinterpret_cast' to class 'I *' from its virtual base 'F *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)reinterpret_cast(f); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" #ifdef MSABI // In MS ABI mode, A is at non-zero offset in H. // expected-warning@+3 {{'reinterpret_cast' to class 'H *' from its base at non-zero offset 'A *' behaves differently from 'static_cast'}} // expected-note@+2 {{use 'static_cast'}} #endif (void)reinterpret_cast(a); // 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'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while downcasting}} (void)reinterpret_cast(e); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" } void different_subobject_upcast(F *f, G *g, H *h, I *i) { // expected-warning@+2 {{'reinterpret_cast' from class 'F *' to its base at non-zero offset 'E *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)reinterpret_cast(f); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" (void)reinterpret_cast(g); // expected-warning@+2 {{'reinterpret_cast' from class 'G *' to its base at non-zero offset 'E *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)reinterpret_cast(g); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" (void)reinterpret_cast(h); #ifdef MSABI // In MS ABI mode, A is at non-zero offset in H. // expected-warning@+3 {{'reinterpret_cast' from class 'H *' to its base at non-zero offset 'A *' behaves differently from 'static_cast'}} // expected-note@+2 {{use 'static_cast'}} #endif (void)reinterpret_cast(h); // expected-warning@+2 {{'reinterpret_cast' from class 'I *' to its virtual base 'F *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)reinterpret_cast(i); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" // expected-warning@+2 {{'reinterpret_cast' from class 'I *' to its virtual base 'E *' behaves differently from 'static_cast'}} // expected-note@+1 {{use 'static_cast' to adjust the pointer correctly while upcasting}} (void)reinterpret_cast(i); // CHECK: fix-it:"{{.*}}":{[[@LINE-1]]:9-[[@LINE-1]]:25}:"static_cast" }