llvm-project/clang/test/OpenMP/masked_messages_attr.cpp

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

87 lines
2.8 KiB
C++
Raw Normal View History

[OpenMP] Support OpenMP 5.1 attributes OpenMP 5.1 added support for writing OpenMP directives using [[]] syntax in addition to using #pragma and this introduces support for the new syntax. In OpenMP, the attributes take one of two forms: [[omp::directive(...)]] or [[omp::sequence(...)]]. A directive attribute contains an OpenMP directive clause that is identical to the analogous #pragma syntax. A sequence attribute can contain either sequence or directive arguments and is used to ensure that the attributes are processed sequentially for situations where the order of the attributes matter (remember: https://eel.is/c++draft/dcl.attr.grammar#4.sentence-4). The approach taken here is somewhat novel and deserves mention. We could refactor much of the OpenMP parsing logic to work for either pragma annotation tokens or for attribute clauses. It would be a fair amount of effort to share the logic for both, but it's certainly doable. However, the semantic attribute system is not designed to handle the arbitrarily complex arguments that OpenMP directives contain. Adding support to thread the novel parsed information until we can produce a semantic attribute would be considerably more effort. What's more, existing OpenMP constructs are not (often) represented as semantic attributes. So doing this through Attr.td would be a massive undertaking that would likely only benefit OpenMP and comes with additional risks. Rather than walk down that path, I am taking advantage of the fact that the syntax of the directives within the directive clause is identical to that of the #pragma form. Once the parser recognizes that we're processing an OpenMP attribute, it caches all of the directive argument tokens and then replays them as though the user wrote a pragma. This reuses the same OpenMP parsing and semantic logic directly, but does come with a risk if the OpenMP committee decides to purposefully diverge their pragma and attribute syntaxes. So, despite this being a novel approach that does token replay, I think it's actually a better approach than trying to do this through the declarative syntax in Attr.td.
2021-07-12 18:51:19 +08:00
// RUN: %clang_cc1 -verify -fopenmp -fopenmp-version=51 %s -Wuninitialized
// RUN: %clang_cc1 -verify -fopenmp-simd -fopenmp-version=51 %s -Wuninitialized
void xxx(int argc) {
int x; // expected-note {{initialize the variable 'x' to silence this warning}}
[[omp::directive(masked)]]
argc = x; // expected-warning {{variable 'x' is uninitialized when used here}}
}
void yyy(int argc) {
int x; // expected-note {{initialize the variable 'x' to silence this warning}}
[[omp::directive(masked filter(1))]]
argc = x; // expected-warning {{variable 'x' is uninitialized when used here}}
}
int foo();
int main() {
[[omp::directive(masked)]]
;
[[omp::directive(masked filter(1) filter(2))]] // expected-error {{directive '#pragma omp masked' cannot contain more than one 'filter' clause}}
;
int x,y,z;
[[omp::directive(masked filter(x) filter(y) filter(z))]] // expected-error 2 {{directive '#pragma omp masked' cannot contain more than one 'filter' clause}}
;
[[omp::directive(masked nowait)]] // expected-error {{unexpected OpenMP clause 'nowait' in directive '#pragma omp masked'}}
[[omp::directive(masked unknown)]] // expected-warning {{extra tokens at the end of '#pragma omp masked' are ignored}}
foo();
{
[[omp::directive(masked)]]
} // expected-error {{expected statement}}
{
[[omp::directive(masked filter(2))]]
} // expected-error {{expected statement}}
[[omp::directive(for)]]
for (int i = 0; i < 10; ++i) {
foo();
[[omp::directive(masked filter(1))]] // expected-error {{region cannot be closely nested inside 'for' region}}
foo();
}
[[omp::directive(sections)]]
{
foo();
[[omp::directive(masked)]] // expected-error {{region cannot be closely nested inside 'sections' region}}
foo();
}
[[omp::directive(single)]]
for (int i = 0; i < 10; ++i) {
foo();
[[omp::directive(masked allocate(i))]] // expected-error {{region cannot be closely nested inside 'single' region}} expected-error {{unexpected OpenMP clause 'allocate' in directive '#pragma omp masked'}}
foo();
}
[[omp::directive(masked)]]
for (int i = 0; i < 10; ++i) {
foo();
[[omp::directive(masked)]]
foo();
}
[[omp::directive(for ordered)]]
for (int i = 0; i < 10; ++i)
[[omp::directive(masked)]] // expected-error {{region cannot be closely nested inside 'for' region}}
{
foo();
}
return 0;
}
int foo() {
L1: // expected-note {{jump exits scope of OpenMP structured block}}
foo();
[[omp::directive(masked filter(0))]]
{
foo();
goto L1; // expected-error {{cannot jump from this goto statement to its label}}
}
goto L2; // expected-error {{cannot jump from this goto statement to its label}}
[[omp::directive(masked filter(-2))]]
{ // expected-note {{jump bypasses OpenMP structured block}}
L2:
foo();
}
return 0;
}