// 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/heap.h" #include "executable_semantics/common/error.h" #include "llvm/ADT/StringExtras.h" namespace Carbon { auto Heap::AllocateValue(Nonnull v) -> Address { // Putting the following two side effects together in this function // ensures that we don't do anything else in between, which is really bad! // Consider whether to include a copy of the input v in this function // or to leave it up to the caller. Address a(values.size()); values.push_back(v); alive.push_back(true); return a; } auto Heap::Read(const Address& a, SourceLocation source_loc) -> Nonnull { this->CheckAlive(a, source_loc); return values[a.index]->GetField(arena, a.field_path, source_loc); } void Heap::Write(const Address& a, Nonnull v, SourceLocation source_loc) { this->CheckAlive(a, source_loc); values[a.index] = values[a.index]->SetField(arena, a.field_path, v, source_loc); } void Heap::CheckAlive(const Address& address, SourceLocation source_loc) { if (!alive[address.index]) { FATAL_RUNTIME_ERROR(source_loc) << "undefined behavior: access to dead value " << *values[address.index]; } } void Heap::Deallocate(const Address& address) { CHECK(address.field_path.IsEmpty()); if (alive[address.index]) { alive[address.index] = false; } else { FATAL_RUNTIME_ERROR_NO_LINE() << "deallocating an already dead value"; } } void Heap::Print(llvm::raw_ostream& out) const { llvm::ListSeparator sep; for (size_t i = 0; i < values.size(); ++i) { out << sep; PrintAddress(Address(i), out); } } void Heap::PrintAddress(const Address& a, llvm::raw_ostream& out) const { if (!alive[a.index]) { out << "!!"; } out << *values[a.index]; } } // namespace Carbon