A store to a module-global array element (`g[i] = v`) was silently dropped: a subsequent `g[i]` read the array's initializer, not `v`. Constant index, variable index, and cross-function stores were all affected, in both `sx run` and `sx build`. Global scalars and local arrays were fine. Root cause: `Lowering.lowerExprAsPtr` (the lvalue/address path) handled only local identifiers. A module-global identifier fell through to the value fallback `lowerExpr`, which emits `global_get` — loading the whole array by value. The LLVM backend's `emitIndexGep` then allocas a throwaway temp, copies the value in, and GEPs into the temp, so the store wrote a discarded copy. Fix: teach `lowerExprAsPtr`'s identifier arm about globals — emit `global_addr` (a pointer into the global's live storage), or `global_get` for a pointer-typed global (mirroring the local pointer case). Route the `address_of(index_expr)` array base through `lowerExprAsPtr` too so `&g[i]` is likewise an lvalue into the global. `index_gep` now GEPs directly into the global for const and variable index, across functions. This also fixes global struct field stores, which shared the same root cause. Regression: examples/0136-types-global-array-element-store.sx (const-index, var-index, cross-function store on a scalar global array; struct-element array for stride; nested-array global for the recursive lvalue). Fails on the pre-fix compiler, passes after.
58 lines
2.0 KiB
Plaintext
58 lines
2.0 KiB
Plaintext
// A store to a module-global array element writes the global's live storage,
|
|
// so a subsequent read sees the stored value — not the array initializer.
|
|
// Covers constant index, variable index, and a cross-function store, on a
|
|
// scalar global array, a struct-element global array (element-stride), and a
|
|
// nested-array global (recursive lvalue).
|
|
// Regression (issue 0079): global-array element stores were silently dropped
|
|
// (read returned the initializer) because the indexed lvalue base loaded the
|
|
// global by value into a temp instead of addressing the global's storage.
|
|
|
|
#import "modules/std.sx";
|
|
|
|
g : [3]s64 = .[10, 20, 30];
|
|
|
|
Pair :: struct { a: s64; b: s64; }
|
|
gp : [2]Pair = .[ .{ a = 1, b = 2 }, .{ a = 3, b = 4 } ];
|
|
|
|
grid : [2][3]s64 = .[ .[0, 0, 0], .[0, 0, 0] ];
|
|
|
|
write_global :: (i: s64, v: s64) { g[i] = v; }
|
|
|
|
main :: () {
|
|
// Scalar global array — const index.
|
|
g[1] = 222;
|
|
print("g[1]={}\n", g[1]); // 222
|
|
|
|
// Scalar global array — variable index.
|
|
k := 2;
|
|
g[k] = 333;
|
|
print("g[k]={}\n", g[k]); // 333
|
|
|
|
// Scalar global array — store from another function.
|
|
write_global(0, 111);
|
|
print("g[0]={}\n", g[0]); // 111
|
|
|
|
// Struct-element global array (16-byte stride) — const and var index.
|
|
gp[0] = .{ a = 10, b = 20 };
|
|
j := 1;
|
|
gp[j] = .{ a = 30, b = 40 };
|
|
print("gp[0]={},{}\n", gp[0].a, gp[0].b); // 10,20
|
|
print("gp[j]={},{}\n", gp[j].a, gp[j].b); // 30,40
|
|
|
|
// Nested-array global — element is [3]s64, recursive indexed lvalue.
|
|
grid[1][2] = 7;
|
|
r := 0;
|
|
grid[r][0] = 5;
|
|
print("grid[1][2]={}\n", grid[1][2]); // 7
|
|
print("grid[0][0]={}\n", grid[r][0]); // 5
|
|
|
|
if g[1] == 222 and g[2] == 333 and g[0] == 111
|
|
and gp[0].a == 10 and gp[0].b == 20
|
|
and gp[1].a == 30 and gp[1].b == 40
|
|
and grid[1][2] == 7 and grid[0][0] == 5 {
|
|
print("PASS\n");
|
|
} else {
|
|
print("FAIL: global array element store dropped\n");
|
|
}
|
|
}
|