mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
4596f79c95
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.
347 lines
7.8 KiB
C
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;
|
|
}
|