// RUN: %clang_cc1 -fsyntax-only -verify %s // RUN: %clang_cc1 -fsyntax-only -verify -std=c++98 %s // RUN: %clang_cc1 -fsyntax-only -verify -std=c++11 %s struct A {}; // See if aliasing can confuse this baby. typedef char c; typedef c *cp; typedef cp *cpp; typedef cpp *cppp; typedef cppp &cpppr; typedef const cppp &cpppcr; typedef const char cc; typedef cc *ccp; typedef volatile ccp ccvp; typedef ccvp *ccvpp; typedef const volatile ccvpp ccvpcvp; typedef ccvpcvp *ccvpcvpp; typedef int iar[100]; typedef iar &iarr; typedef int (*f)(int); char ***good_const_cast_test(ccvpcvpp var) { // Cast away deep consts and volatiles. char ***var2 = const_cast(var); char ***const &var3 = var2; // Const reference to reference. char ***&var4 = const_cast(var3); // Drop reference. Intentionally without qualifier change. char *** var5 = const_cast(var4); // Const array to array reference. const int ar[100] = {0}; int (&rar)[100] = const_cast(ar); // Array decay. Intentionally without qualifier change. int *pi = const_cast(ar); f fp = 0; // Don't misidentify fn** as a function pointer. f *fpp = const_cast(&fp); int const A::* const A::*icapcap = 0; int A::* A::* iapap = const_cast(icapcap); (void)const_cast(A()); #if __cplusplus <= 199711L // C++03 or earlier modes // expected-warning@-2 {{rvalue references are a C++11 extension}} #endif return var4; } short *bad_const_cast_test(char const *volatile *const volatile *var) { // Different pointer levels. char **var2 = const_cast(var); // expected-error {{const_cast from 'const char *volatile *const volatile *' to 'char **' is not allowed}} // Different final type. short ***var3 = const_cast(var); // expected-error {{const_cast from 'const char *volatile *const volatile *' to 'short ***' is not allowed}} // Rvalue to reference. char ***&var4 = const_cast(&var2); // expected-error {{const_cast from rvalue to reference type 'cpppr'}} // Non-pointer. char v = const_cast(**var2); // expected-error {{const_cast to 'char', which is not a reference, pointer-to-object, or pointer-to-data-member}} const int *ar[100] = {0}; extern const int *aub[]; // const_cast looks through arrays as of DR330. (void) const_cast(&ar); // ok (void) const_cast(&aub); // ok // ... but the array bound must exactly match. (void) const_cast(&ar); // expected-error {{const_cast from 'const int *(*)[100]' to 'int *(*)[]' is not allowed}} (void) const_cast(&ar); // expected-error {{const_cast from 'const int *(*)[100]' to 'int *(*)[99]' is not allowed}} (void) const_cast(&aub); // expected-error {{const_cast from 'const int *(*)[]' to 'int *(*)[100]' is not allowed}} f fp1 = 0; // Function pointers. f fp2 = const_cast(fp1); // expected-error {{const_cast to 'f' (aka 'int (*)(int)'), which is not a reference, pointer-to-object, or pointer-to-data-member}} void (A::*mfn)() = 0; (void)const_cast(mfn); // expected-error-re {{const_cast to 'void (A::*)(){{( __attribute__\(\(thiscall\)\))?}}', which is not a reference, pointer-to-object, or pointer-to-data-member}} (void)const_cast(0); // expected-error {{const_cast from rvalue to reference type 'int &&'}} #if __cplusplus <= 199711L // C++03 or earlier modes // expected-warning@-2 {{rvalue references are a C++11 extension}} #endif return **var3; } template char *PR21845() { return const_cast((void)T::x); } // expected-error {{const_cast from 'void' to 'char *' is not allowed}}