mirror of
https://github.com/oven-sh/bun
synced 2026-02-12 20:09:04 +00:00
* Define a generic allocator interface to enable static polymorphism for allocators (see `GenericAllocator` in `src/allocators.zig`). Note that `std.mem.Allocator` itself is considered a generic allocator. * Add utilities to `bun.allocators` for working with generic allocators. * Add a new namespace, `bun.memory`, with basic utilities for working with memory and objects (`create`, `destroy`, `initDefault`, `deinit`). * Add `bun.DefaultAllocator`, a zero-sized generic allocator type whose `allocator` method simply returns `bun.default_allocator`. * Implement the generic allocator interface in `AllocationScope` and `MimallocArena`. * Improve `bun.threading.GuardedValue` (now `bun.threading.Guarded`). * Improve `bun.safety.AllocPtr` (now `bun.safety.CheckedAllocator`). (For internal tracking: fixes STAB-1085, STAB-1086, STAB-1087, STAB-1088, STAB-1089, STAB-1090, STAB-1091)
113 lines
3.8 KiB
Zig
113 lines
3.8 KiB
Zig
/// This type can be used with `bun.ptr.Owned` to model "maybe owned" pointers:
|
|
///
|
|
/// ```
|
|
/// // Either owned by the default allocator, or borrowed
|
|
/// const MaybeOwnedFoo = bun.ptr.Owned(*Foo, bun.allocators.MaybeOwned(bun.DefaultAllocator));
|
|
///
|
|
/// var owned_foo: MaybeOwnedFoo = .new(makeFoo());
|
|
/// var borrowed_foo: MaybeOwnedFoo = .fromRawIn(some_foo_ptr, .initBorrowed());
|
|
///
|
|
/// owned_foo.deinit(); // calls `Foo.deinit` and frees the memory
|
|
/// borrowed_foo.deinit(); // no-op
|
|
/// ```
|
|
///
|
|
/// This type is a `GenericAllocator`; see `src/allocators.zig`.
|
|
pub fn MaybeOwned(comptime Allocator: type) type {
|
|
return struct {
|
|
const Self = @This();
|
|
|
|
_parent: bun.allocators.Nullable(Allocator),
|
|
|
|
/// Same as `.initBorrowed()`. This allocator cannot be used to allocate memory; a panic
|
|
/// will occur.
|
|
pub const borrowed = .initBorrowed();
|
|
|
|
/// Creates a `MaybeOwned` allocator that owns memory.
|
|
///
|
|
/// Allocations are forwarded to a default-initialized `Allocator`.
|
|
pub fn init() Self {
|
|
return .initOwned(bun.memory.initDefault(Allocator));
|
|
}
|
|
|
|
/// Creates a `MaybeOwned` allocator that owns memory, and forwards to a specific
|
|
/// allocator.
|
|
///
|
|
/// Allocations are forwarded to `parent_alloc`.
|
|
pub fn initOwned(parent_alloc: Allocator) Self {
|
|
return .initRaw(parent_alloc);
|
|
}
|
|
|
|
/// Creates a `MaybeOwned` allocator that does not own any memory. This allocator cannot
|
|
/// be used to allocate new memory (a panic will occur), and its implementation of `free`
|
|
/// is a no-op.
|
|
pub fn initBorrowed() Self {
|
|
return .initRaw(null);
|
|
}
|
|
|
|
pub fn deinit(self: *Self) void {
|
|
var maybe_parent = self.intoParent();
|
|
if (maybe_parent) |*parent_alloc| {
|
|
bun.memory.deinit(parent_alloc);
|
|
}
|
|
}
|
|
|
|
pub fn isOwned(self: Self) bool {
|
|
return self.rawParent() != null;
|
|
}
|
|
|
|
pub fn allocator(self: Self) std.mem.Allocator {
|
|
const maybe_parent = self.rawParent();
|
|
return if (maybe_parent) |parent_alloc|
|
|
bun.allocators.asStd(parent_alloc)
|
|
else
|
|
.{ .ptr = undefined, .vtable = &null_vtable };
|
|
}
|
|
|
|
const BorrowedParent = bun.allocators.Borrowed(Allocator);
|
|
|
|
pub fn parent(self: Self) ?BorrowedParent {
|
|
const maybe_parent = self.rawParent();
|
|
return if (maybe_parent) |parent_alloc|
|
|
bun.allocators.borrow(parent_alloc)
|
|
else
|
|
null;
|
|
}
|
|
|
|
pub fn intoParent(self: *Self) ?Allocator {
|
|
defer self.* = undefined;
|
|
return self.rawParent();
|
|
}
|
|
|
|
/// Used by smart pointer types and allocator wrappers. See `bun.allocators.borrow`.
|
|
pub const Borrowed = MaybeOwned(BorrowedParent);
|
|
|
|
pub fn borrow(self: Self) Borrowed {
|
|
return .{ ._parent = bun.allocators.initNullable(BorrowedParent, self.parent()) };
|
|
}
|
|
|
|
fn initRaw(parent_alloc: ?Allocator) Self {
|
|
return .{ ._parent = bun.allocators.initNullable(Allocator, parent_alloc) };
|
|
}
|
|
|
|
fn rawParent(self: Self) ?Allocator {
|
|
return bun.allocators.unpackNullable(Allocator, self._parent);
|
|
}
|
|
};
|
|
}
|
|
|
|
fn nullAlloc(ptr: *anyopaque, len: usize, alignment: Alignment, ret_addr: usize) ?[*]u8 {
|
|
_ = .{ ptr, len, alignment, ret_addr };
|
|
std.debug.panic("cannot allocate with a borrowed `MaybeOwned` allocator", .{});
|
|
}
|
|
|
|
const null_vtable: std.mem.Allocator.VTable = .{
|
|
.alloc = nullAlloc,
|
|
.resize = std.mem.Allocator.noResize,
|
|
.remap = std.mem.Allocator.noRemap,
|
|
.free = std.mem.Allocator.noFree,
|
|
};
|
|
|
|
const bun = @import("bun");
|
|
const std = @import("std");
|
|
const Alignment = std.mem.Alignment;
|