// 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 #ifndef EXECUTABLE_SEMANTICS_INTERPRETER_INTERPRETER_H_ #define EXECUTABLE_SEMANTICS_INTERPRETER_INTERPRETER_H_ #include #include #include #include "executable_semantics/ast/declaration.h" #include "executable_semantics/interpreter/action.h" #include "executable_semantics/interpreter/dictionary.h" #include "executable_semantics/interpreter/stack.h" #include "executable_semantics/interpreter/value.h" namespace Carbon { using Env = Dictionary; /***** Scopes *****/ struct Scope { Scope(Env values, std::list l) : values(values), locals(std::move(l)) {} Env values; std::list locals; }; /***** Frames and State *****/ // A frame represents either a function call or a delimited continuation. struct Frame { // The name of the function. std::string name; // If the frame represents a function call, the bottom scope // contains the parameter-argument bindings for this function // call. The rest of the scopes contain local variables defined by // blocks within the function. The scope at the top of the stack is // the current scope and its environment is the one used for looking // up the value associated with a variable. Stack scopes; // The actions that need to be executed in the future of the // current function call. The top of the stack is the action // that is executed first. Stack todo; // If this frame is the bottom frame of a continuation, then it stores // the address of the continuation. std::optional
continuation; Frame(std::string n, Stack s, Stack c) : name(std::move(std::move(n))), scopes(s), todo(c), continuation() {} }; // A Heap represents the abstract machine's dynamically allocated memory. class Heap { public: // Constructs an empty Heap. Heap() = default; Heap(const Heap&) = delete; Heap& operator=(const Heap&) = delete; // Returns the value at the given address in the heap after // checking that it is alive. auto Read(const Address& a, int line_num) -> const Value*; // Writes the given value at the address in the heap after // checking that the address is alive. auto Write(const Address& a, const Value* v, int line_num) -> void; // Put the given value on the heap and mark it as alive. auto AllocateValue(const Value* v) -> Address; // Marks the object at this address, and all of its sub-objects, as dead. auto Deallocate(const Address& address) -> void; // Print the value at the given address to the stream `out`. auto PrintAddress(const Address& a, std::ostream& out) -> void; // Print all the values on the heap to the stream `out`. auto PrintHeap(std::ostream& out) -> void; private: // Signal an error if the address is no longer alive. void CheckAlive(const Address& address, int line_num); std::vector values_; std::vector alive_; }; struct State { Stack stack; Heap heap; }; extern State* state; void InitEnv(const Declaration& d, Env* env); void PrintFrame(Frame* frame, std::ostream& out); void PrintStack(Stack ls, std::ostream& out); void PrintEnv(Env values); auto CopyVal(const Value* val, int line_num) -> const Value*; auto ToInteger(const Value* v) -> int; /***** Interpreters *****/ auto InterpProgram(std::list* fs) -> int; auto InterpExp(Env values, const Expression* e) -> const Value*; } // namespace Carbon #endif // EXECUTABLE_SEMANTICS_INTERPRETER_INTERPRETER_H_