mirror of
https://github.com/RIOT-OS/RIOT.git
synced 2025-01-18 12:52:44 +01:00
5681befe75
This will play a raw audio (8kHz, 8 Bit) snipped from the 1961 synthesized speech demo. (Can be disabled since the file is rather large with 17k) It will also play sine, square and sawtooth waves with frequencies that can be set on the command line. I tested this with the on-board speaker (connected to P0.26) of the mcb2388 as well as with hooking up a headphone to the ANALOG (PA02) header of the same54-xpro.
324 lines
7.2 KiB
C
324 lines
7.2 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 ENABLE_GREETING
|
|
#define ENABLE_GREETING 1
|
|
#endif
|
|
|
|
#ifndef min
|
|
#define min(a, b) ((a) > (b) ? (b) : (a))
|
|
#endif
|
|
|
|
#define DAC_BUF_SIZE (2048)
|
|
|
|
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 */
|
|
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;
|
|
}
|
|
|
|
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);
|
|
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;
|
|
|
|
/* only work with whole wave periods */
|
|
uint16_t len_aligned = DAC_BUF_SIZE - DAC_BUF_SIZE % period;
|
|
|
|
/* 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;
|
|
|
|
dac_dds_play(DAC_DDS_CHAN, buf, len);
|
|
|
|
/* wait for buffer flip */
|
|
mutex_lock(&lock);
|
|
}
|
|
}
|
|
|
|
#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 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)
|
|
{
|
|
printf("argc: %d\n", argc);
|
|
|
|
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_PARAM_DAC);
|
|
_dac_init();
|
|
|
|
/* start the shell */
|
|
char line_buf[SHELL_DEFAULT_BUFSIZE];
|
|
shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE);
|
|
|
|
return 0;
|
|
}
|