mirror of
https://github.com/Ed94/Odin.git
synced 2026-06-25 23:14:59 -07:00
Allow nested procedures to access @(static) and @(thread_local) variables
This commit is contained in:
+4
-2
@@ -1075,8 +1075,10 @@ Entity *check_ident(CheckerContext *c, Operand *o, Ast *n, Type *named_type, Typ
|
||||
if (e->parent_proc_decl != nullptr &&
|
||||
e->parent_proc_decl != c->curr_proc_decl) {
|
||||
if (e->kind == Entity_Variable) {
|
||||
error(n, "Nested procedures do not capture its parent's variables: %.*s", LIT(name));
|
||||
return nullptr;
|
||||
if ((e->flags & EntityFlag_Static) == 0) {
|
||||
error(n, "Nested procedures do not capture its parent's variables: %.*s", LIT(name));
|
||||
return nullptr;
|
||||
}
|
||||
} else if (e->kind == Entity_Label) {
|
||||
error(n, "Nested procedures do not capture its parent's labels: %.*s", LIT(name));
|
||||
return nullptr;
|
||||
|
||||
+8
-3
@@ -368,9 +368,14 @@ void scope_lookup_parent(Scope *scope, String const &name, Scope **scope_, Entit
|
||||
if (e->kind == Entity_Label) {
|
||||
continue;
|
||||
}
|
||||
if (e->kind == Entity_Variable &&
|
||||
!(e->scope->flags&ScopeFlag_File)) {
|
||||
continue;
|
||||
if (e->kind == Entity_Variable) {
|
||||
if (e->scope->flags&ScopeFlag_File) {
|
||||
// Global variables are file to access
|
||||
} else if (e->flags&EntityFlag_Static) {
|
||||
// Allow static/thread_local variables to be referenced
|
||||
} else {
|
||||
continue;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
+23
-9
@@ -12875,6 +12875,7 @@ void lb_generate_code(lbGenerator *gen) {
|
||||
lbValue var;
|
||||
lbValue init;
|
||||
DeclInfo *decl;
|
||||
bool is_initialized;
|
||||
};
|
||||
auto global_variables = array_make<GlobalVariable>(permanent_allocator(), 0, global_variable_max_count);
|
||||
|
||||
@@ -12937,15 +12938,22 @@ void lb_generate_code(lbGenerator *gen) {
|
||||
var.var = g;
|
||||
var.decl = decl;
|
||||
|
||||
if (decl->init_expr != nullptr && !is_type_any(e->type)) {
|
||||
if (decl->init_expr != nullptr) {
|
||||
TypeAndValue tav = type_and_value_of_expr(decl->init_expr);
|
||||
if (tav.mode != Addressing_Invalid) {
|
||||
if (tav.value.kind != ExactValue_Invalid) {
|
||||
ExactValue v = tav.value;
|
||||
lbValue init = lb_const_value(m, tav.type, v);
|
||||
LLVMSetInitializer(g.value, init.value);
|
||||
if (!is_type_any(e->type)) {
|
||||
if (tav.mode != Addressing_Invalid) {
|
||||
if (tav.value.kind != ExactValue_Invalid) {
|
||||
ExactValue v = tav.value;
|
||||
lbValue init = lb_const_value(m, tav.type, v);
|
||||
LLVMSetInitializer(g.value, init.value);
|
||||
var.is_initialized = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!var.is_initialized &&
|
||||
(is_type_untyped_nil(tav.type) || is_type_untyped_undef(tav.type))) {
|
||||
var.is_initialized = true;
|
||||
}
|
||||
}
|
||||
|
||||
array_add(&global_variables, var);
|
||||
@@ -13146,10 +13154,13 @@ void lb_generate_code(lbGenerator *gen) {
|
||||
auto *var = &global_variables[i];
|
||||
if (var->decl->init_expr != nullptr) {
|
||||
lbValue init = lb_build_expr(p, var->decl->init_expr);
|
||||
if (!lb_is_const(init)) {
|
||||
var->init = init;
|
||||
if (lb_is_const(init)) {
|
||||
if (!var->is_initialized) {
|
||||
LLVMSetInitializer(var->var.value, init.value);
|
||||
var->is_initialized = true;
|
||||
}
|
||||
} else {
|
||||
LLVMSetInitializer(var->var.value, init.value);
|
||||
var->init = init;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -13166,6 +13177,7 @@ void lb_generate_code(lbGenerator *gen) {
|
||||
}
|
||||
|
||||
if (var->init.value != nullptr) {
|
||||
GB_ASSERT(!var->is_initialized);
|
||||
Type *t = type_deref(var->var.type);
|
||||
|
||||
if (is_type_any(t)) {
|
||||
@@ -13182,6 +13194,8 @@ void lb_generate_code(lbGenerator *gen) {
|
||||
} else {
|
||||
lb_emit_store(p, var->var, lb_emit_conv(p, var->init, t));
|
||||
}
|
||||
|
||||
var->is_initialized = true;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
Reference in New Issue
Block a user