223 lines
7.1 KiB
C
223 lines
7.1 KiB
C
|
// 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
|