67 lines
2.7 KiB
C++
67 lines
2.7 KiB
C++
|
// RUN: %clang_cc1 -fsyntax-only -verify %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<cppp>(var);
|
||
|
char ***const &var3 = var2;
|
||
|
// Const reference to reference.
|
||
|
char ***&var4 = const_cast<cpppr>(var3);
|
||
|
// Drop reference. Intentionally without qualifier change.
|
||
|
char *** var5 = const_cast<cppp>(var4);
|
||
|
// Const array to array reference.
|
||
|
const int ar[100] = {0};
|
||
|
int (&rar)[100] = const_cast<iarr>(ar);
|
||
|
// Array decay. Intentionally without qualifier change.
|
||
|
int *pi = const_cast<int*>(ar);
|
||
|
f fp = 0;
|
||
|
// Don't misidentify fn** as a function pointer.
|
||
|
f *fpp = const_cast<f*>(&fp);
|
||
|
int const A::* const A::*icapcap = 0;
|
||
|
int A::* A::* iapap = const_cast<int A::* A::*>(icapcap);
|
||
|
(void)const_cast<A&&>(A()); // expected-warning {{C++11}}
|
||
|
|
||
|
return var4;
|
||
|
}
|
||
|
|
||
|
short *bad_const_cast_test(char const *volatile *const volatile *var)
|
||
|
{
|
||
|
// Different pointer levels.
|
||
|
char **var2 = const_cast<char**>(var); // expected-error {{const_cast from 'const char *volatile *const volatile *' to 'char **' is not allowed}}
|
||
|
// Different final type.
|
||
|
short ***var3 = const_cast<short***>(var); // expected-error {{const_cast from 'const char *volatile *const volatile *' to 'short ***' is not allowed}}
|
||
|
// Rvalue to reference.
|
||
|
char ***&var4 = const_cast<cpppr>(&var2); // expected-error {{const_cast from rvalue to reference type 'cpppr'}}
|
||
|
// Non-pointer.
|
||
|
char v = const_cast<char>(**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};
|
||
|
// Not even lenient g++ accepts this.
|
||
|
int *(*rar)[100] = const_cast<int *(*)[100]>(&ar); // expected-error {{const_cast from 'const int *(*)[100]' to 'int *(*)[100]' is not allowed}}
|
||
|
f fp1 = 0;
|
||
|
// Function pointers.
|
||
|
f fp2 = const_cast<f>(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<void (A::*)()>(mfn); // expected-error {{const_cast to 'void (A::*)()', which is not a reference, pointer-to-object, or pointer-to-data-member}}
|
||
|
(void)const_cast<int&&>(0); // expected-error {{const_cast from rvalue to reference type 'int &&'}} expected-warning {{C++11}}
|
||
|
return **var3;
|
||
|
}
|