diff options
author | Leo Tenenbaum <pommicket@gmail.com> | 2019-11-03 19:08:58 -0500 |
---|---|---|
committer | Leo Tenenbaum <pommicket@gmail.com> | 2019-11-03 19:08:58 -0500 |
commit | 7a135cb2d258fee49880d3dd9f5f43096bd73331 (patch) | |
tree | e1a9a8c02d2911ce3e622e55d801a681ef108bfe | |
parent | 1d4195ce71dcb7c5ca62e68c399af5f091b3e947 (diff) |
fixed problems with eval decls
-rw-r--r-- | eval.c | 58 | ||||
-rw-r--r-- | main.c | 2 | ||||
-rw-r--r-- | parse.c | 5 | ||||
-rw-r--r-- | test.toc | 11 | ||||
-rw-r--r-- | typedefs_cgen.c | 6 | ||||
-rw-r--r-- | types.c | 5 |
6 files changed, 50 insertions, 37 deletions
@@ -731,7 +731,6 @@ static bool eval_address_of(Evaluator *ev, Expression *e, void **ptr) { return true; } - static bool eval_set(Evaluator *ev, Expression *set, Value *to) { switch (set->kind) { case EXPR_IDENT: { @@ -953,7 +952,6 @@ static bool eval_expr(Evaluator *ev, Expression *e, Value *v) { if (!eval_expr(ev, lhs_expr, &lhs)) return false; if (e->binary.op != BINARY_DOT) if (!eval_expr(ev, rhs_expr, &rhs)) return false; - BuiltinType builtin = e->binary.lhs->type.builtin; switch (e->binary.op) { case BINARY_DOT: { @@ -1038,7 +1036,8 @@ static bool eval_expr(Evaluator *ev, Expression *e, Value *v) { while (1) { if (w->cond) { if (!eval_expr(ev, w->cond, &cond)) return false; - if (!val_truthiness(&cond, &w->cond->type)) + Type *cond_type = &w->cond->type; + if (!val_truthiness(&cond, cond_type)) break; } if (!eval_block(ev, &w->body, &e->type, v)) return false; @@ -1069,6 +1068,7 @@ static bool eval_expr(Evaluator *ev, Expression *e, Value *v) { IdentDecl *idecl = ident_decl(e->ident); Declaration *d = idecl->decl; if (!types_decl(ev->typer, d)) return false; + assert(d->type.flags & TYPE_FLAG_RESOLVED); if (idecl->flags & IDECL_FLAG_HAS_VAL) { *v = idecl->val; } else if (d->flags & DECL_FLAG_CONST) { @@ -1087,7 +1087,7 @@ static bool eval_expr(Evaluator *ev, Expression *e, Value *v) { v->type->user.decl = d; v->type->user.index = index; } else { - *v = d->type.kind == TYPE_TUPLE ? d->val.tuple[index] : d->val; + *v = *decl_val_at_index(d, index); } } else { char *s = ident_to_str(e->ident); @@ -1232,42 +1232,46 @@ static bool eval_expr(Evaluator *ev, Expression *e, Value *v) { } static bool eval_decl(Evaluator *ev, Declaration *d) { - Value val = {0}; int has_expr = d->flags & DECL_FLAG_HAS_EXPR; + int is_const = d->flags & DECL_FLAG_CONST; + Value val = {0}; + if (has_expr) { - if (d->flags & DECL_FLAG_CONST) { + if (is_const) { if (!(d->flags & DECL_FLAG_FOUND_VAL)) { if (!eval_expr(ev, &d->expr, &d->val)) return false; d->flags |= DECL_FLAG_FOUND_VAL; } - val = d->val; } else { + /* TODO: tuples allocated here will never be freed! */ if (!eval_expr(ev, &d->expr, &val)) return false; } } - long index = 0; - arr_foreach(d->idents, Identifier, i) { - IdentDecl *id = ident_decl(*i); - Type *type = d->type.kind == TYPE_TUPLE ? &d->type.tuple[index] : &d->type; - Value *thisval = NULL; - if (has_expr) - thisval = d->type.kind == TYPE_TUPLE ? &val.tuple[index] : &val; - Type *inner = type_inner(type); - if (inner->kind == TYPE_STRUCT) { - id->val.struc = err_calloc(1, compiler_sizeof(inner)); - } else if (inner->kind == TYPE_ARR) { - id->val.arr = err_calloc(inner->arr.n, compiler_sizeof(inner->arr.of)); + + if (!is_const) { + int index = 0; + arr_foreach(d->idents, Identifier, i) { + IdentDecl *id = ident_decl(*i); + Type *type = decl_type_at_index(d, index); + Type *inner = type_inner(type); + if (!is_const) { + if (has_expr) { + val_copy(NULL, &id->val, &val, type); + } else { + if (inner->kind == TYPE_STRUCT) { + id->val.struc = err_calloc(1, compiler_sizeof(inner)); + } else if (inner->kind == TYPE_ARR) { + id->val.arr = err_calloc(inner->arr.n, compiler_sizeof(inner->arr.of)); + } else { + id->val = val; /* = (Value)({0}) */ + } + } + } + index++; + id->flags |= IDECL_FLAG_HAS_VAL; } - - if (has_expr) - val_copy(d->flags & DECL_FLAG_CONST ? ev : NULL, &id->val, thisval, type); - index++; - id->flags |= IDECL_FLAG_HAS_VAL; - } - if (has_expr && d->expr.kind == EXPR_TUPLE) { - val_free(&val, &d->type); /* free the tuple */ } return true; } @@ -1,6 +1,6 @@ /* TODO: -pointers to futurely-declared types +pointers to futurely-declared types (or to this type) for +=, -=, *=, /= compile-time arguments @@ -893,7 +893,10 @@ static bool parse_expr(Parser *p, Expression *e, Token *end) { if (parser_is_definitely_type(p, NULL)) { /* it's a type! */ e->kind = EXPR_TYPE; - return parse_type(p, &e->typeval); + if (!parse_type(p, &e->typeval)) + return false; + if (t->token == end) return true; + /* there's more stuff after. maybe it's, e.g. int, float */ } t->token = before; @@ -21,12 +21,21 @@ total @= fn() int { t }; +// Foo @= struct { +// x: &Bar; +// }; + +// Bar @= struct { +// y : &Foo; +// }; + main @= fn() { + // f: Foo; + puti(total()); X @= total(); puti(X); puti(#sizeof(int)); puti(#alignof(int)); - puti(#sizeof({ Foo, Bar @= int, struct { x: i64; y: char; } ; f: Foo; f })); }; diff --git a/typedefs_cgen.c b/typedefs_cgen.c index 58b9767..8228a24 100644 --- a/typedefs_cgen.c +++ b/typedefs_cgen.c @@ -90,10 +90,10 @@ static bool typedefs_expr(CGenerator *g, Expression *e) { } static bool typedefs_decl(CGenerator *g, Declaration *d) { - for (size_t idx = 0; idx < arr_len(d->idents); idx++) { + for (int idx = 0; idx < (int)arr_len(d->idents); idx++) { Identifier i = d->idents[idx]; - Type *type = d->type.kind == TYPE_TUPLE ? &d->type.tuple[idx] : &d->type; - Value *val = d->type.kind == TYPE_TUPLE ? &d->val.tuple[idx] : &d->val; + Type *type = decl_type_at_index(d, idx); + Value *val = decl_val_at_index(d, idx); if (type->kind == TYPE_TYPE) { /* generate typedef */ cgen_write(g, "typedef "); @@ -379,6 +379,7 @@ static bool type_resolve(Typer *tr, Type *t, Location where) { return false; break; case TYPE_USER: { + t->flags |= TYPE_FLAG_RESOLVED; /* pre-resolve type to avoid infinite recursion */ /* find declaration */ Identifier ident = t->user.ident; IdentDecl *idecl = ident_decl(ident); @@ -1333,10 +1334,6 @@ static bool types_decl(Typer *tr, Declaration *d) { success = false; goto ret; } - if (!type_resolve(tr, val->type, d->where)) { - success = false; - goto ret; - } if (val->type->kind == TYPE_TUPLE) { err_print(d->where, "You can't declare a new type to be a tuple."); success = false; |