| // RUN: %clang_cc1 -verify -fblocks -cl-std=CL2.0 %s |
| |
| // OpenCL v2.0 s6.12.5 |
| void f0(int (^const bl)()); |
| // All blocks declarations must be const qualified and initialized. |
| void f1() { |
| int (^bl1)(void) = ^() { |
| return 1; |
| }; |
| int (^const bl2)(void) = ^() { |
| return 1; |
| }; |
| f0(bl1); |
| f0(bl2); |
| bl1 = bl2; // expected-error{{invalid operands to binary expression ('int (__generic ^const)(void)' and 'int (__generic ^const)(void)')}} |
| int (^const bl3)(); // expected-error{{invalid block variable declaration - must be initialized}} |
| } |
| |
| // A block with extern storage class is not allowed. |
| extern int (^bl)(void) = ^() { // expected-error{{invalid block variable declaration - using 'extern' storage class is disallowed}} |
| return 1; |
| }; |
| void f2() { |
| extern int (^bl)(void) = ^() { // expected-error{{invalid block variable declaration - using 'extern' storage class is disallowed}} |
| return 1; |
| }; |
| } |
| |
| // A block cannot be the return value of a function. |
| typedef int (^bl_t)(void); |
| bl_t f3(bl_t bl); // expected-error{{declaring function return value of type 'bl_t' (aka 'int (__generic ^const)(void)') is not allowed}} |
| |
| struct bl_s { |
| int (^bl)(void); // expected-error {{the 'int (__generic ^const)(void)' type cannot be used to declare a structure or union field}} |
| }; |
| |
| void f4() { |
| __block int a = 10; // expected-error {{the __block storage type is not permitted}} |
| } |
| |
| // A block with variadic argument is not allowed. |
| int (^bl)(int, ...) = ^int(int I, ...) { // expected-error {{invalid prototype, variadic arguments are not allowed in OpenCL}} expected-error {{invalid prototype, variadic arguments are not allowed in OpenCL}} |
| return 0; |
| }; |
| typedef int (^bl1_t)(int, ...); // expected-error {{invalid prototype, variadic arguments are not allowed in OpenCL}} |
| |
| // A block can't be used to declare an array |
| typedef int (^bl2_t)(int); |
| void f5(int i) { |
| bl2_t bl1 = ^(int i) { |
| return 1; |
| }; |
| bl2_t bl2 = ^(int i) { |
| return 2; |
| }; |
| bl2_t arr[] = {bl1, bl2}; // expected-error {{array of 'bl2_t' (aka 'int (__generic ^const)(int)') type is invalid in OpenCL}} |
| int tmp = i ? bl1(i) // expected-error {{block type cannot be used as expression in ternary expression in OpenCL}} |
| : bl2(i); // expected-error {{block type cannot be used as expression in ternary expression in OpenCL}} |
| } |
| // A block pointer type and all pointer operations are disallowed |
| void f6(bl2_t *bl_ptr) { // expected-error{{pointer to type '__generic bl2_t' (aka 'int (__generic ^const __generic)(int)') is invalid in OpenCL}} |
| bl2_t bl = ^(int i) { |
| return 1; |
| }; |
| bl2_t *p; // expected-error {{pointer to type '__generic bl2_t' (aka 'int (__generic ^const __generic)(int)') is invalid in OpenCL}} |
| *bl; // expected-error {{invalid argument type 'bl2_t' (aka 'int (__generic ^const)(int)') to unary expression}} |
| &bl; // expected-error {{invalid argument type 'bl2_t' (aka 'int (__generic ^const)(int)') to unary expression}} |
| } |
| // A block can't reference another block |
| kernel void f7() { |
| bl2_t bl1 = ^(int i) { |
| return 1; |
| }; |
| void (^bl2)(void) = ^{ |
| int i = bl1(1); // expected-error {{cannot refer to a block inside block}} |
| }; |
| void (^bl3)(void) = ^{ |
| }; |
| void (^bl4)(void) = ^{ |
| bl3(); // expected-error {{cannot refer to a block inside block}} |
| }; |
| return; |
| } |
| |
| // Taking address of a capture is not allowed |
| int g; |
| kernel void f8(int a1) { |
| int a2; |
| void (^bl)(void) = ^(void) { |
| &g; //expected-warning{{expression result unused}} |
| &a1; //expected-error{{taking address of a capture is not allowed}} |
| &a2; //expected-error{{taking address of a capture is not allowed}} |
| }; |
| } |