76 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			76 lines
		
	
	
		
			2.8 KiB
		
	
	
	
		
			C++
		
	
	
	
| // RUN: %clang_cc1 -DNO_DTORS -DNO_EXCEPTIONS -fno-c++-static-destructors -verify %s
 | |
| // RUN: %clang_cc1 -DNO_EXCEPTIONS -verify %s
 | |
| // RUN: %clang_cc1 -DNO_DTORS -fexceptions -fno-c++-static-destructors -verify %s
 | |
| // RUN: %clang_cc1 -fexceptions -verify %s
 | |
| 
 | |
| struct SecretDestructor {
 | |
| #ifndef NO_DTORS
 | |
|   // expected-note@+2 4 {{private}}
 | |
| #endif
 | |
| private: ~SecretDestructor(); // expected-note + {{private}}
 | |
| };
 | |
| 
 | |
| SecretDestructor sd1;
 | |
| thread_local SecretDestructor sd2;
 | |
| void locals() {
 | |
|   static SecretDestructor sd3;
 | |
|   thread_local SecretDestructor sd4;
 | |
| }
 | |
| 
 | |
| #ifndef NO_DTORS
 | |
| // SecretDestructor sd1;                  // expected-error@-8 {{private}}
 | |
| // thread_local SecretDestructor sd2;     // expected-error@-8 {{private}}
 | |
| // void locals() {
 | |
| //   static SecretDestructor sd3;         // expected-error@-8 {{private}}
 | |
| //   thread_local SecretDestructor sd4;   // expected-error@-8 {{private}}
 | |
| // }
 | |
| #endif
 | |
| 
 | |
| [[clang::always_destroy]] SecretDestructor sd6; // expected-error{{private}}
 | |
| [[clang::always_destroy]] thread_local SecretDestructor sd7; // expected-error{{private}}
 | |
| 
 | |
| [[clang::no_destroy]] SecretDestructor sd8;
 | |
| 
 | |
| int main() {
 | |
|   [[clang::no_destroy]] int p; // expected-error{{no_destroy attribute can only be applied to a variable with static or thread storage duration}}
 | |
|   [[clang::always_destroy]] int p2; // expected-error{{always_destroy attribute can only be applied to a variable with static or thread storage duration}}
 | |
|   [[clang::no_destroy]] static int p3;
 | |
|   [[clang::always_destroy]] static int p4;
 | |
| }
 | |
| 
 | |
| [[clang::always_destroy]] [[clang::no_destroy]] int p; // expected-error{{'no_destroy' and 'always_destroy' attributes are not compatible}} // expected-note{{here}}
 | |
| [[clang::no_destroy]] [[clang::always_destroy]] int p2; // expected-error{{'always_destroy' and 'no_destroy' attributes are not compatible}} // expected-note{{here}}
 | |
| 
 | |
| [[clang::always_destroy]] void f() {} // expected-warning{{'always_destroy' attribute only applies to variables}}
 | |
| struct [[clang::no_destroy]] DoesntApply {};  // expected-warning{{'no_destroy' attribute only applies to variables}}
 | |
| 
 | |
| [[clang::no_destroy(0)]] int no_args; // expected-error{{'no_destroy' attribute takes no arguments}}
 | |
| [[clang::always_destroy(0)]] int no_args2; // expected-error{{'always_destroy' attribute takes no arguments}}
 | |
| 
 | |
| // expected-error@+1 {{temporary of type 'SecretDestructor' has private destructor}}
 | |
| SecretDestructor arr[10];
 | |
| 
 | |
| void local_arrays() {
 | |
|   // expected-error@+1 {{temporary of type 'SecretDestructor' has private destructor}}
 | |
|   static SecretDestructor arr2[10];
 | |
|   // expected-error@+1 {{temporary of type 'SecretDestructor' has private destructor}}
 | |
|   thread_local SecretDestructor arr3[10];
 | |
| }
 | |
| 
 | |
| struct Base {
 | |
|   ~Base();
 | |
| };
 | |
| struct Derived1 {
 | |
|   Derived1(int);
 | |
|   Base b;
 | |
| };
 | |
| struct Derived2 {
 | |
|   Derived1 b;
 | |
| };
 | |
| 
 | |
| void dontcrash() {
 | |
|   [[clang::no_destroy]] static Derived2 d2[] = {0, 0};
 | |
| }
 | |
| 
 | |
| [[clang::no_destroy]] Derived2 d2[] = {0, 0};
 |