1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 17:12:45 +01:00
RIOT/drivers/dac_dds/dac_dds.c
Benjamin Valentin ceb3f8443a drivers/dac_dds: add module to play sample buffer over a DAC
This adds an API function to play a buffer of Audio samples using a DAC.

Double buffered operation is supported by specifying a callback that will
be called when the next buffer can be queued with dac_dds_play().
2020-12-04 23:12:32 +01:00

144 lines
3.8 KiB
C

/*
* Copyright (C) 2020 Beuth Hochschule für Technik Berlin
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @ingroup drivers_periph_dac
* @{
*
* @file
* @brief Common DAC function fallback implementations
*
* @author Benjamin Valentin <benpicco@beuth-hochschule.de>
*
* @}
*/
#include <assert.h>
#include "board.h"
#include "dac_dds.h"
#include "dac_dds_params.h"
#include "irq.h"
#include "kernel_defines.h"
#include "macros/units.h"
#include "periph/timer.h"
static struct dac_ctx {
const uint8_t *buffers[2]; /* The two sample buffers */
size_t buffer_len[2]; /* Size of the sample buffers */
size_t idx; /* Current position in the current buffer */
dac_dds_cb_t cb; /* Called when the current buffer is done */
void *cb_arg; /* Callback argument */
uint16_t sample_ticks; /* Timer ticks per sample */
uint8_t cur; /* Active sample buffer */
uint8_t playing; /* DAC is playing */
uint8_t is_16bit; /* Sample size is 16 instead of 8 bit */
} _ctx[DAC_DDS_NUMOF];
static void _timer_cb(void *arg, int chan)
{
(void)chan;
struct dac_ctx *ctx = arg;
dac_dds_t dac_dds = ctx - _ctx;
dac_t dac = dac_dds_params[dac_dds].dac;
const uint8_t cur = ctx->cur;
const uint8_t *buf = ctx->buffers[cur];
const size_t len = ctx->buffer_len[cur];
if (ctx->is_16bit) {
uint8_t l = buf[ctx->idx++];
uint8_t h = buf[ctx->idx++];
dac_set(dac, (h << 8) | l);
} else {
dac_set(dac, buf[ctx->idx++] << 8);
}
if (ctx->idx >= len) {
/* invalidate old buffer */
ctx->buffer_len[cur] = 0;
ctx->idx = 0;
ctx->cur = !cur;
/* stop playing if no more samples are queued */
if (ctx->buffer_len[!cur] == 0) {
ctx->playing = 0;
timer_stop(dac_dds_params[dac_dds].timer);
/* notify user that next sample buffer can be queued */
} else if (ctx->cb) {
ctx->cb(ctx->cb_arg);
}
}
}
void dac_dds_init(dac_dds_t dac, uint16_t sample_rate, uint8_t flags,
dac_dds_cb_t cb, void *cb_arg)
{
assert(dac < DAC_DDS_NUMOF);
_ctx[dac].cb = cb;
_ctx[dac].cb_arg = cb_arg;
_ctx[dac].sample_ticks = dac_dds_params[dac].timer_hz / sample_rate;
_ctx[dac].is_16bit = flags & DAC_FLAG_16BIT;
timer_init(dac_dds_params[dac].timer, dac_dds_params[dac].timer_hz, _timer_cb, &_ctx[dac]);
}
void dac_dds_set_cb(dac_dds_t dac, dac_dds_cb_t cb, void *cb_arg)
{
unsigned state = irq_disable();
/* allow to update cb_arg independent of cb */
if (cb || cb_arg == NULL) {
_ctx[dac].cb = cb;
}
_ctx[dac].cb_arg = cb_arg;
irq_restore(state);
}
bool dac_dds_play(dac_dds_t dac, const void *buf, size_t len)
{
struct dac_ctx *ctx = &_ctx[dac];
unsigned state = irq_disable();
uint8_t next = !ctx->cur;
ctx->buffers[next] = buf;
ctx->buffer_len[next] = len;
bool is_playing = ctx->playing;
irq_restore(state);
if (is_playing) {
return true;
}
ctx->playing = 1;
ctx->cur = next;
timer_set_periodic(dac_dds_params[dac].timer, 0, ctx->sample_ticks,
TIM_FLAG_RESET_ON_MATCH | TIM_FLAG_RESET_ON_SET);
/* We can already queue the next buffer */
if (ctx->cb) {
ctx->cb(ctx->cb_arg);
}
return false;
}
void dac_dds_stop(dac_dds_t dac)
{
timer_stop(dac_dds_params[dac].timer);
_ctx[dac].playing = 0;
}