| // RUN: %clang_cc1 -fsyntax-only -Wno-pragma-clang-attribute -verify %s |
| // RUN: not %clang_cc1 -fsyntax-only -ast-dump -ast-dump-filter test %s | FileCheck %s |
| |
| // Check for contradictions in rules for attribute without a strict subject set: |
| |
| #pragma clang attribute push (__attribute__((annotate("subRuleContradictions"))), apply_to = any(variable, variable(is_parameter), function(is_member), variable(is_global))) |
| // expected-error@-1 {{redundant attribute subject matcher sub-rule 'variable(is_parameter)'; 'variable' already matches those declarations}} |
| // expected-error@-2 {{redundant attribute subject matcher sub-rule 'variable(is_global)'; 'variable' already matches those declarations}} |
| |
| // Ensure that we've recovered from the error: |
| int testRecoverSubRuleContradiction = 0; |
| // CHECK-LABEL: VarDecl{{.*}} testRecoverSubRuleContradiction |
| // CHECK-NEXT: IntegerLiteral |
| // CHECK-NEXT: AnnotateAttr{{.*}} "subRuleContradictions" |
| |
| #pragma clang attribute pop |
| |
| #pragma clang attribute push (__attribute__((annotate("negatedSubRuleContradictions2"))), apply_to = any(variable(unless(is_parameter)), variable(is_thread_local), function, variable(is_global))) |
| // expected-error@-1 {{negated attribute subject matcher sub-rule 'variable(unless(is_parameter))' contradicts sub-rule 'variable(is_global)'}} |
| // We have just one error, don't error on 'variable(is_global)' |
| |
| // Ensure that we've recovered from the error: |
| int testRecoverNegatedContradiction = 0; |
| // CHECK-LABEL: VarDecl{{.*}} testRecoverNegatedContradiction |
| // CHECK-NEXT: IntegerLiteral |
| // CHECK-NEXT: AnnotateAttr{{.*}} "negatedSubRuleContradictions2" |
| |
| void testRecoverNegatedContradictionFunc(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testRecoverNegatedContradictionFunc |
| // CHECK-NEXT: AnnotateAttr{{.*}} "negatedSubRuleContradictions2" |
| |
| #pragma clang attribute pop |
| |
| // Verify the strict subject set verification. |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = any(function)) |
| |
| int testRecoverStrictnessVar = 0; |
| // CHECK-LABEL: VarDecl{{.*}} testRecoverStrictnessVar |
| // CHECK-NEXT: IntegerLiteral |
| // CHECK-NOT: AbiTagAttr |
| |
| void testRecoverStrictnessFunc(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testRecoverStrictnessFunc |
| // CHECK-NEXT: AbiTagAttr |
| |
| struct testRecoverStrictnessStruct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testRecoverStrictnessStruct |
| // CHECK-NOT: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = any(function, record(unless(is_union)), variable, enum)) |
| // expected-error@-1 {{attribute 'abi_tag' can't be applied to 'enum'}} |
| |
| int testRecoverExtraVar = 0; |
| // CHECK-LABEL: VarDecl{{.*}} testRecoverExtraVar |
| // CHECK-NEXT: IntegerLiteral |
| // CHECK-NEXT: AbiTagAttr |
| |
| void testRecoverExtraFunc(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testRecoverExtraFunc |
| // CHECK-NEXT: AbiTagAttr |
| |
| struct testRecoverExtraStruct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testRecoverExtraStruct |
| // CHECK-NEXT: AbiTagAttr |
| |
| enum testNoEnumAbiTag { CaseCase }; |
| // CHECK-LABEL: EnumDecl{{.*}} testNoEnumAbiTag |
| // CHECK-NO: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| // Verify the non-strict subject set verification. |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = any(function)) |
| |
| int testSubset1Var; |
| // CHECK-LABEL: VarDecl{{.*}} testSubset1Var |
| // CHECK-NOT: AbiTagAttr |
| |
| void testSubset1Func(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testSubset1Func |
| // CHECK-NEXT: AbiTagAttr |
| |
| struct testSubset1Struct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testSubset1Struct |
| // CHECK-NOT: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = variable) |
| |
| int testSubset2Var; |
| // CHECK-LABEL: VarDecl{{.*}} testSubset2Var |
| // CHECK-NEXT: AbiTagAttr |
| |
| void testSubset2Func(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testSubset2Func |
| // CHECK-NOT: AbiTagAttr |
| |
| struct testSubset2Struct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testSubset2Struct |
| // CHECK-NOT: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = any(record(unless(is_union)))) |
| |
| int testSubset3Var; |
| // CHECK-LABEL: VarDecl{{.*}} testSubset3Var |
| // CHECK-NOT: AbiTagAttr |
| |
| void testSubset3Func(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testSubset3Func |
| // CHECK-NOT: AbiTagAttr |
| |
| struct testSubset3Struct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testSubset3Struct |
| // CHECK-NEXT: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = any(function, variable)) |
| |
| int testSubset4Var; |
| // CHECK-LABEL: VarDecl{{.*}} testSubset4Var |
| // CHECK-NEXT: AbiTagAttr |
| |
| void testSubset4Func(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testSubset4Func |
| // CHECK-NEXT: AbiTagAttr |
| |
| struct testSubset4Struct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testSubset4Struct |
| // CHECK-NOT: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = any(variable, record(unless(is_union)))) |
| |
| int testSubset5Var; |
| // CHECK-LABEL: VarDecl{{.*}} testSubset5Var |
| // CHECK-NEXT: AbiTagAttr |
| |
| void testSubset5Func(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testSubset5Func |
| // CHECK-NOT: AbiTagAttr |
| |
| struct testSubset5Struct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testSubset5Struct |
| // CHECK-NEXT: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = any(record(unless(is_union)), function)) |
| |
| int testSubset6Var; |
| // CHECK-LABEL: VarDecl{{.*}} testSubset6Var |
| // CHECK-NOT: AbiTagAttr |
| |
| void testSubset6Func(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testSubset6Func |
| // CHECK-NEXT: AbiTagAttr |
| |
| struct testSubset6Struct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testSubset6Struct |
| // CHECK-NEXT: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = any(record(unless(is_union)), function, variable)) |
| |
| int testSubset7Var; |
| // CHECK-LABEL: VarDecl{{.*}} testSubset7Var |
| // CHECK-NEXT: AbiTagAttr |
| |
| void testSubset7Func(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testSubset7Func |
| // CHECK-NEXT: AbiTagAttr |
| |
| struct testSubset7Struct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testSubset7Struct |
| // CHECK-NEXT: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = any(record(unless(is_union)), function, variable, enum, enum_constant)) |
| // expected-error@-1 {{attribute 'abi_tag' can't be applied to 'enum_constant', and 'enum'}} |
| |
| int testSubsetRecoverVar; |
| // CHECK-LABEL: VarDecl{{.*}} testSubsetRecoverVar |
| // CHECK-NEXT: AbiTagAttr |
| |
| void testSubsetRecoverFunc(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testSubsetRecoverFunc |
| // CHECK-NEXT: AbiTagAttr |
| |
| struct testSubsetRecoverStruct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testSubsetRecoverStruct |
| // CHECK-NEXT: AbiTagAttr |
| |
| #pragma clang attribute pop |
| |
| #pragma clang attribute push (__attribute__((abi_tag("a"))), apply_to = enum) |
| // expected-error@-1 {{attribute 'abi_tag' can't be applied to 'enum'}} |
| |
| int testSubsetNoVar; |
| // CHECK-LABEL: VarDecl{{.*}} testSubsetNoVar |
| // CHECK-NOT: AbiTagAttr |
| |
| void testSubsetNoFunc(void); |
| // CHECK-LABEL: FunctionDecl{{.*}} testSubsetNoFunc |
| // CHECK-NOT: AbiTagAttr |
| |
| struct testSubsetNoStruct { }; |
| // CHECK-LABEL: RecordDecl{{.*}} testSubsetNoStruct |
| // CHECK-NOT: AbiTagAttr |
| |
| #pragma clang attribute pop |