2020-02-25 23:11:52 +08:00
|
|
|
//===-- lib/Semantics/check-purity.cpp ------------------------------------===//
|
2019-11-13 07:43:09 +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
|
2019-11-13 07:43:09 +08:00
|
|
|
//
|
2020-01-11 04:12:03 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2019-11-13 07:43:09 +08:00
|
|
|
|
|
|
|
#include "check-purity.h"
|
2020-02-25 23:11:52 +08:00
|
|
|
#include "flang/Parser/parse-tree.h"
|
|
|
|
#include "flang/Semantics/tools.h"
|
2019-11-13 07:43:09 +08:00
|
|
|
|
|
|
|
namespace Fortran::semantics {
|
|
|
|
void PurityChecker::Enter(const parser::ExecutableConstruct &exec) {
|
|
|
|
if (InPureSubprogram() && IsImageControlStmt(exec)) {
|
|
|
|
context_.Say(GetImageControlStmtLocation(exec),
|
2019-12-24 09:12:53 +08:00
|
|
|
"An image control statement may not appear in a pure subprogram"_err_en_US);
|
2019-11-13 07:43:09 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
void PurityChecker::Enter(const parser::SubroutineSubprogram &subr) {
|
|
|
|
const auto &stmt{std::get<parser::Statement<parser::SubroutineStmt>>(subr.t)};
|
|
|
|
Entered(
|
|
|
|
stmt.source, std::get<std::list<parser::PrefixSpec>>(stmt.statement.t));
|
|
|
|
}
|
|
|
|
|
|
|
|
void PurityChecker::Leave(const parser::SubroutineSubprogram &) { Left(); }
|
|
|
|
|
|
|
|
void PurityChecker::Enter(const parser::FunctionSubprogram &func) {
|
|
|
|
const auto &stmt{std::get<parser::Statement<parser::FunctionStmt>>(func.t)};
|
|
|
|
Entered(
|
|
|
|
stmt.source, std::get<std::list<parser::PrefixSpec>>(stmt.statement.t));
|
|
|
|
}
|
|
|
|
|
|
|
|
void PurityChecker::Leave(const parser::FunctionSubprogram &) { Left(); }
|
|
|
|
|
|
|
|
bool PurityChecker::InPureSubprogram() const {
|
|
|
|
return pureDepth_ >= 0 && depth_ >= pureDepth_;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool PurityChecker::HasPurePrefix(
|
|
|
|
const std::list<parser::PrefixSpec> &prefixes) const {
|
|
|
|
for (const parser::PrefixSpec &prefix : prefixes) {
|
|
|
|
if (std::holds_alternative<parser::PrefixSpec::Pure>(prefix.u)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PurityChecker::Entered(
|
|
|
|
parser::CharBlock source, const std::list<parser::PrefixSpec> &prefixes) {
|
|
|
|
if (depth_ == 2) {
|
|
|
|
context_.messages().Say(source,
|
|
|
|
"An internal subprogram may not contain an internal subprogram"_err_en_US);
|
|
|
|
}
|
|
|
|
if (HasPurePrefix(prefixes)) {
|
|
|
|
if (pureDepth_ < 0) {
|
|
|
|
pureDepth_ = depth_;
|
|
|
|
}
|
|
|
|
} else if (InPureSubprogram()) {
|
|
|
|
context_.messages().Say(source,
|
2019-12-24 09:12:53 +08:00
|
|
|
"An internal subprogram of a pure subprogram must also be pure"_err_en_US);
|
2019-11-13 07:43:09 +08:00
|
|
|
}
|
|
|
|
++depth_;
|
|
|
|
}
|
|
|
|
|
|
|
|
void PurityChecker::Left() {
|
|
|
|
if (pureDepth_ == --depth_) {
|
|
|
|
pureDepth_ = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-29 12:00:16 +08:00
|
|
|
} // namespace Fortran::semantics
|