/** * Looking Glass * Copyright © 2017-2021 The Looking Glass Authors * https://looking-glass.io * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the Free * Software Foundation; either version 2 of the License, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., 59 * Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "common/ringbuffer.h" #include "common/locking.h" #include #include struct RingBuffer { RingBufferValueFn preOverwriteFn; void * preOverwriteUdata; int length; size_t valueSize; LG_Lock lock; int start, pos, count; char values[0]; }; RingBuffer ringbuffer_new(int length, size_t valueSize) { struct RingBuffer * rb = calloc(1, sizeof(*rb) + valueSize * length); rb->length = length; rb->valueSize = valueSize; LG_LOCK_INIT(rb->lock); return rb; } void ringbuffer_free(RingBuffer * rb) { if (!*rb) return; LG_LOCK_FREE(rb->lock); free(*rb); *rb = NULL; } void ringbuffer_push(RingBuffer rb, const void * value) { LG_LOCK(rb->lock); void * dst = rb->values + rb->pos * rb->valueSize; if (rb->count < rb->length) ++rb->count; else { if (++rb->start == rb->length) rb->start = 0; if (rb->preOverwriteFn) rb->preOverwriteFn(dst, rb->preOverwriteUdata); } memcpy(dst, value, rb->valueSize); if (++rb->pos == rb->length) rb->pos = 0; LG_UNLOCK(rb->lock); } bool ringbuffer_shift(RingBuffer rb, void * dst) { if (rb->count == 0) return false; LG_LOCK(rb->lock); memcpy(dst, rb->values + rb->start * rb->valueSize, rb->valueSize); --rb->count; if (++rb->start == rb->length) rb->start = 0; LG_UNLOCK(rb->lock); return true; } void ringbuffer_reset(RingBuffer rb) { LG_LOCK(rb->lock); rb->start = 0; rb->pos = 0; rb->count = 0; LG_UNLOCK(rb->lock); } int ringbuffer_getLength(const RingBuffer rb) { return rb->length; } int ringbuffer_getStart(const RingBuffer rb) { return rb->start; } int ringbuffer_getCount(const RingBuffer rb) { return rb->count; } void * ringbuffer_getValues(const RingBuffer rb) { return rb->values; } void * ringBuffer_getLastValue(const RingBuffer rb) { if (rb->count == 0) return NULL; int index = rb->start + rb->count - 1; if (index >= rb->length) index -= rb->length; return rb->values + index * rb->valueSize; } void ringbuffer_setPreOverwriteFn(const RingBuffer rb, RingBufferValueFn fn, void * udata) { rb->preOverwriteFn = fn; rb->preOverwriteUdata = udata; } void ringbuffer_forEach(const RingBuffer rb, RingBufferIterator fn, void * udata, bool reverse) { LG_LOCK(rb->lock); if (reverse) { int index = rb->start + rb->count - 1; if (index >= rb->length) index -= rb->length; for(int i = 0; i < rb->count; ++i) { void * value = rb->values + index * rb->valueSize; if (--index == -1) index = rb->length - 1; if (!fn(i, value, udata)) break; } } else { int index = rb->start; for(int i = 0; i < rb->count; ++i) { void * value = rb->values + index * rb->valueSize; if (++index == rb->length) index = 0; if (!fn(i, value, udata)) break; } } LG_UNLOCK(rb->lock); }