/* Copyright 2018 The Chromium OS Authors. All rights reserved.
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef FLOAT_BUFFER_H_
#define FLOAT_BUFFER_H_
#include "byte_buffer.h"
/*
* Circular buffer storing deinterleaved floating point data.
* Members:
* fp - Pointer to be filled wtih read/write position of the buffer.
* num_channels - Number of channels.
*/
struct float_buffer {
struct byte_buffer *buf;
float **fp;
unsigned int num_channels;
};
/*
* Creates an float_buffer.
* Args:
* max_size - The max number of frames this buffer may store.
* num_channels - Number of channels of the deinterleaved data.
*/
static inline struct float_buffer *float_buffer_create(
unsigned int max_size,
unsigned int num_channels)
{
struct float_buffer *b;
b = (struct float_buffer *)calloc(1, sizeof(*b));
b->num_channels = num_channels;
b->fp = (float **)malloc(num_channels * sizeof(float *));
b->buf = (struct byte_buffer *)
calloc(1, sizeof(struct byte_buffer) +
max_size * num_channels * sizeof(float));
b->buf->max_size = max_size;
b->buf->used_size = max_size;
return b;
}
/* Destroys the float buffer. */
static inline void float_buffer_destroy(struct float_buffer **b)
{
if (*b == NULL)
return;
byte_buffer_destroy(&(*b)->buf);
free((*b)->fp);
free(*b);
*b = NULL;
}
/* Gets the write pointer of given float_buffer. */
static inline float *const *float_buffer_write_pointer(struct float_buffer *b)
{
unsigned int i;
float *data = (float *)b->buf->bytes;
for (i = 0; i < b->num_channels; i++, data += b->buf->max_size)
b->fp[i] = data + b->buf->write_idx;
return b->fp;
}
/* Gets the number of frames can write to the float_buffer. */
static inline unsigned int float_buffer_writable(struct float_buffer *b)
{
return buf_writable(b->buf);
}
/* Marks |nwritten| of frames as written to float_buffer. */
static inline void float_buffer_written(struct float_buffer *b,
unsigned int nwritten)
{
buf_increment_write(b->buf, nwritten);
}
/* Gets the read pointer of given float_buffer. */
static inline float *const *float_buffer_read_pointer(struct float_buffer *b,
unsigned int offset,
unsigned int *readable)
{
unsigned int i;
float *data = (float *)b->buf->bytes;
unsigned int nread = buf_readable(b->buf);
if (offset >= buf_queued(b->buf)) {
*readable = 0;
offset = 0;
} else if (offset >= nread) {
/* wraps */
offset = offset + b->buf->read_idx - b->buf->max_size;
*readable = MIN(*readable, b->buf->write_idx - offset);
} else {
*readable = MIN(*readable, nread - offset);
offset += b->buf->read_idx;
}
for (i = 0; i < b->num_channels; i++, data += b->buf->max_size)
b->fp[i] = data + offset;
return b->fp;
}
/* Gets the buffer level in frames queued in float_buffer. */
static inline unsigned int float_buffer_level(struct float_buffer *b)
{
return buf_queued(b->buf);
}
/* Resets float_buffer to initial state. */
static inline void float_buffer_reset(struct float_buffer *b)
{
buf_reset(b->buf);
}
/* Marks |nread| frames as read in float_buffer. */
static inline void float_buffer_read(struct float_buffer *b,
unsigned int nread)
{
buf_increment_read(b->buf, nread);
}
#endif /* FLOAT_BUFFER_H_ */