| //===-- lib/Semantics/check-nullify.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 "check-nullify.h" |
| #include "definable.h" |
| #include "flang/Evaluate/expression.h" |
| #include "flang/Parser/message.h" |
| #include "flang/Parser/parse-tree.h" |
| #include "flang/Semantics/expression.h" |
| #include "flang/Semantics/tools.h" |
| |
| namespace Fortran::semantics { |
| |
| void NullifyChecker::Leave(const parser::NullifyStmt &nullifyStmt) { |
| CHECK(context_.location()); |
| const Scope &scope{context_.FindScope(*context_.location())}; |
| for (const parser::PointerObject &pointerObject : nullifyStmt.v) { |
| common::visit( |
| common::visitors{ |
| [&](const parser::Name &name) { |
| if (name.symbol) { |
| if (auto whyNot{WhyNotDefinable(name.source, scope, |
| DefinabilityFlags{DefinabilityFlag::PointerDefinition}, |
| *name.symbol)}) { |
| context_.messages() |
| .Say(name.source, |
| "'%s' may not appear in NULLIFY"_err_en_US, |
| name.source) |
| .Attach(std::move( |
| whyNot->set_severity(parser::Severity::Because))); |
| } |
| } |
| }, |
| [&](const parser::StructureComponent &structureComponent) { |
| const auto &component{structureComponent.component}; |
| SourceName at{component.source}; |
| if (const auto *checkedExpr{GetExpr(context_, pointerObject)}) { |
| if (auto whyNot{WhyNotDefinable(at, scope, |
| DefinabilityFlags{DefinabilityFlag::PointerDefinition}, |
| *checkedExpr)}) { |
| context_.messages() |
| .Say(at, "'%s' may not appear in NULLIFY"_err_en_US, at) |
| .Attach(std::move( |
| whyNot->set_severity(parser::Severity::Because))); |
| } |
| } |
| }, |
| }, |
| pointerObject.u); |
| } |
| // From 9.7.3.1(1) |
| // A pointer-object shall not depend on the value, |
| // bounds, or association status of another pointer- |
| // object in the same NULLIFY statement. |
| // This restriction is the programmer's responsibility. |
| // Some dependencies can be found compile time or at |
| // runtime, but for now we choose to skip such checks. |
| } |
| } // namespace Fortran::semantics |