typedef struct Value { union { Integer intv; Floating floatv; struct Value *points_to; FnExpr fn; bool boolv; }; } Value; static bool eval_truthiness(Expression *e, Value *v) { /* TODO */ (void)e,(void)v; return true; } /* NOTE: expr must be typed before it can be evaluated */ static bool eval_expr(Expression *e, Value *v) { /* TODO: cache eval'd expression values (probably only needed for declarations) */ switch (e->kind) { case EXPR_LITERAL_FLOAT: v->floatv = e->floatl; return false; case EXPR_LITERAL_INT: if (e->intl > (UInteger)INTEGER_MAX) { /* TODO: FIXME */ err_print(e->where, "Overflow when evaluating integer."); return false; } v->intv = (Integer)e->intl; return true; case EXPR_LITERAL_STR: err_print(e->where, "not implemented yet"); /* TODO */ return false; case EXPR_LITERAL_BOOL: v->boolv = e->booll; return true; case EXPR_UNARY_OP: { Expression *of_expr = e->unary.of; switch (e->unary.op) { case UNARY_MINUS: { Value of; if (!eval_expr(of_expr, &of)) return false; assert(e->type.kind != TYPE_BUILTIN); if (type_builtin_is_integer(e->type.builtin)) { v->intv = -of.intv; } else if (type_builtin_is_floating(e->type.builtin)) { v->floatv = -of.floatv; } else { err_print(e->where, "negation of non-numerical types not supported in evaluator yet."); return false; } return true; } case UNARY_NOT: v->boolv = !eval_truthiness(e, v); return true; case UNARY_ADDRESS: v->points_to = err_malloc(sizeof *v->points_to); /* OPTIM */ return eval_expr(e->unary.of, v->points_to); case UNARY_DEREF: { Value ptr; if (!eval_expr(of_expr, &ptr)) return false; *v = *ptr.points_to; return true; } break; } } break; case EXPR_BINARY_OP: { Value lhs, rhs; /* NOTE: this will need to change for short-circuiting */ if (!eval_expr(e->binary.lhs, &lhs)) return false; if (!eval_expr(e->binary.rhs, &rhs)) return false; if (e->type.kind != TYPE_BUILTIN) { err_print(e->where, "Operators can only be applied to builtin types."); return false; } bool is_int = type_builtin_is_integer(e->type.builtin); bool is_float = type_builtin_is_floating(e->type.builtin); bool is_bool = e->type.builtin == BUILTIN_BOOL; switch (e->binary.op) { case BINARY_PLUS: if (is_int) { v->intv = lhs.intv + rhs.intv; } else if (is_float) { v->floatv = lhs.floatv + rhs.floatv; } else assert(0); return true; case BINARY_MINUS: if (is_int) { v->intv = lhs.intv - rhs.intv; } else if (is_float) { v->floatv = lhs.floatv - rhs.floatv; } else assert(0); return true; case BINARY_MUL: if (is_int) { v->intv = lhs.intv * rhs.intv; } else if (is_float) { v->floatv = lhs.floatv * rhs.floatv; } else assert(0); return true; case BINARY_DIV: /* TODO(eventually): check div by 0 */ if (is_int) { v->intv = lhs.intv / rhs.intv; } else if (is_float) { v->floatv = lhs.floatv / rhs.floatv; } else assert(0); return true; case BINARY_EQ: if (is_int) { v->boolv = lhs.intv == rhs.intv; } else if (is_float) { v->boolv = lhs.floatv == rhs.floatv; } else if (is_bool) { v->boolv = lhs.boolv == rhs.boolv; } else assert(0); return true; case BINARY_NE: if (is_int) { v->boolv = lhs.intv != rhs.intv; } else if (is_float) { v->boolv = lhs.floatv != rhs.floatv; } else if (is_bool) { v->boolv = lhs.boolv != rhs.boolv; } else assert(0); return true; case BINARY_GT: if (is_int) { v->boolv = lhs.intv > rhs.intv; } else if (is_float) { v->boolv = lhs.floatv > rhs.floatv; } else assert(0); return true; case BINARY_GE: if (is_int) { v->boolv = lhs.intv >= rhs.intv; } else if (is_float) { v->boolv = lhs.floatv >= rhs.floatv; } else assert(0); return true; case BINARY_LT: if (is_int) { v->boolv = lhs.intv < rhs.intv; } else if (is_float) { v->boolv = lhs.floatv < rhs.floatv; } else assert(0); return true; case BINARY_LE: if (is_int) { v->boolv = lhs.intv <= rhs.intv; } else if (is_float) { v->boolv = lhs.floatv <= rhs.floatv; } else assert(0); return true; case BINARY_SET: return true; case BINARY_COMMA: err_print(e->where, "tuples not supported at compile time yet."); return false; case BINARY_AT_INDEX: err_print(e->where, "Cannot get index of array at compile time yet."); return false; } } break; case EXPR_IDENT: { Identifier id = e->ident; IdentDecl *id_decl = ident_decl(id); if (!id_decl) { char *id_str = ident_to_str(id); err_print(e->where, "Undeclared identifier: %s", id_str); free(id_str); return false; } Declaration *d = id_decl->decl; if (location_after(d->where, e->where)) { err_print(e->where, "Use of constant before its declaration."); info_print(d->where, "Declaration will be here."); return false; } if (!(d->flags & DECL_FLAG_CONST)) { err_print(e->where, "Use of non-constant identifier in a constant expression."); info_print(d->where, "Declaration was here."); return false; } /* TODO: tuples */ if (!d->val) { d->val = err_malloc(sizeof *d->val); /* OPTIM */ if (!eval_expr(&d->expr, d->val)) return false; } *v = *d->val; return true; } break; case EXPR_FN: v->fn = e->fn; return true; case EXPR_CAST: case EXPR_IF: case EXPR_WHILE: case EXPR_CALL: case EXPR_BLOCK: { err_print(e->where, "operation not supported at compile time yet."); /* TODO */ } break; case EXPR_DIRECT: switch (e->direct.which) { case DIRECT_C: err_print(e->where, "Can't run C code at compile time."); return false; case DIRECT_COUNT: assert(0); break; } break; } err_print(e->where, "Not implemented yet"); return false; }