240 lines
5.2 KiB
C
240 lines
5.2 KiB
C
/*
|
|
* Copyright (c) 2011 The WebRTC project authors. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
/*
|
|
* Provides a generic ring buffer that can be written to and read from with
|
|
* arbitrarily sized blocks. The AEC uses this for several different tasks.
|
|
*/
|
|
|
|
#include <stdlib.h>
|
|
#include <string.h>
|
|
#include "ring_buffer.h"
|
|
|
|
typedef struct {
|
|
int readPos;
|
|
int writePos;
|
|
int size;
|
|
char rwWrap;
|
|
bufdata_t *data;
|
|
} buf_t;
|
|
|
|
enum {SAME_WRAP, DIFF_WRAP};
|
|
|
|
int WebRtcApm_CreateBuffer(void **bufInst, int size)
|
|
{
|
|
buf_t *buf = NULL;
|
|
|
|
if (size < 0) {
|
|
return -1;
|
|
}
|
|
|
|
buf = malloc(sizeof(buf_t));
|
|
*bufInst = buf;
|
|
if (buf == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
buf->data = malloc(size*sizeof(bufdata_t));
|
|
if (buf->data == NULL) {
|
|
free(buf);
|
|
buf = NULL;
|
|
return -1;
|
|
}
|
|
|
|
buf->size = size;
|
|
return 0;
|
|
}
|
|
|
|
int WebRtcApm_InitBuffer(void *bufInst)
|
|
{
|
|
buf_t *buf = (buf_t*)bufInst;
|
|
|
|
buf->readPos = 0;
|
|
buf->writePos = 0;
|
|
buf->rwWrap = SAME_WRAP;
|
|
|
|
// Initialize buffer to zeros
|
|
memset(buf->data, 0, sizeof(bufdata_t)*buf->size);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int WebRtcApm_FreeBuffer(void *bufInst)
|
|
{
|
|
buf_t *buf = (buf_t*)bufInst;
|
|
|
|
if (buf == NULL) {
|
|
return -1;
|
|
}
|
|
|
|
free(buf->data);
|
|
free(buf);
|
|
|
|
return 0;
|
|
}
|
|
|
|
int WebRtcApm_ReadBuffer(void *bufInst, bufdata_t *data, int size)
|
|
{
|
|
buf_t *buf = (buf_t*)bufInst;
|
|
int n = 0, margin = 0;
|
|
|
|
if (size <= 0 || size > buf->size) {
|
|
return -1;
|
|
}
|
|
|
|
n = size;
|
|
if (buf->rwWrap == DIFF_WRAP) {
|
|
margin = buf->size - buf->readPos;
|
|
if (n > margin) {
|
|
buf->rwWrap = SAME_WRAP;
|
|
memcpy(data, buf->data + buf->readPos,
|
|
sizeof(bufdata_t)*margin);
|
|
buf->readPos = 0;
|
|
n = size - margin;
|
|
}
|
|
else {
|
|
memcpy(data, buf->data + buf->readPos,
|
|
sizeof(bufdata_t)*n);
|
|
buf->readPos += n;
|
|
return n;
|
|
}
|
|
}
|
|
|
|
if (buf->rwWrap == SAME_WRAP) {
|
|
margin = buf->writePos - buf->readPos;
|
|
if (margin > n)
|
|
margin = n;
|
|
memcpy(data + size - n, buf->data + buf->readPos,
|
|
sizeof(bufdata_t)*margin);
|
|
buf->readPos += margin;
|
|
n -= margin;
|
|
}
|
|
|
|
return size - n;
|
|
}
|
|
|
|
int WebRtcApm_WriteBuffer(void *bufInst, const bufdata_t *data, int size)
|
|
{
|
|
buf_t *buf = (buf_t*)bufInst;
|
|
int n = 0, margin = 0;
|
|
|
|
if (size < 0 || size > buf->size) {
|
|
return -1;
|
|
}
|
|
|
|
n = size;
|
|
if (buf->rwWrap == SAME_WRAP) {
|
|
margin = buf->size - buf->writePos;
|
|
if (n > margin) {
|
|
buf->rwWrap = DIFF_WRAP;
|
|
memcpy(buf->data + buf->writePos, data,
|
|
sizeof(bufdata_t)*margin);
|
|
buf->writePos = 0;
|
|
n = size - margin;
|
|
}
|
|
else {
|
|
memcpy(buf->data + buf->writePos, data,
|
|
sizeof(bufdata_t)*n);
|
|
buf->writePos += n;
|
|
return n;
|
|
}
|
|
}
|
|
|
|
if (buf->rwWrap == DIFF_WRAP) {
|
|
margin = buf->readPos - buf->writePos;
|
|
if (margin > n)
|
|
margin = n;
|
|
memcpy(buf->data + buf->writePos, data + size - n,
|
|
sizeof(bufdata_t)*margin);
|
|
buf->writePos += margin;
|
|
n -= margin;
|
|
}
|
|
|
|
return size - n;
|
|
}
|
|
|
|
int WebRtcApm_FlushBuffer(void *bufInst, int size)
|
|
{
|
|
buf_t *buf = (buf_t*)bufInst;
|
|
int n = 0, margin = 0;
|
|
|
|
if (size <= 0 || size > buf->size) {
|
|
return -1;
|
|
}
|
|
|
|
n = size;
|
|
if (buf->rwWrap == DIFF_WRAP) {
|
|
margin = buf->size - buf->readPos;
|
|
if (n > margin) {
|
|
buf->rwWrap = SAME_WRAP;
|
|
buf->readPos = 0;
|
|
n = size - margin;
|
|
}
|
|
else {
|
|
buf->readPos += n;
|
|
return n;
|
|
}
|
|
}
|
|
|
|
if (buf->rwWrap == SAME_WRAP) {
|
|
margin = buf->writePos - buf->readPos;
|
|
if (margin > n)
|
|
margin = n;
|
|
buf->readPos += margin;
|
|
n -= margin;
|
|
}
|
|
|
|
return size - n;
|
|
}
|
|
|
|
int WebRtcApm_StuffBuffer(void *bufInst, int size)
|
|
{
|
|
buf_t *buf = (buf_t*)bufInst;
|
|
int n = 0, margin = 0;
|
|
|
|
if (size <= 0 || size > buf->size) {
|
|
return -1;
|
|
}
|
|
|
|
n = size;
|
|
if (buf->rwWrap == SAME_WRAP) {
|
|
margin = buf->readPos;
|
|
if (n > margin) {
|
|
buf->rwWrap = DIFF_WRAP;
|
|
buf->readPos = buf->size - 1;
|
|
n -= margin + 1;
|
|
}
|
|
else {
|
|
buf->readPos -= n;
|
|
return n;
|
|
}
|
|
}
|
|
|
|
if (buf->rwWrap == DIFF_WRAP) {
|
|
margin = buf->readPos - buf->writePos;
|
|
if (margin > n)
|
|
margin = n;
|
|
buf->readPos -= margin;
|
|
n -= margin;
|
|
}
|
|
|
|
return size - n;
|
|
}
|
|
|
|
int WebRtcApm_get_buffer_size(const void *bufInst)
|
|
{
|
|
const buf_t *buf = (buf_t*)bufInst;
|
|
|
|
if (buf->rwWrap == SAME_WRAP)
|
|
return buf->writePos - buf->readPos;
|
|
else
|
|
return buf->size - buf->readPos + buf->writePos;
|
|
}
|