/* * 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; } 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; 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; }