Skip to content

Commit 9a97a57

Browse files
authoredOct 11, 2024··
[clang][frontend] Add support for attribute plugins for statement attributes (#110334)
We already have support for declaration attributes; this is just a matter of extending the plugin infrastructure to cover one more case.
1 parent 38b0102 commit 9a97a57

File tree

6 files changed

+121
-12
lines changed

6 files changed

+121
-12
lines changed
 

‎clang/docs/ClangPlugins.rst

+12-5
Original file line numberDiff line numberDiff line change
@@ -92,11 +92,6 @@ The members of ``ParsedAttrInfo`` that a plugin attribute must define are:
9292
attribute, each of which consists of an attribute syntax and how the
9393
attribute name is spelled for that syntax. If the syntax allows a scope then
9494
the spelling must be "scope::attr" if a scope is present or "::attr" if not.
95-
* ``handleDeclAttribute``, which is the function that applies the attribute to
96-
a declaration. It is responsible for checking that the attribute's arguments
97-
are valid, and typically applies the attribute by adding an ``Attr`` to the
98-
``Decl``. It returns either ``AttributeApplied``, to indicate that the
99-
attribute was successfully applied, or ``AttributeNotApplied`` if it wasn't.
10095

10196
The members of ``ParsedAttrInfo`` that may need to be defined, depending on the
10297
attribute, are:
@@ -105,6 +100,18 @@ attribute, are:
105100
arguments to the attribute.
106101
* ``diagAppertainsToDecl``, which checks if the attribute has been used on the
107102
right kind of declaration and issues a diagnostic if not.
103+
* ``handleDeclAttribute``, which is the function that applies the attribute to
104+
a declaration. It is responsible for checking that the attribute's arguments
105+
are valid, and typically applies the attribute by adding an ``Attr`` to the
106+
``Decl``. It returns either ``AttributeApplied``, to indicate that the
107+
attribute was successfully applied, or ``AttributeNotApplied`` if it wasn't.
108+
* ``diagAppertainsToStmt``, which checks if the attribute has been used on the
109+
right kind of statement and issues a diagnostic if not.
110+
* ``handleStmtAttribute``, which is the function that applies the attribute to
111+
a statement. It is responsible for checking that the attribute's arguments
112+
are valid, and typically applies the attribute by adding an ``Attr`` to the
113+
``Stmt``. It returns either ``AttributeApplied``, to indicate that the
114+
attribute was successfully applied, or ``AttributeNotApplied`` if it wasn't.
108115
* ``diagLangOpts``, which checks if the attribute is permitted for the current
109116
language mode and issues a diagnostic if not.
110117
* ``existsInTarget``, which checks if the attribute is permitted for the given

‎clang/docs/ReleaseNotes.rst

+2
Original file line numberDiff line numberDiff line change
@@ -250,6 +250,8 @@ Non-comprehensive list of changes in this release
250250
- The floating point comparison builtins (``__builtin_isgreater``,
251251
``__builtin_isgreaterequal``, ``__builtin_isless``, etc.) and
252252
``__builtin_signbit`` can now be used in constant expressions.
253+
- Plugins can now define custom attributes that apply to statements
254+
as well as declarations.
253255

254256
New Compiler Flags
255257
------------------

‎clang/examples/Attribute/Attribute.cpp

+48
Original file line numberDiff line numberDiff line change
@@ -94,6 +94,54 @@ struct ExampleAttrInfo : public ParsedAttrInfo {
9494
}
9595
return AttributeApplied;
9696
}
97+
98+
bool diagAppertainsToStmt(Sema &S, const ParsedAttr &Attr,
99+
const Stmt *St) const override {
100+
// This attribute appertains to for loop statements only.
101+
if (!isa<ForStmt>(St)) {
102+
S.Diag(Attr.getLoc(), diag::warn_attribute_wrong_decl_type_str)
103+
<< Attr << Attr.isRegularKeywordAttribute() << "for loop statements";
104+
return false;
105+
}
106+
return true;
107+
}
108+
109+
AttrHandling handleStmtAttribute(Sema &S, Stmt *St, const ParsedAttr &Attr,
110+
class Attr *&Result) const override {
111+
// We make some rules here:
112+
// 1. Only accept at most 3 arguments here.
113+
// 2. The first argument must be a string literal if it exists.
114+
if (Attr.getNumArgs() > 3) {
115+
unsigned ID = S.getDiagnostics().getCustomDiagID(
116+
DiagnosticsEngine::Error,
117+
"'example' attribute only accepts at most three arguments");
118+
S.Diag(Attr.getLoc(), ID);
119+
return AttributeNotApplied;
120+
}
121+
// If there are arguments, the first argument should be a string literal.
122+
if (Attr.getNumArgs() > 0) {
123+
auto *Arg0 = Attr.getArgAsExpr(0);
124+
StringLiteral *Literal =
125+
dyn_cast<StringLiteral>(Arg0->IgnoreParenCasts());
126+
if (!Literal) {
127+
unsigned ID = S.getDiagnostics().getCustomDiagID(
128+
DiagnosticsEngine::Error, "first argument to the 'example' "
129+
"attribute must be a string literal");
130+
S.Diag(Attr.getLoc(), ID);
131+
return AttributeNotApplied;
132+
}
133+
SmallVector<Expr *, 16> ArgsBuf;
134+
for (unsigned i = 0; i < Attr.getNumArgs(); i++) {
135+
ArgsBuf.push_back(Attr.getArgAsExpr(i));
136+
}
137+
Result = AnnotateAttr::Create(S.Context, "example", ArgsBuf.data(),
138+
ArgsBuf.size(), Attr.getRange());
139+
} else {
140+
Result = AnnotateAttr::Create(S.Context, "example", nullptr, 0,
141+
Attr.getRange());
142+
}
143+
return AttributeApplied;
144+
}
97145
};
98146

99147
} // namespace

‎clang/include/clang/Basic/ParsedAttrInfo.h

+10
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,7 @@
2424

2525
namespace clang {
2626

27+
class Attr;
2728
class Decl;
2829
class LangOptions;
2930
class ParsedAttr;
@@ -154,6 +155,15 @@ struct ParsedAttrInfo {
154155
const ParsedAttr &Attr) const {
155156
return NotHandled;
156157
}
158+
/// If this ParsedAttrInfo knows how to handle this ParsedAttr applied to this
159+
/// Stmt then do so (referencing the resulting Attr in Result) and return
160+
/// either AttributeApplied if it was applied or AttributeNotApplied if it
161+
/// wasn't. Otherwise return NotHandled.
162+
virtual AttrHandling handleStmtAttribute(Sema &S, Stmt *St,
163+
const ParsedAttr &Attr,
164+
class Attr *&Result) const {
165+
return NotHandled;
166+
}
157167

158168
static const ParsedAttrInfo &get(const AttributeCommonInfo &A);
159169
static ArrayRef<const ParsedAttrInfo *> getAllBuiltin();

‎clang/lib/Sema/SemaStmtAttr.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -682,6 +682,10 @@ static Attr *ProcessStmtAttribute(Sema &S, Stmt *St, const ParsedAttr &A,
682682
case ParsedAttr::AT_Annotate:
683683
return S.CreateAnnotationAttr(A);
684684
default:
685+
if (Attr *AT = nullptr; A.getInfo().handleStmtAttribute(S, St, A, AT) !=
686+
ParsedAttrInfo::NotHandled) {
687+
return AT;
688+
}
685689
// N.B., ClangAttrEmitter.cpp emits a diagnostic helper that ensures a
686690
// declaration attribute is not written on a statement, but this code is
687691
// needed for attributes in Attr.td that do not list any subjects.

‎clang/test/Frontend/plugin-attribute.cpp

+45-7
Original file line numberDiff line numberDiff line change
@@ -4,19 +4,57 @@
44
// REQUIRES: plugins, examples
55
//--- good_attr.cpp
66
// expected-no-diagnostics
7-
void fn1a() __attribute__((example)) {}
8-
[[example]] void fn1b() {}
9-
[[plugin::example]] void fn1c() {}
10-
void fn2() __attribute__((example("somestring", 1, 2.0))) {}
11-
// CHECK-COUNT-4: -AnnotateAttr 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} "example"
7+
void fn1a() __attribute__((example)) {
8+
__attribute__((example)) for (int i = 0; i < 9; ++i) {}
9+
}
10+
[[example]] void fn1b() {
11+
[[example]] for (int i = 0; i < 9; ++i) {}
12+
}
13+
[[plugin::example]] void fn1c() {
14+
[[plugin::example]] for (int i = 0; i < 9; ++i) {}
15+
}
16+
void fn2() __attribute__((example("somestring", 1, 2.0))) {
17+
__attribute__((example("abc", 3, 4.0))) for (int i = 0; i < 9; ++i) {}
18+
}
19+
template <int N> void template_fn() __attribute__((example("template", N))) {
20+
__attribute__((example("def", N + 1))) for (int i = 0; i < 9; ++i) {}
21+
}
22+
void fn3() { template_fn<5>(); }
23+
// CHECK: -AttributedStmt 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}}
24+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} "example"
25+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}} "example"
26+
// CHECK: -AttributedStmt 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}}
27+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} "example"
28+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}} "example"
29+
// CHECK: -AttributedStmt 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}}
30+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} "example"
31+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}} "example"
32+
// CHECK: -AttributedStmt 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}}
33+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} "example"
34+
// CHECK: -StringLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'const char[{{[0-9]+}}]' lvalue "abc"
35+
// CHECK: -IntegerLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'int' 3
36+
// CHECK: -FloatingLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'double' 4.000000e+00
37+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}} "example"
1238
// CHECK: -StringLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'const char[{{[0-9]+}}]' lvalue "somestring"
1339
// CHECK: -IntegerLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'int' 1
1440
// CHECK: -FloatingLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'double' 2.000000e+00
41+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} Implicit "example"
42+
// CHECK: -StringLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'const char[{{[0-9]+}}]' lvalue "def"
43+
// CHECK: -BinaryOperator 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'int' '+'
44+
// CHECK: -IntegerLiteral 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}} 'int' 5
45+
// CHECK: -IntegerLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'int' 1
46+
// CHECK: -AnnotateAttr 0x{{[0-9a-z]+}} {{<line:[0-9]+:[0-9]+(, col:[0-9]+)?>}} "example"
47+
// CHECK: -StringLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'const char[{{[0-9]+}}]' lvalue "template"
48+
// CHECK: -IntegerLiteral 0x{{[0-9a-z]+}} {{<col:[0-9]+(, col:[0-9]+)?>}} 'int' 5
1549

1650
//--- bad_attr.cpp
1751
int var1 __attribute__((example("otherstring"))) = 1; // expected-warning {{'example' attribute only applies to functions}}
1852
class Example {
1953
void __attribute__((example)) fn3(); // expected-error {{'example' attribute only allowed at file scope}}
2054
};
21-
void fn4() __attribute__((example(123))) { } // expected-error {{first argument to the 'example' attribute must be a string literal}}
22-
void fn5() __attribute__((example("a","b", 3, 4.0))) { } // expected-error {{'example' attribute only accepts at most three arguments}}
55+
void fn4() __attribute__((example(123))) { // expected-error {{first argument to the 'example' attribute must be a string literal}}
56+
__attribute__((example("somestring"))) while (true); // expected-warning {{'example' attribute only applies to for loop statements}}
57+
}
58+
void fn5() __attribute__((example("a","b", 3, 4.0))) { // expected-error {{'example' attribute only accepts at most three arguments}}
59+
__attribute__((example("a","b", 3, 4.0))) for (int i = 0; i < 10; ++i) {} // expected-error {{'example' attribute only accepts at most three arguments}}
60+
}

0 commit comments

Comments
 (0)
Please sign in to comment.