2020-02-25 23:11:52 +08:00
|
|
|
//===-- lib/Parser/stmt-parser.h --------------------------------*- C++ -*-===//
|
2018-05-02 03:50:34 +08:00
|
|
|
//
|
2019-12-21 04:52:07 +08:00
|
|
|
// 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
|
2018-05-02 03:50:34 +08:00
|
|
|
//
|
2020-01-11 04:12:03 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2018-05-02 03:50:34 +08:00
|
|
|
|
2018-04-21 06:19:20 +08:00
|
|
|
#ifndef FORTRAN_PARSER_STMT_PARSER_H_
|
|
|
|
#define FORTRAN_PARSER_STMT_PARSER_H_
|
|
|
|
|
|
|
|
// Basic parsing of statements.
|
|
|
|
|
|
|
|
#include "basic-parsers.h"
|
|
|
|
#include "token-parsers.h"
|
|
|
|
|
2018-05-03 04:48:12 +08:00
|
|
|
namespace Fortran::parser {
|
2018-04-21 06:19:20 +08:00
|
|
|
|
|
|
|
// statement(p) parses Statement<P> for some statement type P that is the
|
|
|
|
// result type of the argument parser p, while also handling labels and
|
|
|
|
// end-of-statement markers.
|
|
|
|
|
|
|
|
// R611 label -> digit [digit]...
|
2019-03-23 05:27:18 +08:00
|
|
|
constexpr auto label{space >> digitString64 / spaceCheck};
|
2018-04-21 06:19:20 +08:00
|
|
|
|
2020-03-29 12:00:16 +08:00
|
|
|
template <typename PA>
|
|
|
|
inline constexpr auto unterminatedStatement(const PA &p) {
|
2018-07-13 05:46:23 +08:00
|
|
|
return skipStuffBeforeStatement >>
|
2018-04-24 07:53:16 +08:00
|
|
|
sourced(construct<Statement<typename PA::resultType>>(
|
|
|
|
maybe(label), space >> p));
|
2018-04-21 06:19:20 +08:00
|
|
|
}
|
|
|
|
|
2018-07-13 05:46:23 +08:00
|
|
|
constexpr auto endOfLine{
|
|
|
|
"\n"_ch >> ok || fail("expected end of line"_err_en_US)};
|
2018-04-21 06:19:20 +08:00
|
|
|
|
2018-08-01 03:32:09 +08:00
|
|
|
constexpr auto semicolons{";"_ch >> skipMany(";"_tok) / space / maybe("\n"_ch)};
|
2018-07-11 08:09:07 +08:00
|
|
|
constexpr auto endOfStmt{
|
2018-08-01 03:32:09 +08:00
|
|
|
space >> withMessage("expected end of statement"_err_en_US,
|
|
|
|
semicolons || endOfLine)};
|
|
|
|
constexpr auto forceEndOfStmt{recovery(endOfStmt, SkipPast<'\n'>{})};
|
2018-04-21 06:19:20 +08:00
|
|
|
|
2020-03-29 12:00:16 +08:00
|
|
|
template <typename PA> inline constexpr auto statement(const PA &p) {
|
2018-04-21 06:19:20 +08:00
|
|
|
return unterminatedStatement(p) / endOfStmt;
|
|
|
|
}
|
|
|
|
|
2019-04-02 06:52:49 +08:00
|
|
|
// unlabeledStatement() is basically statement() for those few situations
|
|
|
|
// in Fortran where a statement cannot have a label.
|
2020-03-29 12:00:16 +08:00
|
|
|
template <typename PA> inline constexpr auto unlabeledStatement(const PA &p) {
|
2019-04-02 07:15:47 +08:00
|
|
|
return space >>
|
|
|
|
sourced(construct<UnlabeledStatement<typename PA::resultType>>(p));
|
2019-04-02 06:52:49 +08:00
|
|
|
}
|
|
|
|
|
2018-07-18 03:34:07 +08:00
|
|
|
// This unambiguousStatement() variant of statement() provides better error
|
|
|
|
// recovery for contexts containing statements that might have trailing
|
|
|
|
// garbage, but it must be used only when no instance of the statement in
|
|
|
|
// question could also be a legal prefix of some other statement that might
|
|
|
|
// be valid at that point. It only makes sense to use this within "some()"
|
|
|
|
// or "many()" so as to not end the list of statements.
|
2020-03-29 12:00:16 +08:00
|
|
|
template <typename PA> inline constexpr auto unambiguousStatement(const PA &p) {
|
2018-08-01 03:32:09 +08:00
|
|
|
return unterminatedStatement(p) / forceEndOfStmt;
|
2018-07-18 03:34:07 +08:00
|
|
|
}
|
|
|
|
|
2018-07-11 08:09:07 +08:00
|
|
|
constexpr auto ignoredStatementPrefix{
|
2018-07-13 05:46:23 +08:00
|
|
|
skipStuffBeforeStatement >> maybe(label) >> maybe(name / ":") >> space};
|
2018-04-21 06:19:20 +08:00
|
|
|
|
2018-09-11 05:19:37 +08:00
|
|
|
// Error recovery within a statement() call: skip *to* the end of the line,
|
|
|
|
// unless at an END or CONTAINS statement.
|
|
|
|
constexpr auto inStmtErrorRecovery{!"END"_tok >> !"CONTAINS"_tok >>
|
|
|
|
SkipTo<'\n'>{} >> construct<ErrorRecovery>()};
|
|
|
|
|
|
|
|
// Error recovery within statement sequences: skip *past* the end of the line,
|
2018-04-21 06:19:20 +08:00
|
|
|
// but not over an END or CONTAINS statement.
|
2018-09-11 05:19:37 +08:00
|
|
|
constexpr auto skipStmtErrorRecovery{!"END"_tok >> !"CONTAINS"_tok >>
|
2018-07-13 05:46:23 +08:00
|
|
|
SkipPast<'\n'>{} >> construct<ErrorRecovery>()};
|
2018-04-21 06:19:20 +08:00
|
|
|
|
|
|
|
// Error recovery across statements: skip the line, unless it looks
|
|
|
|
// like it might end the containing construct.
|
2018-07-13 05:46:23 +08:00
|
|
|
constexpr auto stmtErrorRecoveryStart{ignoredStatementPrefix};
|
|
|
|
constexpr auto skipBadLine{SkipPast<'\n'>{} >> construct<ErrorRecovery>()};
|
|
|
|
constexpr auto executionPartErrorRecovery{stmtErrorRecoveryStart >>
|
|
|
|
!"END"_tok >> !"CONTAINS"_tok >> !"ELSE"_tok >> !"CASE"_tok >>
|
2018-07-21 06:49:19 +08:00
|
|
|
!"TYPE IS"_tok >> !"CLASS"_tok >> !"RANK"_tok >>
|
[flang][openacc] OpenACC 3.0 parser
Summary:
This patch introduce the parser for OpenACC 3.0 in Flang. It uses the same TableGen mechanism
than OpenMP.
Reviewers: nvdatian, sscalpone, tskeith, klausler, ichoyjx, jdoerfert, DavidTruby
Reviewed By: klausler
Subscribers: MaskRay, SouraVX, mgorny, hiraditya, jfb, sstefan1, llvm-commits
Tags: #llvm, #flang
Differential Revision: https://reviews.llvm.org/D83649
2020-07-15 02:28:34 +08:00
|
|
|
!("!$ACC "_sptok >> "END"_tok) >>
|
2019-08-14 23:42:28 +08:00
|
|
|
!("!$OMP "_sptok >> ("END"_tok || "SECTION"_id)) >> skipBadLine};
|
2018-04-21 06:19:20 +08:00
|
|
|
|
2018-07-31 06:31:06 +08:00
|
|
|
// END statement error recovery
|
2020-06-13 01:05:04 +08:00
|
|
|
constexpr auto missingOptionalName{pure<std::optional<Name>>()};
|
2018-07-31 06:31:06 +08:00
|
|
|
constexpr auto noNameEnd{"END" >> missingOptionalName};
|
2018-08-01 03:32:09 +08:00
|
|
|
constexpr auto atEndOfStmt{space >>
|
|
|
|
withMessage("expected end of statement"_err_en_US, lookAhead(";\n"_ch))};
|
|
|
|
constexpr auto bareEnd{noNameEnd / recovery(atEndOfStmt, SkipTo<'\n'>{})};
|
|
|
|
|
2018-07-31 06:31:06 +08:00
|
|
|
constexpr auto endStmtErrorRecovery{
|
2018-09-05 02:42:10 +08:00
|
|
|
("END"_tok >> SkipTo<'\n'>{} || ok) >> missingOptionalName};
|
|
|
|
|
2018-07-31 07:33:55 +08:00
|
|
|
constexpr auto progUnitEndStmtErrorRecovery{
|
|
|
|
(many(!"END"_tok >> SkipPast<'\n'>{}) >>
|
|
|
|
("END"_tok >> SkipTo<'\n'>{} || consumedAllInput)) >>
|
|
|
|
missingOptionalName};
|
2020-03-29 12:00:16 +08:00
|
|
|
} // namespace Fortran::parser
|
|
|
|
#endif // FORTRAN_PARSER_STMT_PARSER_H_
|