From 7cf5a8735ecbd37faf5e9f740d5f1a49939eea07 Mon Sep 17 00:00:00 2001 From: Leo Tenenbaum Date: Fri, 27 Mar 2020 21:04:26 -0400 Subject: improved struct field lookup --- main.c | 3 ++- test.toc | 16 ++++++++-------- types.c | 19 ++++++++++--------- 3 files changed, 20 insertions(+), 18 deletions(-) diff --git a/main.c b/main.c index 0551ed9..a54d5c0 100644 --- a/main.c +++ b/main.c @@ -8,7 +8,8 @@ /* TODO: -make sure that struct["member"] still works +do we really need StructDef.constants +all big Statement members should be pointers use - use with a decl, e.g. use p : Point; local structs should not be named in C diff --git a/test.toc b/test.toc index 1323f3f..6e9af49 100644 --- a/test.toc +++ b/test.toc @@ -1,10 +1,10 @@ -main ::= fn() { - Point ::= struct { - x: int; - y, z: int; +s ::= fn(t:: Type) Type { + struct { + m: t; } - p: Point; - p.x = 6; - p.y = 13; - p.z = 43902; +} + +main ::= fn() { + o: s(int); + x := o.m; } diff --git a/types.c b/types.c index d210754..3ecb90b 100644 --- a/types.c +++ b/types.c @@ -2892,16 +2892,16 @@ static Status types_expr(Typer *tr, Expression *e) { /* replace with BINARY_DOT */ e->binary.op = BINARY_DOT; - bool is_field = false; if (!eval_expr(tr->evalr, rhs, &field_name)) return false; - arr_foreach(lhs_type->struc->fields, Field, f) { - if (ident_eq_str(f->name, field_name.slice.data)) { - is_field = true; - *t = *f->type; - e->binary.dot.field = f; - } - } - if (!is_field) { + + /* get the field, if it exists */ + Identifier ident = ident_get_with_len(&lhs_type->struc->body.idents, + field_name.slice.data, (size_t)field_name.slice.n); + if (ident_is_declared(ident)) { + assert(ident->decl_kind == IDECL_DECL); + Field *f = ident->decl->field + ident_index_in_decl(ident, ident->decl); + e->binary.dot.field = f; + } else { char *fstr = err_malloc((size_t)(field_name.slice.n + 1)); memcpy(fstr, field_name.slice.data, (size_t)field_name.slice.n); fstr[field_name.slice.n] = 0; /* null-terminate */ @@ -3013,6 +3013,7 @@ static Status types_expr(Typer *tr, Expression *e) { Field *field = struct_ident->decl->field; field += ident_index_in_decl(struct_ident, struct_ident->decl); e->binary.dot.field = field; + *t = *field->type; } else { if (!get_struct_constant(struct_type->struc, rhs->ident, e)) return false; -- cgit v1.2.3