169 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C++
		
	
	
	
			
		
		
	
	
			169 lines
		
	
	
		
			3.4 KiB
		
	
	
	
		
			C++
		
	
	
	
| // RUN: %clang_cc1 -std=c++11 -fsyntax-only -verify %s -fblocks
 | |
| 
 | |
| void tovoid(void*);
 | |
| 
 | |
| void tovoid_test(int (^f)(int, int)) {
 | |
|   tovoid(f);
 | |
| }
 | |
| 
 | |
| void reference_lvalue_test(int& (^f)()) {
 | |
|   f() = 10;
 | |
| }
 | |
| 
 | |
| // PR 7165
 | |
| namespace test1 {
 | |
|   void g(void (^)());
 | |
|   struct Foo {
 | |
|     void foo();   
 | |
|     void test() {
 | |
|       (void) ^{ foo(); };
 | |
|     }
 | |
|   };
 | |
| }
 | |
| 
 | |
| namespace test2 {
 | |
|   int repeat(int value, int (^block)(int), unsigned n) {
 | |
|     while (n--) value = block(value);
 | |
|     return value;
 | |
|   }
 | |
| 
 | |
|   class Power {
 | |
|     int base;
 | |
| 
 | |
|   public:
 | |
|     Power(int base) : base(base) {}
 | |
|     int calculate(unsigned n) {
 | |
|       return repeat(1, ^(int v) { return v * base; }, n);
 | |
|     }
 | |
|   };
 | |
| 
 | |
|   int test() {
 | |
|     return Power(2).calculate(10);
 | |
|   }
 | |
| }
 | |
| 
 | |
| // rdar: // 8382559
 | |
| namespace radar8382559 {
 | |
|   void func(bool& outHasProperty);
 | |
| 
 | |
|   int test3() {
 | |
|     __attribute__((__blocks__(byref))) bool hasProperty = false;
 | |
|     bool has = true;
 | |
| 
 | |
|     bool (^b)() = ^ {
 | |
|      func(hasProperty);
 | |
|      if (hasProperty)
 | |
|        hasProperty = 0;
 | |
|      if (has)
 | |
|        hasProperty = 1;
 | |
|      return hasProperty;
 | |
|      };
 | |
|     func(hasProperty);
 | |
|     func(has);
 | |
|     b();
 | |
|     if (hasProperty)
 | |
|       hasProperty = 1;
 | |
|     if (has)
 | |
|       has = 2;
 | |
|     return hasProperty = 1;
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Move __block variables to the heap when possible.
 | |
| class MoveOnly {
 | |
| public:
 | |
|   MoveOnly();
 | |
|   MoveOnly(const MoveOnly&) = delete;
 | |
|   MoveOnly(MoveOnly&&);
 | |
| };
 | |
| 
 | |
| void move_block() {
 | |
|   __block MoveOnly mo;
 | |
| }
 | |
| 
 | |
| // Don't crash after failing to build a block due to a capture of an
 | |
| // invalid declaration.
 | |
| namespace test5 {
 | |
|   struct B { // expected-note 2 {{candidate constructor}}
 | |
|     void *p;
 | |
|     B(int); // expected-note {{candidate constructor}}
 | |
|   };
 | |
| 
 | |
|   void use_block(void (^)());
 | |
|   void use_block_2(void (^)(), const B &a);
 | |
| 
 | |
|   void test() {
 | |
|     B x; // expected-error {{no matching constructor for initialization}}
 | |
|     use_block(^{
 | |
|         int y;
 | |
|         use_block_2(^{ (void) y; }, x);
 | |
|       });
 | |
|   }
 | |
| }
 | |
| 
 | |
| 
 | |
| // rdar://16356628
 | |
| //
 | |
| // Ensure that we can end function bodies while parsing an
 | |
| // expression that requires an explicitly-tracked cleanup object
 | |
| // (i.e. a block literal).
 | |
| 
 | |
| // The nested function body in this test case is a template
 | |
| // instantiation.  The template function has to be constexpr because
 | |
| // we'll otherwise delay its instantiation to the end of the
 | |
| // translation unit.
 | |
| namespace test6a {
 | |
|   template <class T> constexpr int func() { return 0; }
 | |
|   void run(void (^)(), int);
 | |
| 
 | |
|   void test() {
 | |
|     int aCapturedVar = 0;
 | |
|     run(^{ (void) aCapturedVar; }, func<int>());
 | |
|   }
 | |
| }
 | |
| 
 | |
| // The nested function body in this test case is a method of a local
 | |
| // class.
 | |
| namespace test6b {
 | |
|   void run(void (^)(), void (^)());
 | |
|   void test() {
 | |
|     int aCapturedVar = 0;
 | |
|     run(^{ (void) aCapturedVar; },
 | |
|         ^{ struct A { static void foo() {} };
 | |
|             A::foo(); });
 | |
|   }
 | |
| }
 | |
| 
 | |
| // The nested function body in this test case is a lambda invocation
 | |
| // function.
 | |
| namespace test6c {
 | |
|   void run(void (^)(), void (^)());
 | |
|   void test() {
 | |
|     int aCapturedVar = 0;
 | |
|     run(^{ (void) aCapturedVar; },
 | |
|         ^{ struct A { static void foo() {} };
 | |
|             A::foo(); });
 | |
|   }
 | |
| }
 | |
| 
 | |
| namespace test7 {
 | |
| struct S {};
 | |
| void f() {
 | |
|   constexpr S s;
 | |
|   auto some_block = ^{ (void)s; };
 | |
| }
 | |
| }
 | |
| 
 | |
| void static_data_member() {
 | |
|   auto block = ^{
 | |
|     class X {
 | |
|       static int x; // expected-error {{static data member 'x' not allowed in local class 'X'}}
 | |
|     };
 | |
|     class Y {
 | |
|       struct Z {
 | |
|         static int z; // expected-error {{static data member 'z' not allowed in local struct 'Z'}}
 | |
|       };
 | |
|     };
 | |
|   };
 | |
| }
 |