commit f09386cce9ad99d77978cee0d15ae7dd422ea50c (tree)
parent 46658257f458b7c3c95d7e10cbde85403f7bdb44
Author: Justus Klausecker <justus@klausecker.de>
Date: Wed, 4 Mar 2026 15:27:54 +0100
std.heap.ArenaAllocator: optimize aligned index calculation
The `alignedIndex` function is very hot (literally every single `alloc`
call invokes it at least once) and `std.mem.alignPointerOffset` seems to
be very slow, so this commit replaces this functions with a custom
implementation that doesn't do any unnecessary validation and doesn't have
any branches as a result of that. The validation `std.mem.alignPointerOffset`
does isn't necessary anyway, we're not actually calculating an offset that
we plan to apply to a pointer directly, but an offset into a valid buffer
that we only apply to a pointer if the result is inside of that buffer.
This leads to a ~4% speedup in a synthetic benchmark that just puts a lot
of concurrent load on an `ArenaAllocator`.
Diffstat:
1 file changed, 4 insertions(+), 2 deletions(-)
diff --git a/lib/std/heap/ArenaAllocator.zig b/lib/std/heap/ArenaAllocator.zig
@@ -315,8 +315,10 @@ fn pushFreeList(arena: *ArenaAllocator, first: *Node, last: *Node) void {
}
fn alignedIndex(buf_ptr: [*]u8, end_index: usize, alignment: Alignment) usize {
- return end_index +
- mem.alignPointerOffset(buf_ptr + end_index, alignment.toByteUnits()).?;
+ // Wrapping arithmetic to avoid overflows since `end_index` isn't bounded by
+ // `size`. This is always ok since the max alignment in byte units is also
+ // the max value of `usize` so wrapped values are correctly aligned anyway.
+ return alignment.forward(@intFromPtr(buf_ptr) +% end_index) -% @intFromPtr(buf_ptr);
}
fn alloc(ctx: *anyopaque, n: usize, alignment: Alignment, ret_addr: usize) ?[*]u8 {