// 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 "executable_semantics/interpreter/action.h" #include #include #include #include #include #include "executable_semantics/ast/declaration.h" #include "executable_semantics/ast/expression.h" #include "executable_semantics/common/arena.h" #include "executable_semantics/interpreter/stack.h" #include "llvm/ADT/StringExtras.h" #include "llvm/Support/Casting.h" namespace Carbon { using llvm::cast; RuntimeScope::RuntimeScope(RuntimeScope&& other) noexcept : locals_(std::move(other.locals_)), // To transfer ownership of other.allocations_, we have to empty it out. allocations_(std::exchange(other.allocations_, {})), heap_(other.heap_) {} auto RuntimeScope::operator=(RuntimeScope&& rhs) noexcept -> RuntimeScope& { locals_ = std::move(rhs.locals_); // To transfer ownership of rhs.allocations_, we have to empty it out. allocations_ = std::exchange(rhs.allocations_, {}); heap_ = rhs.heap_; return *this; } RuntimeScope::~RuntimeScope() { for (AllocationId allocation : allocations_) { heap_->Deallocate(allocation); } } void RuntimeScope::Print(llvm::raw_ostream& out) const { out << "{"; llvm::ListSeparator sep; for (const auto& [value_node, value] : locals_) { out << sep << value_node.base() << ": " << *value; } out << "}"; } void RuntimeScope::Initialize(ValueNodeView value_node, Nonnull value) { CHECK(!value_node.constant_value().has_value()); CHECK(value->kind() != Value::Kind::LValue); allocations_.push_back(heap_->AllocateValue(value)); auto [it, success] = locals_.insert( {value_node, heap_->arena().New(Address(allocations_.back()))}); CHECK(success) << "Duplicate definition of " << value_node.base(); } void RuntimeScope::Merge(RuntimeScope other) { CHECK(heap_ == other.heap_); locals_.merge(other.locals_); CHECK(other.locals_.empty()) << "Duplicate definition of " << other.locals_.size() << " names, including " << other.locals_.begin()->first.base(); allocations_.insert(allocations_.end(), other.allocations_.begin(), other.allocations_.end()); other.allocations_.clear(); } auto RuntimeScope::Get(ValueNodeView value_node) const -> std::optional> { auto it = locals_.find(value_node); if (it != locals_.end()) { return it->second; } else { return std::nullopt; } } auto RuntimeScope::Capture( const std::vector>& scopes) -> RuntimeScope { CHECK(!scopes.empty()); RuntimeScope result(scopes.front()->heap_); for (Nonnull scope : scopes) { CHECK(scope->heap_ == result.heap_); for (const auto& entry : scope->locals_) { // Intentionally disregards duplicates later in the vector. result.locals_.insert(entry); } } return result; } void Action::Print(llvm::raw_ostream& out) const { switch (kind()) { case Action::Kind::LValAction: out << cast(*this).expression(); break; case Action::Kind::ExpressionAction: out << cast(*this).expression(); break; case Action::Kind::PatternAction: out << cast(*this).pattern(); break; case Action::Kind::StatementAction: cast(*this).statement().PrintDepth(1, out); break; case Action::Kind::DeclarationAction: cast(*this).declaration().Print(out); break; case Action::Kind::ScopeAction: out << "ScopeAction"; } out << "<" << pos_ << ">"; if (results_.size() > 0) { out << "("; llvm::ListSeparator sep; for (auto& result : results_) { out << sep << *result; } out << ")"; } } } // namespace Carbon