| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464 |
- // 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/value.h"
- #include <algorithm>
- #include <iostream>
- #include "common/check.h"
- #include "executable_semantics/interpreter/interpreter.h"
- namespace Carbon {
- auto Value::GetIntValue() const -> int {
- return std::get<IntValue>(value).value;
- }
- auto Value::GetBoolValue() const -> bool {
- return std::get<BoolValue>(value).value;
- }
- auto Value::GetFunctionValue() const -> const FunctionValue& {
- return std::get<FunctionValue>(value);
- }
- auto Value::GetStructValue() const -> const StructValue& {
- return std::get<StructValue>(value);
- }
- auto Value::GetAlternativeConstructorValue() const
- -> const AlternativeConstructorValue& {
- return std::get<AlternativeConstructorValue>(value);
- }
- auto Value::GetAlternativeValue() const -> const AlternativeValue& {
- return std::get<AlternativeValue>(value);
- }
- auto Value::GetTupleValue() const -> const TupleValue& {
- return std::get<TupleValue>(value);
- }
- auto Value::GetPointerValue() const -> Address {
- return std::get<PointerValue>(value).value;
- }
- auto Value::GetBindingPlaceholderValue() const
- -> const BindingPlaceholderValue& {
- return std::get<BindingPlaceholderValue>(value);
- }
- auto Value::GetFunctionType() const -> const FunctionType& {
- return std::get<FunctionType>(value);
- }
- auto Value::GetPointerType() const -> const PointerType& {
- return std::get<PointerType>(value);
- }
- auto Value::GetStructType() const -> const StructType& {
- return std::get<StructType>(value);
- }
- auto Value::GetChoiceType() const -> const ChoiceType& {
- return std::get<ChoiceType>(value);
- }
- auto Value::GetContinuationValue() const -> const ContinuationValue& {
- return std::get<ContinuationValue>(value);
- }
- auto FindInVarValues(const std::string& field, const VarValues& inits)
- -> const Value* {
- for (auto& i : inits) {
- if (i.first == field) {
- return i.second;
- }
- }
- return nullptr;
- }
- auto FieldsEqual(const VarValues& ts1, const VarValues& ts2) -> bool {
- if (ts1.size() == ts2.size()) {
- for (auto& iter1 : ts1) {
- auto t2 = FindInVarValues(iter1.first, ts2);
- if (t2 == nullptr) {
- return false;
- }
- if (!TypeEqual(iter1.second, t2)) {
- return false;
- }
- }
- return true;
- } else {
- return false;
- }
- }
- auto FindTupleField(const std::string& name, const Value* tuple)
- -> std::optional<Address> {
- CHECK(tuple->tag() == ValKind::TupleValue);
- for (const TupleElement& element : tuple->GetTupleValue().elements) {
- if (element.name == name) {
- return element.address;
- }
- }
- return std::nullopt;
- }
- auto Value::MakeIntValue(int i) -> const Value* {
- auto* v = new Value();
- v->value = IntValue({.value = i});
- return v;
- }
- auto Value::MakeBoolValue(bool b) -> const Value* {
- auto* v = new Value();
- v->value = BoolValue({.value = b});
- return v;
- }
- auto Value::MakeFunctionValue(std::string name, const Value* param,
- const Statement* body) -> const Value* {
- auto* v = new Value();
- v->value =
- FunctionValue({.name = std::move(name), .param = param, .body = body});
- return v;
- }
- auto Value::MakePointerValue(Address addr) -> const Value* {
- auto* v = new Value();
- v->value = PointerValue({.value = addr});
- return v;
- }
- auto Value::MakeStructValue(const Value* type, const Value* inits)
- -> const Value* {
- auto* v = new Value();
- v->value = StructValue({.type = type, .inits = inits});
- return v;
- }
- auto Value::MakeTupleValue(std::vector<TupleElement> elements) -> const Value* {
- auto* v = new Value();
- v->value = TupleValue({.elements = std::move(elements)});
- return v;
- }
- auto Value::MakeAlternativeValue(std::string alt_name, std::string choice_name,
- Address argument) -> const Value* {
- auto* v = new Value();
- v->value = AlternativeValue({.alt_name = std::move(alt_name),
- .choice_name = std::move(choice_name),
- .argument = argument});
- return v;
- }
- auto Value::MakeAlternativeConstructorValue(std::string alt_name,
- std::string choice_name)
- -> const Value* {
- auto* v = new Value();
- v->value = AlternativeConstructorValue(
- {.alt_name = std::move(alt_name), .choice_name = std::move(choice_name)});
- return v;
- }
- // Return a first-class continuation represented a fragment
- // of the stack.
- auto Value::MakeContinuationValue(std::vector<Frame*> stack) -> Value* {
- auto* v = new Value();
- v->value = ContinuationValue({.stack = std::move(stack)});
- return v;
- }
- auto Value::MakeBindingPlaceholderValue(std::string name, const Value* type)
- -> const Value* {
- auto* v = new Value();
- v->value = BindingPlaceholderValue({.name = std::move(name), .type = type});
- return v;
- }
- auto Value::MakeIntType() -> const Value* {
- auto* v = new Value();
- v->value = IntType();
- return v;
- }
- auto Value::MakeBoolType() -> const Value* {
- auto* v = new Value();
- v->value = BoolType();
- return v;
- }
- auto Value::MakeTypeType() -> const Value* {
- auto* v = new Value();
- v->value = TypeType();
- return v;
- }
- // Return a Continuation type.
- auto Value::MakeContinuationType() -> const Value* {
- auto* v = new Value();
- v->value = ContinuationType();
- return v;
- }
- auto Value::MakeAutoType() -> const Value* {
- auto* v = new Value();
- v->value = AutoType();
- return v;
- }
- auto Value::MakeFunctionType(const Value* param, const Value* ret)
- -> const Value* {
- auto* v = new Value();
- v->value = FunctionType({.param = param, .ret = ret});
- return v;
- }
- auto Value::MakePointerType(const Value* type) -> const Value* {
- auto* v = new Value();
- v->value = PointerType({.type = type});
- return v;
- }
- auto Value::MakeStructType(std::string name, VarValues fields,
- VarValues methods) -> const Value* {
- auto* v = new Value();
- v->value = StructType({.name = std::move(name),
- .fields = std::move(fields),
- .methods = std::move(methods)});
- return v;
- }
- auto Value::MakeUnitTypeVal() -> const Value* {
- auto* v = new Value();
- v->value = TupleValue({.elements = {}});
- return v;
- }
- auto Value::MakeChoiceType(std::string name, VarValues alts) -> const Value* {
- auto* v = new Value();
- v->value =
- ChoiceType({.name = std::move(name), .alternatives = std::move(alts)});
- return v;
- }
- auto PrintValue(const Value* val, std::ostream& out) -> void {
- switch (val->tag()) {
- case ValKind::AlternativeConstructorValue: {
- out << val->GetAlternativeConstructorValue().choice_name << "."
- << val->GetAlternativeConstructorValue().alt_name;
- break;
- }
- case ValKind::BindingPlaceholderValue: {
- PrintValue(val->GetBindingPlaceholderValue().type, out);
- out << ": " << val->GetBindingPlaceholderValue().name;
- break;
- }
- case ValKind::AlternativeValue: {
- out << "alt " << val->GetAlternativeValue().choice_name << "."
- << val->GetAlternativeValue().alt_name << " ";
- state->heap.PrintAddress(val->GetAlternativeValue().argument, out);
- break;
- }
- case ValKind::StructValue: {
- out << val->GetStructValue().type->GetStructType().name;
- PrintValue(val->GetStructValue().inits, out);
- break;
- }
- case ValKind::TupleValue: {
- out << "(";
- bool add_commas = false;
- for (const TupleElement& element : val->GetTupleValue().elements) {
- if (add_commas) {
- out << ", ";
- } else {
- add_commas = true;
- }
- out << element.name << " = ";
- state->heap.PrintAddress(element.address, out);
- }
- out << ")";
- break;
- }
- case ValKind::IntValue:
- out << val->GetIntValue();
- break;
- case ValKind::BoolValue:
- out << std::boolalpha << val->GetBoolValue();
- break;
- case ValKind::FunctionValue:
- out << "fun<" << val->GetFunctionValue().name << ">";
- break;
- case ValKind::PointerValue:
- out << "ptr<" << val->GetPointerValue() << ">";
- break;
- case ValKind::BoolType:
- out << "Bool";
- break;
- case ValKind::IntType:
- out << "Int";
- break;
- case ValKind::TypeType:
- out << "Type";
- break;
- case ValKind::AutoType:
- out << "auto";
- break;
- case ValKind::ContinuationType:
- out << "Continuation";
- break;
- case ValKind::PointerType:
- PrintValue(val->GetPointerType().type, out);
- out << "*";
- break;
- case ValKind::FunctionType:
- out << "fn ";
- PrintValue(val->GetFunctionType().param, out);
- out << " -> ";
- PrintValue(val->GetFunctionType().ret, out);
- break;
- case ValKind::StructType:
- out << "struct " << val->GetStructType().name;
- break;
- case ValKind::ChoiceType:
- out << "choice " << val->GetChoiceType().name;
- break;
- case ValKind::ContinuationValue:
- out << "continuation[[";
- for (Frame* frame : val->GetContinuationValue().stack) {
- PrintFrame(frame, out);
- out << " :: ";
- }
- out << "]]";
- break;
- }
- }
- auto TypeEqual(const Value* t1, const Value* t2) -> bool {
- if (t1->tag() != t2->tag()) {
- return false;
- }
- switch (t1->tag()) {
- case ValKind::PointerType:
- return TypeEqual(t1->GetPointerType().type, t2->GetPointerType().type);
- case ValKind::FunctionType:
- return TypeEqual(t1->GetFunctionType().param,
- t2->GetFunctionType().param) &&
- TypeEqual(t1->GetFunctionType().ret, t2->GetFunctionType().ret);
- case ValKind::StructType:
- return t1->GetStructType().name == t2->GetStructType().name;
- case ValKind::ChoiceType:
- return t1->GetChoiceType().name == t2->GetChoiceType().name;
- case ValKind::TupleValue: {
- if (t1->GetTupleValue().elements.size() !=
- t2->GetTupleValue().elements.size()) {
- return false;
- }
- for (size_t i = 0; i < t1->GetTupleValue().elements.size(); ++i) {
- if (t1->GetTupleValue().elements[i].name !=
- t2->GetTupleValue().elements[i].name) {
- return false;
- }
- if (!TypeEqual(
- state->heap.Read(t1->GetTupleValue().elements[i].address, 0),
- state->heap.Read(t2->GetTupleValue().elements[i].address, 0))) {
- return false;
- }
- }
- return true;
- }
- case ValKind::IntType:
- case ValKind::BoolType:
- case ValKind::ContinuationType:
- case ValKind::TypeType:
- return true;
- default:
- std::cerr << "TypeEqual used to compare non-type values" << std::endl;
- PrintValue(t1, std::cerr);
- std::cerr << std::endl;
- PrintValue(t2, std::cerr);
- exit(-1);
- }
- }
- // Returns true if all the fields of the two tuples contain equal values
- // and returns false otherwise.
- static auto FieldsValueEqual(const std::vector<TupleElement>& ts1,
- const std::vector<TupleElement>& ts2, int line_num)
- -> bool {
- if (ts1.size() != ts2.size()) {
- return false;
- }
- for (const TupleElement& element : ts1) {
- auto iter = std::find_if(
- ts2.begin(), ts2.end(),
- [&](const TupleElement& e2) { return e2.name == element.name; });
- if (iter == ts2.end()) {
- return false;
- }
- if (!ValueEqual(state->heap.Read(element.address, line_num),
- state->heap.Read(iter->address, line_num), line_num)) {
- return false;
- }
- }
- return true;
- }
- // Returns true if the two values are equal and returns false otherwise.
- //
- // This function implements the `==` operator of Carbon.
- auto ValueEqual(const Value* v1, const Value* v2, int line_num) -> bool {
- if (v1->tag() != v2->tag()) {
- return false;
- }
- switch (v1->tag()) {
- case ValKind::IntValue:
- return v1->GetIntValue() == v2->GetIntValue();
- case ValKind::BoolValue:
- return v1->GetBoolValue() == v2->GetBoolValue();
- case ValKind::PointerValue:
- return v1->GetPointerValue() == v2->GetPointerValue();
- case ValKind::FunctionValue:
- return v1->GetFunctionValue().body == v2->GetFunctionValue().body;
- case ValKind::TupleValue:
- return FieldsValueEqual(v1->GetTupleValue().elements,
- v2->GetTupleValue().elements, line_num);
- default:
- case ValKind::IntType:
- case ValKind::BoolType:
- case ValKind::TypeType:
- case ValKind::FunctionType:
- case ValKind::PointerType:
- case ValKind::AutoType:
- case ValKind::StructType:
- case ValKind::ChoiceType:
- case ValKind::ContinuationType:
- return TypeEqual(v1, v2);
- case ValKind::StructValue:
- case ValKind::AlternativeValue:
- case ValKind::BindingPlaceholderValue:
- case ValKind::AlternativeConstructorValue:
- case ValKind::ContinuationValue:
- std::cerr << "ValueEqual does not support this kind of value."
- << std::endl;
- exit(-1);
- }
- }
- auto ToInteger(const Value* v) -> int {
- switch (v->tag()) {
- case ValKind::IntValue:
- return v->GetIntValue();
- default:
- std::cerr << "expected an integer, not ";
- PrintValue(v, std::cerr);
- exit(-1);
- }
- }
- } // namespace Carbon
|