zba-util/src/lib.zig

184 lines
5.5 KiB
Zig
Raw Normal View History

2023-03-10 05:41:46 +00:00
const std = @import("std");
const Log2Int = std.math.Log2Int;
const EmuMessage = enum { Pause, Resume, Quit };
const GuiMessage = enum { Paused, Quit };
pub const TwoWayChannel = struct {
const Self = @This();
emu: Channel(EmuMessage),
gui: Channel(GuiMessage),
pub fn init(items: []u8) Self {
comptime std.debug.assert(@sizeOf(EmuMessage) == @sizeOf(GuiMessage));
comptime std.debug.assert(@sizeOf(@typeInfo([]u8).Pointer.child) == @sizeOf(EmuMessage));
std.debug.assert(items.len % 2 == 0);
const left = @ptrCast([*]EmuMessage, items)[0 .. items.len / 2];
const right = @ptrCast([*]GuiMessage, items)[items.len / 2 .. items.len];
return .{ .emu = Channel(EmuMessage).init(left), .gui = Channel(GuiMessage).init(right) };
}
};
fn Channel(comptime T: type) type {
2023-03-10 05:41:46 +00:00
return struct {
const Self = @This();
const Index = usize;
const Atomic = std.atomic.Atomic;
const max_capacity = (@as(Index, 1) << @typeInfo(Index).Int.bits - 1) - 1; // half the range of index type
const log = std.log.scoped(.Channel);
2023-03-10 05:41:46 +00:00
read: Atomic(Index),
write: Atomic(Index),
buf: []T,
const Error = error{buffer_full};
pub fn init(buf: []T) Self {
std.debug.assert(std.math.isPowerOfTwo(buf.len)); // capacity must be a power of two
std.debug.assert(buf.len <= max_capacity);
return .{
.read = Atomic(Index).init(0),
.write = Atomic(Index).init(0),
.buf = buf,
};
}
pub fn push(self: *Self, value: T) void {
2023-03-10 05:41:46 +00:00
const read_idx = self.read.load(.Acquire);
const write_idx = self.write.load(.Acquire);
// Check to see if Queue is full
if (write_idx - read_idx == self.buf.len) @panic("Channel: Buffer is full");
2023-03-10 05:41:46 +00:00
self.buf[self.mask(write_idx)] = value;
std.atomic.fence(.Release);
self.write.store(write_idx + 1, .Release);
}
pub fn pop(self: *Self) ?T {
const read_idx = self.read.load(.Acquire);
const write_idx = self.write.load(.Acquire);
if (read_idx == write_idx) return null;
std.atomic.fence(.Acquire);
const value = self.buf[self.mask(read_idx)];
2023-03-10 05:41:46 +00:00
std.atomic.fence(.Release);
self.read.store(read_idx + 1, .Release);
return value;
}
pub fn len(self: *const Self) Index {
const read_idx = self.read.load(.Acquire);
const write_idx = self.write.load(.Acquire);
return write_idx - read_idx;
}
fn mask(self: *const Self, idx: Index) Index {
return idx & (self.buf.len - 1);
}
};
}
pub fn RingBuffer(comptime T: type) type {
return struct {
const Self = @This();
const Index = usize;
const max_capacity = (@as(Index, 1) << @typeInfo(Index).Int.bits - 1) - 1; // half the range of index type
const log = std.log.scoped(.RingBuffer);
read: Index,
write: Index,
buf: []T,
const Error = error{buffer_full};
pub fn init(buf: []T) Self {
std.debug.assert(std.math.isPowerOfTwo(buf.len)); // capacity must be a power of two
std.debug.assert(buf.len <= max_capacity);
2023-05-02 05:00:02 +00:00
@memset(buf, 0);
return .{ .read = 0, .write = 0, .buf = buf };
}
pub fn push(self: *Self, value: T) Error!void {
if (self.isFull()) return error.buffer_full;
defer self.write += 1;
self.buf[self.mask(self.write)] = value;
}
pub fn pop(self: *Self) ?T {
if (self.isEmpty()) return null;
defer self.read += 1;
return self.buf[self.mask(self.read)];
}
/// Returns the number of entries read
pub fn copy(self: *const Self, cpy: []T) Index {
const count = std.math.min(self.len(), cpy.len);
var start: Index = self.read;
for (cpy, 0..) |*v, i| {
if (i >= count) break;
v.* = self.buf[self.mask(start)];
start += 1;
}
return count;
}
fn len(self: *const Self) Index {
return self.write - self.read;
}
fn isFull(self: *const Self) bool {
return self.len() == self.buf.len;
}
fn isEmpty(self: *const Self) bool {
return self.read == self.write;
}
fn mask(self: *const Self, idx: Index) Index {
return idx & (self.buf.len - 1);
}
};
}
// Sign-Extend value of type `T` to type `U`
pub fn sext(comptime T: type, comptime U: type, value: T) T {
// U must have less bits than T
comptime std.debug.assert(@typeInfo(U).Int.bits <= @typeInfo(T).Int.bits);
const iT = std.meta.Int(.signed, @typeInfo(T).Int.bits);
const ExtU = if (@typeInfo(U).Int.signedness == .unsigned) T else iT;
const shift_amt = @intCast(Log2Int(T), @typeInfo(T).Int.bits - @typeInfo(U).Int.bits);
return @bitCast(T, @bitCast(iT, @as(ExtU, @truncate(U, value)) << shift_amt) >> shift_amt);
}
/// See https://godbolt.org/z/W3en9Eche
pub inline fn rotr(comptime T: type, x: T, r: anytype) T {
if (@typeInfo(T).Int.signedness == .signed)
@compileError("cannot rotate signed integer");
const ar = @intCast(Log2Int(T), @mod(r, @typeInfo(T).Int.bits));
return x >> ar | x << (1 +% ~ar);
}