aboutsummaryrefslogtreecommitdiff
path: root/src/checker/stmt.cpp
diff options
context:
space:
mode:
authorGinger Bill <bill@gingerbill.org>2016-08-30 23:32:04 +0100
committerGinger Bill <bill@gingerbill.org>2016-08-30 23:32:04 +0100
commitcda0234d487ab73a1c87cbdcd74e300718ca7d0a (patch)
treebf7ccddca7a7c0146cfdba35ab731f93edd25201 /src/checker/stmt.cpp
parenta06f70d5d95bb7889bf9e8b920d70fd10daf7c12 (diff)
Subtyping Polymorphic arguments; `using` procedure parameters
Diffstat (limited to 'src/checker/stmt.cpp')
-rw-r--r--src/checker/stmt.cpp81
1 files changed, 58 insertions, 23 deletions
diff --git a/src/checker/stmt.cpp b/src/checker/stmt.cpp
index 1bbaf1040..752fc43c4 100644
--- a/src/checker/stmt.cpp
+++ b/src/checker/stmt.cpp
@@ -296,6 +296,39 @@ void check_proc_body(Checker *c, Token token, DeclInfo *decl, Type *type, AstNod
c->context.scope = decl->scope;
c->context.decl = decl;
+ GB_ASSERT(type->kind == Type_Proc);
+ if (type->Proc.param_count > 0) {
+ auto *params = &type->Proc.params->Tuple;
+ for (isize i = 0; i < params->variable_count; i++) {
+ Entity *e = params->variables[i];
+ GB_ASSERT(e->kind == Entity_Variable);
+ if (!e->Variable.anonymous)
+ continue;
+ String name = e->token.string;
+ Type *t = get_base_type(type_deref(e->type));
+ if (t->kind == Type_Struct || t->kind == Type_Union) {
+ // IMPORTANT HACK(bill): Entity_(Struct|Union) overlap in some memory allowing
+ // for some variables to accessed to same
+ Scope **found = map_get(&c->info.scopes, hash_pointer(t->Struct.node));
+ GB_ASSERT(found != NULL);
+ gb_for_array(i, (*found)->elements.entries) {
+ Entity *f = (*found)->elements.entries[i].value;
+ if (f->kind == Entity_Variable) {
+ Entity *uvar = make_entity_using_variable(c->allocator, e, f->token, f->type);
+ Entity *prev = scope_insert_entity(c->context.scope, uvar);
+ if (prev != NULL) {
+ error(&c->error_collector, e->token, "Namespace collision while `using` `%.*s` of: %.*s", LIT(name), LIT(prev->token.string));
+ break;
+ }
+ }
+ }
+ } else {
+ error(&c->error_collector, e->token, "`using` can only be applied to variables of type struct or union");
+ break;
+ }
+ }
+ }
+
push_procedure(c, type);
ast_node(bs, BlockStmt, body);
// TODO(bill): Check declarations first (except mutable variable declarations)
@@ -789,6 +822,7 @@ void check_stmt(Checker *c, AstNode *node, u32 flags) {
case_ast_node(us, UsingStmt, node);
switch (us->node->kind) {
case_ast_node(es, ExprStmt, us->node);
+ // TODO(bill): Allow for just a LHS expression list rather than this silly code
Entity *e = NULL;
b32 is_selector = false;
@@ -889,40 +923,41 @@ void check_stmt(Checker *c, AstNode *node, u32 flags) {
} break;
default:
- GB_PANIC("TODO(bill): using Ident");
+ GB_PANIC("TODO(bill): using other expressions?");
}
case_end;
case_ast_node(vd, VarDecl, us->node);
- if (vd->name_count > 1) {
+ if (vd->name_count > 1 && vd->type != NULL) {
error(&c->error_collector, us->token, "`using` can only be applied to one variable of the same type");
}
check_var_decl(c, us->node);
- ast_node(i, Ident, vd->name_list);
-
- String name = i->token.string;
- Entity *e = scope_lookup_entity(c, c->context.scope, name);
- Type *t = get_base_type(type_deref(e->type));
- if (t->kind == Type_Struct || t->kind == Type_Union) {
- // IMPORTANT HACK(bill): Entity_(Struct|Union) overlap in some memory allowing
- // for some variables to accessed to same
- Scope **found = map_get(&c->info.scopes, hash_pointer(t->Struct.node));
- GB_ASSERT(found != NULL);
- gb_for_array(i, (*found)->elements.entries) {
- Entity *f = (*found)->elements.entries[i].value;
- if (f->kind == Entity_Variable) {
- Entity *uvar = make_entity_using_variable(c->allocator, e, f->token, f->type);
- Entity *prev = scope_insert_entity(c->context.scope, uvar);
- if (prev != NULL) {
- error(&c->error_collector, us->token, "Namespace collision while `using` `%.*s` of: %.*s", LIT(name), LIT(prev->token.string));
- return;
+ for (AstNode *item = vd->name_list; item != NULL; item = item->next) {
+ ast_node(i, Ident, item);
+ String name = i->token.string;
+ Entity *e = scope_lookup_entity(c, c->context.scope, name);
+ Type *t = get_base_type(type_deref(e->type));
+ if (t->kind == Type_Struct || t->kind == Type_Union) {
+ // IMPORTANT HACK(bill): Entity_(Struct|Union) overlap in some memory allowing
+ // for some variables to accessed to same
+ Scope **found = map_get(&c->info.scopes, hash_pointer(t->Struct.node));
+ GB_ASSERT(found != NULL);
+ gb_for_array(i, (*found)->elements.entries) {
+ Entity *f = (*found)->elements.entries[i].value;
+ if (f->kind == Entity_Variable) {
+ Entity *uvar = make_entity_using_variable(c->allocator, e, f->token, f->type);
+ Entity *prev = scope_insert_entity(c->context.scope, uvar);
+ if (prev != NULL) {
+ error(&c->error_collector, us->token, "Namespace collision while `using` `%.*s` of: %.*s", LIT(name), LIT(prev->token.string));
+ return;
+ }
}
}
+ } else {
+ error(&c->error_collector, us->token, "`using` can only be applied to variables of type struct or union");
+ return;
}
- } else {
- error(&c->error_collector, us->token, "`using` can only be applied to variables of type struct or union");
- return;
}
case_end;