mirror of
https://github.com/c3lang/c3c.git
synced 2026-02-27 20:11:17 +00:00
Upgrade of mingw in CI. Fix problems using reflection on interface types #1203. Improved debug information on defer. $foreach doesn't create an implicit syntactic scope. Error if `@if` depends on `@if`. Updated Linux stacktrace. Fix of default argument stacktrace. Allow linking libraries directly by file path. Improve inlining warning messages. Added `index_of_char_from`. Compiler crash using enum nameof from different module #1205. Removed unused fields in find_msvc. Use vswhere to find msvc. Update tests for LLVM 19
103 lines
1.9 KiB
C
103 lines
1.9 KiB
C
module std::collections::ringbuffer(<Type, SIZE>);
|
|
|
|
struct RingBuffer
|
|
{
|
|
Type[SIZE] buf;
|
|
usz written;
|
|
usz head;
|
|
}
|
|
|
|
fn void RingBuffer.init(&self) @inline
|
|
{
|
|
*self = {};
|
|
}
|
|
|
|
fn void RingBuffer.push(&self, Type c)
|
|
{
|
|
if (self.written < SIZE)
|
|
{
|
|
self.buf[self.written] = c;
|
|
self.written++;
|
|
}
|
|
else
|
|
{
|
|
self.buf[self.head] = c;
|
|
self.head = (self.head + 1) % SIZE;
|
|
}
|
|
}
|
|
|
|
fn Type RingBuffer.get(&self, usz index) @operator([])
|
|
{
|
|
index %= SIZE;
|
|
usz avail = SIZE - self.head;
|
|
if (index < avail)
|
|
{
|
|
return self.buf[self.head + index];
|
|
}
|
|
return self.buf[index - avail];
|
|
}
|
|
|
|
fn Type! RingBuffer.pop(&self)
|
|
{
|
|
switch
|
|
{
|
|
case self.written == 0:
|
|
return SearchResult.MISSING?;
|
|
case self.written < SIZE:
|
|
self.written--;
|
|
return self.buf[self.written];
|
|
default:
|
|
self.head = (self.head - 1) % SIZE;
|
|
return self.buf[self.head];
|
|
}
|
|
}
|
|
|
|
fn usz RingBuffer.read(&self, usz index, Type[] buffer)
|
|
{
|
|
index %= SIZE;
|
|
if (self.written < SIZE)
|
|
{
|
|
if (index >= self.written) return 0;
|
|
usz end = self.written - index;
|
|
usz n = min(end, buffer.len);
|
|
buffer[:n] = self.buf[index:n];
|
|
return n;
|
|
}
|
|
usz end = SIZE - self.head;
|
|
if (index >= end)
|
|
{
|
|
index -= end;
|
|
if (index >= self.head) return 0;
|
|
usz n = min(self.head - index, buffer.len);
|
|
buffer[:n] = self.buf[index:n];
|
|
return n;
|
|
}
|
|
if (buffer.len <= SIZE - index)
|
|
{
|
|
usz n = buffer.len;
|
|
buffer[:n] = self.buf[self.head + index:n];
|
|
return n;
|
|
}
|
|
usz n1 = SIZE - index;
|
|
buffer[:n1] = self.buf[self.head + index:n1];
|
|
buffer = buffer[n1..];
|
|
index -= n1;
|
|
usz n2 = min(self.head - index, buffer.len);
|
|
buffer[:n2] = self.buf[index:n2];
|
|
return n1 + n2;
|
|
}
|
|
|
|
fn void RingBuffer.write(&self, Type[] buffer)
|
|
{
|
|
usz i;
|
|
while (self.written < SIZE && i < buffer.len)
|
|
{
|
|
self.buf[self.written] = buffer[i++];
|
|
self.written++;
|
|
}
|
|
foreach (c : buffer[i..])
|
|
{
|
|
self.buf[self.head] = c;
|
|
self.head = (self.head + 1) % SIZE;
|
|
}
|
|
} |