16
16
#include " clang/Sema/Sema.h"
17
17
18
18
using namespace clang ;
19
+
20
+ namespace {
21
+ bool diagnoseConstructAppertainment (Sema &S, OpenACCDirectiveKind K,
22
+ SourceLocation StartLoc, bool IsStmt) {
23
+ switch (K) {
24
+ default :
25
+ case OpenACCDirectiveKind::Invalid:
26
+ // Nothing to do here, both invalid and unimplemented don't really need to
27
+ // do anything.
28
+ break ;
29
+ case OpenACCDirectiveKind::Parallel:
30
+ if (!IsStmt)
31
+ return S.Diag (StartLoc, diag::err_acc_construct_appertainment) << K;
32
+ break ;
33
+ }
34
+ return false ;
35
+ }
36
+ } // namespace
37
+
19
38
bool Sema::ActOnOpenACCClause (OpenACCClauseKind ClauseKind,
20
39
SourceLocation StartLoc) {
21
40
if (ClauseKind == OpenACCClauseKind::Invalid)
@@ -35,6 +54,10 @@ void Sema::ActOnOpenACCConstruct(OpenACCDirectiveKind K,
35
54
// ensure that we can still work and don't check any construct-specific
36
55
// rules anywhere.
37
56
break ;
57
+ case OpenACCDirectiveKind::Parallel:
58
+ // Nothing to do here, there is no real legalization that needs to happen
59
+ // here as these constructs do not take any arguments.
60
+ break ;
38
61
default :
39
62
Diag (StartLoc, diag::warn_acc_construct_unimplemented) << K;
40
63
break ;
@@ -43,24 +66,49 @@ void Sema::ActOnOpenACCConstruct(OpenACCDirectiveKind K,
43
66
44
67
bool Sema::ActOnStartOpenACCStmtDirective (OpenACCDirectiveKind K,
45
68
SourceLocation StartLoc) {
46
- return true ;
69
+ return diagnoseConstructAppertainment (* this , K, StartLoc, /* IsStmt= */ true ) ;
47
70
}
48
71
49
72
StmtResult Sema::ActOnEndOpenACCStmtDirective (OpenACCDirectiveKind K,
50
73
SourceLocation StartLoc,
51
74
SourceLocation EndLoc,
52
75
StmtResult AssocStmt) {
53
- return StmtEmpty ();
76
+ switch (K) {
77
+ default :
78
+ return StmtEmpty ();
79
+ case OpenACCDirectiveKind::Invalid:
80
+ return StmtError ();
81
+ case OpenACCDirectiveKind::Parallel:
82
+ return OpenACCComputeConstruct::Create (
83
+ getASTContext (), K, StartLoc, EndLoc,
84
+ AssocStmt.isUsable () ? AssocStmt.get () : nullptr );
85
+ }
86
+ llvm_unreachable (" Unhandled case in directive handling?" );
54
87
}
55
88
56
89
StmtResult Sema::ActOnOpenACCAssociatedStmt (OpenACCDirectiveKind K,
57
90
StmtResult AssocStmt) {
58
- return AssocStmt;
91
+ switch (K) {
92
+ default :
93
+ llvm_unreachable (" Unimplemented associated statement application" );
94
+ case OpenACCDirectiveKind::Parallel:
95
+ // There really isn't any checking here that could happen. As long as we
96
+ // have a statement to associate, this should be fine.
97
+ // OpenACC 3.3 Section 6:
98
+ // Structured Block: in C or C++, an executable statement, possibly
99
+ // compound, with a single entry at the top and a single exit at the
100
+ // bottom.
101
+ // FIXME: Should we reject DeclStmt's here? The standard isn't clear, and
102
+ // an interpretation of it is to allow this and treat the initializer as
103
+ // the 'structured block'.
104
+ return AssocStmt;
105
+ }
106
+ llvm_unreachable (" Invalid associated statement application" );
59
107
}
60
108
61
109
bool Sema::ActOnStartOpenACCDeclDirective (OpenACCDirectiveKind K,
62
110
SourceLocation StartLoc) {
63
- return true ;
111
+ return diagnoseConstructAppertainment (* this , K, StartLoc, /* IsStmt= */ false ) ;
64
112
}
65
113
66
114
DeclGroupRef Sema::ActOnEndOpenACCDeclDirective () { return DeclGroupRef{}; }
0 commit comments