// Part of the Carbon Language project, under the Apache License v2.0 with LLVM // Exceptions. See /LICENSE for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception #include "explorer/ast/static_scope.h" #include #include "explorer/common/error_builders.h" #include "llvm/Support/Error.h" namespace Carbon { auto StaticScope::Add(const std::string& name, ValueNodeView entity, NameStatus status /* = NameStatus::Usable*/) -> ErrorOr { auto [it, inserted] = declared_names_.insert({name, {entity, status}}); if (!inserted) { if (it->second.entity != entity) { return CompilationError(entity.base().source_loc()) << "Duplicate name `" << name << "` also found at " << it->second.entity.base().source_loc(); } if (static_cast(status) > static_cast(it->second.status)) { it->second.status = status; } } return Success(); } void StaticScope::MarkDeclared(const std::string& name) { auto it = declared_names_.find(name); CARBON_CHECK(it != declared_names_.end()) << name << " not found"; if (it->second.status == NameStatus::KnownButNotDeclared) { it->second.status = NameStatus::DeclaredButNotUsable; } } void StaticScope::MarkUsable(const std::string& name) { auto it = declared_names_.find(name); CARBON_CHECK(it != declared_names_.end()) << name << " not found"; it->second.status = NameStatus::Usable; } auto StaticScope::Resolve(const std::string& name, SourceLocation source_loc) const -> ErrorOr { CARBON_ASSIGN_OR_RETURN(std::optional result, TryResolve(name, source_loc)); if (!result) { return CompilationError(source_loc) << "could not resolve '" << name << "'"; } return *result; } auto StaticScope::TryResolve(const std::string& name, SourceLocation source_loc) const -> ErrorOr> { auto it = declared_names_.find(name); if (it != declared_names_.end()) { switch (it->second.status) { case NameStatus::KnownButNotDeclared: return CompilationError(source_loc) << "'" << name << "' has not been declared yet"; case NameStatus::DeclaredButNotUsable: return CompilationError(source_loc) << "'" << name << "' is not usable until after it has been completely declared"; case NameStatus::Usable: return std::make_optional(it->second.entity); } } std::optional result; for (Nonnull parent : parent_scopes_) { CARBON_ASSIGN_OR_RETURN(std::optional parent_result, parent->TryResolve(name, source_loc)); if (parent_result.has_value() && result.has_value() && *parent_result != *result) { return CompilationError(source_loc) << "'" << name << "' is ambiguous between " << result->base().source_loc() << " and " << parent_result->base().source_loc(); } result = parent_result; } return result; } auto StaticScope::AddReturnedVar(ValueNodeView returned_var_def_view) -> ErrorOr { std::optional resolved_returned_var = ResolveReturned(); if (resolved_returned_var.has_value()) { return CompilationError(returned_var_def_view.base().source_loc()) << "Duplicate definition of returned var also found at " << resolved_returned_var->base().source_loc(); } returned_var_def_view_ = std::move(returned_var_def_view); return Success(); } auto StaticScope::ResolveReturned() const -> std::optional { if (returned_var_def_view_.has_value()) { return returned_var_def_view_; } for (Nonnull parent : parent_scopes_) { std::optional parent_returned_var = parent->ResolveReturned(); if (parent_returned_var.has_value()) { return parent_returned_var; } } return std::nullopt; } } // namespace Carbon