aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorGinger Bill <bill@gingerbill.org>2017-06-12 16:26:51 +0100
committerGinger Bill <bill@gingerbill.org>2017-06-12 16:26:51 +0100
commita8e458339b65669d088edf36bb96ffb518f68d2d (patch)
tree19385cdda61504a6323b379994c1f22e10851989 /src
parent6b5e9aec8e0120669cad45056fbf268163b02233 (diff)
foreign_library allow for Pascal-style grouping
Diffstat (limited to 'src')
-rw-r--r--src/check_expr.cpp10
-rw-r--r--src/checker.cpp185
-rw-r--r--src/parser.cpp367
-rw-r--r--src/tokenizer.cpp2
4 files changed, 207 insertions, 357 deletions
diff --git a/src/check_expr.cpp b/src/check_expr.cpp
index cf35c30b3..2d0c94438 100644
--- a/src/check_expr.cpp
+++ b/src/check_expr.cpp
@@ -1123,13 +1123,10 @@ Type *check_get_params(Checker *c, Scope *scope, AstNode *_params, bool *is_vari
AstNode *name = p->names[j];
if (ast_node_expect(name, AstNode_Ident)) {
Entity *param = make_entity_param(c->allocator, scope, name->Ident, type,
- (p->flags&FieldFlag_using) != 0, (p->flags&FieldFlag_immutable) != 0);
+ (p->flags&FieldFlag_using) != 0, false);
if (p->flags&FieldFlag_no_alias) {
param->flags |= EntityFlag_NoAlias;
}
- if (p->flags&FieldFlag_immutable) {
- param->Variable.is_immutable = true;
- }
param->Variable.default_value = value;
param->Variable.default_is_nil = default_is_nil;
@@ -6568,11 +6565,8 @@ gbString write_expr_to_string(gbString str, AstNode *node) {
if (f->flags&FieldFlag_using) {
str = gb_string_appendc(str, "using ");
}
- if (f->flags&FieldFlag_immutable) {
- str = gb_string_appendc(str, "immutable ");
- }
if (f->flags&FieldFlag_no_alias) {
- str = gb_string_appendc(str, "no_alias ");
+ str = gb_string_appendc(str, "#no_alias ");
}
for_array(i, f->names) {
diff --git a/src/checker.cpp b/src/checker.cpp
index 9b6c81622..518b79e3e 100644
--- a/src/checker.cpp
+++ b/src/checker.cpp
@@ -1451,27 +1451,62 @@ void check_collect_entities(Checker *c, Array<AstNode *> nodes, bool is_file_sco
case_end;
case_ast_node(gd, GenDecl, decl);
+ AstNodeValueSpec empty_spec = {};
+ AstNodeValueSpec *last_spec = NULL;
for_array(i, gd->specs) {
AstNode *spec = gd->specs[i];
switch (gd->token.kind) {
+ case Token_const: {
+ ast_node(vs, ValueSpec, spec);
+
+ if (vs->type != NULL || vs->values.count > 0) {
+ last_spec = vs;
+ } else if (last_spec == NULL) {
+ last_spec = &empty_spec;
+ }
+
+ for_array(i, vs->names) {
+ AstNode *name = vs->names[i];
+ if (name->kind != AstNode_Ident) {
+ error_node(name, "A declaration's name must be an identifier, got %.*s", LIT(ast_node_strings[name->kind]));
+ continue;
+ }
+
+ AstNode *init = NULL;
+ if (i < vs->values.count) {
+ init = vs->values[i];
+ }
+
+ DeclInfo *d = make_declaration_info(c->allocator, c->context.scope, c->context.decl);
+ Entity *e = make_entity_constant(c->allocator, d->scope, name->Ident, NULL, empty_exact_value);
+ d->type_expr = last_spec->type;
+ d->init_expr = init;
+ e->identifier = name;
+
+ add_entity_and_decl_info(c, name, e, d);
+ }
+
+ check_arity_match(c, vs);
+ } break;
+
case Token_var:
case Token_let: {
if (!c->context.scope->is_file) {
// NOTE(bill): local scope -> handle later and in order
break;
}
- ast_node(vd, ValueSpec, spec);
+ ast_node(vs, ValueSpec, spec);
// NOTE(bill): You need to store the entity information here unline a constant declaration
- isize entity_cap = vd->names.count;
+ isize entity_cap = vs->names.count;
isize entity_count = 0;
Entity **entities = gb_alloc_array(c->allocator, Entity *, entity_cap);
DeclInfo *di = NULL;
- if (vd->values.count > 0) {
+ if (vs->values.count > 0) {
di = make_declaration_info(heap_allocator(), c->context.scope, c->context.decl);
di->entities = entities;
- di->type_expr = vd->type;
- di->init_expr = vd->values[0];
+ di->type_expr = vs->type;
+ di->init_expr = vs->values[0];
if (gd->flags & VarDeclFlag_thread_local) {
@@ -1480,11 +1515,11 @@ void check_collect_entities(Checker *c, Array<AstNode *> nodes, bool is_file_sco
}
- for_array(i, vd->names) {
- AstNode *name = vd->names[i];
+ for_array(i, vs->names) {
+ AstNode *name = vs->names[i];
AstNode *value = NULL;
- if (i < vd->values.count) {
- value = vd->values[i];
+ if (i < vs->values.count) {
+ value = vs->values[i];
}
if (name->kind != AstNode_Ident) {
error_node(name, "A declaration's name must be an identifier, got %.*s", LIT(ast_node_strings[name->kind]));
@@ -1504,7 +1539,7 @@ void check_collect_entities(Checker *c, Array<AstNode *> nodes, bool is_file_sco
if (d == NULL) {
AstNode *init_expr = value;
d = make_declaration_info(heap_allocator(), e->scope, c->context.decl);
- d->type_expr = vd->type;
+ d->type_expr = vs->type;
d->init_expr = init_expr;
}
@@ -1515,65 +1550,13 @@ void check_collect_entities(Checker *c, Array<AstNode *> nodes, bool is_file_sco
di->entity_count = entity_count;
}
- check_arity_match(c, vd);
- } break;
-
- case Token_const: {
- ast_node(vd, ValueSpec, spec);
-
- for_array(i, vd->names) {
- AstNode *name = vd->names[i];
- if (name->kind != AstNode_Ident) {
- error_node(name, "A declaration's name must be an identifier, got %.*s", LIT(ast_node_strings[name->kind]));
- continue;
- }
-
- AstNode *init = NULL;
- if (i < vd->values.count) {
- init = vd->values[i];
- }
-
- DeclInfo *d = make_declaration_info(c->allocator, c->context.scope, c->context.decl);
- Entity *e = NULL;
-
- AstNode *up_init = unparen_expr(init);
- // if (up_init != NULL && is_ast_node_type(up_init)) {
- // AstNode *type = up_init;
- // e = make_entity_type_name(c->allocator, d->scope, name->Ident, NULL);
- // // TODO(bill): What if vd->type != NULL??? How to handle this case?
- // d->type_expr = type;
- // d->init_expr = type;
- // } else if (up_init != NULL && up_init->kind == AstNode_Alias) {
- // #if 1
- // error_node(up_init, "#alias declarations are not yet supported");
- // continue;
- // #else
- // e = make_entity_alias(c->allocator, d->scope, name->Ident, NULL, EntityAlias_Invalid, NULL);
- // d->type_expr = vd->type;
- // d->init_expr = up_init->Alias.expr;
- // #endif
- // // } else if (init != NULL && up_init->kind == AstNode_ProcLit) {
- // // e = make_entity_procedure(c->allocator, d->scope, name->Ident, NULL, up_init->ProcLit.tags);
- // // d->proc_lit = up_init;
- // // d->type_expr = vd->type;
- // } else {
- e = make_entity_constant(c->allocator, d->scope, name->Ident, NULL, empty_exact_value);
- d->type_expr = vd->type;
- d->init_expr = init;
- // }
- GB_ASSERT(e != NULL);
- e->identifier = name;
-
- add_entity_and_decl_info(c, name, e, d);
- }
-
- check_arity_match(c, vd);
+ check_arity_match(c, vs);
} break;
case Token_type: {
- ast_node(td, TypeSpec, spec);
+ ast_node(ts, TypeSpec, spec);
- AstNode *name = td->name;
+ AstNode *name = ts->name;
if (name->kind != AstNode_Ident) {
error_node(name, "A declaration's name must be an identifier, got %.*s", LIT(ast_node_strings[name->kind]));
break;
@@ -1583,9 +1566,8 @@ void check_collect_entities(Checker *c, Array<AstNode *> nodes, bool is_file_sco
DeclInfo *d = make_declaration_info(c->allocator, c->context.scope, c->context.decl);
Entity *e = NULL;
- AstNode *type = unparen_expr(td->type);
+ AstNode *type = unparen_expr(ts->type);
e = make_entity_type_name(c->allocator, d->scope, name->Ident, NULL);
- // TODO(bill): What if vd->type != NULL??? How to handle this case?
d->type_expr = type;
d->init_expr = type;
@@ -1595,9 +1577,9 @@ void check_collect_entities(Checker *c, Array<AstNode *> nodes, bool is_file_sco
case Token_import:
case Token_import_load: {
- ast_node(id, ImportSpec, spec);
+ ast_node(ts, ImportSpec, spec);
if (!c->context.scope->is_file) {
- if (id->is_import) {
+ if (ts->is_import) {
error_node(decl, "import declarations are only allowed in the file scope");
} else {
error_node(decl, "import_load declarations are only allowed in the file scope");
@@ -1609,6 +1591,37 @@ void check_collect_entities(Checker *c, Array<AstNode *> nodes, bool is_file_sco
DelayedDecl di = {c->context.scope, spec};
array_add(&c->delayed_imports, di);
} break;
+
+ case Token_foreign_library:
+ case Token_foreign_system_library: {
+ ast_node(fl, ForeignLibrarySpec, spec);
+ if (!c->context.scope->is_file) {
+ if (fl->is_system) {
+ error_node(spec, "foreign_system_library declarations are only allowed in the file scope");
+ } else {
+ error_node(spec, "foreign_library declarations are only allowed in the file scope");
+ }
+ // NOTE(bill): _Should_ be caught by the parser
+ // TODO(bill): Better error handling if it isn't
+ continue;
+ }
+
+ if (fl->cond != NULL) {
+ Operand operand = {Addressing_Invalid};
+ check_expr(c, &operand, fl->cond);
+ if (operand.mode != Addressing_Constant || !is_type_boolean(operand.type)) {
+ error_node(fl->cond, "Non-constant boolean `when` condition");
+ continue;
+ }
+ if (operand.value.kind == ExactValue_Bool &&
+ !operand.value.value_bool) {
+ continue;
+ }
+ }
+
+ DelayedDecl di = {c->context.scope, spec};
+ array_add(&c->delayed_foreign_libraries, di);
+ } break;
}
}
case_end;
@@ -1631,34 +1644,6 @@ void check_collect_entities(Checker *c, Array<AstNode *> nodes, bool is_file_sco
add_entity_and_decl_info(c, name, e, d);
case_end;
- case_ast_node(fl, ForeignLibrary, decl);
- if (!c->context.scope->is_file) {
- if (fl->is_system) {
- error_node(decl, "#foreign_system_library declarations are only allowed in the file scope");
- } else {
- error_node(decl, "#foreign_library declarations are only allowed in the file scope");
- }
- // NOTE(bill): _Should_ be caught by the parser
- // TODO(bill): Better error handling if it isn't
- continue;
- }
-
- if (fl->cond != NULL) {
- Operand operand = {Addressing_Invalid};
- check_expr(c, &operand, fl->cond);
- if (operand.mode != Addressing_Constant || !is_type_boolean(operand.type)) {
- error_node(fl->cond, "Non-constant boolean `when` condition");
- continue;
- }
- if (operand.value.kind == ExactValue_Bool &&
- !operand.value.value_bool) {
- continue;
- }
- }
-
- DelayedDecl di = {c->context.scope, decl};
- array_add(&c->delayed_foreign_libraries, di);
- case_end;
default:
if (c->context.scope->is_file) {
error_node(decl, "Only declarations are allowed at file scope");
@@ -1998,8 +1983,8 @@ void check_import_entities(Checker *c, Map<Scope *> *file_scopes) {
for_array(i, c->delayed_foreign_libraries) {
Scope *parent_scope = c->delayed_foreign_libraries[i].parent;
- AstNode *decl = c->delayed_foreign_libraries[i].decl;
- ast_node(fl, ForeignLibrary, decl);
+ AstNode *spec = c->delayed_foreign_libraries[i].decl;
+ ast_node(fl, ForeignLibrarySpec, spec);
String file_str = fl->filepath.string;
String base_dir = fl->base_dir;
@@ -2034,7 +2019,7 @@ void check_import_entities(Checker *c, Map<Scope *> *file_scopes) {
String library_name = path_to_entity_name(fl->library_name.string, file_str);
if (library_name == "_") {
- error(fl->token, "File name, %.*s, cannot be as a library name as it is not a valid identifier", LIT(fl->library_name.string));
+ error_node(spec, "File name, %.*s, cannot be as a library name as it is not a valid identifier", LIT(fl->library_name.string));
} else {
GB_ASSERT(fl->library_name.pos.line != 0);
fl->library_name.string = library_name;
diff --git a/src/parser.cpp b/src/parser.cpp
index 47056373c..c96070044 100644
--- a/src/parser.cpp
+++ b/src/parser.cpp
@@ -100,9 +100,8 @@ enum FieldFlag {
FieldFlag_ellipsis = 1<<0,
FieldFlag_using = 1<<1,
FieldFlag_no_alias = 1<<2,
- FieldFlag_immutable = 1<<3,
- FieldFlag_Signature = FieldFlag_ellipsis|FieldFlag_using|FieldFlag_no_alias|FieldFlag_immutable,
+ FieldFlag_Signature = FieldFlag_ellipsis|FieldFlag_using|FieldFlag_no_alias,
};
enum StmtAllowFlag {
@@ -307,12 +306,12 @@ AST_NODE_KIND(_DeclBegin, "", i32) \
String foreign_name; \
String link_name; \
}) \
- AST_NODE_KIND(ForeignLibrary, "foreign library", struct { \
- Token token, filepath; \
- Token library_name; \
- String base_dir; \
+ AST_NODE_KIND(ForeignLibrarySpec, "foreign library specification", struct { \
+ Token filepath; \
+ Token library_name; \
+ String base_dir; \
AstNode *cond; \
- bool is_system; \
+ bool is_system; \
}) \
AST_NODE_KIND(Label, "label", struct { \
Token token; \
@@ -541,15 +540,15 @@ Token ast_node_token(AstNode *node) {
case AstNode_PushAllocator: return node->PushAllocator.token;
case AstNode_PushContext: return node->PushContext.token;
- case AstNode_BadDecl: return node->BadDecl.begin;
- case AstNode_ProcDecl: return node->ProcDecl.token;
- case AstNode_ForeignLibrary: return node->ForeignLibrary.token;
- case AstNode_Label: return node->Label.token;
+ case AstNode_BadDecl: return node->BadDecl.begin;
+ case AstNode_ProcDecl: return node->ProcDecl.token;
+ case AstNode_ForeignLibrarySpec: return node->ForeignLibrarySpec.filepath;
+ case AstNode_Label: return node->Label.token;
- case AstNode_GenDecl: return node->GenDecl.token;
- case AstNode_ValueSpec: return ast_node_token(node->ValueSpec.names[0]);
- case AstNode_ImportSpec: return node->ImportSpec.import_name;
- case AstNode_TypeSpec: return ast_node_token(node->TypeSpec.name);
+ case AstNode_GenDecl: return node->GenDecl.token;
+ case AstNode_ValueSpec: return ast_node_token(node->ValueSpec.names[0]);
+ case AstNode_ImportSpec: return node->ImportSpec.import_name;
+ case AstNode_TypeSpec: return ast_node_token(node->TypeSpec.name);
case AstNode_Field:
@@ -766,8 +765,8 @@ AstNode *clone_ast_node(gbAllocator a, AstNode *node) {
break;
case AstNode_BadDecl: break;
- case AstNode_ForeignLibrary:
- n->ForeignLibrary.cond = clone_ast_node(a, n->ForeignLibrary.cond);
+ case AstNode_ForeignLibrarySpec:
+ n->ForeignLibrarySpec.cond = clone_ast_node(a, n->ForeignLibrarySpec.cond);
break;
case AstNode_Label:
n->Label.name = clone_ast_node(a, n->Label.name);
@@ -1438,13 +1437,12 @@ AstNode *ast_proc_decl(AstFile *f, Token token, AstNode *name, AstNode *type, As
return result;
}
-AstNode *ast_foreign_library(AstFile *f, Token token, Token filepath, Token library_name, AstNode *cond, bool is_system) {
- AstNode *result = make_ast_node(f, AstNode_ForeignLibrary);
- result->ForeignLibrary.token = token;
- result->ForeignLibrary.filepath = filepath;
- result->ForeignLibrary.library_name = library_name;
- result->ForeignLibrary.cond = cond;
- result->ForeignLibrary.is_system = is_system;
+AstNode *ast_foreign_library_spec(AstFile *f, Token filepath, Token library_name, AstNode *cond, bool is_system) {
+ AstNode *result = make_ast_node(f, AstNode_ForeignLibrarySpec);
+ result->ForeignLibrarySpec.filepath = filepath;
+ result->ForeignLibrarySpec.library_name = library_name;
+ result->ForeignLibrarySpec.cond = cond;
+ result->ForeignLibrarySpec.is_system = is_system;
return result;
}
@@ -2690,6 +2688,72 @@ PARSE_SPEC_FUNC(parse_import_spec) {
}
}
+PARSE_SPEC_FUNC(parse_foreign_library_spec) {
+ if (token.kind == Token_foreign_system_library) {
+ AstNode *cond = NULL;
+ Token lib_name = {};
+
+ switch (f->curr_token.kind) {
+ case Token_Ident:
+ lib_name = f->curr_token;
+ next_token(f);
+ break;
+ default:
+ lib_name.pos = f->curr_token.pos;
+ break;
+ }
+
+ if (lib_name.string == "_") {
+ syntax_error(lib_name, "Illegal foreign_library name: `_`");
+ }
+ Token file_path = expect_token(f, Token_String);
+
+ if (allow_token(f, Token_when)) {
+ cond = parse_expr(f, false);
+ }
+
+ AstNode *spec = NULL;
+ if (f->curr_proc == NULL) {
+ spec = ast_foreign_library_spec(f, file_path, lib_name, cond, true);
+ } else {
+ syntax_error(lib_name, "You cannot use foreign_system_library within a procedure. This must be done at the file scope");
+ spec = ast_bad_decl(f, lib_name, file_path);
+ }
+ return spec;
+ } else {
+ AstNode *cond = NULL;
+ Token lib_name = {};
+
+ switch (f->curr_token.kind) {
+ case Token_Ident:
+ lib_name = f->curr_token;
+ next_token(f);
+ break;
+ default:
+ lib_name.pos = f->curr_token.pos;
+ break;
+ }
+
+ if (lib_name.string == "_") {
+ syntax_error(lib_name, "Illegal foreign_library name: `_`");
+ }
+ Token file_path = expect_token(f, Token_String);
+
+ if (allow_token(f, Token_when)) {
+ cond = parse_expr(f, false);
+ }
+
+ AstNode *spec = NULL;
+ if (f->curr_proc == NULL) {
+ spec = ast_foreign_library_spec(f, file_path, lib_name, cond, false);
+ } else {
+ syntax_error(lib_name, "You cannot use foreign_library within a procedure. This must be done at the file scope");
+ spec = ast_bad_decl(f, lib_name, file_path);
+ }
+ return spec;
+ }
+}
+
AstNode *parse_decl(AstFile *f) {
ParseSpecFunc *func = NULL;
switch (f->curr_token.kind) {
@@ -2708,6 +2772,11 @@ AstNode *parse_decl(AstFile *f) {
func = parse_import_spec;
break;
+ case Token_foreign_library:
+ case Token_foreign_system_library:
+ func = parse_foreign_library_spec;
+ break;
+
case Token_proc:
return parse_proc_decl(f);
@@ -2913,7 +2982,6 @@ enum FieldPrefixKind {
FieldPrefix_Invalid,
FieldPrefix_Using,
- FieldPrefix_Immutable,
FieldPrefix_NoAlias,
};
@@ -2925,9 +2993,6 @@ FieldPrefixKind is_token_field_prefix(AstFile *f) {
case Token_using:
return FieldPrefix_Using;
- case Token_let:
- return FieldPrefix_Immutable;
-
case Token_Hash: {
next_token(f);
switch (f->curr_token.kind) {
@@ -2944,9 +3009,8 @@ FieldPrefixKind is_token_field_prefix(AstFile *f) {
u32 parse_field_prefixes(AstFile *f) {
- i32 using_count = 0;
- i32 no_alias_count = 0;
- i32 immutable_count = 0;
+ i32 using_count = 0;
+ i32 no_alias_count = 0;
for (;;) {
FieldPrefixKind kind = is_token_field_prefix(f);
@@ -2954,20 +3018,17 @@ u32 parse_field_prefixes(AstFile *f) {
break;
}
switch (kind) {
- case FieldPrefix_Using: using_count += 1; next_token(f); break;
- case FieldPrefix_Immutable: immutable_count += 1; next_token(f); break;
- case FieldPrefix_NoAlias: no_alias_count += 1; next_token(f); break;
+ case FieldPrefix_Using: using_count += 1; next_token(f); break;
+ case FieldPrefix_NoAlias: no_alias_count += 1; next_token(f); break;
}
}
- if (using_count > 1) syntax_error(f->curr_token, "Multiple `using` in this field list");
- if (immutable_count > 1) syntax_error(f->curr_token, "Multiple `immutable` in this field list");
+ if (using_count > 1) syntax_error(f->curr_token, "Multiple `using` in this field list");
if (no_alias_count > 1) syntax_error(f->curr_token, "Multiple `#no_alias` in this field list");
u32 field_flags = 0;
if (using_count > 0) field_flags |= FieldFlag_using;
if (no_alias_count > 0) field_flags |= FieldFlag_no_alias;
- if (immutable_count > 0) field_flags |= FieldFlag_immutable;
return field_flags;
}
@@ -2985,10 +3046,6 @@ u32 check_field_prefixes(AstFile *f, isize name_count, u32 allowed_flags, u32 se
syntax_error(f->curr_token, "`no_alias` is not allowed within this field list");
set_flags &= ~FieldFlag_no_alias;
}
- if ((allowed_flags&FieldFlag_immutable) == 0 && (set_flags&FieldFlag_immutable)) {
- syntax_error(f->curr_token, "`immutable` is not allowed within this field list");
- set_flags &= ~FieldFlag_immutable;
- }
return set_flags;
}
@@ -3833,6 +3890,8 @@ AstNode *parse_stmt(AstFile *f) {
case Token_type:
case Token_import:
case Token_import_load:
+ case Token_foreign_library:
+ case Token_foreign_system_library:
return parse_decl(f);
case Token_if: return parse_if_stmt(f);
@@ -3899,24 +3958,6 @@ AstNode *parse_stmt(AstFile *f) {
return ast_bad_stmt(f, token, f->curr_token);
} break;
-#if 0
- case Token_immutable: {
- Token token = expect_token(f, Token_immutable);
- AstNode *node = parse_stmt(f);
-
- if (node->kind == AstNode_ValueDecl) {
- if (node->ValueDecl.token.kind == Token_const) {
- syntax_error(token, "`immutable` may not be applied to constant declarations");
- } else {
- node->ValueDecl.flags |= VarDeclFlag_immutable;
- }
- return node;
- }
- syntax_error(token, "`immutable` may only be applied to a variable declaration");
- return ast_bad_stmt(f, token, f->curr_token);
- } break;
-#endif
-
case Token_push_allocator: {
next_token(f);
isize prev_level = f->expr_level;
@@ -3939,70 +3980,6 @@ AstNode *parse_stmt(AstFile *f) {
return ast_push_context(f, token, expr, body);
} break;
-#if 0
- case Token_import: {
- Token token = expect_token(f, Token_import);
- AstNode *cond = NULL;
- Token import_name = {};
-
- switch (f->curr_token.kind) {
- case Token_Period:
- import_name = f->curr_token;
- import_name.kind = Token_Ident;
- next_token(f);
- break;
- case Token_Ident:
- import_name = f->curr_token;
- next_token(f);
- break;
- default:
- import_name.pos = f->curr_token.pos;
- break;
- }
-
- if (import_name.string == "_") {
- syntax_error(import_name, "Illegal import name: `_`");
- }
-
- Token file_path = expect_token_after(f, Token_String, "import");
- if (allow_token(f, Token_when)) {
- cond = parse_expr(f, false);
- }
-
- AstNode *decl = NULL;
- if (f->curr_proc != NULL) {
- syntax_error(import_name, "You cannot use `import` within a procedure. This must be done at the file scope");
- decl = ast_bad_decl(f, import_name, file_path);
- } else {
- decl = ast_import_decl(f, token, true, file_path, import_name, cond);
- }
- expect_semicolon(f, decl);
- return decl;
- }
-
- case Token_import_load: {
- Token token = expect_token(f, Token_import_load);
- AstNode *cond = NULL;
- Token file_path = expect_token_after(f, Token_String, "import_load");
- Token import_name = file_path;
- import_name.string = str_lit(".");
-
- if (allow_token(f, Token_when)) {
- cond = parse_expr(f, false);
- }
-
- AstNode *decl = NULL;
- if (f->curr_proc != NULL) {
- syntax_error(import_name, "You cannot use `import_load` within a procedure. This must be done at the file scope");
- decl = ast_bad_decl(f, import_name, file_path);
- } else {
- decl = ast_import_decl(f, token, false, file_path, import_name, cond);
- }
- expect_semicolon(f, decl);
- return decl;
- }
-#endif
-
case Token_Hash: {
AstNode *s = NULL;
Token hash_token = expect_token(f, Token_Hash);
@@ -4019,68 +3996,6 @@ AstNode *parse_stmt(AstFile *f) {
}
expect_semicolon(f, s);
return s;
- } else if (tag == "foreign_system_library") {
- AstNode *cond = NULL;
- Token lib_name = {};
-
- switch (f->curr_token.kind) {
- case Token_Ident:
- lib_name = f->curr_token;
- next_token(f);
- break;
- default:
- lib_name.pos = f->curr_token.pos;
- break;
- }
-
- if (lib_name.string == "_") {
- syntax_error(lib_name, "Illegal #foreign_library name: `_`");
- }
- Token file_path = expect_token(f, Token_String);
-
- if (allow_token(f, Token_when)) {
- cond = parse_expr(f, false);
- }
-
- if (f->curr_proc == NULL) {
- s = ast_foreign_library(f, hash_token, file_path, lib_name, cond, true);
- } else {
- syntax_error(token, "You cannot use #foreign_system_library within a procedure. This must be done at the file scope");
- s = ast_bad_decl(f, token, file_path);
- }
- expect_semicolon(f, s);
- return s;
- } else if (tag == "foreign_library") {
- AstNode *cond = NULL;
- Token lib_name = {};
-
- switch (f->curr_token.kind) {
- case Token_Ident:
- lib_name = f->curr_token;
- next_token(f);
- break;
- default:
- lib_name.pos = f->curr_token.pos;
- break;
- }
-
- if (lib_name.string == "_") {
- syntax_error(lib_name, "Illegal #foreign_library name: `_`");
- }
- Token file_path = expect_token(f, Token_String);
-
- if (allow_token(f, Token_when)) {
- cond = parse_expr(f, false);
- }
-
- if (f->curr_proc == NULL) {
- s = ast_foreign_library(f, hash_token, file_path, lib_name, cond, false);
- } else {
- syntax_error(token, "You cannot use #foreign_library within a procedure. This must be done at the file scope");
- s = ast_bad_decl(f, token, file_path);
- }
- expect_semicolon(f, s);
- return s;
} else if (tag == "thread_local") {
AstNode *s = parse_stmt(f);
@@ -4328,55 +4243,6 @@ void parse_setup_file_decls(Parser *p, AstFile *f, String base_dir, Array<AstNod
gbAllocator allocator = heap_allocator(); // TODO(bill): Change this allocator
String import_file = {};
- #if 0
- isize colon_pos = -1;
- for (isize j = 0; j < file_str.len; j++) {
- if (file_str[j] == ':') {
- colon_pos = j;
- break;
- }
- }
- if (colon_pos > 0) {
- collection_name = make_string(file_str.text, colon_pos);
- file_str.text += colon_pos+1;
- file_str.len -= colon_pos+1;
- }
-
- if (collection_name.len == 0) {
- syntax_error_node(node, "Missing import collection for path: `%.*s`", LIT(oirignal_string));
- decls[i] = ast_bad_decl(f, id->relpath, id->relpath);
- continue;
- }
-
-
- if (collection_name == "core") {
- String abs_path = get_fullpath_core(allocator, file_str);
- if (gb_file_exists(cast(char *)abs_path.text)) { // NOTE(bill): This should be null terminated
- import_file = abs_path;
- }
- } else if (collection_name == "local") {
- String rel_path = get_fullpath_relative(allocator, base_dir, file_str);
- if (gb_file_exists(cast(char *)rel_path.text)) { // NOTE(bill): This should be null terminated
- import_file = rel_path;
- }
- } else {
- syntax_error_node(node, "Unknown import collection: `%.*s`", LIT(collection_name));
- decls[i] = ast_bad_decl(f, id->relpath, id->relpath);
- continue;
- }
-
- if (!is_import_path_valid(file_str)) {
- if (id->is_import) {
- syntax_error_node(node, "Invalid import path: `%.*s`", LIT(file_str));
- } else {
- syntax_error_node(node, "Invalid include path: `%.*s`", LIT(file_str));
- }
- // NOTE(bill): It's a naughty name
- decls[i] = ast_bad_decl(f, id->relpath, id->relpath);
- continue;
- }
-
- #else
if (!is_import_path_valid(file_str)) {
if (id->is_import) {
syntax_error_node(node, "Invalid import path: `%.*s`", LIT(file_str));
@@ -4397,28 +4263,31 @@ void parse_setup_file_decls(Parser *p, AstFile *f, String base_dir, Array<AstNod
import_file = abs_path;
}
}
- #endif
id->fullpath = import_file;
try_add_import_path(p, import_file, file_str, ast_node_token(node).pos);
}
- }
- } else if (node->kind == AstNode_ForeignLibrary) {
- AstNodeForeignLibrary *fl = &node->ForeignLibrary;
- String file_str = fl->filepath.string;
+ } else if (gd->token.kind == Token_foreign_library ||
+ gd->token.kind == Token_foreign_system_library) {
+ for_array(spec_index, gd->specs) {
+ AstNode *spec = gd->specs[spec_index];
+ ast_node(fl, ForeignLibrarySpec, spec);
+ String file_str = fl->filepath.string;
- if (!is_import_path_valid(file_str)) {
- if (fl->is_system) {
- syntax_error_node(node, "Invalid `foreign_system_library` path");
- } else {
- syntax_error_node(node, "Invalid `foreign_library` path");
+ if (!is_import_path_valid(file_str)) {
+ if (fl->is_system) {
+ syntax_error_node(node, "Invalid `foreign_system_library` path");
+ } else {
+ syntax_error_node(node, "Invalid `foreign_library` path");
+ }
+ // NOTE(bill): It's a naughty name
+ gd->specs[spec_index] = ast_bad_decl(f, fl->filepath, fl->filepath);
+ continue;
+ }
+
+ fl->base_dir = base_dir;
}
- // NOTE(bill): It's a naughty name
- f->decls[i] = ast_bad_decl(f, fl->token, fl->token);
- continue;
}
-
- fl->base_dir = base_dir;
}
}
}
diff --git a/src/tokenizer.cpp b/src/tokenizer.cpp
index 1bdeb2f10..fb6e2c779 100644
--- a/src/tokenizer.cpp
+++ b/src/tokenizer.cpp
@@ -88,6 +88,8 @@ TOKEN_KIND(Token__KeywordBegin, "_KeywordBegin"), \
TOKEN_KIND(Token_type, "type"), \
TOKEN_KIND(Token_import, "import"), \
TOKEN_KIND(Token_import_load, "import_load"), \
+ TOKEN_KIND(Token_foreign_library, "foreign_library"), \
+ TOKEN_KIND(Token_foreign_system_library, "foreign_system_library"), \
TOKEN_KIND(Token_when, "when"), \
TOKEN_KIND(Token_if, "if"), \
TOKEN_KIND(Token_else, "else"), \