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;
 | |
| }
 |