C++程序  |  129行  |  3.23 KB

/* 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_ */