haiku/src/system/kernel/util/ring_buffer.cpp

209 lines
4.2 KiB
C++
Raw Normal View History

/*
* Copyright 2005, Axel Dörfler, axeld@pinc-software.de. All rights reserved.
* Distributed under the terms of the MIT License.
*/
#include "ring_buffer.h"
#include <KernelExport.h>
#include <lock.h>
#include <stdlib.h>
#include <string.h>
/** This is a light-weight ring_buffer implementation.
* It does not provide any locking - you are supposed to ensure thread-safety
* with the restrictions you choose. Unless you are passing in unsafe buffers,
* the functions are safe to be called with interrupts turned off as well (not
* the user functions).
* They also don't use malloc() or any kind of locking after initialization.
*/
struct ring_buffer {
int32 first;
int32 in;
int32 size;
uint8 buffer[0];
};
static inline int32
space_left_in_buffer(struct ring_buffer *buffer)
{
return buffer->size - buffer->in;
}
static ssize_t
read_from_buffer(struct ring_buffer *buffer, uint8 *data, ssize_t length,
bool user)
{
int32 available = buffer->in;
if (length > available)
length = available;
if (length == 0)
return 0;
ssize_t bytesRead = length;
if (buffer->first + length < buffer->size) {
// simple copy
if (user) {
if (user_memcpy(data, buffer->buffer + buffer->first, length) < B_OK)
return B_BAD_ADDRESS;
} else
memcpy(data, buffer->buffer + buffer->first, length);
} else {
// need to copy both ends
size_t upper = buffer->size - buffer->first;
size_t lower = length - upper;
if (user) {
if (user_memcpy(data, buffer->buffer + buffer->first, upper) < B_OK
|| user_memcpy(data + upper, buffer->buffer, lower) < B_OK)
return B_BAD_ADDRESS;
} else {
memcpy(data, buffer->buffer + buffer->first, upper);
memcpy(data + upper, buffer->buffer, lower);
}
}
buffer->first = (buffer->first + bytesRead) % buffer->size;
buffer->in -= bytesRead;
return bytesRead;
}
static ssize_t
write_to_buffer(struct ring_buffer *buffer, const uint8 *data, ssize_t length,
bool user)
{
int32 left = space_left_in_buffer(buffer);
if (length > left)
length = left;
if (length == 0)
return 0;
ssize_t bytesWritten = length;
int32 position = buffer->first + buffer->in;
if (position + length <= buffer->size) {
// simple copy
if (user) {
if (user_memcpy(buffer->buffer + position, data, length) < B_OK)
return B_BAD_ADDRESS;
} else
memcpy(buffer->buffer + position, data, length);
} else {
// need to copy both ends
size_t upper = buffer->size - position;
size_t lower = length - upper;
if (user) {
if (user_memcpy(buffer->buffer + position, data, upper) < B_OK
|| user_memcpy(buffer->buffer, data + upper, lower) < B_OK)
return B_BAD_ADDRESS;
} else {
memcpy(buffer->buffer + position, data, upper);
memcpy(buffer->buffer, data + upper, lower);
}
}
buffer->in += bytesWritten;
return bytesWritten;
}
// #pragma mark -
struct ring_buffer *
create_ring_buffer(size_t size)
{
struct ring_buffer *buffer = (ring_buffer *)malloc(sizeof(ring_buffer) + size);
if (buffer == NULL)
return NULL;
buffer->size = size;
ring_buffer_clear(buffer);
return buffer;
}
void
delete_ring_buffer(struct ring_buffer *buffer)
{
free(buffer);
}
void
ring_buffer_clear(struct ring_buffer *buffer)
{
buffer->in = 0;
buffer->first = 0;
}
size_t
ring_buffer_readable(struct ring_buffer *buffer)
{
return buffer->in;
}
size_t
ring_buffer_writable(struct ring_buffer *buffer)
{
return buffer->size - buffer->in;
}
void
ring_buffer_flush(struct ring_buffer *buffer, size_t length)
{
// we can't flush more bytes than there are
if (length > (size_t)buffer->in)
length = buffer->in;
buffer->in -= length;
buffer->first = (buffer->first + length) % buffer->size;
}
size_t
ring_buffer_read(struct ring_buffer *buffer, uint8 *data, ssize_t length)
{
return read_from_buffer(buffer, data, length, false);
}
size_t
ring_buffer_write(struct ring_buffer *buffer, const uint8 *data, ssize_t length)
{
return write_to_buffer(buffer, data, length, false);
}
ssize_t
ring_buffer_user_read(struct ring_buffer *buffer, uint8 *data, ssize_t length)
{
return read_from_buffer(buffer, data, length, true);
}
ssize_t
ring_buffer_user_write(struct ring_buffer *buffer, const uint8 *data, ssize_t length)
{
return write_to_buffer(buffer, data, length, true);
}