summaryrefslogtreecommitdiff
path: root/types.c
diff options
context:
space:
mode:
Diffstat (limited to 'types.c')
-rw-r--r--types.c242
1 files changed, 208 insertions, 34 deletions
diff --git a/types.c b/types.c
index 36f6b81..f103f5d 100644
--- a/types.c
+++ b/types.c
@@ -585,6 +585,7 @@ enum {
};
static Status type_of_fn(Typer *tr, FnExpr *f, Type *t, U16 flags) {
+
if (f->flags & FN_EXPR_FOREIGN) {
/* we've already mostly determined the type in parse_expr */
if (!type_resolve(tr, &f->foreign.type, f->where))
@@ -2969,6 +2970,10 @@ static Status types_block(Typer *tr, Block *b) {
arr_foreach(b->stmts, Statement, s) {
if (!types_stmt(tr, s)) {
success = false;
+ if (tr->had_include_err) {
+ /* stop immediately; prevent too many "undeclared identifier" errors */
+ break;
+ }
continue;
}
}
@@ -3031,6 +3036,7 @@ static Status types_decl(Typer *tr, Declaration *d) {
&& tr->fn == NULL) {
e->typeval->struc->name = d->idents[0];
}
+
if (!types_expr(tr, e)) {
success = false;
goto ret;
@@ -3227,6 +3233,33 @@ static Status fix_ident_decls_inline_block(Typer *tr, Statement *stmts) {
return true;
}
+/* introduce identifiers from stmts into current scope, setting their "nms" field to nms */
+static Status include_stmts_link_to_nms(Typer *tr, Namespace *nms, Statement *stmts) {
+ Identifiers *idents = typer_get_idents(tr);
+ arr_foreach(stmts, Statement, s) {
+ if (s->kind == STMT_INLINE_BLOCK) {
+ if (!include_stmts_link_to_nms(tr, nms, s->inline_block))
+ return false;
+ } else if (s->kind == STMT_DECL) {
+ Declaration *d = s->decl;
+ arr_foreach(d->idents, Identifier, ident) {
+ /* @OPTIM: only hash once */
+ Identifier preexisting = ident_translate(*ident, idents);
+ if (preexisting && preexisting->decl != d) {
+ char *istr = ident_to_str(preexisting);
+ err_print(d->where, "Redeclaration of identifier %s.", istr);
+ info_print(preexisting->decl->where, "%s was first declared here.", istr);
+ free(istr);
+ }
+ Identifier i = ident_translate_forced(*ident, idents);
+ i->nms = nms;
+ i->decl = d;
+ }
+ }
+ }
+ return true;
+}
+
static Status types_stmt(Typer *tr, Statement *s) {
top:
if (s->flags & STMT_TYPED) return true;
@@ -3249,14 +3282,8 @@ top:
free(str);
}
}
- if (tr->block == NULL) {
- /* evaluate expression statements at global scope */
- if (e->kind != EXPR_C) {
- if (!eval_stmt(tr->evalr, s))
- return false;
- }
- }
} break;
+
case STMT_FOR: {
bool in_header = true;
Block *prev_block = tr->block; { /* additional block because c++ */
@@ -3711,6 +3738,139 @@ top:
}
}
} break;
+ case STMT_INCLUDE: {
+ Include *inc = s->inc;
+ char *filename = eval_expr_as_cstr(tr, &inc->filename, "import filename");
+ if (!filename)
+ return false;
+ Namespace *prev_nms = tr->nms;
+ Block *prev_block = tr->block;
+ IncludedFile *inc_f = NULL;
+ Namespace *inc_nms = NULL; /* non-NULL if this is an include to nms */
+ bool success = true;
+ if (inc->nms) {
+ inc_nms = typer_calloc(tr, 1, sizeof *inc_nms);
+
+ Block *body = &inc_nms->body;
+ body->kind = BLOCK_NMS;
+ body->where = s->where;
+ idents_create(&body->idents, tr->allocr, body);
+ body->parent = tr->block;
+
+ inc_nms->inc_file = inc_f;
+ /* turn #include "foo", bar into bar ::= nms { ... } */
+ s->kind = STMT_DECL;
+ Declaration *d = s->decl = typer_calloc(tr, 1, sizeof *d);
+ d->flags = DECL_FOUND_TYPE | DECL_HAS_EXPR | DECL_IS_CONST | DECL_FOUND_VAL;
+ construct_resolved_builtin_type(&d->type, BUILTIN_NMS);
+ char *ident_str = inc->nms;
+ Identifier i = ident_insert(typer_get_idents(tr), &ident_str);
+ if (i->decl) {
+ Declaration *d2 = i->decl;
+ /* maybe they included it twice into one namespace */
+ if ((d2->flags & DECL_HAS_EXPR) && (d2->expr.kind == EXPR_NMS) &&
+ (d2->expr.nms->inc_file == inc_f)) {
+ /* that's okay; get rid of this declaration */
+ s->kind = STMT_INLINE_BLOCK;
+ s->inline_block = NULL;
+ break;
+ } else {
+ char *istr = ident_to_str(i);
+ err_print(s->where, "Redeclaration of identifier %s.", istr);
+ info_print(ident_decl_location(i), "Previous declaration was here.");
+ free(istr);
+ return false; /* NOT goto inc_fail; */
+ }
+ }
+ typer_arr_add(tr, d->idents, i);
+ i->decl = d;
+ if (typer_is_at_top_level(tr)) inc_nms->associated_ident = i;
+ typer_gen_nms_prefix(tr, inc_nms);
+
+ d->expr.kind = EXPR_NMS;
+ d->expr.nms = inc_nms;
+ d->expr.flags = EXPR_FOUND_TYPE;
+ d->expr.type = d->type;
+ d->val.nms = inc_nms;
+ d->where = d->expr.where = s->where;
+
+ /* go inside namespace and block (it'll help to be there later on) */
+ tr->nms = inc_nms;
+ typer_block_enter(tr, &inc_nms->body);
+ } else {
+ s->kind = STMT_INLINE_BLOCK;
+ }
+
+ if (!(inc->flags & INC_FORCED)) {
+ size_t filename_len = strlen(filename);
+ if (streq(filename, tr->gctx->main_file->filename)) {
+ err_print(s->where, "Circular #include detected. You can add #force to this #include to force it to be included.");
+ success = false; goto nms_done;
+ }
+ inc_f = str_hash_table_get(&tr->included_files, filename, filename_len);
+ if (inc_f) {
+ /* has already been included */
+ if (inc_f->flags & INC_FILE_INCLUDING) {
+ err_print(s->where, "Circular #include detected. You can add #force to this #include to force it to be included.");
+ success = false; goto nms_done;
+ }
+ if (s->kind == STMT_INLINE_BLOCK) s->inline_block = NULL; /* nothing needed here */
+ /* just set ident declarations */
+ if (!include_stmts_link_to_nms(tr, inc_f->main_nms, inc_f->stmts)) {
+ success = false; goto nms_done;
+ }
+ goto nms_done;
+ }
+ inc_f = str_hash_table_insert(&tr->included_files, filename, filename_len);
+ inc_f->flags |= INC_FILE_INCLUDING;
+ inc_f->main_nms = tr->nms;
+ }
+ {
+ char *contents = read_file_contents(tr->allocr, filename, s->where);
+ if (!contents) {
+ tr->had_include_err = true;
+ success = false; goto nms_done;
+ }
+
+ Tokenizer tokr;
+ tokr_create(&tokr, tr->gctx->err_ctx, tr->allocr);
+ File *file = typer_calloc(tr, 1, sizeof *file);
+ file->filename = filename;
+ file->contents = contents;
+ file->ctx = tr->gctx->err_ctx;
+
+ if (!tokenize_file(&tokr, file)) {
+ success = false; goto nms_done;
+ }
+ Parser parser;
+ parser_create(&parser, tr->globals, &tokr, tr->allocr, tr->gctx);
+ parser.block = tr->block;
+ ParsedFile parsed_file;
+ if (!parse_file(&parser, &parsed_file)) {
+ success = false; goto nms_done;
+ }
+ Statement *stmts_inc = parsed_file.stmts;
+ if (inc_f) {
+ inc_f->stmts = stmts_inc;
+ }
+ if (s->kind == STMT_INLINE_BLOCK) s->inline_block = stmts_inc;
+ arr_foreach(stmts_inc, Statement, s_incd) {
+ if (!types_stmt(tr, s_incd)) {
+ success = false; goto nms_done;
+ }
+ }
+ if (inc_nms) {
+ inc_nms->body.stmts = stmts_inc;
+ }
+ }
+ nms_done:
+ if (inc_nms) {
+ tr->nms = prev_nms;
+ tr->block = prev_block;
+ }
+ if (inc_f) inc_f->flags &= (IncFileFlags)~(IncFileFlags)INC_FILE_INCLUDING;
+ if (!success) return false;
+ } break;
case STMT_MESSAGE: {
Message *m = s->message;
char *text = eval_expr_as_cstr(tr, &m->text, "message");
@@ -3782,51 +3942,65 @@ top:
else
typer_arr_add(tr, tr->uses, u);
} break;
- case STMT_INLINE_BLOCK: {
- bool success = true;
- arr_foreach(s->inline_block, Statement, sub) {
- if (!types_stmt(tr, sub))
- success = false;
- }
- if (!success) return false;
- } break;
+ case STMT_INLINE_BLOCK:
+ assert(0); /* only exists after typing */
+ break;
}
success:
s->flags |= STMT_TYPED;
+ if (tr->block == NULL) {
+ /* evaluate statements at global scope */
+ switch (s->kind) {
+ case STMT_DECL:
+ case STMT_USE:
+ case STMT_INLINE_BLOCK: /* have already been evaluated */
+ case STMT_MESSAGE:
+ break;
+ case STMT_EXPR:
+ case STMT_IF:
+ case STMT_FOR:
+ case STMT_WHILE:
+ case STMT_BLOCK:
+ if (s->expr->kind != EXPR_C) {
+ if (!eval_stmt(tr->evalr, s))
+ return false;
+ }
+ break;
+ case STMT_RET:
+ case STMT_BREAK:
+ case STMT_CONT:
+ case STMT_INCLUDE:
+ case STMT_DEFER:
+ assert(0);
+ break;
+ }
+ }
+ if (s->flags & STMT_IS_INIT) {
+ Initialization *init = typer_arr_add_ptr(tr, tr->gctx->inits);
+ init->stmt = s;
+ }
return true;
}
-static void typer_create(Typer *tr, Evaluator *ev, Allocator *allocr, Identifiers *idents, GlobalCtx *gctx) {
+static void typer_create(Typer *tr, Evaluator *ev, Allocator *allocr, Identifiers *globals, GlobalCtx *gctx) {
memset(tr, 0, sizeof *tr);
tr->evalr = ev;
tr->allocr = allocr;
- tr->globals = idents;
+ tr->globals = globals;
tr->gctx = gctx;
-}
-
-static int compare_inits(const void *av, const void *bv) {
- const Initialization *a = av, *b = bv;
- if (a->priority < b->priority) return -1;
- if (a->priority > b->priority) return +1;
- return 0;
+ str_hash_table_create(&tr->included_files, sizeof(IncludedFile), tr->allocr);
}
static Status types_file(Typer *tr, ParsedFile *f) {
bool ret = true;
tr->parsed_file = f;
tr->uses = NULL;
- Initialization *inits = tr->gctx->inits;
- qsort(inits, arr_len(inits), sizeof *inits, compare_inits);
- arr_foreach(inits, Initialization, init) {
- Statement *s = &init->stmt;
- if (!types_stmt(tr, s))
- return false;
- if (s->kind != STMT_EXPR) /* already evaluated in types_stmt */
- if (!eval_stmt(tr->evalr, s))
- return false;
- }
arr_foreach(f->stmts, Statement, s) {
if (!types_stmt(tr, s)) {
+ if (tr->had_include_err) {
+ /* stop immediately; prevent too many "undeclared identifier" errors */
+ return false;
+ }
ret = false;
}
}