| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340 |
- // 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/ast/expression.h"
- #include <iostream>
- namespace Carbon {
- auto Expression::GetVariable() const -> const Variable& {
- return std::get<Variable>(value);
- }
- auto Expression::GetFieldAccess() const -> const FieldAccess& {
- return std::get<FieldAccess>(value);
- }
- auto Expression::GetIndex() const -> const Index& {
- return std::get<Index>(value);
- }
- auto Expression::GetPatternVariable() const -> const PatternVariable& {
- return std::get<PatternVariable>(value);
- }
- auto Expression::GetInteger() const -> int {
- return std::get<IntLiteral>(value).value;
- }
- auto Expression::GetBoolean() const -> bool {
- return std::get<BoolLiteral>(value).value;
- }
- auto Expression::GetTuple() const -> const Tuple& {
- return std::get<Tuple>(value);
- }
- auto Expression::GetPrimitiveOperator() const -> const PrimitiveOperator& {
- return std::get<PrimitiveOperator>(value);
- }
- auto Expression::GetCall() const -> const Call& {
- return std::get<Call>(value);
- }
- auto Expression::GetFunctionType() const -> const FunctionType& {
- return std::get<FunctionType>(value);
- }
- auto Expression::MakeTypeType(int line_num) -> const Expression* {
- auto* t = new Expression();
- t->line_num = line_num;
- t->value = TypeT();
- return t;
- }
- auto Expression::MakeIntType(int line_num) -> const Expression* {
- auto* t = new Expression();
- t->line_num = line_num;
- t->value = IntT();
- return t;
- }
- auto Expression::MakeBoolType(int line_num) -> const Expression* {
- auto* t = new Expression();
- t->line_num = line_num;
- t->value = BoolT();
- return t;
- }
- auto Expression::MakeAutoType(int line_num) -> const Expression* {
- auto* t = new Expression();
- t->line_num = line_num;
- t->value = AutoT();
- return t;
- }
- // Returns a Continuation type AST node at the given source location.
- auto Expression::MakeContinuationType(int line_num) -> const Expression* {
- auto* type = new Expression();
- type->line_num = line_num;
- type->value = ContinuationT();
- return type;
- }
- auto Expression::MakeFunType(int line_num, const Expression* param,
- const Expression* ret) -> const Expression* {
- auto* t = new Expression();
- t->line_num = line_num;
- t->value = FunctionType({.parameter = param, .return_type = ret});
- return t;
- }
- auto Expression::MakeVar(int line_num, std::string var) -> const Expression* {
- auto* v = new Expression();
- v->line_num = line_num;
- v->value = Variable({.name = std::move(var)});
- return v;
- }
- auto Expression::MakeVarPat(int line_num, std::string var,
- const Expression* type) -> const Expression* {
- auto* v = new Expression();
- v->line_num = line_num;
- v->value = PatternVariable({.name = std::move(var), .type = type});
- return v;
- }
- auto Expression::MakeInt(int line_num, int i) -> const Expression* {
- auto* e = new Expression();
- e->line_num = line_num;
- e->value = IntLiteral({.value = i});
- return e;
- }
- auto Expression::MakeBool(int line_num, bool b) -> const Expression* {
- auto* e = new Expression();
- e->line_num = line_num;
- e->value = BoolLiteral({.value = b});
- return e;
- }
- auto Expression::MakeOp(int line_num, enum Operator op,
- std::vector<const Expression*>* args)
- -> const Expression* {
- auto* e = new Expression();
- e->line_num = line_num;
- e->value = PrimitiveOperator({.op = op, .arguments = args});
- return e;
- }
- auto Expression::MakeUnOp(int line_num, enum Operator op, const Expression* arg)
- -> const Expression* {
- auto* e = new Expression();
- e->line_num = line_num;
- e->value = PrimitiveOperator(
- {.op = op, .arguments = new std::vector<const Expression*>{arg}});
- return e;
- }
- auto Expression::MakeBinOp(int line_num, enum Operator op,
- const Expression* arg1, const Expression* arg2)
- -> const Expression* {
- auto* e = new Expression();
- e->line_num = line_num;
- e->value = PrimitiveOperator(
- {.op = op, .arguments = new std::vector<const Expression*>{arg1, arg2}});
- return e;
- }
- auto Expression::MakeCall(int line_num, const Expression* fun,
- const Expression* arg) -> const Expression* {
- auto* e = new Expression();
- e->line_num = line_num;
- e->value = Call({.function = fun, .argument = arg});
- return e;
- }
- auto Expression::MakeGetField(int line_num, const Expression* exp,
- std::string field) -> const Expression* {
- auto* e = new Expression();
- e->line_num = line_num;
- e->value = FieldAccess({.aggregate = *exp, .field = std::move(field)});
- return e;
- }
- auto Expression::MakeTuple(int line_num, std::vector<FieldInitializer>* args)
- -> const Expression* {
- auto* e = new Expression();
- e->line_num = line_num;
- int i = 0;
- bool seen_named_member = false;
- for (auto& arg : *args) {
- if (arg.name == "") {
- if (seen_named_member) {
- std::cerr << line_num
- << ": positional members must come before named members"
- << std::endl;
- exit(-1);
- }
- arg.name = std::to_string(i);
- ++i;
- } else {
- seen_named_member = true;
- }
- }
- e->value = Tuple({.fields = args});
- return e;
- }
- // Create an AST node for an empty tuple.
- // TODO(geoffromer): remove this and rewrite its callers to use
- // `MakeTuple(line_num, {})`, once that works.
- auto Expression::MakeUnit(int line_num) -> const Expression* {
- auto* unit = new Expression();
- unit->line_num = line_num;
- auto* args = new std::vector<FieldInitializer>();
- unit->value = Tuple({.fields = args});
- return unit;
- }
- auto Expression::MakeIndex(int line_num, const Expression* exp,
- const Expression* i) -> const Expression* {
- auto* e = new Expression();
- e->line_num = line_num;
- e->value = Index({.aggregate = exp, .offset = i});
- return e;
- }
- static void PrintOp(Operator op) {
- switch (op) {
- case Operator::Add:
- std::cout << "+";
- break;
- case Operator::Neg:
- case Operator::Sub:
- std::cout << "-";
- break;
- case Operator::Mul:
- case Operator::Deref:
- case Operator::Ptr:
- std::cout << "*";
- break;
- case Operator::Not:
- std::cout << "not";
- break;
- case Operator::And:
- std::cout << "and";
- break;
- case Operator::Or:
- std::cout << "or";
- break;
- case Operator::Eq:
- std::cout << "==";
- break;
- }
- }
- static void PrintFields(std::vector<FieldInitializer>* fields) {
- int i = 0;
- for (auto iter = fields->begin(); iter != fields->end(); ++iter, ++i) {
- if (i != 0) {
- std::cout << ", ";
- }
- std::cout << iter->name << " = ";
- PrintExp(iter->expression);
- }
- }
- void PrintExp(const Expression* e) {
- switch (e->tag()) {
- case ExpressionKind::Index:
- PrintExp(e->GetIndex().aggregate);
- std::cout << "[";
- PrintExp(e->GetIndex().offset);
- std::cout << "]";
- break;
- case ExpressionKind::GetField:
- PrintExp(e->GetFieldAccess().aggregate.GetPointer());
- std::cout << ".";
- std::cout << e->GetFieldAccess().field;
- break;
- case ExpressionKind::Tuple:
- std::cout << "(";
- PrintFields(e->GetTuple().fields);
- std::cout << ")";
- break;
- case ExpressionKind::Integer:
- std::cout << e->GetInteger();
- break;
- case ExpressionKind::Boolean:
- std::cout << std::boolalpha;
- std::cout << e->GetBoolean();
- break;
- case ExpressionKind::PrimitiveOp: {
- std::cout << "(";
- PrimitiveOperator op = e->GetPrimitiveOperator();
- if (op.arguments->size() == 0) {
- PrintOp(op.op);
- } else if (op.arguments->size() == 1) {
- PrintOp(op.op);
- std::cout << " ";
- auto iter = op.arguments->begin();
- PrintExp(*iter);
- } else if (op.arguments->size() == 2) {
- auto iter = op.arguments->begin();
- PrintExp(*iter);
- std::cout << " ";
- PrintOp(op.op);
- std::cout << " ";
- ++iter;
- PrintExp(*iter);
- }
- std::cout << ")";
- break;
- }
- case ExpressionKind::Variable:
- std::cout << e->GetVariable().name;
- break;
- case ExpressionKind::PatternVariable:
- PrintExp(e->GetPatternVariable().type);
- std::cout << ": ";
- std::cout << e->GetPatternVariable().name;
- break;
- case ExpressionKind::Call:
- PrintExp(e->GetCall().function);
- if (e->GetCall().argument->tag() == ExpressionKind::Tuple) {
- PrintExp(e->GetCall().argument);
- } else {
- std::cout << "(";
- PrintExp(e->GetCall().argument);
- std::cout << ")";
- }
- break;
- case ExpressionKind::BoolT:
- std::cout << "Bool";
- break;
- case ExpressionKind::IntT:
- std::cout << "Int";
- break;
- case ExpressionKind::TypeT:
- std::cout << "Type";
- break;
- case ExpressionKind::AutoT:
- std::cout << "auto";
- break;
- case ExpressionKind::ContinuationT:
- std::cout << "Continuation";
- break;
- case ExpressionKind::FunctionT:
- std::cout << "fn ";
- PrintExp(e->GetFunctionType().parameter);
- std::cout << " -> ";
- PrintExp(e->GetFunctionType().return_type);
- break;
- }
- }
- } // namespace Carbon
|