2018-05-01 21:50:34 +02:00
|
|
|
// Copyright (c) 2018, NVIDIA CORPORATION. All rights reserved.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
// you may not use this file except in compliance with the License.
|
|
|
|
// You may obtain a copy of the License at
|
|
|
|
//
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
//
|
|
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
// See the License for the specific language governing permissions and
|
|
|
|
// limitations under the License.
|
|
|
|
|
2018-04-21 00:19:20 +02: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-02 22:48:12 +02:00
|
|
|
namespace Fortran::parser {
|
2018-04-21 00:19:20 +02: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]...
|
2018-07-11 02:09:07 +02:00
|
|
|
constexpr auto label{space >> digitString / spaceCheck};
|
2018-04-21 00:19:20 +02:00
|
|
|
|
|
|
|
template<typename PA> inline constexpr auto unterminatedStatement(const PA &p) {
|
2018-07-12 23:46:23 +02:00
|
|
|
return skipStuffBeforeStatement >>
|
2018-04-24 01:53:16 +02:00
|
|
|
sourced(construct<Statement<typename PA::resultType>>(
|
|
|
|
maybe(label), space >> p));
|
2018-04-21 00:19:20 +02:00
|
|
|
}
|
|
|
|
|
2018-07-12 23:46:23 +02:00
|
|
|
constexpr auto endOfLine{
|
|
|
|
"\n"_ch >> ok || fail("expected end of line"_err_en_US)};
|
2018-04-21 00:19:20 +02:00
|
|
|
|
2018-07-11 02:09:07 +02:00
|
|
|
constexpr auto endOfStmt{
|
2018-07-12 23:46:23 +02:00
|
|
|
space >> (";"_ch >> skipMany(";"_tok) >> space >> maybe("\n"_ch) >> ok ||
|
|
|
|
endOfLine)};
|
2018-04-21 00:19:20 +02:00
|
|
|
|
|
|
|
template<typename PA> inline constexpr auto statement(const PA &p) {
|
|
|
|
return unterminatedStatement(p) / endOfStmt;
|
|
|
|
}
|
|
|
|
|
2018-07-17 21:34:07 +02: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.
|
|
|
|
template<typename PA> inline constexpr auto unambiguousStatement(const PA &p) {
|
|
|
|
return unterminatedStatement(p) /
|
|
|
|
recovery(space >>
|
|
|
|
withMessage("expected end of statement"_err_en_US, endOfStmt),
|
|
|
|
SkipPast<'\n'>{});
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr auto forceEndOfStmt{recovery(space >>
|
|
|
|
withMessage("expected end of statement"_err_en_US, lookAhead(";\n"_ch)),
|
|
|
|
SkipTo<'\n'>{})};
|
|
|
|
|
2018-07-11 02:09:07 +02:00
|
|
|
constexpr auto ignoredStatementPrefix{
|
2018-07-12 23:46:23 +02:00
|
|
|
skipStuffBeforeStatement >> maybe(label) >> maybe(name / ":") >> space};
|
2018-04-21 00:19:20 +02:00
|
|
|
|
|
|
|
// Error recovery within statements: skip to the end of the line,
|
|
|
|
// but not over an END or CONTAINS statement.
|
2018-07-12 23:46:23 +02:00
|
|
|
constexpr auto stmtErrorRecovery{!"END"_tok >> !"CONTAINS"_tok >>
|
|
|
|
SkipPast<'\n'>{} >> construct<ErrorRecovery>()};
|
2018-04-21 00:19:20 +02:00
|
|
|
|
|
|
|
// Error recovery across statements: skip the line, unless it looks
|
|
|
|
// like it might end the containing construct.
|
2018-07-12 23:46:23 +02:00
|
|
|
constexpr auto stmtErrorRecoveryStart{ignoredStatementPrefix};
|
|
|
|
constexpr auto skipBadLine{SkipPast<'\n'>{} >> construct<ErrorRecovery>()};
|
|
|
|
constexpr auto executionPartErrorRecovery{stmtErrorRecoveryStart >>
|
|
|
|
!"END"_tok >> !"CONTAINS"_tok >> !"ELSE"_tok >> !"CASE"_tok >>
|
|
|
|
!"TYPE IS"_tok >> !"CLASS"_tok >> !"RANK"_tok >> skipBadLine};
|
2018-04-21 00:19:20 +02:00
|
|
|
|
2018-05-02 22:48:12 +02:00
|
|
|
} // namespace Fortran::parser
|
2018-04-21 00:19:20 +02:00
|
|
|
#endif // FORTRAN_PARSER_STMT_PARSER_H_
|