jak-project/goalc/compiler/compilation/Math.cpp
2020-09-13 12:11:49 -04:00

235 lines
7 KiB
C++

#include "goalc/compiler/Compiler.h"
MathMode Compiler::get_math_mode(const TypeSpec& ts) {
if (m_ts.typecheck(m_ts.make_typespec("binteger"), ts, "", false, false)) {
return MATH_BINT;
}
if (m_ts.typecheck(m_ts.make_typespec("integer"), ts, "", false, false)) {
return MATH_INT;
}
if (m_ts.typecheck(m_ts.make_typespec("float"), ts, "", false, false)) {
return MATH_FLOAT;
}
return MATH_INVALID;
}
bool Compiler::is_number(const TypeSpec& ts) {
return m_ts.typecheck(m_ts.make_typespec("number"), ts, "", false, false);
}
bool Compiler::is_float(const TypeSpec& ts) {
return m_ts.typecheck(m_ts.make_typespec("float"), ts, "", false, false);
}
bool Compiler::is_integer(const TypeSpec& ts) {
return m_ts.typecheck(m_ts.make_typespec("integer"), ts, "", false, false) &&
!m_ts.typecheck(m_ts.make_typespec("binteger"), ts, "", false, false);
}
bool Compiler::is_binteger(const TypeSpec& ts) {
return m_ts.typecheck(m_ts.make_typespec("binteger"), ts, "", false, false);
}
bool Compiler::is_singed_integer_or_binteger(const TypeSpec& ts) {
return m_ts.typecheck(m_ts.make_typespec("integer"), ts, "", false, false) &&
!m_ts.typecheck(m_ts.make_typespec("uinteger"), ts, "", false, false);
}
Val* Compiler::number_to_integer(Val* in, Env* env) {
(void)env;
auto ts = in->type();
if (is_binteger(ts)) {
assert(false);
} else if (is_float(ts)) {
assert(false);
} else if (is_integer(ts)) {
return in;
} else {
assert(false);
}
}
Val* Compiler::number_to_binteger(Val* in, Env* env) {
(void)env;
auto ts = in->type();
if (is_binteger(ts)) {
return in;
} else if (is_float(ts)) {
assert(false);
} else if (is_integer(ts)) {
assert(false);
} else {
assert(false);
}
}
Val* Compiler::number_to_float(Val* in, Env* env) {
(void)env;
auto ts = in->type();
if (is_binteger(ts)) {
assert(false);
} else if (is_float(ts)) {
return in;
} else if (is_integer(ts)) {
assert(false);
} else {
assert(false);
}
}
Val* Compiler::to_math_type(Val* in, MathMode mode, Env* env) {
switch (mode) {
case MATH_BINT:
return number_to_binteger(in, env);
case MATH_INT:
return number_to_integer(in, env);
case MATH_FLOAT:
return number_to_float(in, env);
default:
assert(false);
}
}
Val* Compiler::compile_add(const goos::Object& form, const goos::Object& rest, Env* env) {
auto args = get_va(form, rest);
if (!args.named.empty() || args.unnamed.empty()) {
throw_compile_error(form, "Invalid + form");
}
// look at the first value to determine the math mode
auto first_val = compile_error_guard(args.unnamed.at(0), env);
auto first_type = first_val->type();
auto math_type = get_math_mode(first_type);
switch (math_type) {
case MATH_INT: {
auto result = env->make_gpr(first_type);
env->emit(std::make_unique<IR_RegSet>(result, first_val->to_gpr(env)));
for (size_t i = 1; i < args.unnamed.size(); i++) {
env->emit(std::make_unique<IR_IntegerMath>(
IntegerMathKind::ADD_64, result,
to_math_type(compile_error_guard(args.unnamed.at(i), env), math_type, env)
->to_gpr(env)));
}
return result;
}
case MATH_INVALID:
throw_compile_error(
form, "Cannot determine the math mode for object of type " + first_type.print());
break;
default:
assert(false);
}
assert(false);
return get_none();
}
Val* Compiler::compile_mul(const goos::Object& form, const goos::Object& rest, Env* env) {
auto args = get_va(form, rest);
if (!args.named.empty() || args.unnamed.empty()) {
throw_compile_error(form, "Invalid * form");
}
// look at the first value to determine the math mode
auto first_val = compile_error_guard(args.unnamed.at(0), env);
auto first_type = first_val->type();
auto math_type = get_math_mode(first_type);
switch (math_type) {
case MATH_INT: {
auto result = env->make_gpr(first_type);
env->emit(std::make_unique<IR_RegSet>(result, first_val->to_gpr(env)));
for (size_t i = 1; i < args.unnamed.size(); i++) {
env->emit(std::make_unique<IR_IntegerMath>(
IntegerMathKind::IMUL_32, result,
to_math_type(compile_error_guard(args.unnamed.at(i), env), math_type, env)
->to_gpr(env)));
}
return result;
}
case MATH_INVALID:
throw_compile_error(
form, "Cannot determine the math mode for object of type " + first_type.print());
break;
default:
assert(false);
}
assert(false);
return get_none();
}
Val* Compiler::compile_sub(const goos::Object& form, const goos::Object& rest, Env* env) {
auto args = get_va(form, rest);
if (!args.named.empty() || args.unnamed.empty()) {
throw_compile_error(form, "Invalid - form");
}
auto first_val = compile_error_guard(args.unnamed.at(0), env);
auto first_type = first_val->type();
auto math_type = get_math_mode(first_type);
switch (math_type) {
case MATH_INT:
if (args.unnamed.size() == 1) {
auto result = compile_integer(0, env)->to_gpr(env);
env->emit(std::make_unique<IR_IntegerMath>(
IntegerMathKind::SUB_64, result,
to_math_type(compile_error_guard(args.unnamed.at(0), env), math_type, env)
->to_gpr(env)));
return result;
} else {
auto result = env->make_gpr(first_type);
env->emit(std::make_unique<IR_RegSet>(
result, to_math_type(compile_error_guard(args.unnamed.at(0), env), math_type, env)
->to_gpr(env)));
for (size_t i = 1; i < args.unnamed.size(); i++) {
env->emit(std::make_unique<IR_IntegerMath>(
IntegerMathKind::SUB_64, result,
to_math_type(compile_error_guard(args.unnamed.at(i), env), math_type, env)
->to_gpr(env)));
}
return result;
}
case MATH_INVALID:
throw_compile_error(
form, "Cannot determine the math mode for object of type " + first_type.print());
break;
default:
assert(false);
}
assert(false);
return get_none();
}
Val* Compiler::compile_div(const goos::Object& form, const goos::Object& rest, Env* env) {
auto args = get_va(form, rest);
if (!args.named.empty() || args.unnamed.size() != 2) {
throw_compile_error(form, "Invalid / form");
}
auto first_val = compile_error_guard(args.unnamed.at(0), env);
auto first_type = first_val->type();
auto math_type = get_math_mode(first_type);
switch (math_type) {
case MATH_FLOAT:
{
auto result = env->make_xmm(first_type);
env->emit(std::make_unique<IR_RegSet>(result, first_val->to_xmm(env)));
env->emit(std::make_unique<IR_FloatMath>(FloatMathKind::DIV_SS, result, to_math_type(compile_error_guard(args.unnamed.at(1), env), math_type, env)->to_xmm(env))));
}
case MATH_INVALID:
throw_compile_error(
form, "Cannot determine the math mode for object of type " + first_type.print());
break;
default:
assert(false);
}
assert(false);
return get_none();
}