1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-15 21:52:46 +01:00
RIOT/tests/driver_dac_dds/main.c
chrysn 4596f79c95 tests/dac_dds: Fix sine wave to fit in PCM range
The previous sine wave cast signed integers into the PCM range, causing
jumps at zero transitions. This shifts everything up by the respective
maximum signed integer, so that the signed idle zero becomes the
unsigned PCM signal's idle half-point and can continuously cover the
whole unsigned range.
2021-03-24 00:35:19 +01:00

347 lines
7.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 tests
* @{
*
* @file
* @brief DAC (audio) test application
*
* Generates Sine, Square and Sawtooth waves using
* a DAC.
* Connect a speaker or headphones to the DAC output
* pins of your board, you should be able to hear the
* generated sounds.
*
* @author Benjamin Valentin <benpicco@beuth-hochschule.de>
*
* @}
*/
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "mutex.h"
#include "dac_dds.h"
#include "dac_dds_params.h"
#include "shell.h"
#include "blob/hello.raw.h"
#ifndef DAC_DDS_CHAN
#define DAC_DDS_CHAN 0
#endif
#ifndef DAC_DDS_DAC
#define DAC_DDS_DAC DAC_DDS_PARAM_DAC
#endif
#ifndef ENABLE_GREETING
#define ENABLE_GREETING 1
#endif
#ifndef min
#define min(a, b) ((a) > (b) ? (b) : (a))
#endif
#ifndef DAC_BUF_SIZE
#define DAC_BUF_SIZE (2048)
#endif
static bool res_16b = 0;
static unsigned sample_rate = 8000;
#define ISIN_PERIOD (0x7FFF)
#define ISIN_MAX (0x1000)
/**
* @brief A sine approximation via a fourth-order cosine approx.
* source: https://www.coranac.com/2009/07/sines/
*
* @param x angle (with 2^15 units/circle)
* @return sine value (Q12)
*/
static int32_t isin(int32_t x)
{
int32_t c, y;
static const int32_t qN = 13,
qA = 12,
B = 19900,
C = 3516;
c = x << (30 - qN); /* Semi-circle info into carry. */
x -= 1 << qN; /* sine -> cosine calc */
x = x << (31 - qN); /* Mask with PI */
x = x >> (31 - qN); /* Note: SIGNED shift! (to qN) */
x = x * x >> (2 * qN - 14); /* x=x^2 To Q14 */
y = B - (x * C >> 14); /* B - x^2*C */
y = (1 << qA) /* A - x^2*(B-x^2*C) */
- (x * y >> 16);
return c >= 0 ? y : -y;
}
/* simple function to fill buffer with samples
*
* It is up to the caller to ensure that len is always at least period.
* */
typedef void (*sample_gen_t)(uint8_t *dst, size_t len, uint16_t period);
static void _fill_saw_samples_8(uint8_t *buf, size_t len, uint16_t period)
{
uint8_t x = 0;
unsigned step = 0xFF / period;
for (uint16_t i = 0; i < len; ++i) {
x += step;
buf[i] = x;
}
}
static void _fill_saw_samples_16(uint8_t *buf, size_t len, uint16_t period)
{
uint16_t x = 0;
unsigned step = 0xFFFF / period;
for (uint16_t i = 0; i < len; ++i) {
x += step;
buf[i] = x;
buf[++i] = x >> 8;
}
}
static void _fill_sine_samples_8(uint8_t *buf, size_t len, uint16_t period)
{
uint16_t x = 0;
unsigned step = ISIN_PERIOD / period;
for (uint16_t i = 0; i < period; ++i) {
x += step;
buf[i] = isin(x) >> 5;
buf[i] += INT8_MAX + 1;
}
for (uint16_t i = period; i < len; i += period) {
memcpy(&buf[i], &buf[0], period);
}
}
static void _fill_sine_samples_16(uint8_t *buf, size_t len, uint16_t period)
{
uint16_t x = 0;
unsigned step = ISIN_PERIOD / period;
period *= 2;
for (uint16_t i = 0; i < period; ++i) {
x += step;
uint16_t y = isin(x);
y += INT16_MAX + 1;
buf[i] = y;
buf[++i] = y >> 8;
}
for (uint16_t i = period; i < len; i += period) {
memcpy(&buf[i], &buf[0], period);
}
}
static void _fill_square_samples(uint8_t *buf, size_t len, uint16_t period)
{
period /= 2;
if (res_16b) {
period *= 2;
}
for (uint8_t *end = buf + len; buf < end; ) {
memset(buf, 0xFF, period);
buf += period;
memset(buf, 0x0, period);
buf += period;
}
}
static void _unlock(void *arg)
{
mutex_unlock(arg);
}
static void play_function(uint16_t period, uint32_t samples, sample_gen_t fun)
{
static uint8_t buf[DAC_BUF_SIZE];
mutex_t lock = MUTEX_INIT_LOCKED;
if (period > DAC_BUF_SIZE) {
printf("Period duration exceeds sample buffer size.\n");
return;
}
/* only work with whole wave periods */
uint16_t len_aligned = DAC_BUF_SIZE - DAC_BUF_SIZE % period;
/* One underrun indication is expected (for the first sample) */
int underruns = -1;
/* 16 bit samples doubles data rate */
if (res_16b) {
samples *= 2;
}
/* pre-calculate buffer */
fun(buf, len_aligned, period);
/* we want to block until the next buffer can be queued */
dac_dds_set_cb(DAC_DDS_CHAN, _unlock, &lock);
while (samples) {
size_t len = min(samples, len_aligned);
samples -= len;
underruns += !dac_dds_play(DAC_DDS_CHAN, buf, len);
/* wait for buffer flip */
mutex_lock(&lock);
}
if (underruns != 0) {
printf("During playback, %d underruns occurred.\n", underruns);
}
}
#if IS_USED(ENABLE_GREETING)
static int cmd_greeting(int argc, char **argv)
{
(void) argc;
(void) argv;
if (sample_rate != 8000 || res_16b) {
puts("Warning: audio clip was recoded with 8bit/8000 Hz");
}
puts("Play Greeting…");
/* we only want to play a single sample */
dac_dds_set_cb(DAC_DDS_CHAN, NULL, NULL);
dac_dds_play(DAC_DDS_CHAN, hello_raw, hello_raw_len);
return 0;
}
#endif
static void _dac_init(void)
{
printf("init DAC DDS with %d bit, %u Hz\n", res_16b ? 16 : 8, sample_rate);
dac_dds_init(DAC_DDS_CHAN, sample_rate,
res_16b ? DAC_FLAG_16BIT : DAC_FLAG_8BIT, NULL, NULL);
}
static int cmd_init(int argc, char **argv)
{
if (argc < 2) {
printf("usage: %s <freq> <bit>\n", argv[0]);
return 1;
}
if (argc > 2) {
unsigned bit = atoi(argv[2]);
if (bit != 8 && bit != 16) {
printf("Only 8 and 16 bit samples supported.\n");
return 1;
}
res_16b = bit == 16;
}
sample_rate = atoi(argv[1]);
_dac_init();
return 0;
}
static int cmd_saw(int argc, char **argv)
{
if (argc < 3) {
printf("usage: %s <freq> <secs>\n", argv[0]);
return 1;
}
unsigned freq = atoi(argv[1]);
unsigned secs = atoi(argv[2]);
play_function((sample_rate + freq/2) / freq, secs * sample_rate,
res_16b ? _fill_saw_samples_16 : _fill_saw_samples_8);
return 0;
}
static int cmd_sine(int argc, char **argv)
{
if (argc < 3) {
printf("usage: %s <freq> <secs>\n", argv[0]);
return 1;
}
unsigned freq = atoi(argv[1]);
unsigned secs = atoi(argv[2]);
play_function((sample_rate + freq/2) / freq, secs * sample_rate,
res_16b ? _fill_sine_samples_16 : _fill_sine_samples_8);
return 0;
}
static int cmd_square(int argc, char **argv)
{
if (argc < 3) {
printf("usage: %s <freq> <secs>\n", argv[0]);
return 1;
}
unsigned freq = atoi(argv[1]);
unsigned secs = atoi(argv[2]);
play_function((sample_rate + freq/2) / freq, secs * sample_rate,
_fill_square_samples);
return 0;
}
static const shell_command_t shell_commands[] = {
#if IS_USED(ENABLE_GREETING)
{ "hello", "Play Greeting", cmd_greeting },
#endif
{ "init", "Initialize DAC", cmd_init },
{ "saw", "Play sawtooth wave", cmd_saw },
{ "sine", "Play Sine wave", cmd_sine },
{ "square", "Play Square wave", cmd_square },
{ NULL, NULL, NULL }
};
int main(void)
{
dac_init(DAC_DDS_DAC);
/* Initialize to the idle level of 16bit audio */
dac_set(DAC_DDS_DAC, 1 << 15);
_dac_init();
/* start the shell */
char line_buf[SHELL_DEFAULT_BUFSIZE];
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
return 0;
}