diff options
| author | Andrew Kelley <andrew@ziglang.org> | 2023-10-25 20:23:38 -0700 |
|---|---|---|
| committer | Andrew Kelley <andrew@ziglang.org> | 2023-10-25 20:24:05 -0700 |
| commit | cf9735a5e06d302a29d70c737aefd505ca34e0fa (patch) | |
| tree | 03ab9b4214484497990249099af9b41b292f350a /src/link/MachO.zig | |
| parent | 098a07dc45b678af22bb47379e75371767385cbf (diff) | |
| download | zig-cf9735a5e06d302a29d70c737aefd505ca34e0fa.tar.gz zig-cf9735a5e06d302a29d70c737aefd505ca34e0fa.zip | |
link: Coff, MachO, and Wasm all had the same UAF bug
Diffstat (limited to 'src/link/MachO.zig')
| -rw-r--r-- | src/link/MachO.zig | 75 |
1 files changed, 40 insertions, 35 deletions
diff --git a/src/link/MachO.zig b/src/link/MachO.zig index eb82de7374..b863e7a0ac 100644 --- a/src/link/MachO.zig +++ b/src/link/MachO.zig @@ -2866,46 +2866,51 @@ pub fn getDeclVAddr(self: *MachO, decl_index: Module.Decl.Index, reloc_info: Fil return 0; } -pub fn lowerAnonDecl(self: *MachO, decl_val: InternPool.Index, decl_align: InternPool.Alignment, src_loc: Module.SrcLoc) !codegen.Result { - // This is basically the same as lowerUnnamedConst. - // example: - // const ty = mod.intern_pool.typeOf(decl_val).toType(); - // const val = decl_val.toValue(); - // The symbol name can be something like `__anon_{d}` with `@intFromEnum(decl_val)`. - // It doesn't have an owner decl because it's just an unnamed constant that might - // be used by more than one function, however, its address is being used so we need - // to put it in some location. - // ... +pub fn lowerAnonDecl( + self: *MachO, + decl_val: InternPool.Index, + explicit_alignment: InternPool.Alignment, + src_loc: Module.SrcLoc, +) !codegen.Result { const gpa = self.base.allocator; const mod = self.base.options.module.?; const ty = mod.intern_pool.typeOf(decl_val).toType(); - const gop = try self.anon_decls.getOrPut(gpa, decl_val); - const required_alignment = switch (decl_align) { + const decl_alignment = switch (explicit_alignment) { .none => ty.abiAlignment(mod), - else => decl_align, + else => explicit_alignment, }; - if (!gop.found_existing or - !required_alignment.check(self.getAtom(gop.value_ptr.*).getSymbol(self).n_value)) - { - const val = decl_val.toValue(); - const tv = TypedValue{ .ty = ty, .val = val }; - const name = try std.fmt.allocPrint(gpa, "__anon_{d}", .{@intFromEnum(decl_val)}); - defer gpa.free(name); - const res = self.lowerConst(name, tv, required_alignment, self.data_const_section_index.?, src_loc) catch |err| switch (err) { - else => { - // TODO improve error message - const em = try Module.ErrorMsg.create(gpa, src_loc, "lowerAnonDecl failed with error: {s}", .{ - @errorName(err), - }); - return .{ .fail = em }; - }, - }; - const atom_index = switch (res) { - .ok => |atom_index| atom_index, - .fail => |em| return .{ .fail = em }, - }; - gop.value_ptr.* = atom_index; - } + if (self.anon_decls.get(decl_val)) |atom_index| { + const existing_addr = self.getAtom(atom_index).getSymbol(self).n_value; + if (decl_alignment.check(existing_addr)) + return .ok; + } + + const val = decl_val.toValue(); + const tv = TypedValue{ .ty = ty, .val = val }; + var name_buf: [32]u8 = undefined; + const name = std.fmt.bufPrint(&name_buf, "__anon_{d}", .{ + @intFromEnum(decl_val), + }) catch unreachable; + const res = self.lowerConst( + name, + tv, + decl_alignment, + self.data_const_section_index.?, + src_loc, + ) catch |err| switch (err) { + error.OutOfMemory => return error.OutOfMemory, + else => |e| return .{ .fail = try Module.ErrorMsg.create( + gpa, + src_loc, + "unable to lower constant value: {s}", + .{@errorName(e)}, + ) }, + }; + const atom_index = switch (res) { + .ok => |atom_index| atom_index, + .fail => |em| return .{ .fail = em }, + }; + try self.anon_decls.put(gpa, decl_val, atom_index); return .ok; } |
