// Copyright (C) 2020 Free Software Foundation, Inc.
// This file is part of GCC.
// GCC is free software; you can redistribute it and/or modify it under
// the terms of the GNU General Public License as published by the Free
// Software Foundation; either version 3, or (at your option) any later
// version.
// GCC is distributed in the hope that it will be useful, but WITHOUT ANY
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
// FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
// for more details.
// You should have received a copy of the GNU General Public License
// along with GCC; see the file COPYING3. If not see
// .
#ifndef RUST_COMPILE_EXPR
#define RUST_COMPILE_EXPR
#include "rust-compile-base.h"
#include "rust-compile-tyty.h"
#include "rust-compile-resolve-path.h"
namespace Rust {
namespace Compile {
class CompileExpr : public HIRCompileBase
{
public:
static Bexpression *Compile (HIR::Expr *expr, Context *ctx)
{
CompileExpr compiler (ctx);
expr->accept_vis (compiler);
return compiler.translated;
}
virtual ~CompileExpr () {}
void visit (HIR::ReturnExpr &expr)
{
Bexpression *compiled_expr
= CompileExpr::Compile (expr.return_expr.get (), ctx);
rust_assert (compiled_expr != nullptr);
auto fncontext = ctx->peek_fn ();
std::vector retstmts;
retstmts.push_back (compiled_expr);
auto s = ctx->get_backend ()->return_statement (fncontext.fndecl, retstmts,
expr.get_locus ());
ctx->add_statement (s);
}
void visit (HIR::CallExpr &expr)
{
Bexpression *fn = ResolvePath::Compile (expr.get_fnexpr (), ctx);
rust_assert (fn != nullptr);
std::vector args;
expr.iterate_params ([&] (HIR::Expr *p) mutable -> bool {
Bexpression *compiled_expr = CompileExpr::Compile (p, ctx);
rust_assert (compiled_expr != nullptr);
args.push_back (compiled_expr);
return true;
});
auto fncontext = ctx->peek_fn ();
translated
= ctx->get_backend ()->call_expression (fncontext.fndecl, fn, args,
nullptr, expr.get_locus ());
}
void visit (HIR::IdentifierExpr &expr)
{
// need to look up the reference for this identifier
NodeId ref_node_id;
if (!ctx->get_resolver ()->lookup_resolved_name (
expr.get_mappings ().get_nodeid (), &ref_node_id))
{
rust_fatal_error (expr.get_locus (), "failed to look up resolved name");
return;
}
printf ("have ast node id %u ref %u for expr [%s]\n",
expr.get_mappings ().get_nodeid (), ref_node_id,
expr.as_string ().c_str ());
// these ref_node_ids will resolve to a pattern declaration but we are
// interested in the definition that this refers to get the parent id
Resolver::Definition def;
if (!ctx->get_resolver ()->lookup_definition (ref_node_id, &def))
{
rust_error_at (expr.get_locus (), "unknown reference");
return;
}
HirId ref;
if (!ctx->get_mappings ()->lookup_node_to_hir (
expr.get_mappings ().get_crate_num (), def.parent, &ref))
{
rust_fatal_error (expr.get_locus (), "reverse lookup failure");
return;
}
Bvariable *var = nullptr;
if (!ctx->lookup_var_decl (ref, &var))
{
rust_fatal_error (expr.get_locus (),
"failed to lookup compiled variable");
return;
}
translated = ctx->get_backend ()->var_expression (var, expr.get_locus ());
}
void visit (HIR::LiteralExpr &expr)
{
switch (expr.get_lit_type ())
{
case HIR::Literal::BOOL: {
bool bval = expr.as_string ().compare ("true") == 0;
translated = ctx->get_backend ()->boolean_constant_expression (bval);
}
return;
case HIR::Literal::INT: {
mpz_t ival;
if (mpz_init_set_str (ival, expr.as_string ().c_str (), 10) != 0)
{
rust_fatal_error (expr.get_locus (), "bad number in literal");
return;
}
TyTy::TyBase *tyty = nullptr;
if (!ctx->get_tyctx ()->lookup_type (
expr.get_mappings ().get_hirid (), &tyty))
{
rust_fatal_error (expr.get_locus (),
"did not resolve type for this literal expr");
return;
}
Btype *type = TyTyResolveCompile::compile (ctx, tyty);
translated
= ctx->get_backend ()->integer_constant_expression (type, ival);
}
return;
default:
rust_fatal_error (expr.get_locus (), "unknown literal");
return;
}
gcc_unreachable ();
}
void visit (HIR::AssignmentExpr &expr)
{
fncontext fn = ctx->peek_fn ();
auto lhs = CompileExpr::Compile (expr.get_lhs (), ctx);
auto rhs = CompileExpr::Compile (expr.get_rhs (), ctx);
Bstatement *assignment
= ctx->get_backend ()->assignment_statement (fn.fndecl, lhs, rhs,
expr.get_locus ());
ctx->add_statement (assignment);
}
void visit (HIR::ArithmeticOrLogicalExpr &expr)
{
Operator op;
switch (expr.get_expr_type ())
{
case HIR::ArithmeticOrLogicalExpr::ADD:
op = OPERATOR_PLUS;
break;
case HIR::ArithmeticOrLogicalExpr::SUBTRACT:
op = OPERATOR_MINUS;
break;
case HIR::ArithmeticOrLogicalExpr::MULTIPLY:
op = OPERATOR_MULT;
break;
case HIR::ArithmeticOrLogicalExpr::DIVIDE:
op = OPERATOR_DIV;
break;
case HIR::ArithmeticOrLogicalExpr::MODULUS:
op = OPERATOR_MOD;
break;
case HIR::ArithmeticOrLogicalExpr::BITWISE_AND:
op = OPERATOR_AND;
break;
case HIR::ArithmeticOrLogicalExpr::BITWISE_OR:
op = OPERATOR_OR;
break;
case HIR::ArithmeticOrLogicalExpr::BITWISE_XOR:
op = OPERATOR_XOR;
break;
case HIR::ArithmeticOrLogicalExpr::LEFT_SHIFT:
op = OPERATOR_LSHIFT;
break;
case HIR::ArithmeticOrLogicalExpr::RIGHT_SHIFT:
op = OPERATOR_RSHIFT;
break;
default:
rust_fatal_error (expr.get_locus (), "failed to compile operator");
return;
}
auto lhs = CompileExpr::Compile (expr.get_lhs (), ctx);
auto rhs = CompileExpr::Compile (expr.get_rhs (), ctx);
translated = ctx->get_backend ()->binary_expression (op, lhs, rhs,
expr.get_locus ());
}
void visit (HIR::ComparisonExpr &expr)
{
Operator op;
switch (expr.get_expr_type ())
{
case HIR::ComparisonExpr::EQUAL:
op = OPERATOR_EQEQ;
break;
case HIR::ComparisonExpr::NOT_EQUAL:
op = OPERATOR_NOTEQ;
break;
case HIR::ComparisonExpr::GREATER_THAN:
op = OPERATOR_GT;
break;
case HIR::ComparisonExpr::LESS_THAN:
op = OPERATOR_LT;
break;
case HIR::ComparisonExpr::GREATER_OR_EQUAL:
op = OPERATOR_GE;
break;
case HIR::ComparisonExpr::LESS_OR_EQUAL:
op = OPERATOR_LE;
break;
default:
rust_fatal_error (expr.get_locus (), "failed to compile operator");
return;
}
auto lhs = CompileExpr::Compile (expr.get_lhs (), ctx);
auto rhs = CompileExpr::Compile (expr.get_rhs (), ctx);
translated = ctx->get_backend ()->binary_expression (op, lhs, rhs,
expr.get_locus ());
}
void visit (HIR::LazyBooleanExpr &expr)
{
Operator op;
switch (expr.get_expr_type ())
{
case HIR::LazyBooleanExpr::LOGICAL_OR:
op = OPERATOR_OROR;
break;
case HIR::LazyBooleanExpr::LOGICAL_AND:
op = OPERATOR_ANDAND;
break;
default:
rust_fatal_error (expr.get_locus (), "failed to compile operator");
return;
}
auto lhs = CompileExpr::Compile (expr.get_lhs (), ctx);
auto rhs = CompileExpr::Compile (expr.get_rhs (), ctx);
translated = ctx->get_backend ()->binary_expression (op, lhs, rhs,
expr.get_locus ());
}
private:
CompileExpr (Context *ctx) : HIRCompileBase (ctx), translated (nullptr) {}
Bexpression *translated;
};
} // namespace Compile
} // namespace Rust
#endif // RUST_COMPILE_EXPR