| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279 |
- // 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 "toolchain/check/scope_stack.h"
- #include <utility>
- #include "common/check.h"
- #include "toolchain/sem_ir/ids.h"
- namespace Carbon::Check {
- auto ScopeStack::VerifyOnFinish() const -> void {
- CARBON_CHECK(return_scope_stack_.empty(), "{0}", return_scope_stack_.size());
- CARBON_CHECK(break_continue_stack_.empty(), "{0}",
- break_continue_stack_.size());
- CARBON_CHECK(scope_stack_.empty(), "{0}", scope_stack_.size());
- CARBON_CHECK(destroy_id_stack_.empty(), "{0}",
- destroy_id_stack_.all_values_size());
- CARBON_CHECK(non_lexical_scope_stack_.empty(), "{0}",
- non_lexical_scope_stack_.size());
- CARBON_CHECK(compile_time_binding_stack_.empty(), "{0}",
- compile_time_binding_stack_.all_values_size());
- full_pattern_stack_.VerifyOnFinish();
- }
- auto ScopeStack::VerifyNextCompileTimeBindIndex(llvm::StringLiteral label,
- const ScopeStackEntry& scope)
- -> void {
- CARBON_CHECK(
- static_cast<int32_t>(compile_time_binding_stack_.all_values_size()) ==
- scope.next_compile_time_bind_index.index,
- "Wrong number of entries in compile-time binding stack after {0}: have "
- "{1}, expected {2}",
- label, compile_time_binding_stack_.all_values_size(),
- scope.next_compile_time_bind_index.index);
- }
- auto ScopeStack::Push(SemIR::InstId scope_inst_id, SemIR::NameScopeId scope_id,
- SemIR::SpecificId specific_id,
- bool lexical_lookup_has_load_error) -> void {
- // If this scope doesn't have a specific of its own, it lives in the enclosing
- // scope's specific, if any.
- auto enclosing_specific_id = specific_id;
- if (!specific_id.has_value() && !scope_stack_.empty()) {
- enclosing_specific_id = PeekSpecificId();
- }
- compile_time_binding_stack_.PushArray();
- scope_stack_.push_back(
- {.index = next_scope_index_,
- .scope_inst_id = scope_inst_id,
- .scope_id = scope_id,
- .specific_id = enclosing_specific_id,
- .next_compile_time_bind_index = SemIR::CompileTimeBindIndex(
- compile_time_binding_stack_.all_values_size()),
- .lexical_lookup_has_load_error =
- LexicalLookupHasLoadError() || lexical_lookup_has_load_error});
- if (scope_stack_.back().is_lexical_scope()) {
- // For lexical lookups, unqualified lookup doesn't know how to find the
- // associated specific, so if we start adding lexical scopes associated with
- // specifics, we'll need to somehow track them in lookup.
- CARBON_CHECK(!specific_id.has_value(),
- "Lexical scope should not have an associated specific.");
- destroy_id_stack_.PushArray();
- } else {
- non_lexical_scope_stack_.push_back({.scope_index = next_scope_index_,
- .name_scope_id = scope_id,
- .specific_id = enclosing_specific_id});
- }
- // TODO: Handle this case more gracefully.
- CARBON_CHECK(next_scope_index_.index != std::numeric_limits<int32_t>::max(),
- "Ran out of scopes");
- ++next_scope_index_.index;
- VerifyNextCompileTimeBindIndex("Push", scope_stack_.back());
- }
- auto ScopeStack::Pop() -> void {
- auto scope = scope_stack_.pop_back_val();
- scope.names.ForEach([&](SemIR::NameId str_id) {
- auto& lexical_results = lexical_lookup_.Get(str_id);
- CARBON_CHECK(lexical_results.back().scope_index == scope.index,
- "Inconsistent scope index for name {0}", str_id);
- lexical_results.pop_back();
- });
- if (scope.is_lexical_scope()) {
- destroy_id_stack_.PopArray();
- } else {
- CARBON_CHECK(non_lexical_scope_stack_.back().scope_index == scope.index);
- non_lexical_scope_stack_.pop_back();
- }
- if (scope.has_returned_var) {
- CARBON_CHECK(!return_scope_stack_.empty());
- CARBON_CHECK(return_scope_stack_.back().returned_var.has_value());
- return_scope_stack_.back().returned_var = SemIR::InstId::None;
- }
- VerifyNextCompileTimeBindIndex("Pop", scope);
- compile_time_binding_stack_.PopArray();
- }
- auto ScopeStack::PopTo(ScopeIndex index) -> void {
- while (PeekIndex() > index) {
- Pop();
- }
- CARBON_CHECK(PeekIndex() == index,
- "Scope index {0} does not enclose the current scope {1}", index,
- PeekIndex());
- }
- auto ScopeStack::LookupInLexicalScopesWithin(SemIR::NameId name_id,
- ScopeIndex scope_index)
- -> SemIR::InstId {
- auto& lexical_results = lexical_lookup_.Get(name_id);
- if (lexical_results.empty()) {
- return SemIR::InstId::None;
- }
- auto result = lexical_results.back();
- if (result.scope_index < scope_index) {
- return SemIR::InstId::None;
- }
- return result.inst_id;
- }
- auto ScopeStack::LookupInLexicalScopes(SemIR::NameId name_id)
- -> std::pair<SemIR::InstId, llvm::ArrayRef<NonLexicalScope>> {
- // Find the results from lexical scopes. These will be combined with results
- // from non-lexical scopes such as namespaces and classes.
- llvm::ArrayRef<LexicalLookup::Result> lexical_results =
- lexical_lookup_.Get(name_id);
- // If we have no lexical results, check all non-lexical scopes.
- if (lexical_results.empty()) {
- return {LexicalLookupHasLoadError() ? SemIR::ErrorInst::InstId
- : SemIR::InstId::None,
- non_lexical_scope_stack_};
- }
- // Find the first non-lexical scope that is within the scope of the lexical
- // lookup result.
- auto* first_non_lexical_scope = llvm::lower_bound(
- non_lexical_scope_stack_, lexical_results.back().scope_index,
- [](const NonLexicalScope& scope, ScopeIndex index) {
- return scope.scope_index < index;
- });
- return {
- lexical_results.back().inst_id,
- llvm::ArrayRef(first_non_lexical_scope, non_lexical_scope_stack_.end())};
- }
- auto ScopeStack::LookupOrAddName(SemIR::NameId name_id, SemIR::InstId target_id,
- ScopeIndex scope_index) -> SemIR::InstId {
- // Find the corresponding scope depth.
- //
- // TODO: Consider passing in the depth rather than performing a scan for it.
- // We only do this scan when declaring an entity such as a class within a
- // function, so it should be relatively rare, but it's still not necesasry to
- // recompute this.
- int scope_depth = scope_stack_.size() - 1;
- if (scope_index.has_value()) {
- scope_depth =
- llvm::lower_bound(scope_stack_, scope_index,
- [](const ScopeStackEntry& entry, ScopeIndex index) {
- return entry.index < index;
- }) -
- scope_stack_.begin();
- CARBON_CHECK(scope_stack_[scope_depth].index == scope_index,
- "Declaring name in scope that has already ended");
- } else {
- scope_index = scope_stack_[scope_depth].index;
- }
- // If this name has already been declared in this scope or an inner scope,
- // return the existing result.
- auto& lexical_results = lexical_lookup_.Get(name_id);
- if (!lexical_results.empty() &&
- lexical_results.back().scope_index >= scope_index) {
- return lexical_results.back().inst_id;
- }
- // Add the name into the scope.
- bool inserted = scope_stack_[scope_depth].names.Insert(name_id).is_inserted();
- CARBON_CHECK(inserted, "Name in scope but not in lexical lookups");
- ++scope_stack_[scope_depth].num_names;
- // Add a corresponding lexical lookup result.
- lexical_results.push_back({.inst_id = target_id, .scope_index = scope_index});
- return SemIR::InstId::None;
- }
- auto ScopeStack::SetReturnedVarOrGetExisting(SemIR::InstId inst_id)
- -> SemIR::InstId {
- CARBON_CHECK(!return_scope_stack_.empty(), "`returned var` in no function");
- auto& returned_var = return_scope_stack_.back().returned_var;
- if (returned_var.has_value()) {
- return returned_var;
- }
- returned_var = inst_id;
- CARBON_CHECK(!scope_stack_.back().has_returned_var,
- "Scope has returned var but none is set");
- if (inst_id.has_value()) {
- scope_stack_.back().has_returned_var = true;
- }
- return SemIR::InstId::None;
- }
- auto ScopeStack::Suspend() -> SuspendedScope {
- CARBON_CHECK(!scope_stack_.empty(), "No scope to suspend");
- SuspendedScope result = {.entry = scope_stack_.pop_back_val(),
- .suspended_items = {}};
- if (result.entry.is_lexical_scope()) {
- CARBON_CHECK(destroy_id_stack_.PeekArray().empty(),
- "Missing support to suspend scopes with destructed storage");
- destroy_id_stack_.PopArray();
- } else {
- non_lexical_scope_stack_.pop_back();
- }
- auto peek_compile_time_bindings = compile_time_binding_stack_.PeekArray();
- result.suspended_items.reserve(result.entry.num_names +
- peek_compile_time_bindings.size());
- result.entry.names.ForEach([&](SemIR::NameId name_id) {
- auto [index, inst_id] = lexical_lookup_.Suspend(name_id);
- CARBON_CHECK(index !=
- SuspendedScope::ScopeItem::IndexForCompileTimeBinding);
- result.suspended_items.push_back({.index = index, .inst_id = inst_id});
- });
- CARBON_CHECK(static_cast<int>(result.suspended_items.size()) ==
- result.entry.num_names);
- // Move any compile-time bindings into the suspended scope.
- for (auto inst_id : peek_compile_time_bindings) {
- result.suspended_items.push_back(
- {.index = SuspendedScope::ScopeItem::IndexForCompileTimeBinding,
- .inst_id = inst_id});
- }
- compile_time_binding_stack_.PopArray();
- // This would be easy to support if we had a need, but currently we do not.
- CARBON_CHECK(!result.entry.has_returned_var,
- "Should not suspend a scope with a returned var.");
- return result;
- }
- auto ScopeStack::Restore(SuspendedScope scope) -> void {
- compile_time_binding_stack_.PushArray();
- for (auto [index, inst_id] : scope.suspended_items) {
- if (index == SuspendedScope::ScopeItem::IndexForCompileTimeBinding) {
- compile_time_binding_stack_.AppendToTop(inst_id);
- } else {
- lexical_lookup_.Restore({.index = index, .inst_id = inst_id},
- scope.entry.index);
- }
- }
- VerifyNextCompileTimeBindIndex("Restore", scope.entry);
- if (scope.entry.is_lexical_scope()) {
- destroy_id_stack_.PushArray();
- } else {
- non_lexical_scope_stack_.push_back(
- {.scope_index = scope.entry.index,
- .name_scope_id = scope.entry.scope_id,
- .specific_id = scope.entry.specific_id});
- }
- scope_stack_.push_back(std::move(scope.entry));
- }
- } // namespace Carbon::Check
|