fix(0117): pointer-to-array indexing auto-derefs

A '*[N]T' receiver in an index expression reached LLVM emission with an
unresolved element type and tripped the panic sentinel — no read or
write spelling worked. ptrToArrayElem on Lowering recognises the shape;
the index READ path GEPs the pointee array through the pointer value
and loads the element; the write / compound-assign / lvalue /
addr-of-element paths and the expression typer resolve the element type
through the same helper (their GEP machinery already handled a pointer
base). Kept out of getElementType so slice paths don't half-accept a
raw pointer base.

Regression: examples/0176 (read, write, compound, element ptr + deref).
This commit is contained in:
agra
2026-06-11 12:15:45 +03:00
parent 57979ed8e6
commit 82d6b8da0e
9 changed files with 63 additions and 5 deletions

View File

@@ -364,6 +364,7 @@ pub const ExprTyper = struct {
return self.l.inferExprType(arg_node);
}
const obj_ty = self.l.inferExprType(ie.object);
if (self.l.ptrToArrayElem(obj_ty)) |elem| return elem;
return self.l.getElementType(obj_ty);
},
.slice_expr => |se| {

View File

@@ -1106,6 +1106,20 @@ pub const Lowering = struct {
};
}
/// The element type when `ty` is a POINTER TO AN ARRAY (`*[N]T` → T),
/// else null. Indexing auto-derefs this shape (GEP the pointee array
/// through the pointer value); kept OUT of `getElementType` so the
/// slice/subslice paths don't half-accept a raw pointer base.
pub fn ptrToArrayElem(self: *Lowering, ty: TypeId) ?TypeId {
if (ty.isBuiltin()) return null;
const info = self.module.types.get(ty);
if (info != .pointer) return null;
const pointee = info.pointer.pointee;
if (pointee.isBuiltin()) return null;
const pi = self.module.types.get(pointee);
return if (pi == .array) pi.array.element else null;
}
pub fn isFloat(ty: TypeId) bool {
return ty == .f32 or ty == .f64;
}

View File

@@ -1263,6 +1263,12 @@ pub fn lowerIndexExpr(self: *Lowering, ie: *const ast.IndexExpr) Ref {
const idx = self.lowerExpr(ie.index);
// Infer element type from the object's slice/array type
const obj_ty = self.inferExprType(ie.object);
// `*[N]T` receiver auto-derefs (issue 0117): `obj` IS the pointer
// value — GEP the pointee array and load the element.
if (self.ptrToArrayElem(obj_ty)) |elem| {
const gep = self.builder.emit(.{ .index_gep = .{ .lhs = obj, .rhs = idx } }, self.module.types.ptrTo(elem));
return self.builder.load(gep, elem);
}
const elem_ty = self.getElementType(obj_ty);
return self.builder.emit(.{ .index_get = .{ .lhs = obj, .rhs = idx } }, elem_ty);
}
@@ -1843,7 +1849,7 @@ pub fn lowerExpr(self: *Lowering, node: *const Node) Ref {
const ie = &uop.operand.data.index_expr;
const idx = self.lowerExpr(ie.index);
const obj_ty = self.inferExprType(ie.object);
const elem_ty = self.getElementType(obj_ty);
const elem_ty = self.ptrToArrayElem(obj_ty) orelse self.getElementType(obj_ty);
const ptr_ty = self.module.types.ptrTo(elem_ty);
// For array targets, use the storage pointer (alloca for a
// local, global_addr for a module global) so the resulting

View File

@@ -564,7 +564,8 @@ pub fn lowerAssignment(self: *Lowering, asgn: *const ast.Assignment) void {
}
} else if (asgn.target.data == .index_expr) {
// For array[i] = val, set target_type to the element type
const elem_ty = self.getElementType(self.inferExprType(asgn.target.data.index_expr.object));
const tgt_obj_ty = self.inferExprType(asgn.target.data.index_expr.object);
const elem_ty = self.ptrToArrayElem(tgt_obj_ty) orelse self.getElementType(tgt_obj_ty);
if (elem_ty != .void) self.target_type = elem_ty;
} else if (asgn.target.data == .field_access) {
// For obj.field = val, set target_type to the field's type so RHS
@@ -722,7 +723,7 @@ pub fn lowerAssignment(self: *Lowering, asgn: *const ast.Assignment) void {
.index_expr => |ie| {
const idx = self.lowerExpr(ie.index);
const obj_ty = self.inferExprType(ie.object);
const elem_ty = self.getElementType(obj_ty);
const elem_ty = self.ptrToArrayElem(obj_ty) orelse self.getElementType(obj_ty);
const ptr_ty = self.module.types.ptrTo(elem_ty);
// For fixed-size array assignment targets, use the alloca pointer directly
// so that the store modifies the original variable (not a loaded copy).
@@ -939,7 +940,7 @@ pub fn lowerExprAsPtr(self: *Lowering, node: *const Node) Ref {
.index_expr => |ie| {
const idx = self.lowerExpr(ie.index);
const obj_ty = self.inferExprType(ie.object);
const elem_ty = self.getElementType(obj_ty);
const elem_ty = self.ptrToArrayElem(obj_ty) orelse self.getElementType(obj_ty);
const ptr_ty = self.module.types.ptrTo(elem_ty);
// For fixed-size arrays, use the alloca so GEP addresses the original memory
const is_array = !obj_ty.isBuiltin() and self.module.types.get(obj_ty) == .array;
@@ -1153,7 +1154,7 @@ pub fn lowerMultiAssign(self: *Lowering, ma: *const ast.MultiAssign) void {
.index_expr => |ie| {
const idx = self.lowerExpr(ie.index);
const obj_ty = self.inferExprType(ie.object);
const elem_ty = self.getElementType(obj_ty);
const elem_ty = self.ptrToArrayElem(obj_ty) orelse self.getElementType(obj_ty);
const ptr_ty = self.module.types.ptrTo(elem_ty);
const val_ty = self.builder.getRefType(val);
const store_val = if (val_ty != elem_ty and val_ty != .void and elem_ty != .void)