summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--package.c55
-rw-r--r--test.toc19
-rw-r--r--types.c2
-rw-r--r--types.h6
4 files changed, 44 insertions, 38 deletions
diff --git a/package.c b/package.c
index a425ecb..ccb4ffc 100644
--- a/package.c
+++ b/package.c
@@ -5,7 +5,7 @@
*/
static bool export_decl(Exporter *ex, Declaration *d);
static bool export_block(Exporter *ex, Block *b);
-
+static bool export_expr(Exporter *ex, Expression *e);
static void exptr_create(Exporter *ex, FILE *out) {
ex->out = out;
@@ -223,6 +223,16 @@ static bool export_val(Exporter *ex, Value val, Type *type, Location where) {
return export_val_ptr(ex, val_get_ptr(&val, type), type, where);
}
+/* e can be NULL! */
+static inline bool export_optional_expr(Exporter *ex, Expression *e) {
+ bool has_e = e != NULL;
+ export_bool(ex, has_e);
+ if (has_e)
+ return export_expr(ex, e);
+ else
+ return true;
+}
+
static bool export_expr(Exporter *ex, Expression *e) {
assert(e->flags & EXPR_FOUND_TYPE);
if (!export_type(ex, &e->type, e->where))
@@ -303,11 +313,8 @@ static bool export_expr(Exporter *ex, Expression *e) {
case EXPR_NEW:
if (!export_type(ex, &e->new.type, e->where))
return false;
- bool has_n = e->new.n != NULL;
- export_bool(ex, has_n);
- if (has_n)
- if (!export_expr(ex, e->new.n))
- return false;
+ if (!export_optional_expr(ex, e->new.n))
+ return false;
break;
case EXPR_CAST:
if (!export_expr(ex, e->cast.expr)
@@ -325,17 +332,26 @@ static bool export_expr(Exporter *ex, Expression *e) {
} break;
case EXPR_IF: {
IfExpr *i = &e->if_;
- bool has_cond = i->cond != NULL;
- export_bool(ex, has_cond);
- if (has_cond) {
- if (!export_expr(ex, i->cond))
- return false;
- }
+ if (!export_optional_expr(ex, i->cond))
+ return false;
if (!export_block(ex, &i->body)) return false;
- bool has_next = i->next_elif != NULL;
- if (has_next)
- if (!export_expr(ex, i->next_elif))
- return false;
+ if (!export_optional_expr(ex, i->next_elif))
+ return false;
+ } break;
+ case EXPR_WHILE: {
+ WhileExpr *w = &e->while_;
+ if (!export_optional_expr(ex, w->cond))
+ return false;
+ if (!export_block(ex, &w->body))
+ return false;
+ } break;
+ case EXPR_SLICE: {
+ SliceExpr *s = &e->slice;
+ if (!export_expr(ex, s->of)) return false;
+ if (!export_optional_expr(ex, s->from))
+ return false;
+ if (!export_optional_expr(ex, s->to))
+ return false;
} break;
case EXPR_DSIZEOF:
case EXPR_DALIGNOF:
@@ -417,11 +433,8 @@ static bool export_block(Exporter *ex, Block *b) {
if (!export_stmt(ex, s))
return false;
}
- bool has_ret_expr = b->ret_expr != NULL;
- export_bool(ex, has_ret_expr);
- if (has_ret_expr)
- if (!export_expr(ex, b->ret_expr))
- return false;
+ if (!export_optional_expr(ex, b->ret_expr))
+ return false;
return true;
}
diff --git a/test.toc b/test.toc
index 79e0d43..d4726ff 100644
--- a/test.toc
+++ b/test.toc
@@ -2,17 +2,10 @@ Foo ::= struct {
x: u8;
};
#export main ::= fn() Foo {
- g ::= fn() int { 3 };
-
- f: Foo;
- if f.x == 0 {
- f.x = g() as u8;
- } elif f.x == 1 {
- f.x = 2;
- } elif f.x == 3 {
- f.x = 17;
- } else {
- f.x = 112;
- }
- f
+ g ::= fn() int { 3 };
+ a : [3]int;
+ b := a[1:3];
+ b[0] = 7;
+ f: Foo;
+ f
}; \ No newline at end of file
diff --git a/types.c b/types.c
index d823d15..e2a1bf1 100644
--- a/types.c
+++ b/types.c
@@ -1284,7 +1284,7 @@ static bool types_expr(Typer *tr, Expression *e) {
if (!infer_ident_vals(tr, decl_types, arg_types, inferred_idents, inferred_vals, inferred_types))
return false;
- allocr_free(tr->allocr, inferred_idents, ninferred_idents * sizeof *inferred_idents);
+ arr_cleara(&inferred_idents, tr->allocr);
{
Type *type = inferred_types;
diff --git a/types.h b/types.h
index 26efd86..9100652 100644
--- a/types.h
+++ b/types.h
@@ -590,9 +590,9 @@ typedef struct NewExpr {
} NewExpr;
typedef struct SliceExpr {
- struct Expression *of;
- struct Expression *from;
- struct Expression *to;
+ struct Expression *of; /* required */
+ struct Expression *from; /* optional */
+ struct Expression *to; /* optional */
struct {
IdentID id;
} c;