//===-- lib/Semantics/canonicalize-acc.cpp --------------------------------===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "canonicalize-acc.h" #include "flang/Parser/parse-tree-visitor.h" #include "flang/Semantics/tools.h" // After Loop Canonicalization, rewrite OpenACC parse tree to make OpenACC // Constructs more structured which provide explicit scopes for later // structural checks and semantic analysis. // 1. move structured DoConstruct into // OpenACCLoopConstruct. Compilation will not proceed in case of errors // after this pass. // 2. move structured DoConstruct into OpenACCCombinedConstruct. Move // AccEndCombinedConstruct into OpenACCCombinedConstruct if present. // Compilation will not proceed in case of errors after this pass. namespace Fortran::semantics { using namespace parser::literals; class CanonicalizationOfAcc { public: template bool Pre(T &) { return true; } template void Post(T &) {} CanonicalizationOfAcc(parser::Messages &messages) : messages_{messages} {} void Post(parser::Block &block) { for (auto it{block.begin()}; it != block.end(); ++it) { if (auto *accLoop{parser::Unwrap(*it)}) { RewriteOpenACCLoopConstruct(*accLoop, block, it); } else if (auto *accCombined{ parser::Unwrap(*it)}) { RewriteOpenACCCombinedConstruct(*accCombined, block, it); } else if (auto *endDir{ parser::Unwrap(*it)}) { // Unmatched AccEndCombinedDirective messages_.Say(endDir->v.source, "The %s directive must follow the DO loop associated with the " "loop construct"_err_en_US, parser::ToUpperCaseLetters(endDir->v.source.ToString())); } } // Block list } private: // Check constraint in 2.9.7 // If there are n tile sizes in the list, the loop construct must be // immediately followed by n tightly-nested loops. template void CheckTileClauseRestriction( const C &x, const parser::DoConstruct &outer) { const auto &beginLoopDirective = std::get(x.t); const auto &accClauseList = std::get(beginLoopDirective.t); for (const auto &clause : accClauseList.v) { if (const auto *tileClause = std::get_if(&clause.u)) { const parser::AccTileExprList &tileExprList = tileClause->v; const std::list &listTileExpr = tileExprList.v; std::size_t tileArgNb = listTileExpr.size(); if (outer.IsDoConcurrent()) { return; // Tile is not allowed on DO CONCURRENT } for (const parser::DoConstruct *loop{&outer}; loop && tileArgNb > 0; --tileArgNb) { const auto &block{std::get(loop->t)}; const auto it{block.begin()}; loop = it != block.end() ? parser::Unwrap(*it) : nullptr; } if (tileArgNb > 0) { messages_.Say(beginLoopDirective.source, "The loop construct with the TILE clause must be followed by %d " "tightly-nested loops"_err_en_US, listTileExpr.size()); } } } } // Check constraint on line 1835 in Section 2.9 // A tile and collapse clause may not appear on loop that is associated with // do concurrent. template void CheckDoConcurrentClauseRestriction( const C &x, const parser::DoConstruct &doCons) { if (!doCons.IsDoConcurrent()) { return; } const auto &beginLoopDirective = std::get(x.t); const auto &accClauseList = std::get(beginLoopDirective.t); for (const auto &clause : accClauseList.v) { if (std::holds_alternative(clause.u) || std::holds_alternative(clause.u)) { messages_.Say(beginLoopDirective.source, "TILE and COLLAPSE clause may not appear on loop construct " "associated with DO CONCURRENT"_err_en_US); } } } void RewriteOpenACCLoopConstruct(parser::OpenACCLoopConstruct &x, parser::Block &block, parser::Block::iterator it) { parser::Block::iterator nextIt; auto &beginDir{std::get(x.t)}; auto &dir{std::get(beginDir.t)}; auto &nestedDo{std::get>(x.t)}; if (!nestedDo) { nextIt = it; if (++nextIt != block.end()) { if (auto *doCons{parser::Unwrap(*nextIt)}) { nestedDo = std::move(*doCons); nextIt = block.erase(nextIt); } } } if (nestedDo) { if (!nestedDo->GetLoopControl()) { messages_.Say(dir.source, "DO loop after the %s directive must have loop control"_err_en_US, parser::ToUpperCaseLetters(dir.source.ToString())); return; } CheckDoConcurrentClauseRestriction(x, *nestedDo); CheckTileClauseRestriction(x, *nestedDo); return; } messages_.Say(dir.source, "A DO loop must follow the %s directive"_err_en_US, parser::ToUpperCaseLetters(dir.source.ToString())); } void RewriteOpenACCCombinedConstruct(parser::OpenACCCombinedConstruct &x, parser::Block &block, parser::Block::iterator it) { // Check the sequence of DoConstruct in the same iteration. parser::Block::iterator nextIt; auto &beginDir{std::get(x.t)}; auto &dir{std::get(beginDir.t)}; auto &nestedDo{std::get>(x.t)}; if (!nestedDo) { nextIt = it; if (++nextIt != block.end()) { if (auto *doCons{parser::Unwrap(*nextIt)}) { nestedDo = std::move(*doCons); nextIt = block.erase(nextIt); } } } if (nestedDo) { CheckDoConcurrentClauseRestriction(x, *nestedDo); CheckTileClauseRestriction(x, *nestedDo); if (!nestedDo->GetLoopControl()) { messages_.Say(dir.source, "DO loop after the %s directive must have loop control"_err_en_US, parser::ToUpperCaseLetters(dir.source.ToString())); return; } return; } messages_.Say(dir.source, "A DO loop must follow the %s directive"_err_en_US, parser::ToUpperCaseLetters(dir.source.ToString())); } parser::Messages &messages_; }; bool CanonicalizeAcc(parser::Messages &messages, parser::Program &program) { CanonicalizationOfAcc acc{messages}; Walk(program, acc); return !messages.AnyFatalError(); } } // namespace Fortran::semantics