2024-02-04 12:23:59 -08:00
|
|
|
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <gc/cord.h>
|
|
|
|
#include <gc.h>
|
|
|
|
#include <stdio.h>
|
2024-03-03 16:12:53 -08:00
|
|
|
#include <uninorm.h>
|
2024-02-04 12:23:59 -08:00
|
|
|
|
|
|
|
#include "ast.h"
|
2024-03-03 15:15:45 -08:00
|
|
|
#include "builtins/text.h"
|
2024-02-04 15:04:41 -08:00
|
|
|
#include "compile.h"
|
2024-02-24 12:24:44 -08:00
|
|
|
#include "enums.h"
|
2024-02-24 10:27:49 -08:00
|
|
|
#include "structs.h"
|
2024-02-17 13:56:19 -08:00
|
|
|
#include "environment.h"
|
2024-02-17 14:00:21 -08:00
|
|
|
#include "typecheck.h"
|
2024-02-04 12:23:59 -08:00
|
|
|
#include "util.h"
|
|
|
|
|
2024-02-17 23:34:39 -08:00
|
|
|
CORD compile_type_ast(type_ast_t *t)
|
2024-02-04 12:23:59 -08:00
|
|
|
{
|
|
|
|
switch (t->tag) {
|
2024-02-07 21:52:18 -08:00
|
|
|
case VarTypeAST: return CORD_cat(Match(t, VarTypeAST)->name, "_t");
|
2024-02-17 23:34:39 -08:00
|
|
|
case PointerTypeAST: return CORD_cat(compile_type_ast(Match(t, PointerTypeAST)->pointed), "*");
|
|
|
|
case TableTypeAST: return "table_t";
|
|
|
|
case ArrayTypeAST: return "array_t";
|
|
|
|
case FunctionTypeAST: return "const void*";
|
2024-02-17 13:56:19 -08:00
|
|
|
default: code_err(t, "Not implemented");
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-17 23:34:39 -08:00
|
|
|
CORD compile_type(type_t *t)
|
|
|
|
{
|
|
|
|
switch (t->tag) {
|
|
|
|
case AbortType: return "void";
|
|
|
|
case VoidType: return "void";
|
|
|
|
case MemoryType: return "void";
|
|
|
|
case BoolType: return "Bool_t";
|
2024-03-03 10:37:05 -08:00
|
|
|
case IntType: return Match(t, IntType)->bits == 64 ? "Int_t" : CORD_asprintf("Int%ld_t", Match(t, IntType)->bits);
|
|
|
|
case NumType: return Match(t, NumType)->bits == 64 ? "Num_t" : CORD_asprintf("Num%ld_t", Match(t, NumType)->bits);
|
2024-03-03 15:15:45 -08:00
|
|
|
case TextType: {
|
|
|
|
const char *dsl = Match(t, TextType)->dsl;
|
|
|
|
return dsl ? CORD_cat(dsl, "_t") : "Text_t";
|
2024-02-17 23:34:39 -08:00
|
|
|
}
|
|
|
|
case ArrayType: return "array_t";
|
|
|
|
case TableType: return "table_t";
|
|
|
|
case FunctionType: return "const void*";
|
|
|
|
case ClosureType: compiler_err(NULL, NULL, NULL, "Not implemented");
|
|
|
|
case PointerType: return CORD_cat(compile_type(Match(t, PointerType)->pointed), "*");
|
|
|
|
case StructType: return CORD_cat(Match(t, StructType)->name, "_t");
|
|
|
|
case EnumType: return CORD_cat(Match(t, EnumType)->name, "_t");
|
|
|
|
case TypeInfoType: return "TypeInfo";
|
|
|
|
default: compiler_err(NULL, NULL, NULL, "Not implemented");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-13 11:42:33 -08:00
|
|
|
CORD compile_statement(env_t *env, ast_t *ast)
|
2024-02-04 15:04:41 -08:00
|
|
|
{
|
2024-02-13 21:19:16 -08:00
|
|
|
CORD stmt;
|
2024-02-04 15:04:41 -08:00
|
|
|
switch (ast->tag) {
|
2024-02-22 09:45:12 -08:00
|
|
|
case If: case When: case For: case While: case FunctionDef: case Return: case StructDef: case EnumDef:
|
2024-02-25 12:28:46 -08:00
|
|
|
case Declare: case Assign: case UpdateAssign: case DocTest: case Block:
|
2024-02-13 21:19:16 -08:00
|
|
|
stmt = compile(env, ast);
|
|
|
|
break;
|
2024-02-04 15:04:41 -08:00
|
|
|
default:
|
2024-02-13 21:19:16 -08:00
|
|
|
stmt = CORD_asprintf("(void)%r;", compile(env, ast));
|
|
|
|
break;
|
2024-02-04 15:04:41 -08:00
|
|
|
}
|
2024-02-17 20:43:55 -08:00
|
|
|
// int64_t line = get_line_number(ast->file, ast->start);
|
|
|
|
// return stmt ? CORD_asprintf("#line %ld\n%r", line, stmt) : stmt;
|
|
|
|
return stmt;
|
2024-02-04 15:04:41 -08:00
|
|
|
}
|
|
|
|
|
2024-03-03 15:16:33 -08:00
|
|
|
CORD expr_as_texting(env_t *env, CORD expr, type_t *t, CORD color)
|
2024-02-17 17:47:43 -08:00
|
|
|
{
|
|
|
|
switch (t->tag) {
|
2024-03-03 15:16:33 -08:00
|
|
|
case MemoryType: return CORD_asprintf("Memory__as_text($stack(%r), %r, &Memory)", expr, color);
|
|
|
|
case BoolType: return CORD_asprintf("Bool__as_text($stack(%r), %r, &Bool)", expr, color);
|
2024-03-03 10:37:05 -08:00
|
|
|
case IntType: {
|
|
|
|
CORD name = type_to_cord(t);
|
2024-03-03 15:16:33 -08:00
|
|
|
return CORD_asprintf("%r__as_text($stack(%r), %r, &%r)", name, expr, color, name);
|
2024-03-03 10:37:05 -08:00
|
|
|
}
|
|
|
|
case NumType: {
|
|
|
|
CORD name = type_to_cord(t);
|
2024-03-03 15:16:33 -08:00
|
|
|
return CORD_asprintf("%r__as_text($stack(%r), %r, &Num%r)", name, expr, color, name);
|
|
|
|
}
|
|
|
|
case TextType: return CORD_asprintf("Text__as_text($stack(%r), %r, &Text)", expr, color);
|
|
|
|
case ArrayType: return CORD_asprintf("Array__as_text($stack(%r), %r, %r)", expr, color, compile_type_info(env, t));
|
|
|
|
case TableType: return CORD_asprintf("Table_as_text($stack(%r), %r, %r)", expr, color, compile_type_info(env, t));
|
|
|
|
case FunctionType: return CORD_asprintf("Func__as_text($stack(%r), %r, %r)", expr, color, compile_type_info(env, t));
|
|
|
|
case PointerType: return CORD_asprintf("Pointer__as_text($stack(%r), %r, %r)", expr, color, compile_type_info(env, t));
|
|
|
|
case StructType: case EnumType: return CORD_asprintf("(%r)->CustomInfo.as_text($stack(%r), %r, %r)",
|
2024-02-17 18:04:35 -08:00
|
|
|
compile_type_info(env, t), expr, color, compile_type_info(env, t));
|
2024-02-17 17:47:43 -08:00
|
|
|
default: compiler_err(NULL, NULL, NULL, "Stringifying is not supported for %T", t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
CORD compile_string(env_t *env, ast_t *ast, CORD color)
|
|
|
|
{
|
|
|
|
type_t *t = get_type(env, ast);
|
|
|
|
CORD expr = compile(env, ast);
|
2024-03-03 15:16:33 -08:00
|
|
|
return expr_as_texting(env, expr, t, color);
|
2024-02-17 17:47:43 -08:00
|
|
|
}
|
|
|
|
|
2024-02-20 10:06:03 -08:00
|
|
|
static CORD compile_to_pointer_depth(env_t *env, ast_t *ast, int64_t target_depth, bool allow_optional)
|
2024-02-18 11:28:35 -08:00
|
|
|
{
|
|
|
|
CORD val = compile(env, ast);
|
|
|
|
type_t *t = get_type(env, ast);
|
|
|
|
int64_t depth = 0;
|
|
|
|
for (type_t *tt = t; tt->tag == PointerType; tt = Match(tt, PointerType)->pointed)
|
|
|
|
++depth;
|
|
|
|
|
|
|
|
while (depth != target_depth) {
|
|
|
|
if (depth < target_depth) {
|
|
|
|
if (ast->tag == Var && target_depth == 1)
|
2024-03-03 11:34:13 -08:00
|
|
|
val = CORD_all("(&", val, ")");
|
2024-02-18 11:28:35 -08:00
|
|
|
else
|
|
|
|
val = CORD_all("$stack(", val, ")");
|
|
|
|
t = Type(PointerType, .pointed=t, .is_stack=true);
|
|
|
|
++depth;
|
|
|
|
} else {
|
|
|
|
auto ptr = Match(t, PointerType);
|
|
|
|
if (ptr->is_optional)
|
|
|
|
code_err(ast, "You can't dereference this value, since it's not guaranteed to be non-null");
|
|
|
|
val = CORD_all("*(", val, ")");
|
|
|
|
t = ptr->pointed;
|
|
|
|
--depth;
|
|
|
|
}
|
|
|
|
}
|
2024-02-20 10:06:03 -08:00
|
|
|
if (!allow_optional) {
|
|
|
|
while (t->tag == PointerType) {
|
|
|
|
auto ptr = Match(t, PointerType);
|
|
|
|
if (ptr->is_optional)
|
|
|
|
code_err(ast, "You can't dereference this value, since it's not guaranteed to be non-null");
|
|
|
|
t = ptr->pointed;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-18 11:28:35 -08:00
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2024-02-25 10:13:36 -08:00
|
|
|
static void check_assignable(env_t *env, ast_t *ast)
|
|
|
|
{
|
|
|
|
if (!can_be_mutated(env, ast)) {
|
|
|
|
if (ast->tag == Index || ast->tag == FieldAccess) {
|
|
|
|
ast_t *subject = ast->tag == Index ? Match(ast, Index)->indexed : Match(ast, FieldAccess)->fielded;
|
|
|
|
code_err(subject, "This is a readonly pointer, which can't be assigned to");
|
|
|
|
} else {
|
|
|
|
code_err(ast, "This is a value of type %T and can't be assigned to", get_type(env, ast));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-08 11:23:16 -08:00
|
|
|
static CORD compile_arguments(env_t *env, ast_t *call_ast, arg_t *spec_args, arg_ast_t *call_args)
|
|
|
|
{
|
|
|
|
table_t used_args = {};
|
|
|
|
CORD code = CORD_EMPTY;
|
|
|
|
env_t *default_scope = fresh_scope(env);
|
|
|
|
default_scope->locals->fallback = env->globals;
|
|
|
|
for (arg_t *spec_arg = spec_args; spec_arg; spec_arg = spec_arg->next) {
|
|
|
|
// Find keyword:
|
|
|
|
if (spec_arg->name) {
|
|
|
|
for (arg_ast_t *call_arg = call_args; call_arg; call_arg = call_arg->next) {
|
|
|
|
if (call_arg->name && streq(call_arg->name, spec_arg->name)) {
|
|
|
|
type_t *actual_t = get_type(env, call_arg->value);
|
|
|
|
if (!can_promote(actual_t, spec_arg->type))
|
|
|
|
code_err(call_arg->value, "This argument is supposed to be a %T, but this value is a %T", spec_arg->type, actual_t);
|
|
|
|
Table_str_set(&used_args, call_arg->name, call_arg);
|
|
|
|
if (code) code = CORD_cat(code, ", ");
|
|
|
|
code = CORD_cat(code, compile(env, call_arg->value));
|
|
|
|
goto found_it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Find positional:
|
|
|
|
int64_t i = 1;
|
|
|
|
for (arg_ast_t *call_arg = call_args; call_arg; call_arg = call_arg->next) {
|
|
|
|
if (call_arg->name) continue;
|
|
|
|
const char *pseudoname = heap_strf("%ld", i++);
|
|
|
|
if (!Table_str_get(&used_args, pseudoname)) {
|
|
|
|
type_t *actual_t = get_type(env, call_arg->value);
|
|
|
|
if (!can_promote(actual_t, spec_arg->type))
|
|
|
|
code_err(call_arg->value, "This argument is supposed to be a %T, but this value is a %T", spec_arg->type, actual_t);
|
|
|
|
Table_str_set(&used_args, pseudoname, call_arg);
|
|
|
|
if (code) code = CORD_cat(code, ", ");
|
|
|
|
code = CORD_cat(code, compile(env, call_arg->value));
|
|
|
|
goto found_it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (spec_arg->default_val) {
|
|
|
|
if (code) code = CORD_cat(code, ", ");
|
|
|
|
code = CORD_cat(code, compile(default_scope, spec_arg->default_val));
|
|
|
|
goto found_it;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(spec_arg->name);
|
|
|
|
code_err(call_ast, "The required argument '%s' was not provided", spec_arg->name);
|
|
|
|
found_it: continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
int64_t i = 1;
|
|
|
|
for (arg_ast_t *call_arg = call_args; call_arg; call_arg = call_arg->next) {
|
|
|
|
if (call_arg->name) {
|
|
|
|
if (!Table_str_get(&used_args, call_arg->name))
|
|
|
|
code_err(call_arg->value, "There is no argument with the name '%s'", call_arg->name);
|
|
|
|
} else {
|
|
|
|
const char *pseudoname = heap_strf("%ld", i++);
|
|
|
|
if (!Table_str_get(&used_args, pseudoname))
|
|
|
|
code_err(call_arg->value, "This is one argument too many!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return code;
|
|
|
|
}
|
|
|
|
|
2024-02-13 11:42:33 -08:00
|
|
|
CORD compile(env_t *env, ast_t *ast)
|
2024-02-04 12:23:59 -08:00
|
|
|
{
|
|
|
|
switch (ast->tag) {
|
2024-02-17 23:34:39 -08:00
|
|
|
case Nil: return CORD_asprintf("$Null(%r)", compile_type_ast(Match(ast, Nil)->type));
|
2024-02-12 00:00:31 -08:00
|
|
|
case Bool: return Match(ast, Bool)->b ? "yes" : "no";
|
2024-02-29 10:28:39 -08:00
|
|
|
case Var: {
|
|
|
|
binding_t *b = get_binding(env, Match(ast, Var)->name);
|
|
|
|
if (b)
|
|
|
|
return b->code ? b->code : Match(ast, Var)->name;
|
2024-02-29 10:49:24 -08:00
|
|
|
return Match(ast, Var)->name;
|
|
|
|
// code_err(ast, "I don't know of any variable by this name");
|
2024-02-29 10:28:39 -08:00
|
|
|
}
|
2024-02-17 16:32:30 -08:00
|
|
|
case Int: return CORD_asprintf("I%ld(%ld)", Match(ast, Int)->bits, Match(ast, Int)->i);
|
2024-02-13 20:37:24 -08:00
|
|
|
case Num: {
|
|
|
|
// HACK: since the cord library doesn't support the '%a' specifier, this workaround
|
|
|
|
// is necessary:
|
2024-02-17 16:32:30 -08:00
|
|
|
char *buf = asprintfa(Match(ast, Num)->bits == 64 ? "%a" : "%af", Match(ast, Num)->n);
|
2024-02-13 20:37:24 -08:00
|
|
|
return CORD_from_char_star(buf);
|
|
|
|
}
|
2024-02-17 22:27:25 -08:00
|
|
|
case Length: {
|
|
|
|
ast_t *expr = Match(ast, Length)->value;
|
|
|
|
type_t *t = get_type(env, expr);
|
2024-02-18 11:28:35 -08:00
|
|
|
switch (value_type(t)->tag) {
|
2024-03-03 15:15:45 -08:00
|
|
|
case TextType: {
|
2024-02-20 10:06:03 -08:00
|
|
|
CORD str = compile_to_pointer_depth(env, expr, 0, false);
|
2024-03-03 22:34:12 -08:00
|
|
|
return CORD_all("Text__num_clusters(", str, ")");
|
2024-02-18 11:28:35 -08:00
|
|
|
}
|
|
|
|
case ArrayType: {
|
|
|
|
if (t->tag == PointerType) {
|
2024-02-20 10:06:03 -08:00
|
|
|
CORD arr = compile_to_pointer_depth(env, expr, 1, false);
|
2024-02-18 11:28:35 -08:00
|
|
|
return CORD_all("I64((", arr, ")->length)");
|
|
|
|
} else {
|
2024-02-20 10:06:03 -08:00
|
|
|
CORD arr = compile_to_pointer_depth(env, expr, 0, false);
|
2024-02-18 11:28:35 -08:00
|
|
|
return CORD_all("I64((", arr, ").length)");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case TableType: {
|
|
|
|
if (t->tag == PointerType) {
|
2024-02-20 10:06:03 -08:00
|
|
|
CORD table = compile_to_pointer_depth(env, expr, 1, false);
|
2024-02-18 11:28:35 -08:00
|
|
|
return CORD_all("I64((", table, ")->entries.length)");
|
|
|
|
} else {
|
2024-02-20 10:06:03 -08:00
|
|
|
CORD table = compile_to_pointer_depth(env, expr, 0, false);
|
2024-02-18 11:28:35 -08:00
|
|
|
return CORD_all("I64((", table, ").entries.length)");
|
|
|
|
}
|
2024-02-17 22:27:25 -08:00
|
|
|
}
|
|
|
|
default: code_err(ast, "Length is only supported for strings, arrays, and tables, not: %T", t);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2024-02-17 13:56:19 -08:00
|
|
|
case Not: return CORD_asprintf("not(%r)", compile(env, Match(ast, Not)->value));
|
2024-02-17 20:27:02 -08:00
|
|
|
case Negative: return CORD_asprintf("-(%r)", compile(env, Match(ast, Negative)->value));
|
|
|
|
case HeapAllocate: return CORD_asprintf("$heap(%r)", compile(env, Match(ast, HeapAllocate)->value));
|
2024-02-25 09:24:38 -08:00
|
|
|
case StackReference: {
|
2024-02-25 10:04:35 -08:00
|
|
|
ast_t *subject = Match(ast, StackReference)->value;
|
|
|
|
if (can_be_mutated(env, subject))
|
2024-03-03 11:34:13 -08:00
|
|
|
return CORD_all("(&", compile(env, subject), ")");
|
2024-02-25 10:04:35 -08:00
|
|
|
return CORD_all("$stack(", compile(env, subject), ")");
|
2024-02-25 09:24:38 -08:00
|
|
|
}
|
2024-02-04 12:23:59 -08:00
|
|
|
case BinaryOp: {
|
|
|
|
auto binop = Match(ast, BinaryOp);
|
2024-02-13 11:42:33 -08:00
|
|
|
CORD lhs = compile(env, binop->lhs);
|
|
|
|
CORD rhs = compile(env, binop->rhs);
|
2024-02-17 17:21:01 -08:00
|
|
|
|
2024-02-22 10:18:47 -08:00
|
|
|
type_t *lhs_t = get_type(env, binop->lhs);
|
|
|
|
type_t *rhs_t = get_type(env, binop->rhs);
|
|
|
|
type_t *operand_t;
|
|
|
|
if (can_promote(rhs_t, lhs_t))
|
|
|
|
operand_t = lhs_t;
|
|
|
|
else if (can_promote(lhs_t, rhs_t))
|
|
|
|
operand_t = rhs_t;
|
|
|
|
else
|
2024-02-22 10:26:43 -08:00
|
|
|
code_err(ast, "I can't do operations between %T and %T", lhs_t, rhs_t);
|
2024-02-17 17:21:01 -08:00
|
|
|
|
2024-02-04 12:23:59 -08:00
|
|
|
switch (binop->op) {
|
2024-02-22 10:09:46 -08:00
|
|
|
case BINOP_POWER: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag != NumType && operand_t->tag != IntType)
|
|
|
|
code_err(ast, "Exponentiation is only supported for numeric types");
|
|
|
|
if (operand_t->tag == NumType && Match(operand_t, NumType)->bits == 32)
|
2024-02-22 10:09:46 -08:00
|
|
|
return CORD_all("powf(", lhs, ", ", rhs, ")");
|
|
|
|
else
|
|
|
|
return CORD_all("pow(", lhs, ", ", rhs, ")");
|
|
|
|
}
|
2024-02-17 17:25:31 -08:00
|
|
|
case BINOP_MULT: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag != IntType && operand_t->tag != NumType)
|
2024-02-17 17:25:31 -08:00
|
|
|
code_err(ast, "Math operations are only supported for numeric types");
|
|
|
|
return CORD_asprintf("(%r * %r)", lhs, rhs);
|
|
|
|
}
|
|
|
|
case BINOP_DIVIDE: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag != IntType && operand_t->tag != NumType)
|
2024-02-17 17:25:31 -08:00
|
|
|
code_err(ast, "Math operations are only supported for numeric types");
|
|
|
|
return CORD_asprintf("(%r / %r)", lhs, rhs);
|
|
|
|
}
|
|
|
|
case BINOP_MOD: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag != IntType && operand_t->tag != NumType)
|
2024-02-17 17:25:31 -08:00
|
|
|
code_err(ast, "Math operations are only supported for numeric types");
|
|
|
|
return CORD_asprintf("mod(%r, %r)", lhs, rhs);
|
|
|
|
}
|
|
|
|
case BINOP_MOD1: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag != IntType && operand_t->tag != NumType)
|
2024-02-17 17:25:31 -08:00
|
|
|
code_err(ast, "Math operations are only supported for numeric types");
|
|
|
|
return CORD_asprintf("mod1(%r, %r)", lhs, rhs);
|
|
|
|
}
|
|
|
|
case BINOP_PLUS: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag != IntType && operand_t->tag != NumType)
|
2024-02-17 17:25:31 -08:00
|
|
|
code_err(ast, "Math operations are only supported for numeric types");
|
|
|
|
return CORD_asprintf("(%r + %r)", lhs, rhs);
|
|
|
|
}
|
|
|
|
case BINOP_MINUS: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag != IntType && operand_t->tag != NumType)
|
2024-02-17 17:25:31 -08:00
|
|
|
code_err(ast, "Math operations are only supported for numeric types");
|
|
|
|
return CORD_asprintf("(%r - %r)", lhs, rhs);
|
|
|
|
}
|
|
|
|
case BINOP_LSHIFT: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag != IntType && operand_t->tag != NumType)
|
2024-02-17 17:25:31 -08:00
|
|
|
code_err(ast, "Math operations are only supported for numeric types");
|
|
|
|
return CORD_asprintf("(%r << %r)", lhs, rhs);
|
|
|
|
}
|
|
|
|
case BINOP_RSHIFT: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag != IntType && operand_t->tag != NumType)
|
2024-02-17 17:25:31 -08:00
|
|
|
code_err(ast, "Math operations are only supported for numeric types");
|
|
|
|
return CORD_asprintf("(%r >> %r)", lhs, rhs);
|
|
|
|
}
|
2024-02-17 17:21:01 -08:00
|
|
|
case BINOP_EQ: {
|
2024-02-22 10:18:47 -08:00
|
|
|
switch (operand_t->tag) {
|
2024-02-17 17:21:01 -08:00
|
|
|
case BoolType: case IntType: case NumType: case PointerType: case FunctionType:
|
|
|
|
return CORD_asprintf("(%r == %r)", lhs, rhs);
|
|
|
|
default:
|
2024-02-22 10:18:47 -08:00
|
|
|
return CORD_asprintf("generic_equal($stack(%r), $stack(%r), %r)", lhs, rhs, compile_type_info(env, operand_t));
|
2024-02-17 17:21:01 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case BINOP_NE: {
|
2024-02-22 10:18:47 -08:00
|
|
|
switch (operand_t->tag) {
|
2024-02-17 17:21:01 -08:00
|
|
|
case BoolType: case IntType: case NumType: case PointerType: case FunctionType:
|
|
|
|
return CORD_asprintf("(%r != %r)", lhs, rhs);
|
|
|
|
default:
|
2024-02-22 10:18:47 -08:00
|
|
|
return CORD_asprintf("!generic_equal($stack(%r), $stack(%r), %r)", lhs, rhs, compile_type_info(env, operand_t));
|
2024-02-17 17:21:01 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case BINOP_LT: {
|
2024-02-22 10:18:47 -08:00
|
|
|
switch (operand_t->tag) {
|
2024-02-17 17:21:01 -08:00
|
|
|
case BoolType: case IntType: case NumType: case PointerType: case FunctionType:
|
|
|
|
return CORD_asprintf("(%r < %r)", lhs, rhs);
|
|
|
|
default:
|
2024-02-22 10:18:47 -08:00
|
|
|
return CORD_asprintf("(generic_compare($stack(%r), $stack(%r), %r) < 0)", lhs, rhs, compile_type_info(env, operand_t));
|
2024-02-17 17:21:01 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case BINOP_LE: {
|
2024-02-22 10:18:47 -08:00
|
|
|
switch (operand_t->tag) {
|
2024-02-17 17:21:01 -08:00
|
|
|
case BoolType: case IntType: case NumType: case PointerType: case FunctionType:
|
|
|
|
return CORD_asprintf("(%r <= %r)", lhs, rhs);
|
|
|
|
default:
|
2024-02-22 10:18:47 -08:00
|
|
|
return CORD_asprintf("(generic_compare($stack(%r), $stack(%r), %r) <= 0)", lhs, rhs, compile_type_info(env, operand_t));
|
2024-02-17 17:21:01 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case BINOP_GT: {
|
2024-02-22 10:18:47 -08:00
|
|
|
switch (operand_t->tag) {
|
2024-02-17 17:21:01 -08:00
|
|
|
case BoolType: case IntType: case NumType: case PointerType: case FunctionType:
|
|
|
|
return CORD_asprintf("(%r > %r)", lhs, rhs);
|
|
|
|
default:
|
2024-02-22 10:18:47 -08:00
|
|
|
return CORD_asprintf("(generic_compare($stack(%r), $stack(%r), %r) > 0)", lhs, rhs, compile_type_info(env, operand_t));
|
2024-02-17 17:21:01 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case BINOP_GE: {
|
2024-02-22 10:18:47 -08:00
|
|
|
switch (operand_t->tag) {
|
2024-02-17 17:21:01 -08:00
|
|
|
case BoolType: case IntType: case NumType: case PointerType: case FunctionType:
|
|
|
|
return CORD_asprintf("(%r >= %r)", lhs, rhs);
|
|
|
|
default:
|
2024-02-22 10:18:47 -08:00
|
|
|
return CORD_asprintf("(generic_compare($stack(%r), $stack(%r), %r) >= 0)", lhs, rhs, compile_type_info(env, operand_t));
|
2024-02-17 17:21:01 -08:00
|
|
|
}
|
|
|
|
}
|
2024-02-17 17:25:31 -08:00
|
|
|
case BINOP_AND: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag == BoolType)
|
2024-02-17 17:25:31 -08:00
|
|
|
return CORD_asprintf("(%r && %r)", lhs, rhs);
|
2024-02-22 10:18:47 -08:00
|
|
|
else if (operand_t->tag == IntType)
|
2024-02-17 17:25:31 -08:00
|
|
|
return CORD_asprintf("(%r & %r)", lhs, rhs);
|
|
|
|
else
|
|
|
|
code_err(ast, "Boolean operators are only supported for Bool and integer types");
|
|
|
|
}
|
|
|
|
case BINOP_OR: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag == BoolType)
|
2024-02-17 17:25:31 -08:00
|
|
|
return CORD_asprintf("(%r || %r)", lhs, rhs);
|
2024-02-22 10:18:47 -08:00
|
|
|
else if (operand_t->tag == IntType)
|
2024-02-17 17:25:31 -08:00
|
|
|
return CORD_asprintf("(%r | %r)", lhs, rhs);
|
|
|
|
else
|
|
|
|
code_err(ast, "Boolean operators are only supported for Bool and integer types");
|
|
|
|
}
|
|
|
|
case BINOP_XOR: {
|
2024-02-22 10:18:47 -08:00
|
|
|
if (operand_t->tag == BoolType || operand_t->tag == IntType)
|
2024-02-17 17:25:31 -08:00
|
|
|
return CORD_asprintf("(%r ^ %r)", lhs, rhs);
|
|
|
|
else
|
|
|
|
code_err(ast, "Boolean operators are only supported for Bool and integer types");
|
|
|
|
}
|
2024-02-22 10:00:27 -08:00
|
|
|
case BINOP_CONCAT: {
|
2024-02-22 10:18:47 -08:00
|
|
|
switch (operand_t->tag) {
|
2024-03-03 15:15:45 -08:00
|
|
|
case TextType: {
|
2024-02-22 10:00:27 -08:00
|
|
|
return CORD_all("CORD_cat(", lhs, ", ", rhs, ")");
|
|
|
|
}
|
|
|
|
case ArrayType: {
|
2024-02-23 10:24:06 -08:00
|
|
|
return CORD_all("Array__concat(", lhs, ", ", rhs, ", ", compile_type_info(env, operand_t), ")");
|
2024-02-22 10:00:27 -08:00
|
|
|
}
|
|
|
|
default:
|
2024-02-22 10:18:47 -08:00
|
|
|
code_err(ast, "Concatenation isn't supported for %T types", operand_t);
|
2024-02-22 10:00:27 -08:00
|
|
|
}
|
|
|
|
}
|
2024-02-04 12:23:59 -08:00
|
|
|
default: break;
|
|
|
|
}
|
2024-02-17 13:56:19 -08:00
|
|
|
code_err(ast, "unimplemented binop");
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
|
|
|
case UpdateAssign: {
|
|
|
|
auto update = Match(ast, UpdateAssign);
|
2024-02-25 10:13:36 -08:00
|
|
|
check_assignable(env, update->lhs);
|
2024-02-13 11:42:33 -08:00
|
|
|
CORD lhs = compile(env, update->lhs);
|
|
|
|
CORD rhs = compile(env, update->rhs);
|
2024-02-22 10:26:43 -08:00
|
|
|
|
|
|
|
type_t *lhs_t = get_type(env, update->lhs);
|
|
|
|
type_t *rhs_t = get_type(env, update->rhs);
|
|
|
|
type_t *operand_t;
|
|
|
|
if (can_promote(rhs_t, lhs_t))
|
|
|
|
operand_t = lhs_t;
|
|
|
|
else if (can_promote(lhs_t, rhs_t))
|
|
|
|
operand_t = rhs_t;
|
2024-02-23 10:47:06 -08:00
|
|
|
else if (lhs_t->tag == ArrayType && can_promote(rhs_t, Match(lhs_t, ArrayType)->item_type))
|
|
|
|
operand_t = lhs_t;
|
2024-02-22 10:26:43 -08:00
|
|
|
else
|
|
|
|
code_err(ast, "I can't do operations between %T and %T", lhs_t, rhs_t);
|
|
|
|
|
2024-02-04 12:23:59 -08:00
|
|
|
switch (update->op) {
|
2024-02-04 18:13:50 -08:00
|
|
|
case BINOP_MULT: return CORD_asprintf("%r *= %r;", lhs, rhs);
|
|
|
|
case BINOP_DIVIDE: return CORD_asprintf("%r /= %r;", lhs, rhs);
|
2024-02-22 10:26:43 -08:00
|
|
|
case BINOP_MOD: return CORD_asprintf("%r = mod(%r, %r);", lhs, lhs, rhs);
|
|
|
|
case BINOP_MOD1: return CORD_asprintf("%r = mod1(%r, %r);", lhs, lhs, rhs);
|
2024-02-04 18:13:50 -08:00
|
|
|
case BINOP_PLUS: return CORD_asprintf("%r += %r;", lhs, rhs);
|
|
|
|
case BINOP_MINUS: return CORD_asprintf("%r -= %r;", lhs, rhs);
|
2024-02-22 10:26:43 -08:00
|
|
|
case BINOP_POWER: {
|
|
|
|
if (lhs_t->tag != NumType)
|
|
|
|
code_err(ast, "'^=' is only supported for Num types");
|
|
|
|
if (lhs_t->tag == NumType && Match(lhs_t, NumType)->bits == 32)
|
|
|
|
return CORD_all(lhs, " = powf(", lhs, ", ", rhs, ")");
|
|
|
|
else
|
|
|
|
return CORD_all(lhs, " = pow(", lhs, ", ", rhs, ")");
|
|
|
|
}
|
2024-02-04 18:13:50 -08:00
|
|
|
case BINOP_LSHIFT: return CORD_asprintf("%r <<= %r;", lhs, rhs);
|
|
|
|
case BINOP_RSHIFT: return CORD_asprintf("%r >>= %r;", lhs, rhs);
|
2024-02-22 10:26:43 -08:00
|
|
|
case BINOP_AND: {
|
|
|
|
if (operand_t->tag == BoolType)
|
|
|
|
return CORD_asprintf("if (%r) %r = %r;", lhs, lhs, rhs);
|
|
|
|
else if (operand_t->tag == IntType)
|
|
|
|
return CORD_asprintf("%r &= %r;", lhs, rhs);
|
|
|
|
else
|
|
|
|
code_err(ast, "'or=' is not implemented for %T types", operand_t);
|
|
|
|
}
|
|
|
|
case BINOP_OR: {
|
|
|
|
if (operand_t->tag == BoolType)
|
|
|
|
return CORD_asprintf("if (!(%r)) %r = %r;", lhs, lhs, rhs);
|
|
|
|
else if (operand_t->tag == IntType)
|
|
|
|
return CORD_asprintf("%r |= %r;", lhs, rhs);
|
|
|
|
else
|
|
|
|
code_err(ast, "'or=' is not implemented for %T types", operand_t);
|
|
|
|
}
|
|
|
|
case BINOP_XOR: return CORD_asprintf("%r ^= %r;", lhs, rhs);
|
2024-02-23 10:24:06 -08:00
|
|
|
case BINOP_CONCAT: {
|
2024-03-03 15:15:45 -08:00
|
|
|
if (operand_t->tag == TextType) {
|
2024-02-23 10:24:06 -08:00
|
|
|
return CORD_asprintf("%r = CORD_cat(%r, %r);", lhs, lhs, rhs);
|
2024-02-23 10:29:20 -08:00
|
|
|
} else if (operand_t->tag == ArrayType) {
|
2024-02-23 10:47:06 -08:00
|
|
|
if (can_promote(rhs_t, Match(lhs_t, ArrayType)->item_type)) {
|
|
|
|
// arr ++= item
|
|
|
|
if (update->lhs->tag == Var)
|
|
|
|
return CORD_all("Array__insert(&", lhs, ", $stack(", rhs, "), 0, ", compile_type_info(env, operand_t), ")");
|
|
|
|
else
|
|
|
|
return CORD_all(lhs, "Array__concat(", lhs, ", $Array(", rhs, "), ", compile_type_info(env, operand_t), ")");
|
|
|
|
} else {
|
|
|
|
// arr ++= [...]
|
|
|
|
if (update->lhs->tag == Var)
|
|
|
|
return CORD_all("Array__insert_all(&", lhs, ", ", rhs, ", 0, ", compile_type_info(env, operand_t), ")");
|
|
|
|
else
|
|
|
|
return CORD_all(lhs, "Array__concat(", lhs, ", ", rhs, ", ", compile_type_info(env, operand_t), ")");
|
|
|
|
}
|
2024-02-23 10:29:20 -08:00
|
|
|
} else {
|
2024-02-23 10:24:06 -08:00
|
|
|
code_err(ast, "'++=' is not implemented for %T types", operand_t);
|
2024-02-23 10:29:20 -08:00
|
|
|
}
|
2024-02-23 10:24:06 -08:00
|
|
|
}
|
2024-02-22 10:26:43 -08:00
|
|
|
default: code_err(ast, "Update assignments are not implemented for this operation");
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
|
|
|
}
|
2024-03-03 15:15:45 -08:00
|
|
|
case TextLiteral: {
|
|
|
|
CORD literal = Match(ast, TextLiteral)->cord;
|
2024-02-11 12:31:30 -08:00
|
|
|
if (literal == CORD_EMPTY)
|
2024-02-11 12:22:32 -08:00
|
|
|
return "(CORD)CORD_EMPTY";
|
2024-02-24 11:36:08 -08:00
|
|
|
CORD code = "(CORD)\"";
|
2024-02-11 12:31:30 -08:00
|
|
|
CORD_pos i;
|
|
|
|
CORD_FOR(i, literal) {
|
2024-02-16 10:29:02 -08:00
|
|
|
char c = CORD_pos_fetch(i);
|
2024-02-11 12:31:30 -08:00
|
|
|
switch (c) {
|
2024-02-04 15:04:41 -08:00
|
|
|
case '\\': code = CORD_cat(code, "\\\\"); break;
|
|
|
|
case '"': code = CORD_cat(code, "\\\""); break;
|
|
|
|
case '\a': code = CORD_cat(code, "\\a"); break;
|
|
|
|
case '\b': code = CORD_cat(code, "\\b"); break;
|
|
|
|
case '\n': code = CORD_cat(code, "\\n"); break;
|
|
|
|
case '\r': code = CORD_cat(code, "\\r"); break;
|
|
|
|
case '\t': code = CORD_cat(code, "\\t"); break;
|
|
|
|
case '\v': code = CORD_cat(code, "\\v"); break;
|
2024-02-04 12:23:59 -08:00
|
|
|
default: {
|
2024-02-11 12:31:30 -08:00
|
|
|
if (isprint(c))
|
|
|
|
code = CORD_cat_char(code, c);
|
2024-02-04 12:23:59 -08:00
|
|
|
else
|
2024-03-03 14:49:40 -08:00
|
|
|
CORD_sprintf(&code, "%r\\x%02X", code, (uint8_t)c);
|
2024-02-04 12:23:59 -08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-02-04 15:04:41 -08:00
|
|
|
return CORD_cat_char(code, '"');
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
2024-03-03 15:15:45 -08:00
|
|
|
case TextJoin: {
|
|
|
|
ast_list_t *chunks = Match(ast, TextJoin)->children;
|
2024-02-10 12:33:35 -08:00
|
|
|
if (!chunks) {
|
2024-02-11 12:22:32 -08:00
|
|
|
return "(CORD)CORD_EMPTY";
|
2024-02-10 12:33:35 -08:00
|
|
|
} else if (!chunks->next) {
|
2024-02-17 17:47:43 -08:00
|
|
|
type_t *t = get_type(env, chunks->ast);
|
2024-03-03 15:15:45 -08:00
|
|
|
if (t->tag == TextType)
|
2024-02-17 17:47:43 -08:00
|
|
|
return compile(env, chunks->ast);
|
|
|
|
return compile_string(env, chunks->ast, "no");
|
2024-02-10 12:33:35 -08:00
|
|
|
} else {
|
2024-02-18 01:26:26 -08:00
|
|
|
CORD code = "CORD_all(";
|
2024-02-10 12:33:35 -08:00
|
|
|
for (ast_list_t *chunk = chunks; chunk; chunk = chunk->next) {
|
2024-02-18 01:26:26 -08:00
|
|
|
type_t *chunk_t = get_type(env, chunk->ast);
|
2024-03-03 15:15:45 -08:00
|
|
|
CORD chunk_str = (chunk_t->tag == TextType) ?
|
2024-02-17 17:47:43 -08:00
|
|
|
compile(env, chunk->ast) : compile_string(env, chunk->ast, "no");
|
2024-02-18 01:26:26 -08:00
|
|
|
code = CORD_cat(code, chunk_str);
|
|
|
|
if (chunk->next) code = CORD_cat(code, ", ");
|
2024-02-10 12:33:35 -08:00
|
|
|
}
|
2024-02-18 01:26:26 -08:00
|
|
|
return CORD_cat(code, ")");
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case Block: {
|
2024-02-04 15:04:41 -08:00
|
|
|
ast_list_t *stmts = Match(ast, Block)->statements;
|
|
|
|
if (stmts && !stmts->next)
|
2024-02-13 11:42:33 -08:00
|
|
|
return compile_statement(env, stmts->ast);
|
2024-02-04 15:04:41 -08:00
|
|
|
|
|
|
|
CORD code = "{\n";
|
2024-02-17 14:00:21 -08:00
|
|
|
env = fresh_scope(env);
|
2024-02-04 15:04:41 -08:00
|
|
|
for (ast_list_t *stmt = stmts; stmt; stmt = stmt->next) {
|
2024-02-24 11:29:40 -08:00
|
|
|
bind_statement(env, stmt->ast);
|
2024-02-13 11:42:33 -08:00
|
|
|
code = CORD_cat(code, compile_statement(env, stmt->ast));
|
2024-02-04 15:04:41 -08:00
|
|
|
code = CORD_cat(code, "\n");
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
2024-02-04 15:04:41 -08:00
|
|
|
return CORD_cat(code, "}");
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
|
|
|
case Declare: {
|
|
|
|
auto decl = Match(ast, Declare);
|
2024-02-17 23:42:40 -08:00
|
|
|
type_t *t = get_type(env, decl->value);
|
2024-02-17 20:43:55 -08:00
|
|
|
// return CORD_asprintf("auto %r = %r;", compile(env, decl->var), compile(env, decl->value));
|
2024-02-17 23:42:40 -08:00
|
|
|
return CORD_asprintf("%r %r = %r;", compile_type(t), compile(env, decl->var), compile(env, decl->value));
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
|
|
|
case Assign: {
|
|
|
|
auto assign = Match(ast, Assign);
|
2024-02-13 10:08:00 -08:00
|
|
|
// Single assignment:
|
|
|
|
if (assign->targets && !assign->targets->next)
|
2024-02-25 12:28:46 -08:00
|
|
|
return CORD_asprintf("%r = %r;", compile(env, assign->targets->ast), compile(env, assign->values->ast));
|
2024-02-13 10:08:00 -08:00
|
|
|
|
|
|
|
CORD code = "{ // Assignment\n";
|
|
|
|
int64_t i = 1;
|
|
|
|
for (ast_list_t *value = assign->values; value; value = value->next)
|
2024-02-17 23:42:40 -08:00
|
|
|
CORD_appendf(&code, "%r $%ld = %r;\n", compile_type(get_type(env, value->ast)), i++, compile(env, value->ast));
|
2024-02-13 10:08:00 -08:00
|
|
|
i = 1;
|
2024-02-25 10:13:36 -08:00
|
|
|
for (ast_list_t *target = assign->targets; target; target = target->next) {
|
|
|
|
check_assignable(env, target->ast);
|
2024-02-14 10:28:28 -08:00
|
|
|
CORD_appendf(&code, "%r = $%ld;\n", compile(env, target->ast), i++);
|
2024-02-25 10:13:36 -08:00
|
|
|
}
|
2024-02-13 10:08:00 -08:00
|
|
|
return CORD_cat(code, "\n}");
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
2024-03-05 11:46:01 -08:00
|
|
|
case Min: case Max: {
|
|
|
|
type_t *t = get_type(env, ast);
|
|
|
|
ast_t *key = ast->tag == Min ? Match(ast, Min)->key : Match(ast, Max)->key;
|
|
|
|
ast_t *lhs = ast->tag == Min ? Match(ast, Min)->lhs : Match(ast, Max)->lhs;
|
|
|
|
ast_t *rhs = ast->tag == Min ? Match(ast, Min)->rhs : Match(ast, Max)->rhs;
|
2024-03-07 09:15:38 -08:00
|
|
|
const char *key_name = "$";
|
2024-03-05 11:46:01 -08:00
|
|
|
if (key == NULL) key = FakeAST(Var, key_name);
|
|
|
|
|
|
|
|
env_t *expr_env = fresh_scope(env);
|
|
|
|
set_binding(expr_env, key_name, new(binding_t, .type=t, .code="$ternary$lhs"));
|
|
|
|
CORD lhs_key = compile(expr_env, key);
|
|
|
|
|
|
|
|
set_binding(expr_env, key_name, new(binding_t, .type=t, .code="$ternary$rhs"));
|
|
|
|
CORD rhs_key = compile(expr_env, key);
|
|
|
|
|
|
|
|
type_t *key_t = get_type(expr_env, key);
|
|
|
|
CORD comparison;
|
|
|
|
if (key_t->tag == IntType || key_t->tag == NumType || key_t->tag == BoolType || key_t->tag == PointerType)
|
|
|
|
comparison = CORD_all("((", lhs_key, ")", (ast->tag == Min ? "<=" : ">="), "(", rhs_key, "))");
|
|
|
|
else if (key_t->tag == TextType)
|
|
|
|
comparison = CORD_all("CORD_cmp(", lhs_key, ", ", rhs_key, ")", (ast->tag == Min ? "<=" : ">="), "0");
|
|
|
|
else
|
|
|
|
comparison = CORD_all("generic_compare($stack(", lhs_key, "), $stack(", rhs_key, "), ", compile_type_info(env, key_t), ")",
|
|
|
|
(ast->tag == Min ? "<=" : ">="), "0");
|
|
|
|
|
|
|
|
return CORD_all(
|
|
|
|
"({\n",
|
|
|
|
compile_type(t), " $ternary$lhs = ", compile(env, lhs), ", $ternary$rhs = ", compile(env, rhs), ";\n",
|
|
|
|
comparison, " ? $ternary$lhs : $ternary$rhs;\n"
|
|
|
|
"})");
|
2024-02-10 14:36:04 -08:00
|
|
|
}
|
2024-02-04 18:13:50 -08:00
|
|
|
case Array: {
|
|
|
|
auto array = Match(ast, Array);
|
|
|
|
if (!array->items)
|
2024-02-25 12:28:46 -08:00
|
|
|
return "(array_t){.length=0}";
|
2024-03-05 23:27:01 -08:00
|
|
|
|
|
|
|
int64_t n = 0;
|
|
|
|
for (ast_list_t *item = array->items; item; item = item->next)
|
|
|
|
++n;
|
|
|
|
|
|
|
|
type_t *item_type = Match(get_type(env, ast), ArrayType)->item_type;
|
|
|
|
CORD code = CORD_all("$TypedArrayN(", compile_type(item_type), CORD_asprintf(", %ld", n));
|
|
|
|
for (ast_list_t *item = array->items; item; item = item->next)
|
|
|
|
code = CORD_all(code, ", ", compile(env, item->ast));
|
2024-02-17 21:12:23 -08:00
|
|
|
return CORD_cat(code, ")");
|
2024-02-04 18:13:50 -08:00
|
|
|
}
|
2024-02-17 21:12:23 -08:00
|
|
|
case Table: {
|
|
|
|
auto table = Match(ast, Table);
|
|
|
|
if (!table->entries) {
|
|
|
|
CORD code = "(table_t){";
|
|
|
|
if (table->fallback)
|
|
|
|
code = CORD_all(code, ".fallback=", compile(env, table->fallback),",");
|
|
|
|
if (table->default_value)
|
2024-02-24 11:36:08 -08:00
|
|
|
code = CORD_all(code, ".default_value=$heap(", compile(env, table->default_value),"),");
|
2024-02-17 21:12:23 -08:00
|
|
|
return CORD_cat(code, "}");
|
|
|
|
}
|
|
|
|
|
|
|
|
type_t *table_t = get_type(env, ast);
|
2024-02-18 01:26:26 -08:00
|
|
|
type_t *key_t = Match(table_t, TableType)->key_type;
|
|
|
|
type_t *value_t = Match(table_t, TableType)->value_type;
|
|
|
|
CORD code = CORD_all("$Table(",
|
|
|
|
compile_type(key_t), ", ",
|
|
|
|
compile_type(value_t), ", ",
|
|
|
|
compile_type_info(env, key_t), ", ",
|
2024-02-24 12:39:33 -08:00
|
|
|
compile_type_info(env, value_t));
|
2024-02-18 01:26:26 -08:00
|
|
|
if (table->fallback)
|
2024-02-24 12:39:33 -08:00
|
|
|
code = CORD_all(code, ", /*fallback:*/ $heap(", compile(env, table->fallback), ")");
|
2024-02-18 01:26:26 -08:00
|
|
|
else
|
2024-02-24 12:39:33 -08:00
|
|
|
code = CORD_all(code, ", /*fallback:*/ NULL");
|
2024-02-18 01:26:26 -08:00
|
|
|
|
|
|
|
if (table->default_value)
|
2024-02-24 12:39:33 -08:00
|
|
|
code = CORD_all(code, ", /*default:*/ $heap(", compile(env, table->default_value), ")");
|
2024-02-18 01:26:26 -08:00
|
|
|
else
|
2024-02-24 12:39:33 -08:00
|
|
|
code = CORD_all(code, ", /*default:*/ NULL");
|
2024-02-18 01:26:26 -08:00
|
|
|
|
2024-03-03 13:08:38 -08:00
|
|
|
size_t n = 0;
|
|
|
|
for (ast_list_t *entry = table->entries; entry; entry = entry->next)
|
|
|
|
++n;
|
|
|
|
CORD_appendf(&code, ", %zu", n);
|
|
|
|
|
2024-02-17 21:12:23 -08:00
|
|
|
for (ast_list_t *entry = table->entries; entry; entry = entry->next) {
|
2024-02-22 09:45:12 -08:00
|
|
|
auto e = Match(entry->ast, TableEntry);
|
2024-03-03 13:08:38 -08:00
|
|
|
code = CORD_all(code, ",\n\t{", compile(env, e->key), ", ", compile(env, e->value), "}");
|
2024-02-17 21:12:23 -08:00
|
|
|
}
|
2024-02-18 01:26:26 -08:00
|
|
|
return CORD_cat(code, ")");
|
2024-02-17 21:12:23 -08:00
|
|
|
|
|
|
|
}
|
2024-02-04 12:23:59 -08:00
|
|
|
case FunctionDef: {
|
|
|
|
auto fndef = Match(ast, FunctionDef);
|
2024-02-13 19:13:54 -08:00
|
|
|
CORD name = compile(env, fndef->name);
|
2024-03-04 10:51:47 -08:00
|
|
|
CORD signature = CORD_all(fndef->ret_type ? compile_type_ast(fndef->ret_type) : "void", " ", name, "(");
|
2024-02-07 21:52:18 -08:00
|
|
|
for (arg_ast_t *arg = fndef->args; arg; arg = arg->next) {
|
2024-02-24 11:29:40 -08:00
|
|
|
type_t *arg_type = get_arg_ast_type(env, arg);
|
2024-03-04 10:51:47 -08:00
|
|
|
CORD_appendf(&signature, "%r %s", compile_type(arg_type), arg->name);
|
|
|
|
if (arg->next) signature = CORD_cat(signature, ", ");
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
2024-03-04 10:51:47 -08:00
|
|
|
signature = CORD_cat(signature, ")");
|
|
|
|
if (fndef->is_private)
|
|
|
|
env->code->staticdefs = CORD_all(env->code->staticdefs, "static ", signature, ";\n");
|
|
|
|
else
|
|
|
|
env->code->fndefs = CORD_all(env->code->fndefs, signature, ";\n");
|
|
|
|
|
|
|
|
CORD code = signature;
|
|
|
|
if (fndef->is_inline)
|
|
|
|
code = CORD_cat("inline ", code);
|
|
|
|
if (!fndef->is_private)
|
|
|
|
code = CORD_cat("public ", code);
|
2024-02-13 11:42:33 -08:00
|
|
|
|
2024-02-17 14:00:21 -08:00
|
|
|
env_t *body_scope = fresh_scope(env);
|
|
|
|
body_scope->locals->fallback = env->globals;
|
2024-02-13 11:42:33 -08:00
|
|
|
for (arg_ast_t *arg = fndef->args; arg; arg = arg->next) {
|
2024-02-24 11:29:40 -08:00
|
|
|
type_t *arg_type = get_arg_ast_type(env, arg);
|
2024-03-04 10:51:47 -08:00
|
|
|
set_binding(body_scope, arg->name, new(binding_t, .type=arg_type, .code=arg->name));
|
|
|
|
}
|
2024-02-13 19:13:54 -08:00
|
|
|
|
2024-02-22 10:35:28 -08:00
|
|
|
CORD body = compile(body_scope, fndef->body);
|
2024-02-13 11:42:33 -08:00
|
|
|
if (CORD_fetch(body, 0) != '{')
|
|
|
|
body = CORD_asprintf("{\n%r\n}", body);
|
2024-03-04 10:51:47 -08:00
|
|
|
env->code->funcs = CORD_all(env->code->funcs, code, " ", body);
|
2024-02-13 11:42:33 -08:00
|
|
|
return CORD_EMPTY;
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
2024-03-08 11:23:16 -08:00
|
|
|
case MethodCall: {
|
|
|
|
auto call = Match(ast, MethodCall);
|
|
|
|
type_t *self_t = get_type(env, call->self);
|
|
|
|
type_t *self_value_t = value_type(self_t);
|
|
|
|
switch (self_value_t->tag) {
|
|
|
|
case ArrayType: {
|
|
|
|
// TODO: check for readonly
|
|
|
|
if (streq(call->name, "insert")) {
|
|
|
|
type_t *item_t = Match(self_value_t, ArrayType)->item_type;
|
|
|
|
CORD self = compile_to_pointer_depth(env, call->self, 1, false);
|
|
|
|
arg_t *arg_spec = new(arg_t, .name="item", .type=Type(PointerType, .pointed=item_t, .is_stack=true, .is_readonly=true),
|
|
|
|
.next=new(arg_t, .name="at", .type=Type(IntType, .bits=64), .default_val=FakeAST(Int, .i=0, .bits=64)));
|
|
|
|
return CORD_all("Array__insert(", self, ", ", compile_arguments(env, ast, arg_spec, call->args), ", ",
|
|
|
|
compile_type_info(env, self_value_t), ")");
|
|
|
|
} else if (streq(call->name, "remove")) {
|
|
|
|
CORD self = compile_to_pointer_depth(env, call->self, 1, false);
|
|
|
|
arg_t *arg_spec = new(arg_t, .name="index", .type=Type(IntType, .bits=64), .default_val=FakeAST(Int, .i=-1, .bits=64),
|
|
|
|
.next=new(arg_t, .name="count", .type=Type(IntType, .bits=64), .default_val=FakeAST(Int, .i=1, .bits=64)));
|
|
|
|
return CORD_all("Array__remove(", self, ", ", compile_arguments(env, ast, arg_spec, call->args), ", ",
|
|
|
|
compile_type_info(env, self_value_t), ")");
|
|
|
|
} else if (streq(call->name, "random")) {
|
|
|
|
CORD self = compile_to_pointer_depth(env, call->self, 0, false);
|
|
|
|
return CORD_all("Array__random(", self, ")");
|
|
|
|
} else if (streq(call->name, "shuffle")) {
|
|
|
|
CORD self = compile_to_pointer_depth(env, call->self, 1, false);
|
|
|
|
return CORD_all("Array__shuffle(", self, ", ", compile_type_info(env, self_value_t), ")");
|
|
|
|
} else if (streq(call->name, "sort")) {
|
|
|
|
CORD self = compile_to_pointer_depth(env, call->self, 1, false);
|
|
|
|
return CORD_all("Array__sort(", self, ", ", compile_type_info(env, self_value_t), ")");
|
|
|
|
} else if (streq(call->name, "clear")) {
|
|
|
|
CORD self = compile_to_pointer_depth(env, call->self, 1, false);
|
|
|
|
return CORD_all("Array__compact(", self, ")");
|
|
|
|
} else if (streq(call->name, "slice")) {
|
|
|
|
CORD self = compile_to_pointer_depth(env, call->self, 1, false);
|
|
|
|
arg_t *arg_spec = new(arg_t, .name="first", .type=Type(IntType, .bits=64), .default_val=FakeAST(Int, .i=1, .bits=64),
|
|
|
|
.next=new(arg_t, .name="length", .type=Type(IntType, .bits=64), .default_val=FakeAST(Int, .i=INT64_MAX, .bits=64),
|
|
|
|
.next=new(arg_t, .name="stride", .type=Type(IntType, .bits=64), .default_val=FakeAST(Int, .i=1, .bits=64))));
|
|
|
|
return CORD_all("Array__slice(", self, ", ", compile_arguments(env, ast, arg_spec, call->args), ", ",
|
|
|
|
compile_type_info(env, self_value_t), ")");
|
|
|
|
} else code_err(ast, "There is no '%s' method for arrays", call->name);
|
|
|
|
}
|
|
|
|
case TableType: {
|
|
|
|
goto fncall;
|
|
|
|
}
|
|
|
|
default: goto fncall;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
case FunctionCall: {
|
|
|
|
fncall:;
|
2024-02-29 10:49:24 -08:00
|
|
|
type_t *fn_t;
|
|
|
|
arg_ast_t *args;
|
|
|
|
CORD fn;
|
|
|
|
if (ast->tag == FunctionCall) {
|
|
|
|
auto call = Match(ast, FunctionCall);
|
|
|
|
fn_t = get_type(env, call->fn);
|
2024-03-03 10:04:50 -08:00
|
|
|
if (fn_t->tag == TypeInfoType) {
|
|
|
|
type_t *t = Match(fn_t, TypeInfoType)->type;
|
|
|
|
if (!(t->tag == StructType))
|
|
|
|
code_err(call->fn, "This is not a type that has a constructor");
|
|
|
|
fn_t = Type(FunctionType, .args=Match(t, StructType)->fields, .ret=t);
|
|
|
|
} else if (fn_t->tag != FunctionType) {
|
2024-02-29 10:49:24 -08:00
|
|
|
code_err(call->fn, "This is not a function, it's a %T", fn_t);
|
2024-03-03 10:04:50 -08:00
|
|
|
}
|
2024-02-29 10:49:24 -08:00
|
|
|
args = call->args;
|
|
|
|
fn = compile(env, call->fn);
|
|
|
|
} else {
|
|
|
|
auto method = Match(ast, MethodCall);
|
|
|
|
fn_t = get_method_type(env, method->self, method->name);
|
|
|
|
args = new(arg_ast_t, .value=method->self, .next=method->args);
|
2024-03-03 10:04:50 -08:00
|
|
|
binding_t *b = get_namespace_binding(env, method->self, method->name);
|
2024-02-29 10:49:24 -08:00
|
|
|
if (!b) code_err(ast, "No such method");
|
|
|
|
fn = b->code;
|
|
|
|
}
|
|
|
|
|
2024-03-08 11:23:16 -08:00
|
|
|
CORD code = CORD_all(fn, "(", compile_arguments(env, ast, Match(fn_t, FunctionType)->args, args), ")");
|
|
|
|
return code;
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
2024-02-04 15:04:41 -08:00
|
|
|
case If: {
|
|
|
|
auto if_ = Match(ast, If);
|
2024-03-06 10:36:36 -08:00
|
|
|
if (if_->condition->tag == Declare) {
|
|
|
|
auto decl = Match(if_->condition, Declare);
|
|
|
|
env_t *true_scope = fresh_scope(env);
|
|
|
|
const char *name = Match(decl->var, Var)->name;
|
|
|
|
CORD var_code = CORD_cat(env->scope_prefix, name);
|
|
|
|
type_t *var_t = get_type(env, decl->value);
|
|
|
|
if (var_t->tag == PointerType) {
|
|
|
|
auto ptr = Match(var_t, PointerType);
|
|
|
|
if (!ptr->is_optional)
|
|
|
|
code_err(if_->condition, "This pointer will always be non-null, so it should not be used in a conditional.");
|
|
|
|
var_t = Type(PointerType, .pointed=ptr->pointed, .is_optional=false, .is_stack=ptr->is_stack, .is_readonly=ptr->is_readonly);
|
|
|
|
} else {
|
|
|
|
code_err(if_->condition, "Only optional pointer types can be used in 'if var := ...' statements (this is a %T)", var_t);
|
|
|
|
}
|
|
|
|
set_binding(true_scope, name, new(binding_t, .type=var_t, .code=var_code));
|
|
|
|
CORD code = CORD_all("{\n",
|
|
|
|
compile_type(var_t), " ", var_code, " = ", compile(env, decl->value), ";\n"
|
|
|
|
"if (", var_code, ") ", compile_statement(true_scope, if_->body));
|
|
|
|
if (if_->else_body)
|
|
|
|
code = CORD_all(code, "\nelse ", compile_statement(env, if_->else_body));
|
|
|
|
code = CORD_cat(code, "\n}");
|
|
|
|
return code;
|
|
|
|
} else {
|
|
|
|
type_t *cond_t = get_type(env, if_->condition);
|
|
|
|
if (cond_t->tag == PointerType) {
|
|
|
|
if (!Match(cond_t, PointerType)->is_optional)
|
|
|
|
code_err(if_->condition, "This pointer will always be non-null, so it should not be used in a conditional.");
|
|
|
|
} else if (cond_t->tag != BoolType) {
|
|
|
|
code_err(if_->condition, "Only boolean values and optional pointers can be used in conditionals (this is a %T)", cond_t);
|
|
|
|
}
|
|
|
|
CORD code;
|
|
|
|
CORD_sprintf(&code, "if (%r) %r", compile(env, if_->condition), compile_statement(env, if_->body));
|
|
|
|
if (if_->else_body)
|
|
|
|
code = CORD_all(code, "\nelse ", compile_statement(env, if_->else_body));
|
|
|
|
return code;
|
|
|
|
}
|
2024-02-04 15:04:41 -08:00
|
|
|
}
|
2024-02-22 09:45:12 -08:00
|
|
|
case When: {
|
|
|
|
auto when = Match(ast, When);
|
|
|
|
type_t *subject_t = get_type(env, when->subject);
|
|
|
|
auto enum_t = Match(subject_t, EnumType);
|
|
|
|
CORD code = CORD_all("{ ", compile_type(subject_t), " $subject = ", compile(env, when->subject), ";\n"
|
|
|
|
"switch ($subject.$tag) {");
|
|
|
|
type_t *result_t = get_type(env, ast);
|
|
|
|
(void)result_t;
|
|
|
|
for (when_clause_t *clause = when->clauses; clause; clause = clause->next) {
|
|
|
|
const char *clause_tag_name = Match(clause->tag_name, Var)->name;
|
|
|
|
code = CORD_all(code, "case $tag$", enum_t->name, "$", clause_tag_name, ": {\n");
|
|
|
|
type_t *tag_type = NULL;
|
|
|
|
for (tag_t *tag = enum_t->tags; tag; tag = tag->next) {
|
|
|
|
if (streq(tag->name, clause_tag_name)) {
|
|
|
|
tag_type = tag->type;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
assert(tag_type);
|
|
|
|
env_t *scope = env;
|
|
|
|
if (clause->var) {
|
|
|
|
code = CORD_all(code, compile_type(tag_type), " ", compile(env, clause->var), " = $subject.", clause_tag_name, ";\n");
|
|
|
|
scope = fresh_scope(env);
|
|
|
|
set_binding(scope, Match(clause->var, Var)->name, new(binding_t, .type=tag_type));
|
|
|
|
}
|
|
|
|
code = CORD_all(code, compile(scope, clause->body), "\nbreak;\n}\n");
|
|
|
|
}
|
|
|
|
if (when->else_body) {
|
|
|
|
code = CORD_all(code, "default: {\n", compile(env, when->else_body), "\nbreak;\n}");
|
|
|
|
}
|
|
|
|
code = CORD_all(code, "\n}\n}");
|
|
|
|
return code;
|
|
|
|
}
|
2024-02-04 18:13:50 -08:00
|
|
|
case While: {
|
|
|
|
auto while_ = Match(ast, While);
|
2024-02-13 11:42:33 -08:00
|
|
|
return CORD_asprintf("while (%r) %r", compile(env, while_->condition), compile(env, while_->body));
|
2024-02-04 18:13:50 -08:00
|
|
|
}
|
|
|
|
case For: {
|
|
|
|
auto for_ = Match(ast, For);
|
2024-02-18 01:26:26 -08:00
|
|
|
type_t *iter_t = get_type(env, for_->iter);
|
|
|
|
switch (iter_t->tag) {
|
|
|
|
case ArrayType: {
|
|
|
|
type_t *item_t = Match(iter_t, ArrayType)->item_type;
|
|
|
|
env_t *scope = fresh_scope(env);
|
|
|
|
CORD index = for_->index ? compile(env, for_->index) : "$i";
|
|
|
|
if (for_->index)
|
|
|
|
set_binding(scope, CORD_to_const_char_star(index), new(binding_t, .type=Type(IntType, .bits=64)));
|
|
|
|
CORD value = compile(env, for_->value);
|
|
|
|
set_binding(scope, CORD_to_const_char_star(value), new(binding_t, .type=item_t));
|
2024-03-05 23:15:23 -08:00
|
|
|
CORD empty_handling = for_->empty ? CORD_all("if ($arr.length == 0) ", compile(env, for_->empty), "\nelse ") : CORD_EMPTY;
|
|
|
|
return CORD_all("{\n"
|
|
|
|
"array_t $arr = ", compile(env, for_->iter), ";\n",
|
|
|
|
// "$ARRAY_INCREF($arr);\n",
|
|
|
|
empty_handling,
|
|
|
|
"for (int64_t ", index, " = 1; ", index, " <= $arr.length; ", index, "++) {\n",
|
|
|
|
compile_type(item_t), " ", value, " = *(", compile_type(item_t), "*)($arr.data + (", index, "-1)*$arr.stride);\n",
|
|
|
|
compile(scope, for_->body),
|
|
|
|
"}\n"
|
|
|
|
// "$ARRAY_DECREF($arr);\n"
|
|
|
|
"}");
|
2024-02-18 01:26:26 -08:00
|
|
|
}
|
|
|
|
case TableType: {
|
|
|
|
type_t *key_t = Match(iter_t, TableType)->key_type;
|
|
|
|
type_t *value_t = Match(iter_t, TableType)->value_type;
|
|
|
|
env_t *scope = fresh_scope(env);
|
|
|
|
CORD key, value;
|
|
|
|
if (for_->index) {
|
|
|
|
key = compile(env, for_->index);
|
|
|
|
value = compile(env, for_->value);
|
|
|
|
set_binding(scope, CORD_to_const_char_star(key), new(binding_t, .type=key_t));
|
|
|
|
set_binding(scope, CORD_to_const_char_star(value), new(binding_t, .type=value_t));
|
|
|
|
|
|
|
|
size_t value_offset = type_size(key_t);
|
|
|
|
if (type_align(value_t) > 1 && value_offset % type_align(value_t))
|
|
|
|
value_offset += type_align(value_t) - (value_offset % type_align(value_t)); // padding
|
2024-02-29 09:45:10 -08:00
|
|
|
return CORD_all("$TABLE_FOREACH(", compile(env, for_->iter), ", ", compile_type(key_t), ", ", key, ", ",
|
|
|
|
compile_type(value_t), ", ", value, ", ", heap_strf("%zu", value_offset),
|
2024-02-29 10:49:24 -08:00
|
|
|
", ", compile(scope, for_->body), ", ", for_->empty ? compile(env, for_->empty) : "{}", ")");
|
2024-02-18 01:26:26 -08:00
|
|
|
} else {
|
2024-02-18 01:27:52 -08:00
|
|
|
key = compile(env, for_->value);
|
2024-02-18 01:26:26 -08:00
|
|
|
set_binding(scope, CORD_to_const_char_star(key), new(binding_t, .type=key_t));
|
2024-02-29 09:45:10 -08:00
|
|
|
return CORD_all("$ARRAY_FOREACH((", compile(env, for_->iter), ").entries, $i, ", compile_type(key_t), ", ", key, ", ",
|
2024-02-29 10:49:24 -08:00
|
|
|
compile(scope, for_->body), ", ", for_->empty ? compile(env, for_->empty) : "{}", ")");
|
2024-02-18 01:26:26 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
case IntType: {
|
|
|
|
type_t *item_t = iter_t;
|
|
|
|
env_t *scope = fresh_scope(env);
|
|
|
|
CORD value = compile(env, for_->value);
|
2024-03-06 09:56:30 -08:00
|
|
|
set_binding(scope, CORD_to_const_char_star(value), new(binding_t, .type=item_t, .code=value));
|
|
|
|
|
|
|
|
CORD n = compile(env, for_->iter);
|
|
|
|
CORD index = CORD_EMPTY;
|
|
|
|
if (for_->index) {
|
|
|
|
index = compile(env, for_->index);
|
|
|
|
set_binding(scope, CORD_to_const_char_star(index), new(binding_t, .type=Type(IntType, .bits=64), .code=index));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (for_->empty && index) {
|
|
|
|
return CORD_all(
|
|
|
|
"{\n"
|
|
|
|
"int64_t $n = ", n, ";\n"
|
|
|
|
"if ($n > 0) {\n"
|
|
|
|
"for (int64_t ", index, " = 1, ", value, "; (", value, "=", index,") <= $n; ++", index, ")\n"
|
|
|
|
"\t", compile(scope, for_->body), "\n"
|
|
|
|
"}\n else ", compile(env, for_->empty),
|
|
|
|
"\n}");
|
|
|
|
} else if (for_->empty) {
|
|
|
|
return CORD_all(
|
|
|
|
"{\n"
|
|
|
|
"int64_t $n = ", n, ";\n"
|
|
|
|
"if ($n > 0) {\n"
|
|
|
|
"for (int64_t ", value, " = 1; ", value, " <= $n; ++", value, ")\n"
|
|
|
|
"\t", compile(scope, for_->body), "\n"
|
|
|
|
"}\n else ", compile(env, for_->empty),
|
|
|
|
"\n}");
|
|
|
|
} else if (index) {
|
|
|
|
return CORD_all(
|
|
|
|
"for (int64_t ", value, ", ", index, " = 1, $n = ", n, "; (", value, "=", index,") <= $n; ++", value, ")\n"
|
|
|
|
"\t", compile(scope, for_->body), "\n");
|
|
|
|
} else {
|
|
|
|
return CORD_all(
|
|
|
|
"for (int64_t ", value, " = 1, $n = ", compile(env, for_->iter), "; ", value, " <= $n; ++", value, ")\n"
|
|
|
|
"\t", compile(scope, for_->body), "\n");
|
|
|
|
}
|
2024-02-18 01:26:26 -08:00
|
|
|
}
|
|
|
|
default: code_err(for_->iter, "Iteration is not implemented for type: %T", iter_t);
|
|
|
|
}
|
2024-02-04 18:13:50 -08:00
|
|
|
}
|
2024-02-25 12:28:46 -08:00
|
|
|
case Reduction: {
|
|
|
|
auto reduction = Match(ast, Reduction);
|
|
|
|
type_t *t = get_type(env, ast);
|
|
|
|
CORD code = CORD_all(
|
|
|
|
"({ // Reduction:\n",
|
2024-03-05 11:46:01 -08:00
|
|
|
compile_type(t), " $reduction;\n"
|
2024-02-25 12:28:46 -08:00
|
|
|
);
|
|
|
|
env_t *scope = fresh_scope(env);
|
2024-03-05 11:46:01 -08:00
|
|
|
ast_t *result = FakeAST(Var, "$reduction");
|
|
|
|
set_binding(scope, "$reduction", new(binding_t, .type=t));
|
2024-02-25 13:02:36 -08:00
|
|
|
ast_t *empty = NULL;
|
2024-02-25 12:28:46 -08:00
|
|
|
if (reduction->fallback) {
|
|
|
|
type_t *fallback_type = get_type(scope, reduction->fallback);
|
|
|
|
if (fallback_type->tag == AbortType) {
|
2024-02-25 13:02:36 -08:00
|
|
|
empty = reduction->fallback;
|
2024-02-25 12:28:46 -08:00
|
|
|
} else {
|
2024-02-25 13:02:36 -08:00
|
|
|
empty = FakeAST(Assign, .targets=new(ast_list_t, .ast=result), .values=new(ast_list_t, .ast=reduction->fallback));
|
2024-02-25 12:28:46 -08:00
|
|
|
}
|
|
|
|
} else {
|
2024-02-25 13:02:36 -08:00
|
|
|
empty = FakeAST(
|
|
|
|
InlineCCode,
|
|
|
|
CORD_asprintf("fail_source(%s, %ld, %ld, \"This collection was empty!\");\n",
|
2024-03-03 15:15:45 -08:00
|
|
|
Text__quoted(ast->file->filename, false), (long)(reduction->iter->start - reduction->iter->file->text),
|
2024-02-25 13:02:36 -08:00
|
|
|
(long)(reduction->iter->end - reduction->iter->file->text)));
|
2024-02-25 12:28:46 -08:00
|
|
|
}
|
|
|
|
ast_t *i = FakeAST(Var, "$i");
|
2024-03-05 11:46:01 -08:00
|
|
|
ast_t *item = FakeAST(Var, "$iter_value");
|
2024-02-25 12:28:46 -08:00
|
|
|
ast_t *body = FakeAST(
|
|
|
|
If, .condition=FakeAST(BinaryOp, .lhs=i, .op=BINOP_EQ, .rhs=FakeAST(Int, .i=1, .bits=64)),
|
|
|
|
.body=FakeAST(Assign, .targets=new(ast_list_t, .ast=result), .values=new(ast_list_t, .ast=item)),
|
|
|
|
.else_body=FakeAST(Assign, .targets=new(ast_list_t, .ast=result), .values=new(ast_list_t, .ast=reduction->combination)));
|
2024-02-25 13:02:36 -08:00
|
|
|
ast_t *loop = FakeAST(For, .index=i, .value=item, .iter=reduction->iter, .body=body, .empty=empty);
|
2024-03-05 11:46:01 -08:00
|
|
|
set_binding(scope, "$iter_value", new(binding_t, .type=t));
|
|
|
|
code = CORD_all(code, compile(scope, loop), "\n$reduction;})");
|
2024-02-25 12:28:46 -08:00
|
|
|
return code;
|
|
|
|
}
|
2024-02-04 18:13:50 -08:00
|
|
|
case Skip: {
|
2024-02-17 13:56:19 -08:00
|
|
|
if (Match(ast, Skip)->target) code_err(ast, "Named skips not yet implemented");
|
2024-02-04 18:13:50 -08:00
|
|
|
return "continue";
|
|
|
|
}
|
|
|
|
case Stop: {
|
2024-02-17 13:56:19 -08:00
|
|
|
if (Match(ast, Stop)->target) code_err(ast, "Named stops not yet implemented");
|
2024-02-04 18:13:50 -08:00
|
|
|
return "break";
|
|
|
|
}
|
|
|
|
case Pass: return ";";
|
|
|
|
case Return: {
|
|
|
|
auto ret = Match(ast, Return)->value;
|
2024-02-13 11:42:33 -08:00
|
|
|
return ret ? CORD_asprintf("return %r;", compile(env, ret)) : "return;";
|
2024-02-04 18:13:50 -08:00
|
|
|
}
|
2024-02-04 12:23:59 -08:00
|
|
|
// Extern,
|
2024-02-07 21:52:18 -08:00
|
|
|
case StructDef: {
|
2024-02-24 10:27:49 -08:00
|
|
|
compile_struct_def(env, ast);
|
2024-02-13 11:42:33 -08:00
|
|
|
return CORD_EMPTY;
|
2024-02-07 21:52:18 -08:00
|
|
|
}
|
|
|
|
case EnumDef: {
|
2024-02-24 12:24:44 -08:00
|
|
|
compile_enum_def(env, ast);
|
2024-02-13 11:42:33 -08:00
|
|
|
return CORD_EMPTY;
|
2024-02-04 18:13:50 -08:00
|
|
|
}
|
2024-02-11 20:09:00 -08:00
|
|
|
case DocTest: {
|
|
|
|
auto test = Match(ast, DocTest);
|
|
|
|
CORD src = heap_strn(test->expr->start, (size_t)(test->expr->end - test->expr->start));
|
2024-02-17 17:07:04 -08:00
|
|
|
type_t *expr_t = get_type(env, test->expr);
|
2024-02-26 20:02:09 -08:00
|
|
|
if (!expr_t)
|
|
|
|
code_err(test->expr, "I couldn't figure out the type of this expression");
|
|
|
|
|
2024-03-03 16:12:53 -08:00
|
|
|
CORD output = NULL;
|
|
|
|
if (test->output) {
|
|
|
|
const uint8_t *raw = (const uint8_t*)CORD_to_const_char_star(test->output);
|
|
|
|
uint8_t buf[128] = {0};
|
2024-03-03 22:34:12 -08:00
|
|
|
size_t norm_len = sizeof(buf);
|
|
|
|
uint8_t *norm = u8_normalize(UNINORM_NFD, (uint8_t*)raw, strlen((char*)raw)+1, buf, &norm_len);
|
|
|
|
assert(norm[norm_len-1] == 0);
|
2024-03-03 16:12:53 -08:00
|
|
|
output = CORD_from_char_star((char*)norm);
|
|
|
|
if (norm && norm != buf) free(norm);
|
|
|
|
}
|
|
|
|
|
2024-02-11 23:07:12 -08:00
|
|
|
if (test->expr->tag == Declare) {
|
|
|
|
auto decl = Match(test->expr, Declare);
|
|
|
|
return CORD_asprintf(
|
2024-02-17 20:43:55 -08:00
|
|
|
"%r\n"
|
2024-02-17 17:07:04 -08:00
|
|
|
"__doctest(&%r, %r, %r, %r, %ld, %ld);",
|
2024-02-17 20:43:55 -08:00
|
|
|
compile(env, test->expr),
|
2024-02-13 11:42:33 -08:00
|
|
|
compile(env, decl->var),
|
2024-02-17 16:32:30 -08:00
|
|
|
compile_type_info(env, get_type(env, decl->value)),
|
2024-03-03 16:12:53 -08:00
|
|
|
compile(env, WrapAST(test->expr, TextLiteral, .cord=output)),
|
2024-03-03 15:15:45 -08:00
|
|
|
compile(env, WrapAST(test->expr, TextLiteral, .cord=test->expr->file->filename)),
|
2024-02-17 16:32:30 -08:00
|
|
|
(int64_t)(test->expr->start - test->expr->file->text),
|
|
|
|
(int64_t)(test->expr->end - test->expr->file->text));
|
2024-02-11 23:07:12 -08:00
|
|
|
} else if (test->expr->tag == Assign) {
|
2024-02-13 10:32:08 -08:00
|
|
|
auto assign = Match(test->expr, Assign);
|
|
|
|
CORD code = "{ // Assignment\n";
|
|
|
|
int64_t i = 1;
|
|
|
|
for (ast_list_t *value = assign->values; value; value = value->next)
|
2024-02-17 23:42:40 -08:00
|
|
|
CORD_appendf(&code, "%r $%ld = %r;\n", compile_type(get_type(env, value->ast)), i++, compile(env, value->ast));
|
2024-02-13 10:32:08 -08:00
|
|
|
i = 1;
|
2024-02-25 10:13:36 -08:00
|
|
|
for (ast_list_t *target = assign->targets; target; target = target->next) {
|
|
|
|
check_assignable(env, target->ast);
|
2024-02-14 10:28:28 -08:00
|
|
|
CORD_appendf(&code, "%r = $%ld;\n", compile(env, target->ast), i++);
|
2024-02-25 10:13:36 -08:00
|
|
|
}
|
2024-02-13 10:32:08 -08:00
|
|
|
|
2024-02-17 23:42:40 -08:00
|
|
|
CORD expr_cord = "CORD_all(";
|
2024-02-13 10:32:08 -08:00
|
|
|
i = 1;
|
2024-02-17 23:42:40 -08:00
|
|
|
for (ast_list_t *target = assign->targets; target; target = target->next) {
|
2024-03-03 15:16:33 -08:00
|
|
|
CORD item = expr_as_texting(env, CORD_asprintf("$%ld", i++), get_type(env, target->ast), "USE_COLOR");
|
2024-02-17 23:42:40 -08:00
|
|
|
expr_cord = CORD_all(expr_cord, item, target->next ? ", \", \", " : CORD_EMPTY);
|
|
|
|
}
|
2024-02-13 10:32:08 -08:00
|
|
|
expr_cord = CORD_cat(expr_cord, ")");
|
|
|
|
|
2024-02-14 10:28:28 -08:00
|
|
|
CORD_appendf(&code, "$test(%r, %r, %r);",
|
2024-03-03 15:15:45 -08:00
|
|
|
compile(env, WrapAST(test->expr, TextLiteral, .cord=src)),
|
2024-02-13 10:32:08 -08:00
|
|
|
expr_cord,
|
2024-03-03 16:12:53 -08:00
|
|
|
compile(env, WrapAST(test->expr, TextLiteral, .cord=output)));
|
2024-02-13 10:32:08 -08:00
|
|
|
return CORD_cat(code, "\n}");
|
2024-02-17 17:07:04 -08:00
|
|
|
} else if (expr_t->tag == VoidType || expr_t->tag == AbortType) {
|
|
|
|
return CORD_asprintf(
|
2024-02-20 10:06:03 -08:00
|
|
|
"%r;\n"
|
|
|
|
"__doctest(NULL, NULL, NULL, %r, %ld, %ld);",
|
2024-02-17 17:07:04 -08:00
|
|
|
compile(env, test->expr),
|
2024-03-03 15:15:45 -08:00
|
|
|
compile(env, WrapAST(test->expr, TextLiteral, .cord=test->expr->file->filename)),
|
2024-02-17 17:07:04 -08:00
|
|
|
(int64_t)(test->expr->start - test->expr->file->text),
|
|
|
|
(int64_t)(test->expr->end - test->expr->file->text));
|
2024-02-11 23:07:12 -08:00
|
|
|
} else {
|
|
|
|
return CORD_asprintf(
|
2024-02-22 09:45:12 -08:00
|
|
|
"{ // Test:\n%r $expr = %r;\n"
|
2024-02-20 10:06:03 -08:00
|
|
|
"__doctest(&$expr, %r, %r, %r, %ld, %ld);\n"
|
|
|
|
"}",
|
|
|
|
compile_type(expr_t),
|
2024-02-13 11:42:33 -08:00
|
|
|
compile(env, test->expr),
|
2024-02-17 17:07:04 -08:00
|
|
|
compile_type_info(env, expr_t),
|
2024-03-03 16:12:53 -08:00
|
|
|
compile(env, WrapAST(test->expr, TextLiteral, .cord=output)),
|
2024-03-03 15:15:45 -08:00
|
|
|
compile(env, WrapAST(test->expr, TextLiteral, .cord=test->expr->file->filename)),
|
2024-02-17 17:07:04 -08:00
|
|
|
(int64_t)(test->expr->start - test->expr->file->text),
|
|
|
|
(int64_t)(test->expr->end - test->expr->file->text));
|
2024-02-11 23:07:12 -08:00
|
|
|
}
|
2024-02-11 20:09:00 -08:00
|
|
|
}
|
2024-02-11 22:46:32 -08:00
|
|
|
case FieldAccess: {
|
|
|
|
auto f = Match(ast, FieldAccess);
|
2024-02-17 17:47:43 -08:00
|
|
|
type_t *fielded_t = get_type(env, f->fielded);
|
2024-02-20 10:06:03 -08:00
|
|
|
type_t *value_t = value_type(fielded_t);
|
|
|
|
switch (value_t->tag) {
|
2024-03-03 10:04:50 -08:00
|
|
|
case TypeInfoType: {
|
|
|
|
auto info = Match(value_t, TypeInfoType);
|
|
|
|
table_t *namespace = Table_str_get(env->type_namespaces, info->name);
|
|
|
|
if (!namespace) code_err(f->fielded, "I couldn't find a namespace for this type");
|
|
|
|
binding_t *b = Table_str_get(namespace, f->field);
|
|
|
|
if (!b) code_err(ast, "I couldn't find the field '%s' on this type", f->field);
|
|
|
|
if (!b->code) code_err(ast, "I couldn't figure out how to compile this field");
|
|
|
|
return b->code;
|
|
|
|
}
|
2024-02-20 10:06:03 -08:00
|
|
|
case StructType: {
|
|
|
|
for (arg_t *field = Match(value_t, StructType)->fields; field; field = field->next) {
|
|
|
|
if (streq(field->name, f->field)) {
|
|
|
|
if (fielded_t->tag == PointerType) {
|
|
|
|
CORD fielded = compile_to_pointer_depth(env, f->fielded, 1, false);
|
|
|
|
return CORD_asprintf("(%r)->%s", fielded, f->field);
|
|
|
|
} else {
|
|
|
|
CORD fielded = compile(env, f->fielded);
|
|
|
|
return CORD_asprintf("(%r).%s", fielded, f->field);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
code_err(ast, "The field '%s' is not a valid field name of %T", f->field, value_t);
|
|
|
|
}
|
|
|
|
case EnumType: {
|
|
|
|
auto enum_ = Match(value_t, EnumType);
|
|
|
|
for (tag_t *tag = enum_->tags; tag; tag = tag->next) {
|
|
|
|
if (streq(tag->name, f->field)) {
|
|
|
|
CORD fielded = compile_to_pointer_depth(env, f->fielded, 0, false);
|
|
|
|
return CORD_asprintf("$tagged(%r, %s, %s)", fielded, enum_->name, f->field);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
code_err(ast, "The field '%s' is not a valid field name of %T", f->field, value_t);
|
|
|
|
}
|
2024-02-25 11:35:25 -08:00
|
|
|
case TableType: {
|
|
|
|
if (streq(f->field, "keys")) {
|
2024-02-29 09:54:29 -08:00
|
|
|
return CORD_all("({ table_t *$t = ", compile_to_pointer_depth(env, f->fielded, 1, false), ";\n"
|
|
|
|
"$t->entries.data_refcount = 3;\n"
|
|
|
|
"$t->entries; })");
|
2024-02-25 11:35:25 -08:00
|
|
|
} else if (streq(f->field, "values")) {
|
|
|
|
auto table = Match(value_t, TableType);
|
|
|
|
size_t offset = type_size(table->key_type);
|
|
|
|
size_t align = type_align(table->value_type);
|
|
|
|
if (align > 1 && offset % align > 0)
|
|
|
|
offset += align - (offset % align);
|
2024-02-29 09:54:29 -08:00
|
|
|
return CORD_all("({ table_t *$t = ", compile_to_pointer_depth(env, f->fielded, 1, false), ";\n"
|
|
|
|
"$t->entries.data_refcount = 3;\n"
|
|
|
|
"(array_t){.data = $t->entries.data + ", CORD_asprintf("%zu", offset),
|
|
|
|
",\n .length=$t->entries.length,\n .stride=$t->entries.stride,\n .data_refcount=3};})");
|
2024-02-25 11:35:25 -08:00
|
|
|
} else if (streq(f->field, "fallback")) {
|
|
|
|
return CORD_all("(", compile_to_pointer_depth(env, f->fielded, 0, false), ").fallback");
|
|
|
|
} else if (streq(f->field, "default")) {
|
|
|
|
return CORD_all("(", compile_to_pointer_depth(env, f->fielded, 0, false), ").default_value");
|
|
|
|
}
|
|
|
|
code_err(ast, "There is no '%s' field on tables", f->field);
|
|
|
|
}
|
2024-02-20 10:06:03 -08:00
|
|
|
default:
|
|
|
|
code_err(ast, "Field accesses are only supported on struct and enum values");
|
2024-02-17 17:47:43 -08:00
|
|
|
}
|
2024-02-11 22:46:32 -08:00
|
|
|
}
|
2024-02-18 11:28:35 -08:00
|
|
|
case Index: {
|
|
|
|
auto indexing = Match(ast, Index);
|
2024-02-25 09:20:25 -08:00
|
|
|
type_t *indexed_type = get_type(env, indexing->indexed);
|
|
|
|
if (!indexing->index && indexed_type->tag == PointerType) {
|
2024-02-29 10:00:28 -08:00
|
|
|
auto ptr = Match(indexed_type, PointerType);
|
|
|
|
if (ptr->is_optional)
|
|
|
|
code_err(ast, "This pointer is potentially null, so it can't be safely dereferenced");
|
|
|
|
if (ptr->pointed->tag == ArrayType) {
|
|
|
|
return CORD_all("({ array_t *$arr = ", compile(env, indexing->indexed), "; $arr->data_refcount = 3; *$arr; })");
|
|
|
|
} else if (ptr->pointed->tag == TableType) {
|
|
|
|
return CORD_all("({ table_t *$t = ", compile(env, indexing->indexed), "; Table_mark_copy_on_write($t); *$t; })");
|
|
|
|
} else {
|
|
|
|
return CORD_all("*(", compile(env, indexing->indexed), ")");
|
|
|
|
}
|
2024-02-25 09:20:25 -08:00
|
|
|
}
|
|
|
|
type_t *container_t = value_type(indexed_type);
|
2024-02-18 11:28:35 -08:00
|
|
|
type_t *index_t = get_type(env, indexing->index);
|
|
|
|
switch (container_t->tag) {
|
|
|
|
case ArrayType: {
|
|
|
|
if (index_t->tag != IntType)
|
|
|
|
code_err(indexing->index, "Arrays can only be indexed by integers, not %T", index_t);
|
|
|
|
type_t *item_type = Match(container_t, ArrayType)->item_type;
|
2024-02-25 10:04:35 -08:00
|
|
|
CORD arr = compile_to_pointer_depth(env, indexing->indexed, 0, false);
|
2024-02-18 11:28:35 -08:00
|
|
|
CORD index = compile(env, indexing->index);
|
2024-02-18 11:53:52 -08:00
|
|
|
file_t *f = indexing->index->file;
|
|
|
|
if (indexing->unchecked)
|
|
|
|
return CORD_all("$Array_get_unchecked", compile_type(item_type), ", ", arr, ", ", index, ")");
|
|
|
|
else
|
|
|
|
return CORD_all("$Array_get(", compile_type(item_type), ", ", arr, ", ", index, ", ",
|
2024-03-03 15:15:45 -08:00
|
|
|
Text__quoted(f->filename, false), ", ", CORD_asprintf("%ld", (int64_t)(indexing->index->start - f->text)), ", ",
|
2024-02-18 11:53:52 -08:00
|
|
|
CORD_asprintf("%ld", (int64_t)(indexing->index->end - f->text)),
|
|
|
|
")");
|
2024-02-18 11:28:35 -08:00
|
|
|
}
|
|
|
|
case TableType: {
|
|
|
|
type_t *key_t = Match(container_t, TableType)->key_type;
|
|
|
|
type_t *value_t = Match(container_t, TableType)->value_type;
|
|
|
|
if (!can_promote(index_t, key_t))
|
|
|
|
code_err(indexing->index, "This value has type %T, but this table can only be index with keys of type %T", index_t, key_t);
|
2024-02-20 10:06:03 -08:00
|
|
|
CORD table = compile_to_pointer_depth(env, indexing->indexed, 1, false);
|
2024-02-18 11:28:35 -08:00
|
|
|
CORD key = compile(env, indexing->index);
|
2024-02-18 11:53:52 -08:00
|
|
|
file_t *f = indexing->index->file;
|
2024-02-18 11:28:35 -08:00
|
|
|
return CORD_all("$Table_get(", table, ", ", compile_type(key_t), ", ", compile_type(value_t), ", ",
|
2024-02-18 11:53:52 -08:00
|
|
|
key, ", ", compile_type_info(env, container_t), ", ",
|
2024-03-03 15:15:45 -08:00
|
|
|
Text__quoted(f->filename, false), ", ", CORD_asprintf("%ld", (int64_t)(indexing->index->start - f->text)), ", ",
|
2024-02-18 11:53:52 -08:00
|
|
|
CORD_asprintf("%ld", (int64_t)(indexing->index->end - f->text)),
|
|
|
|
")");
|
2024-02-18 11:28:35 -08:00
|
|
|
}
|
|
|
|
default: code_err(ast, "Indexing is not supported for type: %T", container_t);
|
|
|
|
}
|
|
|
|
}
|
2024-02-04 12:23:59 -08:00
|
|
|
// Use,
|
|
|
|
// LinkerDirective,
|
2024-02-25 13:02:36 -08:00
|
|
|
case InlineCCode: return Match(ast, InlineCCode)->code;
|
2024-02-17 13:56:19 -08:00
|
|
|
case Unknown: code_err(ast, "Unknown AST");
|
2024-02-29 10:49:24 -08:00
|
|
|
case Lambda: code_err(ast, "Lambdas are not supported yet");
|
|
|
|
case Use: code_err(ast, "Uses are not supported yet");
|
|
|
|
case LinkerDirective: code_err(ast, "Linker directives are not supported yet");
|
|
|
|
case Extern: code_err(ast, "Externs are not supported yet");
|
|
|
|
case TableEntry: code_err(ast, "Table entries should not be compiled directly");
|
2024-02-04 12:23:59 -08:00
|
|
|
}
|
2024-02-29 10:49:24 -08:00
|
|
|
code_err(ast, "Unknown AST: %W", ast);
|
2024-02-04 12:23:59 -08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2024-03-06 09:41:18 -08:00
|
|
|
void compile_namespace(env_t *env, const char *ns_name, ast_t *block)
|
|
|
|
{
|
|
|
|
env_t *ns_env = namespace_env(env, ns_name);
|
|
|
|
for (ast_list_t *stmt = block ? Match(block, Block)->statements : NULL; stmt; stmt = stmt->next) {
|
|
|
|
ast_t *ast = stmt->ast;
|
|
|
|
switch (ast->tag) {
|
|
|
|
case FunctionDef:
|
|
|
|
CORD code = compile_statement(ns_env, ast);
|
|
|
|
env->code->funcs = CORD_cat(env->code->funcs, code);
|
|
|
|
break;
|
|
|
|
case Declare: {
|
|
|
|
auto decl = Match(ast, Declare);
|
|
|
|
type_t *t = get_type(ns_env, decl->value);
|
|
|
|
|
|
|
|
CORD var_decl = CORD_all(compile_type(t), " ", compile(ns_env, decl->var), ";\n");
|
|
|
|
env->code->staticdefs = CORD_cat(env->code->staticdefs, var_decl);
|
|
|
|
|
|
|
|
CORD init = CORD_all(compile(ns_env, decl->var), " = ", compile(ns_env, decl->value), ";\n");
|
|
|
|
env->code->main = CORD_cat(env->code->main, init);
|
|
|
|
|
|
|
|
env->code->fndefs = CORD_all(env->code->fndefs, "extern ", compile_type(t), " ", compile(ns_env, decl->var), ";\n");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
CORD code = compile_statement(ns_env, ast);
|
|
|
|
env->code->main = CORD_cat(env->code->main, code);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-17 15:38:29 -08:00
|
|
|
CORD compile_type_info(env_t *env, type_t *t)
|
|
|
|
{
|
|
|
|
switch (t->tag) {
|
2024-03-03 10:37:05 -08:00
|
|
|
case BoolType: case IntType: case NumType: return CORD_asprintf("&%r", type_to_cord(t));
|
2024-03-03 15:15:45 -08:00
|
|
|
case TextType: return CORD_all("(&", Match(t, TextType)->dsl ? Match(t, TextType)->dsl : "Text", ")");
|
2024-03-03 11:34:13 -08:00
|
|
|
case StructType: return CORD_all("(&", Match(t, StructType)->name, ")");
|
|
|
|
case EnumType: return CORD_all("(&", Match(t, EnumType)->name, ")");
|
2024-02-17 15:38:29 -08:00
|
|
|
case ArrayType: {
|
|
|
|
type_t *item_t = Match(t, ArrayType)->item_type;
|
2024-02-20 10:06:03 -08:00
|
|
|
return CORD_asprintf("$ArrayInfo(%r)", compile_type_info(env, item_t));
|
2024-02-17 15:38:29 -08:00
|
|
|
}
|
|
|
|
case TableType: {
|
|
|
|
type_t *key_type = Match(t, TableType)->key_type;
|
|
|
|
type_t *value_type = Match(t, TableType)->value_type;
|
2024-02-20 10:06:03 -08:00
|
|
|
return CORD_asprintf("$TableInfo(%r, %r)", compile_type_info(env, key_type), compile_type_info(env, value_type));
|
2024-02-17 15:38:29 -08:00
|
|
|
}
|
|
|
|
case PointerType: {
|
|
|
|
auto ptr = Match(t, PointerType);
|
|
|
|
CORD sigil = ptr->is_stack ? "&" : (ptr->is_optional ? "?" : "@");
|
|
|
|
if (ptr->is_readonly) sigil = CORD_cat(sigil, "(readonly)");
|
2024-03-03 15:15:45 -08:00
|
|
|
return CORD_asprintf("$PointerInfo(%r, %r)", Text__quoted(sigil, false), compile_type_info(env, ptr->pointed));
|
2024-02-17 15:38:29 -08:00
|
|
|
}
|
|
|
|
case FunctionType: {
|
2024-03-03 15:15:45 -08:00
|
|
|
return CORD_asprintf("$FunctionInfo(%r)", Text__quoted(type_to_cord(t), false));
|
2024-02-17 15:38:29 -08:00
|
|
|
}
|
|
|
|
case ClosureType: {
|
|
|
|
errx(1, "No typeinfo for closures yet");
|
|
|
|
}
|
2024-02-27 10:39:12 -08:00
|
|
|
case TypeInfoType: return "&TypeInfo_info";
|
2024-02-26 20:10:19 -08:00
|
|
|
default:
|
|
|
|
compiler_err(NULL, 0, 0, "I couldn't convert to a type info: %T", t);
|
2024-02-17 15:38:29 -08:00
|
|
|
}
|
|
|
|
}
|
2024-02-17 14:29:56 -08:00
|
|
|
|
2024-02-17 23:22:31 -08:00
|
|
|
module_code_t compile_file(ast_t *ast)
|
2024-02-17 23:17:44 -08:00
|
|
|
{
|
|
|
|
env_t *env = new_compilation_unit();
|
2024-03-03 13:08:38 -08:00
|
|
|
CORD_appendf(&env->code->imports, "#include <tomo.h>\n");
|
2024-02-17 23:17:44 -08:00
|
|
|
|
|
|
|
for (ast_list_t *stmt = Match(ast, Block)->statements; stmt; stmt = stmt->next) {
|
2024-02-24 11:29:40 -08:00
|
|
|
bind_statement(env, stmt->ast);
|
2024-02-17 23:17:44 -08:00
|
|
|
CORD code = compile_statement(env, stmt->ast);
|
|
|
|
if (code)
|
|
|
|
CORD_appendf(&env->code->main, "%r\n", code);
|
|
|
|
}
|
|
|
|
|
2024-03-03 12:18:15 -08:00
|
|
|
const char *slash = strrchr(ast->file->filename, '/');
|
|
|
|
const char *name = slash ? slash+1 : ast->file->filename;
|
2024-03-03 12:04:36 -08:00
|
|
|
size_t name_len = 0;
|
|
|
|
while (name[name_len] && (isalnum(name[name_len]) || name[name_len] == '_'))
|
|
|
|
++name_len;
|
2024-03-03 12:18:15 -08:00
|
|
|
if (name_len == 0)
|
|
|
|
errx(1, "No module name found for: %s", ast->file->filename);
|
2024-03-03 12:04:36 -08:00
|
|
|
const char *module_name = heap_strn(name, name_len);
|
|
|
|
|
2024-02-17 23:22:31 -08:00
|
|
|
return (module_code_t){
|
2024-03-03 12:04:36 -08:00
|
|
|
.module_name=module_name,
|
2024-02-17 23:22:31 -08:00
|
|
|
.header=CORD_all(
|
2024-03-03 15:15:45 -08:00
|
|
|
// CORD_asprintf("#line 0 %r\n", Text__quoted(ast->file->filename, false)),
|
2024-02-17 23:22:31 -08:00
|
|
|
env->code->imports, "\n",
|
|
|
|
env->code->typedefs, "\n",
|
2024-03-03 11:12:40 -08:00
|
|
|
env->code->typecode, "\n",
|
2024-03-03 12:04:36 -08:00
|
|
|
env->code->fndefs, "\n",
|
2024-03-06 21:44:57 -08:00
|
|
|
"public void use(void);\n"
|
2024-03-03 12:04:36 -08:00
|
|
|
),
|
2024-02-17 23:22:31 -08:00
|
|
|
.c_file=CORD_all(
|
2024-03-03 15:15:45 -08:00
|
|
|
// CORD_asprintf("#line 0 %r\n", Text__quoted(ast->file->filename, false)),
|
2024-02-17 23:22:31 -08:00
|
|
|
env->code->staticdefs, "\n",
|
|
|
|
env->code->funcs, "\n",
|
|
|
|
env->code->typeinfos, "\n",
|
|
|
|
"\n"
|
2024-03-06 21:44:57 -08:00
|
|
|
"public void use(void) {\n",
|
2024-02-17 23:22:31 -08:00
|
|
|
env->code->main,
|
|
|
|
"}\n"
|
|
|
|
),
|
|
|
|
};
|
2024-02-17 23:17:44 -08:00
|
|
|
}
|
|
|
|
|
2024-02-04 12:23:59 -08:00
|
|
|
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1,\:0
|