forked from OSchip/llvm-project
[OPENMP 4.1] Initial support for extended 'ordered' clause.
OpenMP 4.1 introduces optional argument '(n)' for 'ordered' clause, where 'n' is a number of loops that immediately follow the directive. 'n' must be constant positive integer expressions and it must be less or equal than the number of the loops in the resulting loop nest. Patch adds parsing and semantic analysis for this optional argument. llvm-svn: 243635
This commit is contained in:
parent
3cc4a250a6
commit
10e775f4a8
|
@ -2480,7 +2480,8 @@ RecursiveASTVisitor<Derived>::VisitOMPScheduleClause(OMPScheduleClause *C) {
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Derived>
|
template <typename Derived>
|
||||||
bool RecursiveASTVisitor<Derived>::VisitOMPOrderedClause(OMPOrderedClause *) {
|
bool RecursiveASTVisitor<Derived>::VisitOMPOrderedClause(OMPOrderedClause *C) {
|
||||||
|
TRY_TO(TraverseStmt(C->getNumForLoops()));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -694,32 +694,53 @@ public:
|
||||||
/// \brief This represents 'ordered' clause in the '#pragma omp ...' directive.
|
/// \brief This represents 'ordered' clause in the '#pragma omp ...' directive.
|
||||||
///
|
///
|
||||||
/// \code
|
/// \code
|
||||||
/// #pragma omp for ordered
|
/// #pragma omp for ordered (2)
|
||||||
/// \endcode
|
/// \endcode
|
||||||
/// In this example directive '#pragma omp for' has 'ordered' clause.
|
/// In this example directive '#pragma omp for' has 'ordered' clause with
|
||||||
|
/// parameter 2.
|
||||||
///
|
///
|
||||||
class OMPOrderedClause : public OMPClause {
|
class OMPOrderedClause : public OMPClause {
|
||||||
|
friend class OMPClauseReader;
|
||||||
|
/// \brief Location of '('.
|
||||||
|
SourceLocation LParenLoc;
|
||||||
|
/// \brief Number of for-loops.
|
||||||
|
Stmt *NumForLoops;
|
||||||
|
|
||||||
|
/// \brief Set the number of associated for-loops.
|
||||||
|
void setNumForLoops(Expr *Num) { NumForLoops = Num; }
|
||||||
|
|
||||||
public:
|
public:
|
||||||
/// \brief Build 'ordered' clause.
|
/// \brief Build 'ordered' clause.
|
||||||
///
|
///
|
||||||
|
/// \param Num Expression, possibly associated with this clause.
|
||||||
/// \param StartLoc Starting location of the clause.
|
/// \param StartLoc Starting location of the clause.
|
||||||
|
/// \param LParenLoc Location of '('.
|
||||||
/// \param EndLoc Ending location of the clause.
|
/// \param EndLoc Ending location of the clause.
|
||||||
///
|
///
|
||||||
OMPOrderedClause(SourceLocation StartLoc, SourceLocation EndLoc)
|
OMPOrderedClause(Expr *Num, SourceLocation StartLoc,
|
||||||
: OMPClause(OMPC_ordered, StartLoc, EndLoc) {}
|
SourceLocation LParenLoc, SourceLocation EndLoc)
|
||||||
|
: OMPClause(OMPC_ordered, StartLoc, EndLoc), LParenLoc(LParenLoc),
|
||||||
|
NumForLoops(Num) {}
|
||||||
|
|
||||||
/// \brief Build an empty clause.
|
/// \brief Build an empty clause.
|
||||||
///
|
///
|
||||||
OMPOrderedClause()
|
explicit OMPOrderedClause()
|
||||||
: OMPClause(OMPC_ordered, SourceLocation(), SourceLocation()) {}
|
: OMPClause(OMPC_ordered, SourceLocation(), SourceLocation()),
|
||||||
|
LParenLoc(SourceLocation()), NumForLoops(nullptr) {}
|
||||||
|
|
||||||
|
/// \brief Sets the location of '('.
|
||||||
|
void setLParenLoc(SourceLocation Loc) { LParenLoc = Loc; }
|
||||||
|
/// \brief Returns the location of '('.
|
||||||
|
SourceLocation getLParenLoc() const { return LParenLoc; }
|
||||||
|
|
||||||
|
/// \brief Return the number of associated for-loops.
|
||||||
|
Expr *getNumForLoops() const { return cast_or_null<Expr>(NumForLoops); }
|
||||||
|
|
||||||
static bool classof(const OMPClause *T) {
|
static bool classof(const OMPClause *T) {
|
||||||
return T->getClauseKind() == OMPC_ordered;
|
return T->getClauseKind() == OMPC_ordered;
|
||||||
}
|
}
|
||||||
|
|
||||||
child_range children() {
|
child_range children() { return child_range(&NumForLoops, &NumForLoops + 1); }
|
||||||
return child_range(child_iterator(), child_iterator());
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/// \brief This represents 'nowait' clause in the '#pragma omp ...' directive.
|
/// \brief This represents 'nowait' clause in the '#pragma omp ...' directive.
|
||||||
|
|
|
@ -2512,7 +2512,8 @@ RecursiveASTVisitor<Derived>::VisitOMPScheduleClause(OMPScheduleClause *C) {
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Derived>
|
template <typename Derived>
|
||||||
bool RecursiveASTVisitor<Derived>::VisitOMPOrderedClause(OMPOrderedClause *) {
|
bool RecursiveASTVisitor<Derived>::VisitOMPOrderedClause(OMPOrderedClause *C) {
|
||||||
|
TRY_TO(TraverseStmt(C->getNumForLoops()));
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -7501,8 +7501,8 @@ def err_omp_loop_var_dsa : Error<
|
||||||
def err_omp_not_for : Error<
|
def err_omp_not_for : Error<
|
||||||
"%select{statement after '#pragma omp %1' must be a for loop|"
|
"%select{statement after '#pragma omp %1' must be a for loop|"
|
||||||
"expected %2 for loops after '#pragma omp %1'%select{|, but found only %4}3}0">;
|
"expected %2 for loops after '#pragma omp %1'%select{|, but found only %4}3}0">;
|
||||||
def note_omp_collapse_expr : Note<
|
def note_omp_collapse_ordered_expr : Note<
|
||||||
"as specified in 'collapse' clause">;
|
"as specified in %select{'collapse'|'ordered'|'collapse' and 'ordered'}0 clause%select{||s}0">;
|
||||||
def err_omp_negative_expression_in_clause : Error<
|
def err_omp_negative_expression_in_clause : Error<
|
||||||
"argument to '%0' clause must be a positive integer value">;
|
"argument to '%0' clause must be a positive integer value">;
|
||||||
def err_omp_not_integral : Error<
|
def err_omp_not_integral : Error<
|
||||||
|
|
|
@ -7903,6 +7903,11 @@ public:
|
||||||
SourceLocation StartLoc,
|
SourceLocation StartLoc,
|
||||||
SourceLocation LParenLoc,
|
SourceLocation LParenLoc,
|
||||||
SourceLocation EndLoc);
|
SourceLocation EndLoc);
|
||||||
|
/// \brief Called on well-formed 'ordered' clause.
|
||||||
|
OMPClause *
|
||||||
|
ActOnOpenMPOrderedClause(SourceLocation StartLoc, SourceLocation EndLoc,
|
||||||
|
SourceLocation LParenLoc = SourceLocation(),
|
||||||
|
Expr *NumForLoops = nullptr);
|
||||||
|
|
||||||
OMPClause *ActOnOpenMPSimpleClause(OpenMPClauseKind Kind,
|
OMPClause *ActOnOpenMPSimpleClause(OpenMPClauseKind Kind,
|
||||||
unsigned Argument,
|
unsigned Argument,
|
||||||
|
@ -7940,9 +7945,6 @@ public:
|
||||||
|
|
||||||
OMPClause *ActOnOpenMPClause(OpenMPClauseKind Kind, SourceLocation StartLoc,
|
OMPClause *ActOnOpenMPClause(OpenMPClauseKind Kind, SourceLocation StartLoc,
|
||||||
SourceLocation EndLoc);
|
SourceLocation EndLoc);
|
||||||
/// \brief Called on well-formed 'ordered' clause.
|
|
||||||
OMPClause *ActOnOpenMPOrderedClause(SourceLocation StartLoc,
|
|
||||||
SourceLocation EndLoc);
|
|
||||||
/// \brief Called on well-formed 'nowait' clause.
|
/// \brief Called on well-formed 'nowait' clause.
|
||||||
OMPClause *ActOnOpenMPNowaitClause(SourceLocation StartLoc,
|
OMPClause *ActOnOpenMPNowaitClause(SourceLocation StartLoc,
|
||||||
SourceLocation EndLoc);
|
SourceLocation EndLoc);
|
||||||
|
|
|
@ -651,8 +651,13 @@ void OMPClausePrinter::VisitOMPScheduleClause(OMPScheduleClause *Node) {
|
||||||
OS << ")";
|
OS << ")";
|
||||||
}
|
}
|
||||||
|
|
||||||
void OMPClausePrinter::VisitOMPOrderedClause(OMPOrderedClause *) {
|
void OMPClausePrinter::VisitOMPOrderedClause(OMPOrderedClause *Node) {
|
||||||
OS << "ordered";
|
OS << "ordered";
|
||||||
|
if (auto *Num = Node->getNumForLoops()) {
|
||||||
|
OS << "(";
|
||||||
|
Num->printPretty(OS, nullptr, Policy, 0);
|
||||||
|
OS << ")";
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void OMPClausePrinter::VisitOMPNowaitClause(OMPNowaitClause *) {
|
void OMPClausePrinter::VisitOMPNowaitClause(OMPNowaitClause *) {
|
||||||
|
|
|
@ -306,7 +306,10 @@ void OMPClauseProfiler::VisitOMPScheduleClause(const OMPScheduleClause *C) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void OMPClauseProfiler::VisitOMPOrderedClause(const OMPOrderedClause *) {}
|
void OMPClauseProfiler::VisitOMPOrderedClause(const OMPOrderedClause *C) {
|
||||||
|
if (auto *Num = C->getNumForLoops())
|
||||||
|
Profiler->VisitStmt(Num);
|
||||||
|
}
|
||||||
|
|
||||||
void OMPClauseProfiler::VisitOMPNowaitClause(const OMPNowaitClause *) {}
|
void OMPClauseProfiler::VisitOMPNowaitClause(const OMPNowaitClause *) {}
|
||||||
|
|
||||||
|
|
|
@ -415,6 +415,7 @@ OMPClause *Parser::ParseOpenMPClause(OpenMPDirectiveKind DKind,
|
||||||
case OMPC_num_threads:
|
case OMPC_num_threads:
|
||||||
case OMPC_safelen:
|
case OMPC_safelen:
|
||||||
case OMPC_collapse:
|
case OMPC_collapse:
|
||||||
|
case OMPC_ordered:
|
||||||
// OpenMP [2.5, Restrictions]
|
// OpenMP [2.5, Restrictions]
|
||||||
// At most one if clause can appear on the directive.
|
// At most one if clause can appear on the directive.
|
||||||
// At most one num_threads clause can appear on the directive.
|
// At most one num_threads clause can appear on the directive.
|
||||||
|
@ -430,6 +431,9 @@ OMPClause *Parser::ParseOpenMPClause(OpenMPDirectiveKind DKind,
|
||||||
ErrorFound = true;
|
ErrorFound = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (CKind == OMPC_ordered && PP.LookAhead(/*N=*/0).isNot(tok::l_paren))
|
||||||
|
Clause = ParseOpenMPClause(CKind);
|
||||||
|
else
|
||||||
Clause = ParseOpenMPSingleExprClause(CKind);
|
Clause = ParseOpenMPSingleExprClause(CKind);
|
||||||
break;
|
break;
|
||||||
case OMPC_default:
|
case OMPC_default:
|
||||||
|
@ -458,7 +462,6 @@ OMPClause *Parser::ParseOpenMPClause(OpenMPDirectiveKind DKind,
|
||||||
|
|
||||||
Clause = ParseOpenMPSingleExprWithArgClause(CKind);
|
Clause = ParseOpenMPSingleExprWithArgClause(CKind);
|
||||||
break;
|
break;
|
||||||
case OMPC_ordered:
|
|
||||||
case OMPC_nowait:
|
case OMPC_nowait:
|
||||||
case OMPC_untied:
|
case OMPC_untied:
|
||||||
case OMPC_mergeable:
|
case OMPC_mergeable:
|
||||||
|
|
|
@ -2751,7 +2751,7 @@ void Sema::ActOnOpenMPLoopInitialization(SourceLocation ForLoc, Stmt *Init) {
|
||||||
static bool CheckOpenMPIterationSpace(
|
static bool CheckOpenMPIterationSpace(
|
||||||
OpenMPDirectiveKind DKind, Stmt *S, Sema &SemaRef, DSAStackTy &DSA,
|
OpenMPDirectiveKind DKind, Stmt *S, Sema &SemaRef, DSAStackTy &DSA,
|
||||||
unsigned CurrentNestedLoopCount, unsigned NestedLoopCount,
|
unsigned CurrentNestedLoopCount, unsigned NestedLoopCount,
|
||||||
Expr *NestedLoopCountExpr,
|
Expr *CollapseLoopCountExpr, Expr *OrderedLoopCountExpr,
|
||||||
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA,
|
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA,
|
||||||
LoopIterationSpace &ResultIterSpace) {
|
LoopIterationSpace &ResultIterSpace) {
|
||||||
// OpenMP [2.6, Canonical Loop Form]
|
// OpenMP [2.6, Canonical Loop Form]
|
||||||
|
@ -2759,13 +2759,24 @@ static bool CheckOpenMPIterationSpace(
|
||||||
auto For = dyn_cast_or_null<ForStmt>(S);
|
auto For = dyn_cast_or_null<ForStmt>(S);
|
||||||
if (!For) {
|
if (!For) {
|
||||||
SemaRef.Diag(S->getLocStart(), diag::err_omp_not_for)
|
SemaRef.Diag(S->getLocStart(), diag::err_omp_not_for)
|
||||||
<< (NestedLoopCountExpr != nullptr) << getOpenMPDirectiveName(DKind)
|
<< (CollapseLoopCountExpr != nullptr || OrderedLoopCountExpr != nullptr)
|
||||||
<< NestedLoopCount << (CurrentNestedLoopCount > 0)
|
<< getOpenMPDirectiveName(DKind) << NestedLoopCount
|
||||||
<< CurrentNestedLoopCount;
|
<< (CurrentNestedLoopCount > 0) << CurrentNestedLoopCount;
|
||||||
if (NestedLoopCount > 1)
|
if (NestedLoopCount > 1) {
|
||||||
SemaRef.Diag(NestedLoopCountExpr->getExprLoc(),
|
if (CollapseLoopCountExpr && OrderedLoopCountExpr)
|
||||||
diag::note_omp_collapse_expr)
|
SemaRef.Diag(DSA.getConstructLoc(),
|
||||||
<< NestedLoopCountExpr->getSourceRange();
|
diag::note_omp_collapse_ordered_expr)
|
||||||
|
<< 2 << CollapseLoopCountExpr->getSourceRange()
|
||||||
|
<< OrderedLoopCountExpr->getSourceRange();
|
||||||
|
else if (CollapseLoopCountExpr)
|
||||||
|
SemaRef.Diag(CollapseLoopCountExpr->getExprLoc(),
|
||||||
|
diag::note_omp_collapse_ordered_expr)
|
||||||
|
<< 0 << CollapseLoopCountExpr->getSourceRange();
|
||||||
|
else
|
||||||
|
SemaRef.Diag(OrderedLoopCountExpr->getExprLoc(),
|
||||||
|
diag::note_omp_collapse_ordered_expr)
|
||||||
|
<< 1 << OrderedLoopCountExpr->getSourceRange();
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
assert(For->getBody());
|
assert(For->getBody());
|
||||||
|
@ -2840,7 +2851,8 @@ static bool CheckOpenMPIterationSpace(
|
||||||
} else if (LoopVarRefExpr != nullptr) {
|
} else if (LoopVarRefExpr != nullptr) {
|
||||||
// Make the loop iteration variable private (for worksharing constructs),
|
// Make the loop iteration variable private (for worksharing constructs),
|
||||||
// linear (for simd directives with the only one associated loop) or
|
// linear (for simd directives with the only one associated loop) or
|
||||||
// lastprivate (for simd directives with several collapsed loops).
|
// lastprivate (for simd directives with several collapsed or ordered
|
||||||
|
// loops).
|
||||||
if (DVar.CKind == OMPC_unknown)
|
if (DVar.CKind == OMPC_unknown)
|
||||||
DVar = DSA.hasDSA(Var, isOpenMPPrivate, MatchesAlways(),
|
DVar = DSA.hasDSA(Var, isOpenMPPrivate, MatchesAlways(),
|
||||||
/*FromParent=*/false);
|
/*FromParent=*/false);
|
||||||
|
@ -2942,16 +2954,23 @@ static bool FitsInto(unsigned Bits, bool Signed, Expr *E, Sema &SemaRef) {
|
||||||
/// \return Returns 0 if one of the collapsed stmts is not canonical for loop,
|
/// \return Returns 0 if one of the collapsed stmts is not canonical for loop,
|
||||||
/// number of collapsed loops otherwise.
|
/// number of collapsed loops otherwise.
|
||||||
static unsigned
|
static unsigned
|
||||||
CheckOpenMPLoop(OpenMPDirectiveKind DKind, Expr *NestedLoopCountExpr,
|
CheckOpenMPLoop(OpenMPDirectiveKind DKind, Expr *CollapseLoopCountExpr,
|
||||||
Stmt *AStmt, Sema &SemaRef, DSAStackTy &DSA,
|
Expr *OrderedLoopCountExpr, Stmt *AStmt, Sema &SemaRef,
|
||||||
|
DSAStackTy &DSA,
|
||||||
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA,
|
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA,
|
||||||
OMPLoopDirective::HelperExprs &Built) {
|
OMPLoopDirective::HelperExprs &Built) {
|
||||||
unsigned NestedLoopCount = 1;
|
unsigned NestedLoopCount = 1;
|
||||||
if (NestedLoopCountExpr) {
|
if (CollapseLoopCountExpr) {
|
||||||
// Found 'collapse' clause - calculate collapse number.
|
// Found 'collapse' clause - calculate collapse number.
|
||||||
llvm::APSInt Result;
|
llvm::APSInt Result;
|
||||||
if (NestedLoopCountExpr->EvaluateAsInt(Result, SemaRef.getASTContext()))
|
if (CollapseLoopCountExpr->EvaluateAsInt(Result, SemaRef.getASTContext()))
|
||||||
NestedLoopCount = Result.getLimitedValue();
|
NestedLoopCount += Result.getLimitedValue() - 1;
|
||||||
|
}
|
||||||
|
if (OrderedLoopCountExpr) {
|
||||||
|
// Found 'ordered' clause - calculate collapse number.
|
||||||
|
llvm::APSInt Result;
|
||||||
|
if (OrderedLoopCountExpr->EvaluateAsInt(Result, SemaRef.getASTContext()))
|
||||||
|
NestedLoopCount += Result.getLimitedValue() - 1;
|
||||||
}
|
}
|
||||||
// This is helper routine for loop directives (e.g., 'for', 'simd',
|
// This is helper routine for loop directives (e.g., 'for', 'simd',
|
||||||
// 'for simd', etc.).
|
// 'for simd', etc.).
|
||||||
|
@ -2960,8 +2979,9 @@ CheckOpenMPLoop(OpenMPDirectiveKind DKind, Expr *NestedLoopCountExpr,
|
||||||
Stmt *CurStmt = AStmt->IgnoreContainers(/* IgnoreCaptured */ true);
|
Stmt *CurStmt = AStmt->IgnoreContainers(/* IgnoreCaptured */ true);
|
||||||
for (unsigned Cnt = 0; Cnt < NestedLoopCount; ++Cnt) {
|
for (unsigned Cnt = 0; Cnt < NestedLoopCount; ++Cnt) {
|
||||||
if (CheckOpenMPIterationSpace(DKind, CurStmt, SemaRef, DSA, Cnt,
|
if (CheckOpenMPIterationSpace(DKind, CurStmt, SemaRef, DSA, Cnt,
|
||||||
NestedLoopCount, NestedLoopCountExpr,
|
NestedLoopCount, CollapseLoopCountExpr,
|
||||||
VarsWithImplicitDSA, IterSpaces[Cnt]))
|
OrderedLoopCountExpr, VarsWithImplicitDSA,
|
||||||
|
IterSpaces[Cnt]))
|
||||||
return 0;
|
return 0;
|
||||||
// Move on to the next nested for loop, or to the loop body.
|
// Move on to the next nested for loop, or to the loop body.
|
||||||
// OpenMP [2.8.1, simd construct, Restrictions]
|
// OpenMP [2.8.1, simd construct, Restrictions]
|
||||||
|
@ -2978,8 +2998,9 @@ CheckOpenMPLoop(OpenMPDirectiveKind DKind, Expr *NestedLoopCountExpr,
|
||||||
|
|
||||||
// An example of what is generated for the following code:
|
// An example of what is generated for the following code:
|
||||||
//
|
//
|
||||||
// #pragma omp simd collapse(2)
|
// #pragma omp simd collapse(2) ordered(2)
|
||||||
// for (i = 0; i < NI; ++i)
|
// for (i = 0; i < NI; ++i)
|
||||||
|
// for (k = 0; k < NK; ++k)
|
||||||
// for (j = J0; j < NJ; j+=2) {
|
// for (j = J0; j < NJ; j+=2) {
|
||||||
// <loop body>
|
// <loop body>
|
||||||
// }
|
// }
|
||||||
|
@ -3292,7 +3313,7 @@ CheckOpenMPLoop(OpenMPDirectiveKind DKind, Expr *NestedLoopCountExpr,
|
||||||
return NestedLoopCount;
|
return NestedLoopCount;
|
||||||
}
|
}
|
||||||
|
|
||||||
static Expr *GetCollapseNumberExpr(ArrayRef<OMPClause *> Clauses) {
|
static Expr *getCollapseNumberExpr(ArrayRef<OMPClause *> Clauses) {
|
||||||
auto &&CollapseFilter = [](const OMPClause *C) -> bool {
|
auto &&CollapseFilter = [](const OMPClause *C) -> bool {
|
||||||
return C->getClauseKind() == OMPC_collapse;
|
return C->getClauseKind() == OMPC_collapse;
|
||||||
};
|
};
|
||||||
|
@ -3303,15 +3324,27 @@ static Expr *GetCollapseNumberExpr(ArrayRef<OMPClause *> Clauses) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static Expr *getOrderedNumberExpr(ArrayRef<OMPClause *> Clauses) {
|
||||||
|
auto &&OrderedFilter = [](const OMPClause *C) -> bool {
|
||||||
|
return C->getClauseKind() == OMPC_ordered;
|
||||||
|
};
|
||||||
|
OMPExecutableDirective::filtered_clause_iterator<decltype(OrderedFilter)> I(
|
||||||
|
Clauses, std::move(OrderedFilter));
|
||||||
|
if (I)
|
||||||
|
return cast<OMPOrderedClause>(*I)->getNumForLoops();
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
StmtResult Sema::ActOnOpenMPSimdDirective(
|
StmtResult Sema::ActOnOpenMPSimdDirective(
|
||||||
ArrayRef<OMPClause *> Clauses, Stmt *AStmt, SourceLocation StartLoc,
|
ArrayRef<OMPClause *> Clauses, Stmt *AStmt, SourceLocation StartLoc,
|
||||||
SourceLocation EndLoc,
|
SourceLocation EndLoc,
|
||||||
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA) {
|
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA) {
|
||||||
OMPLoopDirective::HelperExprs B;
|
OMPLoopDirective::HelperExprs B;
|
||||||
// In presence of clause 'collapse', it will define the nested loops number.
|
// In presence of clause 'collapse' or 'ordered' with number of loops, it will
|
||||||
unsigned NestedLoopCount =
|
// define the nested loops number.
|
||||||
CheckOpenMPLoop(OMPD_simd, GetCollapseNumberExpr(Clauses), AStmt, *this,
|
unsigned NestedLoopCount = CheckOpenMPLoop(
|
||||||
*DSAStack, VarsWithImplicitDSA, B);
|
OMPD_simd, getCollapseNumberExpr(Clauses), getOrderedNumberExpr(Clauses),
|
||||||
|
AStmt, *this, *DSAStack, VarsWithImplicitDSA, B);
|
||||||
if (NestedLoopCount == 0)
|
if (NestedLoopCount == 0)
|
||||||
return StmtError();
|
return StmtError();
|
||||||
|
|
||||||
|
@ -3338,10 +3371,11 @@ StmtResult Sema::ActOnOpenMPForDirective(
|
||||||
SourceLocation EndLoc,
|
SourceLocation EndLoc,
|
||||||
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA) {
|
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA) {
|
||||||
OMPLoopDirective::HelperExprs B;
|
OMPLoopDirective::HelperExprs B;
|
||||||
// In presence of clause 'collapse', it will define the nested loops number.
|
// In presence of clause 'collapse' or 'ordered' with number of loops, it will
|
||||||
unsigned NestedLoopCount =
|
// define the nested loops number.
|
||||||
CheckOpenMPLoop(OMPD_for, GetCollapseNumberExpr(Clauses), AStmt, *this,
|
unsigned NestedLoopCount = CheckOpenMPLoop(
|
||||||
*DSAStack, VarsWithImplicitDSA, B);
|
OMPD_for, getCollapseNumberExpr(Clauses), getOrderedNumberExpr(Clauses),
|
||||||
|
AStmt, *this, *DSAStack, VarsWithImplicitDSA, B);
|
||||||
if (NestedLoopCount == 0)
|
if (NestedLoopCount == 0)
|
||||||
return StmtError();
|
return StmtError();
|
||||||
|
|
||||||
|
@ -3358,10 +3392,12 @@ StmtResult Sema::ActOnOpenMPForSimdDirective(
|
||||||
SourceLocation EndLoc,
|
SourceLocation EndLoc,
|
||||||
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA) {
|
llvm::DenseMap<VarDecl *, Expr *> &VarsWithImplicitDSA) {
|
||||||
OMPLoopDirective::HelperExprs B;
|
OMPLoopDirective::HelperExprs B;
|
||||||
// In presence of clause 'collapse', it will define the nested loops number.
|
// In presence of clause 'collapse' or 'ordered' with number of loops, it will
|
||||||
|
// define the nested loops number.
|
||||||
unsigned NestedLoopCount =
|
unsigned NestedLoopCount =
|
||||||
CheckOpenMPLoop(OMPD_for_simd, GetCollapseNumberExpr(Clauses), AStmt,
|
CheckOpenMPLoop(OMPD_for_simd, getCollapseNumberExpr(Clauses),
|
||||||
*this, *DSAStack, VarsWithImplicitDSA, B);
|
getOrderedNumberExpr(Clauses), AStmt, *this, *DSAStack,
|
||||||
|
VarsWithImplicitDSA, B);
|
||||||
if (NestedLoopCount == 0)
|
if (NestedLoopCount == 0)
|
||||||
return StmtError();
|
return StmtError();
|
||||||
|
|
||||||
|
@ -3490,10 +3526,12 @@ StmtResult Sema::ActOnOpenMPParallelForDirective(
|
||||||
CS->getCapturedDecl()->setNothrow();
|
CS->getCapturedDecl()->setNothrow();
|
||||||
|
|
||||||
OMPLoopDirective::HelperExprs B;
|
OMPLoopDirective::HelperExprs B;
|
||||||
// In presence of clause 'collapse', it will define the nested loops number.
|
// In presence of clause 'collapse' or 'ordered' with number of loops, it will
|
||||||
|
// define the nested loops number.
|
||||||
unsigned NestedLoopCount =
|
unsigned NestedLoopCount =
|
||||||
CheckOpenMPLoop(OMPD_parallel_for, GetCollapseNumberExpr(Clauses), AStmt,
|
CheckOpenMPLoop(OMPD_parallel_for, getCollapseNumberExpr(Clauses),
|
||||||
*this, *DSAStack, VarsWithImplicitDSA, B);
|
getOrderedNumberExpr(Clauses), AStmt, *this, *DSAStack,
|
||||||
|
VarsWithImplicitDSA, B);
|
||||||
if (NestedLoopCount == 0)
|
if (NestedLoopCount == 0)
|
||||||
return StmtError();
|
return StmtError();
|
||||||
|
|
||||||
|
@ -3519,10 +3557,12 @@ StmtResult Sema::ActOnOpenMPParallelForSimdDirective(
|
||||||
CS->getCapturedDecl()->setNothrow();
|
CS->getCapturedDecl()->setNothrow();
|
||||||
|
|
||||||
OMPLoopDirective::HelperExprs B;
|
OMPLoopDirective::HelperExprs B;
|
||||||
// In presence of clause 'collapse', it will define the nested loops number.
|
// In presence of clause 'collapse' or 'ordered' with number of loops, it will
|
||||||
|
// define the nested loops number.
|
||||||
unsigned NestedLoopCount =
|
unsigned NestedLoopCount =
|
||||||
CheckOpenMPLoop(OMPD_parallel_for_simd, GetCollapseNumberExpr(Clauses),
|
CheckOpenMPLoop(OMPD_parallel_for_simd, getCollapseNumberExpr(Clauses),
|
||||||
AStmt, *this, *DSAStack, VarsWithImplicitDSA, B);
|
getOrderedNumberExpr(Clauses), AStmt, *this, *DSAStack,
|
||||||
|
VarsWithImplicitDSA, B);
|
||||||
if (NestedLoopCount == 0)
|
if (NestedLoopCount == 0)
|
||||||
return StmtError();
|
return StmtError();
|
||||||
|
|
||||||
|
@ -4426,6 +4466,9 @@ OMPClause *Sema::ActOnOpenMPSingleExprClause(OpenMPClauseKind Kind, Expr *Expr,
|
||||||
case OMPC_collapse:
|
case OMPC_collapse:
|
||||||
Res = ActOnOpenMPCollapseClause(Expr, StartLoc, LParenLoc, EndLoc);
|
Res = ActOnOpenMPCollapseClause(Expr, StartLoc, LParenLoc, EndLoc);
|
||||||
break;
|
break;
|
||||||
|
case OMPC_ordered:
|
||||||
|
Res = ActOnOpenMPOrderedClause(StartLoc, EndLoc, LParenLoc, Expr);
|
||||||
|
break;
|
||||||
case OMPC_default:
|
case OMPC_default:
|
||||||
case OMPC_proc_bind:
|
case OMPC_proc_bind:
|
||||||
case OMPC_schedule:
|
case OMPC_schedule:
|
||||||
|
@ -4438,7 +4481,6 @@ OMPClause *Sema::ActOnOpenMPSingleExprClause(OpenMPClauseKind Kind, Expr *Expr,
|
||||||
case OMPC_aligned:
|
case OMPC_aligned:
|
||||||
case OMPC_copyin:
|
case OMPC_copyin:
|
||||||
case OMPC_copyprivate:
|
case OMPC_copyprivate:
|
||||||
case OMPC_ordered:
|
|
||||||
case OMPC_nowait:
|
case OMPC_nowait:
|
||||||
case OMPC_untied:
|
case OMPC_untied:
|
||||||
case OMPC_mergeable:
|
case OMPC_mergeable:
|
||||||
|
@ -4588,7 +4630,11 @@ ExprResult Sema::VerifyPositiveIntegerConstantInClause(Expr *E,
|
||||||
return ExprError();
|
return ExprError();
|
||||||
}
|
}
|
||||||
if (CKind == OMPC_collapse) {
|
if (CKind == OMPC_collapse) {
|
||||||
DSAStack->setCollapseNumber(Result.getExtValue());
|
DSAStack->setCollapseNumber(DSAStack->getCollapseNumber() - 1 +
|
||||||
|
Result.getExtValue());
|
||||||
|
} else if (CKind == OMPC_ordered) {
|
||||||
|
DSAStack->setCollapseNumber(DSAStack->getCollapseNumber() - 1 +
|
||||||
|
Result.getExtValue());
|
||||||
}
|
}
|
||||||
return ICE;
|
return ICE;
|
||||||
}
|
}
|
||||||
|
@ -4623,6 +4669,27 @@ OMPClause *Sema::ActOnOpenMPCollapseClause(Expr *NumForLoops,
|
||||||
OMPCollapseClause(NumForLoopsResult.get(), StartLoc, LParenLoc, EndLoc);
|
OMPCollapseClause(NumForLoopsResult.get(), StartLoc, LParenLoc, EndLoc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
OMPClause *Sema::ActOnOpenMPOrderedClause(SourceLocation StartLoc,
|
||||||
|
SourceLocation EndLoc,
|
||||||
|
SourceLocation LParenLoc,
|
||||||
|
Expr *NumForLoops) {
|
||||||
|
DSAStack->setOrderedRegion();
|
||||||
|
// OpenMP [2.7.1, loop construct, Description]
|
||||||
|
// OpenMP [2.8.1, simd construct, Description]
|
||||||
|
// OpenMP [2.9.6, distribute construct, Description]
|
||||||
|
// The parameter of the ordered clause must be a constant
|
||||||
|
// positive integer expression if any.
|
||||||
|
if (NumForLoops && LParenLoc.isValid()) {
|
||||||
|
ExprResult NumForLoopsResult =
|
||||||
|
VerifyPositiveIntegerConstantInClause(NumForLoops, OMPC_ordered);
|
||||||
|
if (NumForLoopsResult.isInvalid())
|
||||||
|
return nullptr;
|
||||||
|
NumForLoops = NumForLoopsResult.get();
|
||||||
|
}
|
||||||
|
return new (Context)
|
||||||
|
OMPOrderedClause(NumForLoops, StartLoc, LParenLoc, EndLoc);
|
||||||
|
}
|
||||||
|
|
||||||
OMPClause *Sema::ActOnOpenMPSimpleClause(
|
OMPClause *Sema::ActOnOpenMPSimpleClause(
|
||||||
OpenMPClauseKind Kind, unsigned Argument, SourceLocation ArgumentLoc,
|
OpenMPClauseKind Kind, unsigned Argument, SourceLocation ArgumentLoc,
|
||||||
SourceLocation StartLoc, SourceLocation LParenLoc, SourceLocation EndLoc) {
|
SourceLocation StartLoc, SourceLocation LParenLoc, SourceLocation EndLoc) {
|
||||||
|
@ -4915,12 +4982,6 @@ OMPClause *Sema::ActOnOpenMPClause(OpenMPClauseKind Kind,
|
||||||
return Res;
|
return Res;
|
||||||
}
|
}
|
||||||
|
|
||||||
OMPClause *Sema::ActOnOpenMPOrderedClause(SourceLocation StartLoc,
|
|
||||||
SourceLocation EndLoc) {
|
|
||||||
DSAStack->setOrderedRegion();
|
|
||||||
return new (Context) OMPOrderedClause(StartLoc, EndLoc);
|
|
||||||
}
|
|
||||||
|
|
||||||
OMPClause *Sema::ActOnOpenMPNowaitClause(SourceLocation StartLoc,
|
OMPClause *Sema::ActOnOpenMPNowaitClause(SourceLocation StartLoc,
|
||||||
SourceLocation EndLoc) {
|
SourceLocation EndLoc) {
|
||||||
DSAStack->setNowaitRegion();
|
DSAStack->setNowaitRegion();
|
||||||
|
|
|
@ -1447,6 +1447,16 @@ public:
|
||||||
Kind, ChunkSize, StartLoc, LParenLoc, KindLoc, CommaLoc, EndLoc);
|
Kind, ChunkSize, StartLoc, LParenLoc, KindLoc, CommaLoc, EndLoc);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// \brief Build a new OpenMP 'ordered' clause.
|
||||||
|
///
|
||||||
|
/// By default, performs semantic analysis to build the new OpenMP clause.
|
||||||
|
/// Subclasses may override this routine to provide different behavior.
|
||||||
|
OMPClause *RebuildOMPOrderedClause(SourceLocation StartLoc,
|
||||||
|
SourceLocation EndLoc,
|
||||||
|
SourceLocation LParenLoc, Expr *Num) {
|
||||||
|
return getSema().ActOnOpenMPOrderedClause(StartLoc, EndLoc, LParenLoc, Num);
|
||||||
|
}
|
||||||
|
|
||||||
/// \brief Build a new OpenMP 'private' clause.
|
/// \brief Build a new OpenMP 'private' clause.
|
||||||
///
|
///
|
||||||
/// By default, performs semantic analysis to build the new OpenMP clause.
|
/// By default, performs semantic analysis to build the new OpenMP clause.
|
||||||
|
@ -7237,8 +7247,14 @@ TreeTransform<Derived>::TransformOMPScheduleClause(OMPScheduleClause *C) {
|
||||||
template <typename Derived>
|
template <typename Derived>
|
||||||
OMPClause *
|
OMPClause *
|
||||||
TreeTransform<Derived>::TransformOMPOrderedClause(OMPOrderedClause *C) {
|
TreeTransform<Derived>::TransformOMPOrderedClause(OMPOrderedClause *C) {
|
||||||
// No need to rebuild this clause, no template-dependent parameters.
|
ExprResult E;
|
||||||
return C;
|
if (auto *Num = C->getNumForLoops()) {
|
||||||
|
E = getDerived().TransformExpr(Num);
|
||||||
|
if (E.isInvalid())
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
|
return getDerived().RebuildOMPOrderedClause(C->getLocStart(), C->getLocEnd(),
|
||||||
|
C->getLParenLoc(), E.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename Derived>
|
template <typename Derived>
|
||||||
|
|
|
@ -1845,7 +1845,10 @@ void OMPClauseReader::VisitOMPScheduleClause(OMPScheduleClause *C) {
|
||||||
C->setCommaLoc(Reader->ReadSourceLocation(Record, Idx));
|
C->setCommaLoc(Reader->ReadSourceLocation(Record, Idx));
|
||||||
}
|
}
|
||||||
|
|
||||||
void OMPClauseReader::VisitOMPOrderedClause(OMPOrderedClause *) {}
|
void OMPClauseReader::VisitOMPOrderedClause(OMPOrderedClause *C) {
|
||||||
|
C->setNumForLoops(Reader->Reader.ReadSubExpr());
|
||||||
|
C->setLParenLoc(Reader->ReadSourceLocation(Record, Idx));
|
||||||
|
}
|
||||||
|
|
||||||
void OMPClauseReader::VisitOMPNowaitClause(OMPNowaitClause *) {}
|
void OMPClauseReader::VisitOMPNowaitClause(OMPNowaitClause *) {}
|
||||||
|
|
||||||
|
|
|
@ -1763,7 +1763,10 @@ void OMPClauseWriter::VisitOMPScheduleClause(OMPScheduleClause *C) {
|
||||||
Writer->Writer.AddSourceLocation(C->getCommaLoc(), Record);
|
Writer->Writer.AddSourceLocation(C->getCommaLoc(), Record);
|
||||||
}
|
}
|
||||||
|
|
||||||
void OMPClauseWriter::VisitOMPOrderedClause(OMPOrderedClause *) {}
|
void OMPClauseWriter::VisitOMPOrderedClause(OMPOrderedClause *C) {
|
||||||
|
Writer->Writer.AddStmt(C->getNumForLoops());
|
||||||
|
Writer->Writer.AddSourceLocation(C->getLParenLoc(), Record);
|
||||||
|
}
|
||||||
|
|
||||||
void OMPClauseWriter::VisitOMPNowaitClause(OMPNowaitClause *) {}
|
void OMPClauseWriter::VisitOMPNowaitClause(OMPNowaitClause *) {}
|
||||||
|
|
||||||
|
|
|
@ -20,7 +20,12 @@ T tmain(T argc) {
|
||||||
// CHECK-NEXT: for (int i = 0; i < 2; ++i)
|
// CHECK-NEXT: for (int i = 0; i < 2; ++i)
|
||||||
// CHECK-NEXT: a = 2;
|
// CHECK-NEXT: a = 2;
|
||||||
#pragma omp parallel
|
#pragma omp parallel
|
||||||
#pragma omp for private(argc, b), firstprivate(c, d), lastprivate(d, f) collapse(N) schedule(static, N) ordered nowait
|
#pragma omp for private(argc, b), firstprivate(c, d), lastprivate(d, f) collapse(N) schedule(static, N) ordered(N) nowait
|
||||||
|
for (int i = 0; i < 10; ++i)
|
||||||
|
for (int j = 0; j < 10; ++j)
|
||||||
|
for (int j = 0; j < 10; ++j)
|
||||||
|
for (int j = 0; j < 10; ++j)
|
||||||
|
for (int j = 0; j < 10; ++j)
|
||||||
for (int i = 0; i < 10; ++i)
|
for (int i = 0; i < 10; ++i)
|
||||||
for (int j = 0; j < 10; ++j)
|
for (int j = 0; j < 10; ++j)
|
||||||
for (int j = 0; j < 10; ++j)
|
for (int j = 0; j < 10; ++j)
|
||||||
|
@ -28,7 +33,12 @@ T tmain(T argc) {
|
||||||
for (int j = 0; j < 10; ++j)
|
for (int j = 0; j < 10; ++j)
|
||||||
foo();
|
foo();
|
||||||
// CHECK-NEXT: #pragma omp parallel
|
// CHECK-NEXT: #pragma omp parallel
|
||||||
// CHECK-NEXT: #pragma omp for private(argc,b) firstprivate(c,d) lastprivate(d,f) collapse(N) schedule(static, N) ordered nowait
|
// CHECK-NEXT: #pragma omp for private(argc,b) firstprivate(c,d) lastprivate(d,f) collapse(N) schedule(static, N) ordered(N) nowait
|
||||||
|
// CHECK-NEXT: for (int i = 0; i < 10; ++i)
|
||||||
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
// CHECK-NEXT: for (int i = 0; i < 10; ++i)
|
// CHECK-NEXT: for (int i = 0; i < 10; ++i)
|
||||||
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
|
|
|
@ -0,0 +1,104 @@
|
||||||
|
// RUN: %clang_cc1 -verify -fopenmp %s
|
||||||
|
|
||||||
|
void foo() {
|
||||||
|
}
|
||||||
|
|
||||||
|
bool foobool(int argc) {
|
||||||
|
return argc;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct S1; // expected-note {{declared here}}
|
||||||
|
|
||||||
|
template <class T, typename S, int N, int ST> // expected-note {{declared here}}
|
||||||
|
T tmain(T argc, S **argv) { //expected-note 2 {{declared here}}
|
||||||
|
#pragma omp for ordered
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp for ordered( // expected-error {{expected expression}} expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp for ordered() // expected-error {{expected expression}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
// expected-error@+3 {{expected ')'}} expected-note@+3 {{to match this '('}}
|
||||||
|
// expected-error@+2 2 {{expression is not an integral constant expression}}
|
||||||
|
// expected-note@+1 2 {{read of non-const variable 'argc' is not allowed in a constant expression}}
|
||||||
|
#pragma omp for ordered(argc
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
// expected-error@+1 2 {{argument to 'ordered' clause must be a positive integer value}}
|
||||||
|
#pragma omp for ordered(ST // expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp for ordered(1)) // expected-warning {{extra tokens at the end of '#pragma omp for' are ignored}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp for ordered((ST > 0) ? 1 + ST : 2) // expected-note 2 {{as specified in 'ordered' clause}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST]; // expected-error 2 {{expected 2 for loops after '#pragma omp for', but found only 1}}
|
||||||
|
// expected-error@+3 2 {{directive '#pragma omp for' cannot contain more than one 'ordered' clause}}
|
||||||
|
// expected-error@+2 2 {{argument to 'ordered' clause must be a positive integer value}}
|
||||||
|
// expected-error@+1 2 {{expression is not an integral constant expression}}
|
||||||
|
#pragma omp for ordered(foobool(argc)), ordered(true), ordered(-5)
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp for ordered(S) // expected-error {{'S' does not refer to a value}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
// expected-error@+1 2 {{expression is not an integral constant expression}}
|
||||||
|
#pragma omp for ordered(argv[1] = 2) // expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp for ordered(1)
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp for ordered(N) // expected-error {{argument to 'ordered' clause must be a positive integer value}}
|
||||||
|
for (T i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp for ordered(2) // expected-note {{as specified in 'ordered' clause}}
|
||||||
|
foo(); // expected-error {{expected 2 for loops after '#pragma omp for'}}
|
||||||
|
return argc;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char **argv) {
|
||||||
|
#pragma omp for ordered
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
#pragma omp for ordered( // expected-error {{expected expression}} expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
#pragma omp for ordered() // expected-error {{expected expression}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
#pragma omp for ordered(4 // expected-error {{expected ')'}} expected-note {{to match this '('}} expected-note {{as specified in 'ordered' clause}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4]; // expected-error {{expected 4 for loops after '#pragma omp for', but found only 1}}
|
||||||
|
#pragma omp for ordered(2 + 2)) // expected-warning {{extra tokens at the end of '#pragma omp for' are ignored}} expected-note {{as specified in 'ordered' clause}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4]; // expected-error {{expected 4 for loops after '#pragma omp for', but found only 1}}
|
||||||
|
#pragma omp for ordered(foobool(1) > 0 ? 1 : 2) // expected-error {{expression is not an integral constant expression}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
// expected-error@+3 {{expression is not an integral constant expression}}
|
||||||
|
// expected-error@+2 2 {{directive '#pragma omp for' cannot contain more than one 'ordered' clause}}
|
||||||
|
// expected-error@+1 2 {{argument to 'ordered' clause must be a positive integer value}}
|
||||||
|
#pragma omp for ordered(foobool(argc)), ordered(true), ordered(-5)
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
#pragma omp for ordered(S1) // expected-error {{'S1' does not refer to a value}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
// expected-error@+1 {{expression is not an integral constant expression}}
|
||||||
|
#pragma omp for ordered(argv[1] = 2) // expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
// expected-error@+3 {{statement after '#pragma omp for' must be a for loop}}
|
||||||
|
// expected-note@+1 {{in instantiation of function template specialization 'tmain<int, char, -1, -2>' requested here}}
|
||||||
|
#pragma omp for ordered(ordered(tmain < int, char, -1, -2 > (argc, argv) // expected-error 2 {{expected ')'}} expected-note 2 {{to match this '('}}
|
||||||
|
foo();
|
||||||
|
#pragma omp for ordered(2) // expected-note {{as specified in 'ordered' clause}}
|
||||||
|
foo(); // expected-error {{expected 2 for loops after '#pragma omp for'}}
|
||||||
|
// expected-note@+1 {{in instantiation of function template specialization 'tmain<int, char, 1, 0>' requested here}}
|
||||||
|
return tmain<int, char, 1, 0>(argc, argv);
|
||||||
|
}
|
||||||
|
|
|
@ -21,14 +21,24 @@ T tmain(T argc) {
|
||||||
a = 2;
|
a = 2;
|
||||||
// CHECK-NEXT: for (int i = 0; i < 2; ++i)
|
// CHECK-NEXT: for (int i = 0; i < 2; ++i)
|
||||||
// CHECK-NEXT: a = 2;
|
// CHECK-NEXT: a = 2;
|
||||||
#pragma omp parallel for private(argc, b), firstprivate(c, d), lastprivate(d, f) collapse(N) schedule(static, N) ordered if (argc) num_threads(N) default(shared) shared(e) reduction(+ : h)
|
#pragma omp parallel for private(argc, b), firstprivate(c, d), lastprivate(d, f) collapse(N) schedule(static, N) ordered(N) if (argc) num_threads(N) default(shared) shared(e) reduction(+ : h)
|
||||||
|
for (int i = 0; i < 10; ++i)
|
||||||
|
for (int j = 0; j < 10; ++j)
|
||||||
|
for (int j = 0; j < 10; ++j)
|
||||||
|
for (int j = 0; j < 10; ++j)
|
||||||
|
for (int j = 0; j < 10; ++j)
|
||||||
for (int i = 0; i < 10; ++i)
|
for (int i = 0; i < 10; ++i)
|
||||||
for (int j = 0; j < 10; ++j)
|
for (int j = 0; j < 10; ++j)
|
||||||
for (int j = 0; j < 10; ++j)
|
for (int j = 0; j < 10; ++j)
|
||||||
for (int j = 0; j < 10; ++j)
|
for (int j = 0; j < 10; ++j)
|
||||||
for (int j = 0; j < 10; ++j)
|
for (int j = 0; j < 10; ++j)
|
||||||
foo();
|
foo();
|
||||||
// CHECK-NEXT: #pragma omp parallel for private(argc,b) firstprivate(c,d) lastprivate(d,f) collapse(N) schedule(static, N) ordered if(argc) num_threads(N) default(shared) shared(e) reduction(+: h)
|
// CHECK-NEXT: #pragma omp parallel for private(argc,b) firstprivate(c,d) lastprivate(d,f) collapse(N) schedule(static, N) ordered(N) if(argc) num_threads(N) default(shared) shared(e) reduction(+: h)
|
||||||
|
// CHECK-NEXT: for (int i = 0; i < 10; ++i)
|
||||||
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
// CHECK-NEXT: for (int i = 0; i < 10; ++i)
|
// CHECK-NEXT: for (int i = 0; i < 10; ++i)
|
||||||
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
// CHECK-NEXT: for (int j = 0; j < 10; ++j)
|
||||||
|
|
|
@ -0,0 +1,104 @@
|
||||||
|
// RUN: %clang_cc1 -verify -fopenmp %s
|
||||||
|
|
||||||
|
void foo() {
|
||||||
|
}
|
||||||
|
|
||||||
|
bool foobool(int argc) {
|
||||||
|
return argc;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct S1; // expected-note {{declared here}}
|
||||||
|
|
||||||
|
template <class T, typename S, int N, int ST> // expected-note {{declared here}}
|
||||||
|
T tmain(T argc, S **argv) { //expected-note 2 {{declared here}}
|
||||||
|
#pragma omp parallel for ordered
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp parallel for ordered( // expected-error {{expected expression}} expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp parallel for ordered() // expected-error {{expected expression}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
// expected-error@+3 {{expected ')'}} expected-note@+3 {{to match this '('}}
|
||||||
|
// expected-error@+2 2 {{expression is not an integral constant expression}}
|
||||||
|
// expected-note@+1 2 {{read of non-const variable 'argc' is not allowed in a constant expression}}
|
||||||
|
#pragma omp parallel for ordered(argc
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
// expected-error@+1 2 {{argument to 'ordered' clause must be a positive integer value}}
|
||||||
|
#pragma omp parallel for ordered(ST // expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp parallel for ordered(1)) // expected-warning {{extra tokens at the end of '#pragma omp parallel for' are ignored}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp parallel for ordered((ST > 0) ? 1 + ST : 2) // expected-note 2 {{as specified in 'ordered' clause}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST]; // expected-error 2 {{expected 2 for loops after '#pragma omp parallel for', but found only 1}}
|
||||||
|
// expected-error@+3 2 {{directive '#pragma omp parallel for' cannot contain more than one 'ordered' clause}}
|
||||||
|
// expected-error@+2 2 {{argument to 'ordered' clause must be a positive integer value}}
|
||||||
|
// expected-error@+1 2 {{expression is not an integral constant expression}}
|
||||||
|
#pragma omp parallel for ordered(foobool(argc)), ordered(true), ordered(-5)
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp parallel for ordered(S) // expected-error {{'S' does not refer to a value}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
// expected-error@+1 2 {{expression is not an integral constant expression}}
|
||||||
|
#pragma omp parallel for ordered(argv[1] = 2) // expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp parallel for ordered(1)
|
||||||
|
for (int i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp parallel for ordered(N) // expected-error {{argument to 'ordered' clause must be a positive integer value}}
|
||||||
|
for (T i = ST; i < N; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - ST];
|
||||||
|
#pragma omp parallel for ordered(2) // expected-note {{as specified in 'ordered' clause}}
|
||||||
|
foo(); // expected-error {{expected 2 for loops after '#pragma omp parallel for'}}
|
||||||
|
return argc;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char **argv) {
|
||||||
|
#pragma omp parallel for ordered
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
#pragma omp parallel for ordered( // expected-error {{expected expression}} expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
#pragma omp parallel for ordered() // expected-error {{expected expression}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
#pragma omp parallel for ordered(4 // expected-error {{expected ')'}} expected-note {{to match this '('}} expected-note {{as specified in 'ordered' clause}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4]; // expected-error {{expected 4 for loops after '#pragma omp parallel for', but found only 1}}
|
||||||
|
#pragma omp parallel for ordered(2 + 2)) // expected-warning {{extra tokens at the end of '#pragma omp parallel for' are ignored}} expected-note {{as specified in 'ordered' clause}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4]; // expected-error {{expected 4 for loops after '#pragma omp parallel for', but found only 1}}
|
||||||
|
#pragma omp parallel for ordered(foobool(1) > 0 ? 1 : 2) // expected-error {{expression is not an integral constant expression}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
// expected-error@+3 {{expression is not an integral constant expression}}
|
||||||
|
// expected-error@+2 2 {{directive '#pragma omp parallel for' cannot contain more than one 'ordered' clause}}
|
||||||
|
// expected-error@+1 2 {{argument to 'ordered' clause must be a positive integer value}}
|
||||||
|
#pragma omp parallel for ordered(foobool(argc)), ordered(true), ordered(-5)
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
#pragma omp parallel for ordered(S1) // expected-error {{'S1' does not refer to a value}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
// expected-error@+1 {{expression is not an integral constant expression}}
|
||||||
|
#pragma omp parallel for ordered(argv[1] = 2) // expected-error {{expected ')'}} expected-note {{to match this '('}}
|
||||||
|
for (int i = 4; i < 12; i++)
|
||||||
|
argv[0][i] = argv[0][i] - argv[0][i - 4];
|
||||||
|
// expected-error@+3 {{statement after '#pragma omp parallel for' must be a for loop}}
|
||||||
|
// expected-note@+1 {{in instantiation of function template specialization 'tmain<int, char, -1, -2>' requested here}}
|
||||||
|
#pragma omp parallel for ordered(ordered(tmain < int, char, -1, -2 > (argc, argv) // expected-error 2 {{expected ')'}} expected-note 2 {{to match this '('}}
|
||||||
|
foo();
|
||||||
|
#pragma omp parallel for ordered(2) // expected-note {{as specified in 'ordered' clause}}
|
||||||
|
foo(); // expected-error {{expected 2 for loops after '#pragma omp parallel for'}}
|
||||||
|
// expected-note@+1 {{in instantiation of function template specialization 'tmain<int, char, 1, 0>' requested here}}
|
||||||
|
return tmain<int, char, 1, 0>(argc, argv);
|
||||||
|
}
|
||||||
|
|
|
@ -2033,7 +2033,9 @@ void OMPClauseEnqueue::VisitOMPScheduleClause(const OMPScheduleClause *C) {
|
||||||
Visitor->AddStmt(C->getHelperChunkSize());
|
Visitor->AddStmt(C->getHelperChunkSize());
|
||||||
}
|
}
|
||||||
|
|
||||||
void OMPClauseEnqueue::VisitOMPOrderedClause(const OMPOrderedClause *) {}
|
void OMPClauseEnqueue::VisitOMPOrderedClause(const OMPOrderedClause *C) {
|
||||||
|
Visitor->AddStmt(C->getNumForLoops());
|
||||||
|
}
|
||||||
|
|
||||||
void OMPClauseEnqueue::VisitOMPNowaitClause(const OMPNowaitClause *) {}
|
void OMPClauseEnqueue::VisitOMPNowaitClause(const OMPNowaitClause *) {}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue