Added a simple ring buffer implementation to be used by the mouse
and keyboard drivers, as well as pipefs. git-svn-id: file:///srv/svn/repos/haiku/haiku/trunk@12357 a95241bf-73f2-0310-859d-f6bbb57e9c96
This commit is contained in:
parent
709bbb9ea3
commit
ef2b38e7a9
34
headers/private/kernel/util/ring_buffer.h
Normal file
34
headers/private/kernel/util/ring_buffer.h
Normal file
@ -0,0 +1,34 @@
|
||||
/*
|
||||
* Copyright 2005, Axel Dörfler, axeld@pinc-software.de. All rights reserved.
|
||||
* Distributed under the terms of the MIT License.
|
||||
*/
|
||||
#ifndef RING_BUFFER_H
|
||||
#define RING_BUFFER_H
|
||||
|
||||
|
||||
#include <OS.h>
|
||||
|
||||
|
||||
struct ring_buffer;
|
||||
|
||||
#ifdef __cplusplus
|
||||
extern "C" {
|
||||
#endif
|
||||
|
||||
struct ring_buffer *create_ring_buffer(size_t size);
|
||||
void delete_ring_buffer(struct ring_buffer *buffer);
|
||||
|
||||
void ring_buffer_clear(struct ring_buffer *buffer);
|
||||
size_t ring_buffer_readable(struct ring_buffer *buffer);
|
||||
size_t ring_buffer_writable(struct ring_buffer *buffer);
|
||||
void ring_buffer_flush(struct ring_buffer *buffer, size_t bytes);
|
||||
size_t ring_buffer_read(struct ring_buffer *buffer, uint8 *data, ssize_t length);
|
||||
size_t ring_buffer_write(struct ring_buffer *buffer, const uint8 *data, ssize_t length);
|
||||
ssize_t ring_buffer_user_read(struct ring_buffer *buffer, uint8 *data, ssize_t length);
|
||||
ssize_t ring_buffer_user_write(struct ring_buffer *buffer, const uint8 *data, ssize_t length);
|
||||
|
||||
#ifdef __cplusplus
|
||||
}
|
||||
#endif
|
||||
|
||||
#endif /* RING_BUFFER_H */
|
@ -8,6 +8,7 @@ KernelMergeObject kernel_util.o :
|
||||
khash.c
|
||||
list.c
|
||||
queue.c
|
||||
ring_buffer.cpp
|
||||
|
||||
: -fno-pic -Wno-unused -DUSING_LIBGCC
|
||||
;
|
||||
|
208
src/kernel/core/util/ring_buffer.cpp
Normal file
208
src/kernel/core/util/ring_buffer.cpp
Normal file
@ -0,0 +1,208 @@
|
||||
/*
|
||||
* 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);
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user