Type Function StackFallbackAllocator [src]

An allocator that attempts to allocate using a FixedBufferAllocator using an array of size size. If the allocation fails, it will fall back to using fallback_allocator. Easily created with stackFallback.

Prototype

pub fn StackFallbackAllocator(comptime size: usize) type

Parameters

size: usize

Source

pub fn StackFallbackAllocator(comptime size: usize) type { return struct { const Self = @This(); buffer: [size]u8, fallback_allocator: Allocator, fixed_buffer_allocator: FixedBufferAllocator, get_called: if (std.debug.runtime_safety) bool else void = if (std.debug.runtime_safety) false else {}, /// This function both fetches a `Allocator` interface to this /// allocator *and* resets the internal buffer allocator. pub fn get(self: *Self) Allocator { if (std.debug.runtime_safety) { assert(!self.get_called); // `get` called multiple times; instead use `const allocator = stackFallback(N).get();` self.get_called = true; } self.fixed_buffer_allocator = FixedBufferAllocator.init(self.buffer[0..]); return .{ .ptr = self, .vtable = &.{ .alloc = alloc, .resize = resize, .remap = remap, .free = free, }, }; } /// Unlike most std allocators `StackFallbackAllocator` modifies /// its internal state before returning an implementation of /// the`Allocator` interface and therefore also doesn't use /// the usual `.allocator()` method. pub const allocator = @compileError("use 'const allocator = stackFallback(N).get();' instead"); fn alloc( ctx: *anyopaque, len: usize, alignment: mem.Alignment, ra: usize, ) ?[*]u8 { const self: *Self = @ptrCast(@alignCast(ctx)); return FixedBufferAllocator.alloc(&self.fixed_buffer_allocator, len, alignment, ra) orelse return self.fallback_allocator.rawAlloc(len, alignment, ra); } fn resize( ctx: *anyopaque, buf: []u8, alignment: mem.Alignment, new_len: usize, ra: usize, ) bool { const self: *Self = @ptrCast(@alignCast(ctx)); if (self.fixed_buffer_allocator.ownsPtr(buf.ptr)) { return FixedBufferAllocator.resize(&self.fixed_buffer_allocator, buf, alignment, new_len, ra); } else { return self.fallback_allocator.rawResize(buf, alignment, new_len, ra); } } fn remap( context: *anyopaque, memory: []u8, alignment: mem.Alignment, new_len: usize, return_address: usize, ) ?[*]u8 { const self: *Self = @ptrCast(@alignCast(context)); if (self.fixed_buffer_allocator.ownsPtr(memory.ptr)) { return FixedBufferAllocator.remap(&self.fixed_buffer_allocator, memory, alignment, new_len, return_address); } else { return self.fallback_allocator.rawRemap(memory, alignment, new_len, return_address); } } fn free( ctx: *anyopaque, buf: []u8, alignment: mem.Alignment, ra: usize, ) void { const self: *Self = @ptrCast(@alignCast(ctx)); if (self.fixed_buffer_allocator.ownsPtr(buf.ptr)) { return FixedBufferAllocator.free(&self.fixed_buffer_allocator, buf, alignment, ra); } else { return self.fallback_allocator.rawFree(buf, alignment, ra); } } }; }