commit
e1181fbe59
@ -1,8 +0,0 @@ |
|||||||
# Copyright 2023 jacqueline <me@jacqueline.id.au> |
|
||||||
# |
|
||||||
# SPDX-License-Identifier: GPL-3.0-only |
|
||||||
|
|
||||||
idf_component_register( |
|
||||||
SRCS "stb_vorbis.c" |
|
||||||
INCLUDE_DIRS "include" |
|
||||||
) |
|
@ -1,418 +0,0 @@ |
|||||||
// Ogg Vorbis audio decoder - v1.22 - public domain
|
|
||||||
// http://nothings.org/stb_vorbis/
|
|
||||||
//
|
|
||||||
// Original version written by Sean Barrett in 2007.
|
|
||||||
//
|
|
||||||
// Originally sponsored by RAD Game Tools. Seeking implementation
|
|
||||||
// sponsored by Phillip Bennefall, Marc Andersen, Aaron Baker,
|
|
||||||
// Elias Software, Aras Pranckevicius, and Sean Barrett.
|
|
||||||
//
|
|
||||||
// LICENSE
|
|
||||||
//
|
|
||||||
// See end of file for license information.
|
|
||||||
//
|
|
||||||
// Limitations:
|
|
||||||
//
|
|
||||||
// - floor 0 not supported (used in old ogg vorbis files pre-2004)
|
|
||||||
// - lossless sample-truncation at beginning ignored
|
|
||||||
// - cannot concatenate multiple vorbis streams
|
|
||||||
// - sample positions are 32-bit, limiting seekable 192Khz
|
|
||||||
// files to around 6 hours (Ogg supports 64-bit)
|
|
||||||
//
|
|
||||||
// Feature contributors:
|
|
||||||
// Dougall Johnson (sample-exact seeking)
|
|
||||||
//
|
|
||||||
// Bugfix/warning contributors:
|
|
||||||
// Terje Mathisen Niklas Frykholm Andy Hill
|
|
||||||
// Casey Muratori John Bolton Gargaj
|
|
||||||
// Laurent Gomila Marc LeBlanc Ronny Chevalier
|
|
||||||
// Bernhard Wodo Evan Balster github:alxprd
|
|
||||||
// Tom Beaumont Ingo Leitgeb Nicolas Guillemot
|
|
||||||
// Phillip Bennefall Rohit Thiago Goulart
|
|
||||||
// github:manxorist Saga Musix github:infatum
|
|
||||||
// Timur Gagiev Maxwell Koo Peter Waller
|
|
||||||
// github:audinowho Dougall Johnson David Reid
|
|
||||||
// github:Clownacy Pedro J. Estebanez Remi Verschelde
|
|
||||||
// AnthoFoxo github:morlat Gabriel Ravier
|
|
||||||
//
|
|
||||||
// Partial history:
|
|
||||||
// 1.22 - 2021-07-11 - various small fixes
|
|
||||||
// 1.21 - 2021-07-02 - fix bug for files with no comments
|
|
||||||
// 1.20 - 2020-07-11 - several small fixes
|
|
||||||
// 1.19 - 2020-02-05 - warnings
|
|
||||||
// 1.18 - 2020-02-02 - fix seek bugs; parse header comments; misc warnings etc.
|
|
||||||
// 1.17 - 2019-07-08 - fix CVE-2019-13217..CVE-2019-13223 (by ForAllSecure)
|
|
||||||
// 1.16 - 2019-03-04 - fix warnings
|
|
||||||
// 1.15 - 2019-02-07 - explicit failure if Ogg Skeleton data is found
|
|
||||||
// 1.14 - 2018-02-11 - delete bogus dealloca usage
|
|
||||||
// 1.13 - 2018-01-29 - fix truncation of last frame (hopefully)
|
|
||||||
// 1.12 - 2017-11-21 - limit residue begin/end to blocksize/2 to avoid large temp allocs in bad/corrupt files
|
|
||||||
// 1.11 - 2017-07-23 - fix MinGW compilation
|
|
||||||
// 1.10 - 2017-03-03 - more robust seeking; fix negative ilog(); clear error in open_memory
|
|
||||||
// 1.09 - 2016-04-04 - back out 'truncation of last frame' fix from previous version
|
|
||||||
// 1.08 - 2016-04-02 - warnings; setup memory leaks; truncation of last frame
|
|
||||||
// 1.07 - 2015-01-16 - fixes for crashes on invalid files; warning fixes; const
|
|
||||||
// 1.06 - 2015-08-31 - full, correct support for seeking API (Dougall Johnson)
|
|
||||||
// some crash fixes when out of memory or with corrupt files
|
|
||||||
// fix some inappropriately signed shifts
|
|
||||||
// 1.05 - 2015-04-19 - don't define __forceinline if it's redundant
|
|
||||||
// 1.04 - 2014-08-27 - fix missing const-correct case in API
|
|
||||||
// 1.03 - 2014-08-07 - warning fixes
|
|
||||||
// 1.02 - 2014-07-09 - declare qsort comparison as explicitly _cdecl in Windows
|
|
||||||
// 1.01 - 2014-06-18 - fix stb_vorbis_get_samples_float (interleaved was correct)
|
|
||||||
// 1.0 - 2014-05-26 - fix memory leaks; fix warnings; fix bugs in >2-channel;
|
|
||||||
// (API change) report sample rate for decode-full-file funcs
|
|
||||||
//
|
|
||||||
// See end of file for full version history.
|
|
||||||
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////
|
|
||||||
//
|
|
||||||
// HEADER BEGINS HERE
|
|
||||||
//
|
|
||||||
|
|
||||||
#ifndef STB_VORBIS_INCLUDE_STB_VORBIS_H |
|
||||||
#define STB_VORBIS_INCLUDE_STB_VORBIS_H |
|
||||||
|
|
||||||
#if defined(STB_VORBIS_NO_CRT) && !defined(STB_VORBIS_NO_STDIO) |
|
||||||
#define STB_VORBIS_NO_STDIO 1 |
|
||||||
#endif |
|
||||||
|
|
||||||
#ifndef STB_VORBIS_NO_STDIO |
|
||||||
#include <stdio.h> |
|
||||||
#endif |
|
||||||
|
|
||||||
#ifdef __cplusplus |
|
||||||
extern "C" { |
|
||||||
#endif |
|
||||||
|
|
||||||
/////////// THREAD SAFETY
|
|
||||||
|
|
||||||
// Individual stb_vorbis* handles are not thread-safe; you cannot decode from
|
|
||||||
// them from multiple threads at the same time. However, you can have multiple
|
|
||||||
// stb_vorbis* handles and decode from them independently in multiple thrads.
|
|
||||||
|
|
||||||
|
|
||||||
/////////// MEMORY ALLOCATION
|
|
||||||
|
|
||||||
// normally stb_vorbis uses malloc() to allocate memory at startup,
|
|
||||||
// and alloca() to allocate temporary memory during a frame on the
|
|
||||||
// stack. (Memory consumption will depend on the amount of setup
|
|
||||||
// data in the file and how you set the compile flags for speed
|
|
||||||
// vs. size. In my test files the maximal-size usage is ~150KB.)
|
|
||||||
//
|
|
||||||
// You can modify the wrapper functions in the source (setup_malloc,
|
|
||||||
// setup_temp_malloc, temp_malloc) to change this behavior, or you
|
|
||||||
// can use a simpler allocation model: you pass in a buffer from
|
|
||||||
// which stb_vorbis will allocate _all_ its memory (including the
|
|
||||||
// temp memory). "open" may fail with a VORBIS_outofmem if you
|
|
||||||
// do not pass in enough data; there is no way to determine how
|
|
||||||
// much you do need except to succeed (at which point you can
|
|
||||||
// query get_info to find the exact amount required. yes I know
|
|
||||||
// this is lame).
|
|
||||||
//
|
|
||||||
// If you pass in a non-NULL buffer of the type below, allocation
|
|
||||||
// will occur from it as described above. Otherwise just pass NULL
|
|
||||||
// to use malloc()/alloca()
|
|
||||||
|
|
||||||
typedef struct |
|
||||||
{ |
|
||||||
char *alloc_buffer; |
|
||||||
int alloc_buffer_length_in_bytes; |
|
||||||
} stb_vorbis_alloc; |
|
||||||
|
|
||||||
|
|
||||||
/////////// FUNCTIONS USEABLE WITH ALL INPUT MODES
|
|
||||||
|
|
||||||
typedef struct stb_vorbis stb_vorbis; |
|
||||||
|
|
||||||
typedef struct |
|
||||||
{ |
|
||||||
unsigned int sample_rate; |
|
||||||
int channels; |
|
||||||
|
|
||||||
unsigned int setup_memory_required; |
|
||||||
unsigned int setup_temp_memory_required; |
|
||||||
unsigned int temp_memory_required; |
|
||||||
|
|
||||||
int max_frame_size; |
|
||||||
} stb_vorbis_info; |
|
||||||
|
|
||||||
typedef struct |
|
||||||
{ |
|
||||||
char *vendor; |
|
||||||
|
|
||||||
int comment_list_length; |
|
||||||
char **comment_list; |
|
||||||
} stb_vorbis_comment; |
|
||||||
|
|
||||||
// get general information about the file
|
|
||||||
extern stb_vorbis_info stb_vorbis_get_info(stb_vorbis *f); |
|
||||||
|
|
||||||
// get ogg comments
|
|
||||||
extern stb_vorbis_comment stb_vorbis_get_comment(stb_vorbis *f); |
|
||||||
|
|
||||||
// get the last error detected (clears it, too)
|
|
||||||
extern int stb_vorbis_get_error(stb_vorbis *f); |
|
||||||
|
|
||||||
// close an ogg vorbis file and free all memory in use
|
|
||||||
extern void stb_vorbis_close(stb_vorbis *f); |
|
||||||
|
|
||||||
// this function returns the offset (in samples) from the beginning of the
|
|
||||||
// file that will be returned by the next decode, if it is known, or -1
|
|
||||||
// otherwise. after a flush_pushdata() call, this may take a while before
|
|
||||||
// it becomes valid again.
|
|
||||||
// NOT WORKING YET after a seek with PULLDATA API
|
|
||||||
extern int stb_vorbis_get_sample_offset(stb_vorbis *f); |
|
||||||
|
|
||||||
// returns the current seek point within the file, or offset from the beginning
|
|
||||||
// of the memory buffer. In pushdata mode it returns 0.
|
|
||||||
extern unsigned int stb_vorbis_get_file_offset(stb_vorbis *f); |
|
||||||
|
|
||||||
/////////// PUSHDATA API
|
|
||||||
|
|
||||||
#ifndef STB_VORBIS_NO_PUSHDATA_API |
|
||||||
|
|
||||||
// this API allows you to get blocks of data from any source and hand
|
|
||||||
// them to stb_vorbis. you have to buffer them; stb_vorbis will tell
|
|
||||||
// you how much it used, and you have to give it the rest next time;
|
|
||||||
// and stb_vorbis may not have enough data to work with and you will
|
|
||||||
// need to give it the same data again PLUS more. Note that the Vorbis
|
|
||||||
// specification does not bound the size of an individual frame.
|
|
||||||
|
|
||||||
extern stb_vorbis *stb_vorbis_open_pushdata( |
|
||||||
const unsigned char * datablock, int datablock_length_in_bytes, |
|
||||||
int *datablock_memory_consumed_in_bytes, |
|
||||||
int *error, |
|
||||||
const stb_vorbis_alloc *alloc_buffer); |
|
||||||
// create a vorbis decoder by passing in the initial data block containing
|
|
||||||
// the ogg&vorbis headers (you don't need to do parse them, just provide
|
|
||||||
// the first N bytes of the file--you're told if it's not enough, see below)
|
|
||||||
// on success, returns an stb_vorbis *, does not set error, returns the amount of
|
|
||||||
// data parsed/consumed on this call in *datablock_memory_consumed_in_bytes;
|
|
||||||
// on failure, returns NULL on error and sets *error, does not change *datablock_memory_consumed
|
|
||||||
// if returns NULL and *error is VORBIS_need_more_data, then the input block was
|
|
||||||
// incomplete and you need to pass in a larger block from the start of the file
|
|
||||||
|
|
||||||
extern int stb_vorbis_decode_frame_pushdata( |
|
||||||
stb_vorbis *f, |
|
||||||
const unsigned char *datablock, int datablock_length_in_bytes, |
|
||||||
int *channels, // place to write number of float * buffers
|
|
||||||
float ***output, // place to write float ** array of float * buffers
|
|
||||||
int *samples // place to write number of output samples
|
|
||||||
); |
|
||||||
// decode a frame of audio sample data if possible from the passed-in data block
|
|
||||||
//
|
|
||||||
// return value: number of bytes we used from datablock
|
|
||||||
//
|
|
||||||
// possible cases:
|
|
||||||
// 0 bytes used, 0 samples output (need more data)
|
|
||||||
// N bytes used, 0 samples output (resynching the stream, keep going)
|
|
||||||
// N bytes used, M samples output (one frame of data)
|
|
||||||
// note that after opening a file, you will ALWAYS get one N-bytes,0-sample
|
|
||||||
// frame, because Vorbis always "discards" the first frame.
|
|
||||||
//
|
|
||||||
// Note that on resynch, stb_vorbis will rarely consume all of the buffer,
|
|
||||||
// instead only datablock_length_in_bytes-3 or less. This is because it wants
|
|
||||||
// to avoid missing parts of a page header if they cross a datablock boundary,
|
|
||||||
// without writing state-machiney code to record a partial detection.
|
|
||||||
//
|
|
||||||
// The number of channels returned are stored in *channels (which can be
|
|
||||||
// NULL--it is always the same as the number of channels reported by
|
|
||||||
// get_info). *output will contain an array of float* buffers, one per
|
|
||||||
// channel. In other words, (*output)[0][0] contains the first sample from
|
|
||||||
// the first channel, and (*output)[1][0] contains the first sample from
|
|
||||||
// the second channel.
|
|
||||||
//
|
|
||||||
// *output points into stb_vorbis's internal output buffer storage; these
|
|
||||||
// buffers are owned by stb_vorbis and application code should not free
|
|
||||||
// them or modify their contents. They are transient and will be overwritten
|
|
||||||
// once you ask for more data to get decoded, so be sure to grab any data
|
|
||||||
// you need before then.
|
|
||||||
|
|
||||||
extern void stb_vorbis_flush_pushdata(stb_vorbis *f); |
|
||||||
// inform stb_vorbis that your next datablock will not be contiguous with
|
|
||||||
// previous ones (e.g. you've seeked in the data); future attempts to decode
|
|
||||||
// frames will cause stb_vorbis to resynchronize (as noted above), and
|
|
||||||
// once it sees a valid Ogg page (typically 4-8KB, as large as 64KB), it
|
|
||||||
// will begin decoding the _next_ frame.
|
|
||||||
//
|
|
||||||
// if you want to seek using pushdata, you need to seek in your file, then
|
|
||||||
// call stb_vorbis_flush_pushdata(), then start calling decoding, then once
|
|
||||||
// decoding is returning you data, call stb_vorbis_get_sample_offset, and
|
|
||||||
// if you don't like the result, seek your file again and repeat.
|
|
||||||
#endif |
|
||||||
|
|
||||||
|
|
||||||
////////// PULLING INPUT API
|
|
||||||
|
|
||||||
#ifndef STB_VORBIS_NO_PULLDATA_API |
|
||||||
// This API assumes stb_vorbis is allowed to pull data from a source--
|
|
||||||
// either a block of memory containing the _entire_ vorbis stream, or a
|
|
||||||
// FILE * that you or it create, or possibly some other reading mechanism
|
|
||||||
// if you go modify the source to replace the FILE * case with some kind
|
|
||||||
// of callback to your code. (But if you don't support seeking, you may
|
|
||||||
// just want to go ahead and use pushdata.)
|
|
||||||
|
|
||||||
#if !defined(STB_VORBIS_NO_STDIO) && !defined(STB_VORBIS_NO_INTEGER_CONVERSION) |
|
||||||
extern int stb_vorbis_decode_filename(const char *filename, int *channels, int *sample_rate, short **output); |
|
||||||
#endif |
|
||||||
#if !defined(STB_VORBIS_NO_INTEGER_CONVERSION) |
|
||||||
extern int stb_vorbis_decode_memory(const unsigned char *mem, int len, int *channels, int *sample_rate, short **output); |
|
||||||
#endif |
|
||||||
// decode an entire file and output the data interleaved into a malloc()ed
|
|
||||||
// buffer stored in *output. The return value is the number of samples
|
|
||||||
// decoded, or -1 if the file could not be opened or was not an ogg vorbis file.
|
|
||||||
// When you're done with it, just free() the pointer returned in *output.
|
|
||||||
|
|
||||||
extern stb_vorbis * stb_vorbis_open_memory(const unsigned char *data, int len, |
|
||||||
int *error, const stb_vorbis_alloc *alloc_buffer); |
|
||||||
// create an ogg vorbis decoder from an ogg vorbis stream in memory (note
|
|
||||||
// this must be the entire stream!). on failure, returns NULL and sets *error
|
|
||||||
|
|
||||||
#ifndef STB_VORBIS_NO_STDIO |
|
||||||
extern stb_vorbis * stb_vorbis_open_filename(const char *filename, |
|
||||||
int *error, const stb_vorbis_alloc *alloc_buffer); |
|
||||||
// create an ogg vorbis decoder from a filename via fopen(). on failure,
|
|
||||||
// returns NULL and sets *error (possibly to VORBIS_file_open_failure).
|
|
||||||
|
|
||||||
extern stb_vorbis * stb_vorbis_open_file(FILE *f, int close_handle_on_close, |
|
||||||
int *error, const stb_vorbis_alloc *alloc_buffer); |
|
||||||
// create an ogg vorbis decoder from an open FILE *, looking for a stream at
|
|
||||||
// the _current_ seek point (ftell). on failure, returns NULL and sets *error.
|
|
||||||
// note that stb_vorbis must "own" this stream; if you seek it in between
|
|
||||||
// calls to stb_vorbis, it will become confused. Moreover, if you attempt to
|
|
||||||
// perform stb_vorbis_seek_*() operations on this file, it will assume it
|
|
||||||
// owns the _entire_ rest of the file after the start point. Use the next
|
|
||||||
// function, stb_vorbis_open_file_section(), to limit it.
|
|
||||||
|
|
||||||
extern stb_vorbis * stb_vorbis_open_file_section(FILE *f, int close_handle_on_close, |
|
||||||
int *error, const stb_vorbis_alloc *alloc_buffer, unsigned int len); |
|
||||||
// create an ogg vorbis decoder from an open FILE *, looking for a stream at
|
|
||||||
// the _current_ seek point (ftell); the stream will be of length 'len' bytes.
|
|
||||||
// on failure, returns NULL and sets *error. note that stb_vorbis must "own"
|
|
||||||
// this stream; if you seek it in between calls to stb_vorbis, it will become
|
|
||||||
// confused.
|
|
||||||
#endif |
|
||||||
|
|
||||||
extern int stb_vorbis_seek_frame(stb_vorbis *f, unsigned int sample_number); |
|
||||||
extern int stb_vorbis_seek(stb_vorbis *f, unsigned int sample_number); |
|
||||||
// these functions seek in the Vorbis file to (approximately) 'sample_number'.
|
|
||||||
// after calling seek_frame(), the next call to get_frame_*() will include
|
|
||||||
// the specified sample. after calling stb_vorbis_seek(), the next call to
|
|
||||||
// stb_vorbis_get_samples_* will start with the specified sample. If you
|
|
||||||
// do not need to seek to EXACTLY the target sample when using get_samples_*,
|
|
||||||
// you can also use seek_frame().
|
|
||||||
|
|
||||||
extern int stb_vorbis_seek_start(stb_vorbis *f); |
|
||||||
// this function is equivalent to stb_vorbis_seek(f,0)
|
|
||||||
|
|
||||||
extern unsigned int stb_vorbis_stream_length_in_samples(stb_vorbis *f); |
|
||||||
extern float stb_vorbis_stream_length_in_seconds(stb_vorbis *f); |
|
||||||
// these functions return the total length of the vorbis stream
|
|
||||||
|
|
||||||
extern int stb_vorbis_get_frame_float(stb_vorbis *f, int *channels, float ***output); |
|
||||||
// decode the next frame and return the number of samples. the number of
|
|
||||||
// channels returned are stored in *channels (which can be NULL--it is always
|
|
||||||
// the same as the number of channels reported by get_info). *output will
|
|
||||||
// contain an array of float* buffers, one per channel. These outputs will
|
|
||||||
// be overwritten on the next call to stb_vorbis_get_frame_*.
|
|
||||||
//
|
|
||||||
// You generally should not intermix calls to stb_vorbis_get_frame_*()
|
|
||||||
// and stb_vorbis_get_samples_*(), since the latter calls the former.
|
|
||||||
|
|
||||||
#ifndef STB_VORBIS_NO_INTEGER_CONVERSION |
|
||||||
extern int stb_vorbis_get_frame_short_interleaved(stb_vorbis *f, int num_c, short *buffer, int num_shorts); |
|
||||||
extern int stb_vorbis_get_frame_short (stb_vorbis *f, int num_c, short **buffer, int num_samples); |
|
||||||
#endif |
|
||||||
// decode the next frame and return the number of *samples* per channel.
|
|
||||||
// Note that for interleaved data, you pass in the number of shorts (the
|
|
||||||
// size of your array), but the return value is the number of samples per
|
|
||||||
// channel, not the total number of samples.
|
|
||||||
//
|
|
||||||
// The data is coerced to the number of channels you request according to the
|
|
||||||
// channel coercion rules (see below). You must pass in the size of your
|
|
||||||
// buffer(s) so that stb_vorbis will not overwrite the end of the buffer.
|
|
||||||
// The maximum buffer size needed can be gotten from get_info(); however,
|
|
||||||
// the Vorbis I specification implies an absolute maximum of 4096 samples
|
|
||||||
// per channel.
|
|
||||||
|
|
||||||
// Channel coercion rules:
|
|
||||||
// Let M be the number of channels requested, and N the number of channels present,
|
|
||||||
// and Cn be the nth channel; let stereo L be the sum of all L and center channels,
|
|
||||||
// and stereo R be the sum of all R and center channels (channel assignment from the
|
|
||||||
// vorbis spec).
|
|
||||||
// M N output
|
|
||||||
// 1 k sum(Ck) for all k
|
|
||||||
// 2 * stereo L, stereo R
|
|
||||||
// k l k > l, the first l channels, then 0s
|
|
||||||
// k l k <= l, the first k channels
|
|
||||||
// Note that this is not _good_ surround etc. mixing at all! It's just so
|
|
||||||
// you get something useful.
|
|
||||||
|
|
||||||
extern int stb_vorbis_get_samples_float_interleaved(stb_vorbis *f, int channels, float *buffer, int num_floats); |
|
||||||
extern int stb_vorbis_get_samples_float(stb_vorbis *f, int channels, float **buffer, int num_samples); |
|
||||||
// gets num_samples samples, not necessarily on a frame boundary--this requires
|
|
||||||
// buffering so you have to supply the buffers. DOES NOT APPLY THE COERCION RULES.
|
|
||||||
// Returns the number of samples stored per channel; it may be less than requested
|
|
||||||
// at the end of the file. If there are no more samples in the file, returns 0.
|
|
||||||
|
|
||||||
#ifndef STB_VORBIS_NO_INTEGER_CONVERSION |
|
||||||
extern int stb_vorbis_get_samples_short_interleaved(stb_vorbis *f, int channels, short *buffer, int num_shorts); |
|
||||||
extern int stb_vorbis_get_samples_short(stb_vorbis *f, int channels, short **buffer, int num_samples); |
|
||||||
#endif |
|
||||||
// gets num_samples samples, not necessarily on a frame boundary--this requires
|
|
||||||
// buffering so you have to supply the buffers. Applies the coercion rules above
|
|
||||||
// to produce 'channels' channels. Returns the number of samples stored per channel;
|
|
||||||
// it may be less than requested at the end of the file. If there are no more
|
|
||||||
// samples in the file, returns 0.
|
|
||||||
|
|
||||||
#endif |
|
||||||
|
|
||||||
//////// ERROR CODES
|
|
||||||
|
|
||||||
enum STBVorbisError |
|
||||||
{ |
|
||||||
VORBIS__no_error, |
|
||||||
|
|
||||||
VORBIS_need_more_data=1, // not a real error
|
|
||||||
|
|
||||||
VORBIS_invalid_api_mixing, // can't mix API modes
|
|
||||||
VORBIS_outofmem, // not enough memory
|
|
||||||
VORBIS_feature_not_supported, // uses floor 0
|
|
||||||
VORBIS_too_many_channels, // STB_VORBIS_MAX_CHANNELS is too small
|
|
||||||
VORBIS_file_open_failure, // fopen() failed
|
|
||||||
VORBIS_seek_without_length, // can't seek in unknown-length file
|
|
||||||
|
|
||||||
VORBIS_unexpected_eof=10, // file is truncated?
|
|
||||||
VORBIS_seek_invalid, // seek past EOF
|
|
||||||
|
|
||||||
// decoding errors (corrupt/invalid stream) -- you probably
|
|
||||||
// don't care about the exact details of these
|
|
||||||
|
|
||||||
// vorbis errors:
|
|
||||||
VORBIS_invalid_setup=20, |
|
||||||
VORBIS_invalid_stream, |
|
||||||
|
|
||||||
// ogg errors:
|
|
||||||
VORBIS_missing_capture_pattern=30, |
|
||||||
VORBIS_invalid_stream_structure_version, |
|
||||||
VORBIS_continued_packet_flag_invalid, |
|
||||||
VORBIS_incorrect_stream_serial_number, |
|
||||||
VORBIS_invalid_first_page, |
|
||||||
VORBIS_bad_packet_type, |
|
||||||
VORBIS_cant_find_last_page, |
|
||||||
VORBIS_seek_failed, |
|
||||||
VORBIS_ogg_skeleton_not_supported |
|
||||||
}; |
|
||||||
|
|
||||||
|
|
||||||
#ifdef __cplusplus |
|
||||||
} |
|
||||||
#endif |
|
||||||
|
|
||||||
#endif // STB_VORBIS_INCLUDE_STB_VORBIS_H
|
|
||||||
//
|
|
||||||
// HEADER ENDS HERE
|
|
||||||
//
|
|
||||||
//////////////////////////////////////////////////////////////////////////////
|
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,131 @@ |
|||||||
|
/*
|
||||||
|
* FIR filter coefficients from resample-1.x smallfilter.h
|
||||||
|
* see Digital Audio Resampling Home Page located at |
||||||
|
* http://ccrma.stanford.edu/~jos/resample/
|
||||||
|
*/ |
||||||
|
32767, 32766, 32764, 32760, 32755, 32749, 32741, 32731, 32721, 32708, |
||||||
|
32695, 32679, 32663, 32645, 32625, 32604, 32582, 32558, 32533, 32506, |
||||||
|
32478, 32448, 32417, 32385, 32351, 32316, 32279, 32241, 32202, 32161, |
||||||
|
32119, 32075, 32030, 31984, 31936, 31887, 31836, 31784, 31731, 31676, |
||||||
|
31620, 31563, 31504, 31444, 31383, 31320, 31256, 31191, 31124, 31056, |
||||||
|
30987, 30916, 30845, 30771, 30697, 30621, 30544, 30466, 30387, 30306, |
||||||
|
30224, 30141, 30057, 29971, 29884, 29796, 29707, 29617, 29525, 29433, |
||||||
|
29339, 29244, 29148, 29050, 28952, 28852, 28752, 28650, 28547, 28443, |
||||||
|
28338, 28232, 28125, 28017, 27908, 27797, 27686, 27574, 27461, 27346, |
||||||
|
27231, 27115, 26998, 26879, 26760, 26640, 26519, 26398, 26275, 26151, |
||||||
|
26027, 25901, 25775, 25648, 25520, 25391, 25262, 25131, 25000, 24868, |
||||||
|
24735, 24602, 24467, 24332, 24197, 24060, 23923, 23785, 23647, 23507, |
||||||
|
23368, 23227, 23086, 22944, 22802, 22659, 22515, 22371, 22226, 22081, |
||||||
|
21935, 21789, 21642, 21494, 21346, 21198, 21049, 20900, 20750, 20600, |
||||||
|
20449, 20298, 20146, 19995, 19842, 19690, 19537, 19383, 19230, 19076, |
||||||
|
18922, 18767, 18612, 18457, 18302, 18146, 17990, 17834, 17678, 17521, |
||||||
|
17365, 17208, 17051, 16894, 16737, 16579, 16422, 16264, 16106, 15949, |
||||||
|
15791, 15633, 15475, 15317, 15159, 15001, 14843, 14685, 14527, 14369, |
||||||
|
14212, 14054, 13896, 13739, 13581, 13424, 13266, 13109, 12952, 12795, |
||||||
|
12639, 12482, 12326, 12170, 12014, 11858, 11703, 11548, 11393, 11238, |
||||||
|
11084, 10929, 10776, 10622, 10469, 10316, 10164, 10011, 9860, 9708, |
||||||
|
9557, 9407, 9256, 9106, 8957, 8808, 8659, 8511, 8364, 8216, 8070, |
||||||
|
7924, 7778, 7633, 7488, 7344, 7200, 7057, 6914, 6773, 6631, 6490, |
||||||
|
6350, 6210, 6071, 5933, 5795, 5658, 5521, 5385, 5250, 5115, 4981, |
||||||
|
4848, 4716, 4584, 4452, 4322, 4192, 4063, 3935, 3807, 3680, 3554, |
||||||
|
3429, 3304, 3180, 3057, 2935, 2813, 2692, 2572, 2453, 2335, 2217, |
||||||
|
2101, 1985, 1870, 1755, 1642, 1529, 1418, 1307, 1197, 1088, 979, 872, |
||||||
|
765, 660, 555, 451, 348, 246, 145, 44, -54, -153, -250, -347, -443, |
||||||
|
-537, -631, -724, -816, -908, -998, -1087, -1175, -1263, -1349, -1435, |
||||||
|
-1519, -1603, -1685, -1767, -1848, -1928, -2006, -2084, -2161, -2237, |
||||||
|
-2312, -2386, -2459, -2531, -2603, -2673, -2742, -2810, -2878, -2944, |
||||||
|
-3009, -3074, -3137, -3200, -3261, -3322, -3381, -3440, -3498, -3554, |
||||||
|
-3610, -3665, -3719, -3772, -3824, -3875, -3925, -3974, -4022, -4069, |
||||||
|
-4116, -4161, -4205, -4249, -4291, -4333, -4374, -4413, -4452, -4490, |
||||||
|
-4527, -4563, -4599, -4633, -4666, -4699, -4730, -4761, -4791, -4820, |
||||||
|
-4848, -4875, -4901, -4926, -4951, -4974, -4997, -5019, -5040, -5060, |
||||||
|
-5080, -5098, -5116, -5133, -5149, -5164, -5178, -5192, -5205, -5217, |
||||||
|
-5228, -5238, -5248, -5257, -5265, -5272, -5278, -5284, -5289, -5293, |
||||||
|
-5297, -5299, -5301, -5303, -5303, -5303, -5302, -5300, -5298, -5295, |
||||||
|
-5291, -5287, -5282, -5276, -5270, -5263, -5255, -5246, -5237, -5228, |
||||||
|
-5217, -5206, -5195, -5183, -5170, -5157, -5143, -5128, -5113, -5097, |
||||||
|
-5081, -5064, -5047, -5029, -5010, -4991, -4972, -4952, -4931, -4910, |
||||||
|
-4889, -4867, -4844, -4821, -4797, -4774, -4749, -4724, -4699, -4673, |
||||||
|
-4647, -4620, -4593, -4566, -4538, -4510, -4481, -4452, -4422, -4393, |
||||||
|
-4363, -4332, -4301, -4270, -4238, -4206, -4174, -4142, -4109, -4076, |
||||||
|
-4042, -4009, -3975, -3940, -3906, -3871, -3836, -3801, -3765, -3729, |
||||||
|
-3693, -3657, -3620, -3584, -3547, -3510, -3472, -3435, -3397, -3360, |
||||||
|
-3322, -3283, -3245, -3207, -3168, -3129, -3091, -3052, -3013, -2973, |
||||||
|
-2934, -2895, -2855, -2816, -2776, -2736, -2697, -2657, -2617, -2577, |
||||||
|
-2537, -2497, -2457, -2417, -2377, -2337, -2297, -2256, -2216, -2176, |
||||||
|
-2136, -2096, -2056, -2016, -1976, -1936, -1896, -1856, -1817, -1777, |
||||||
|
-1737, -1698, -1658, -1619, -1579, -1540, -1501, -1462, -1423, -1384, |
||||||
|
-1345, -1306, -1268, -1230, -1191, -1153, -1115, -1077, -1040, -1002, |
||||||
|
-965, -927, -890, -854, -817, -780, -744, -708, -672, -636, -600, |
||||||
|
-565, -530, -494, -460, -425, -391, -356, -322, -289, -255, -222, |
||||||
|
-189, -156, -123, -91, -59, -27, 4, 35, 66, 97, 127, 158, 188, 218, |
||||||
|
247, 277, 306, 334, 363, 391, 419, 447, 474, 501, 528, 554, 581, 606, |
||||||
|
632, 657, 683, 707, 732, 756, 780, 803, 827, 850, 872, 895, 917, 939, |
||||||
|
960, 981, 1002, 1023, 1043, 1063, 1082, 1102, 1121, 1139, 1158, 1176, |
||||||
|
1194, 1211, 1228, 1245, 1262, 1278, 1294, 1309, 1325, 1340, 1354, |
||||||
|
1369, 1383, 1397, 1410, 1423, 1436, 1448, 1461, 1473, 1484, 1496, |
||||||
|
1507, 1517, 1528, 1538, 1548, 1557, 1566, 1575, 1584, 1592, 1600, |
||||||
|
1608, 1616, 1623, 1630, 1636, 1643, 1649, 1654, 1660, 1665, 1670, |
||||||
|
1675, 1679, 1683, 1687, 1690, 1694, 1697, 1700, 1702, 1704, 1706, |
||||||
|
1708, 1709, 1711, 1712, 1712, 1713, 1713, 1713, 1713, 1712, 1711, |
||||||
|
1710, 1709, 1708, 1706, 1704, 1702, 1700, 1697, 1694, 1691, 1688, |
||||||
|
1685, 1681, 1677, 1673, 1669, 1664, 1660, 1655, 1650, 1644, 1639, |
||||||
|
1633, 1627, 1621, 1615, 1609, 1602, 1596, 1589, 1582, 1575, 1567, |
||||||
|
1560, 1552, 1544, 1536, 1528, 1520, 1511, 1503, 1494, 1485, 1476, |
||||||
|
1467, 1458, 1448, 1439, 1429, 1419, 1409, 1399, 1389, 1379, 1368, |
||||||
|
1358, 1347, 1337, 1326, 1315, 1304, 1293, 1282, 1271, 1260, 1248, |
||||||
|
1237, 1225, 1213, 1202, 1190, 1178, 1166, 1154, 1142, 1130, 1118, |
||||||
|
1106, 1094, 1081, 1069, 1057, 1044, 1032, 1019, 1007, 994, 981, 969, |
||||||
|
956, 943, 931, 918, 905, 892, 879, 867, 854, 841, 828, 815, 802, 790, |
||||||
|
777, 764, 751, 738, 725, 713, 700, 687, 674, 662, 649, 636, 623, 611, |
||||||
|
598, 585, 573, 560, 548, 535, 523, 510, 498, 486, 473, 461, 449, 437, |
||||||
|
425, 413, 401, 389, 377, 365, 353, 341, 330, 318, 307, 295, 284, 272, |
||||||
|
261, 250, 239, 228, 217, 206, 195, 184, 173, 163, 152, 141, 131, 121, |
||||||
|
110, 100, 90, 80, 70, 60, 51, 41, 31, 22, 12, 3, -5, -14, -23, -32, |
||||||
|
-41, -50, -59, -67, -76, -84, -93, -101, -109, -117, -125, -133, -140, |
||||||
|
-148, -156, -163, -170, -178, -185, -192, -199, -206, -212, -219, |
||||||
|
-226, -232, -239, -245, -251, -257, -263, -269, -275, -280, -286, |
||||||
|
-291, -297, -302, -307, -312, -317, -322, -327, -332, -336, -341, |
||||||
|
-345, -349, -354, -358, -362, -366, -369, -373, -377, -380, -384, |
||||||
|
-387, -390, -394, -397, -400, -402, -405, -408, -411, -413, -416, |
||||||
|
-418, -420, -422, -424, -426, -428, -430, -432, -433, -435, -436, |
||||||
|
-438, -439, -440, -442, -443, -444, -445, -445, -446, -447, -447, |
||||||
|
-448, -448, -449, -449, -449, -449, -449, -449, -449, -449, -449, |
||||||
|
-449, -449, -448, -448, -447, -447, -446, -445, -444, -443, -443, |
||||||
|
-442, -441, -440, -438, -437, -436, -435, -433, -432, -430, -429, |
||||||
|
-427, -426, -424, -422, -420, -419, -417, -415, -413, -411, -409, |
||||||
|
-407, -405, -403, -400, -398, -396, -393, -391, -389, -386, -384, |
||||||
|
-381, -379, -376, -374, -371, -368, -366, -363, -360, -357, -355, |
||||||
|
-352, -349, -346, -343, -340, -337, -334, -331, -328, -325, -322, |
||||||
|
-319, -316, -313, -310, -307, -304, -301, -298, -294, -291, -288, |
||||||
|
-285, -282, -278, -275, -272, -269, -265, -262, -259, -256, -252, |
||||||
|
-249, -246, -243, -239, -236, -233, -230, -226, -223, -220, -217, |
||||||
|
-213, -210, -207, -204, -200, -197, -194, -191, -187, -184, -181, |
||||||
|
-178, -175, -172, -168, -165, -162, -159, -156, -153, -150, -147, |
||||||
|
-143, -140, -137, -134, -131, -128, -125, -122, -120, -117, -114, |
||||||
|
-111, -108, -105, -102, -99, -97, -94, -91, -88, -86, -83, -80, -78, |
||||||
|
-75, -72, -70, -67, -65, -62, -59, -57, -55, -52, -50, -47, -45, -43, |
||||||
|
-40, -38, -36, -33, -31, -29, -27, -25, -22, -20, -18, -16, -14, -12, |
||||||
|
-10, -8, -6, -4, -2, 0, 0, 2, 4, 6, 8, 9, 11, 13, 14, 16, 17, 19, 21, |
||||||
|
22, 24, 25, 27, 28, 29, 31, 32, 33, 35, 36, 37, 38, 40, 41, 42, 43, |
||||||
|
44, 45, 46, 47, 48, 49, 50, 51, 52, 53, 54, 55, 56, 56, 57, 58, 59, |
||||||
|
59, 60, 61, 62, 62, 63, 63, 64, 64, 65, 66, 66, 66, 67, 67, 68, 68, |
||||||
|
69, 69, 69, 70, 70, 70, 70, 71, 71, 71, 71, 71, 72, 72, 72, 72, 72, |
||||||
|
72, 72, 72, 72, 72, 72, 72, 72, 72, 72, 72, 72, 72, 72, 72, 72, 72, |
||||||
|
72, 71, 71, 71, 71, 71, 70, 70, 70, 70, 69, 69, 69, 69, 68, 68, 68, |
||||||
|
67, 67, 67, 66, 66, 66, 65, 65, 64, 64, 64, 63, 63, 62, 62, 62, 61, |
||||||
|
61, 60, 60, 59, 59, 58, 58, 58, 57, 57, 56, 56, 55, 55, 54, 54, 53, |
||||||
|
53, 52, 52, 51, 51, 50, 50, 49, 48, 48, 47, 47, 46, 46, 45, 45, 44, |
||||||
|
44, 43, 43, 42, 42, 41, 41, 40, 39, 39, 38, 38, 37, 37, 36, 36, 35, |
||||||
|
35, 34, 34, 33, 33, 32, 32, 31, 31, 30, 30, 29, 29, 28, 28, 27, 27, |
||||||
|
26, 26, 25, 25, 24, 24, 23, 23, 23, 22, 22, 21, 21, 20, 20, 20, 19, |
||||||
|
19, 18, 18, 17, 17, 17, 16, 16, 15, 15, 15, 14, 14, 14, 13, 13, 12, |
||||||
|
12, 12, 11, 11, 11, 10, 10, 10, 9, 9, 9, 9, 8, 8, 8, 7, 7, 7, 7, 6, 6, |
||||||
|
6, 6, 5, 5, 5, 5, 4, 4, 4, 4, 3, 3, 3, 3, 3, 2, 2, 2, 2, 2, 1, 1, 1, |
||||||
|
1, 1, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, -1, -1, -1, |
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -2, -2, -2, -2, -2, -2, -2, |
||||||
|
-2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, |
||||||
|
-2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, |
||||||
|
-2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, -2, |
||||||
|
-2, -2, -2, -2, -2, -2, -2, -2, -1, -1, -1, -1, -1, -1, -1, -1, -1, |
||||||
|
-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, |
@ -0,0 +1,44 @@ |
|||||||
|
#pragma once |
||||||
|
|
||||||
|
#include <sys/_stdint.h> |
||||||
|
#include <vector> |
||||||
|
|
||||||
|
#include "span.hpp" |
||||||
|
|
||||||
|
#include "sample.hpp" |
||||||
|
|
||||||
|
namespace audio { |
||||||
|
|
||||||
|
class Resampler { |
||||||
|
public: |
||||||
|
Resampler(uint32_t source_sample_rate, |
||||||
|
uint32_t target_sample_rate, |
||||||
|
uint8_t num_channels); |
||||||
|
|
||||||
|
~Resampler(); |
||||||
|
|
||||||
|
auto source_sample_rate() -> uint32_t { return source_sample_rate_; } |
||||||
|
auto target_sample_rate() -> uint32_t { return target_sample_rate_; } |
||||||
|
auto channels() -> uint_fast8_t { return num_channels_; } |
||||||
|
|
||||||
|
auto Process(cpp::span<const sample::Sample> input, |
||||||
|
cpp::span<sample::Sample> output, |
||||||
|
bool end_of_data) -> std::pair<size_t, size_t>; |
||||||
|
|
||||||
|
private: |
||||||
|
auto Subsample(int channel) -> float; |
||||||
|
auto ApplyFilter(cpp::span<float> filter, cpp::span<float> input) -> float; |
||||||
|
|
||||||
|
uint32_t source_sample_rate_; |
||||||
|
uint32_t target_sample_rate_; |
||||||
|
float factor_; |
||||||
|
uint8_t num_channels_; |
||||||
|
|
||||||
|
std::vector<float*> channel_buffers_; |
||||||
|
size_t channel_buffer_size_; |
||||||
|
|
||||||
|
float output_offset_; |
||||||
|
int32_t input_index_; |
||||||
|
}; |
||||||
|
|
||||||
|
} // namespace audio
|
@ -0,0 +1,71 @@ |
|||||||
|
/*
|
||||||
|
* Copyright 2023 jacqueline <me@jacqueline.id.au> |
||||||
|
* |
||||||
|
* SPDX-License-Identifier: GPL-3.0-only |
||||||
|
*/ |
||||||
|
|
||||||
|
#pragma once |
||||||
|
|
||||||
|
#include <sys/_stdint.h> |
||||||
|
#include <cstdint> |
||||||
|
#include <memory> |
||||||
|
|
||||||
|
#include "resample.hpp" |
||||||
|
#include "sample.hpp" |
||||||
|
|
||||||
|
#include "audio_decoder.hpp" |
||||||
|
#include "audio_sink.hpp" |
||||||
|
#include "audio_source.hpp" |
||||||
|
#include "codec.hpp" |
||||||
|
#include "pipeline.hpp" |
||||||
|
#include "stream_info.hpp" |
||||||
|
|
||||||
|
namespace audio { |
||||||
|
|
||||||
|
/*
|
||||||
|
* Handles the final downmix + resample + quantisation stage of audio, |
||||||
|
* generation sending the result directly to an IAudioSink. |
||||||
|
*/ |
||||||
|
class SinkMixer { |
||||||
|
public: |
||||||
|
SinkMixer(StreamBufferHandle_t dest); |
||||||
|
~SinkMixer(); |
||||||
|
|
||||||
|
auto MixAndSend(InputStream&, const StreamInfo::Pcm&) -> std::size_t; |
||||||
|
|
||||||
|
private: |
||||||
|
auto Main() -> void; |
||||||
|
|
||||||
|
auto SetTargetFormat(const StreamInfo::Pcm& format) -> void; |
||||||
|
auto HandleBytes() -> void; |
||||||
|
|
||||||
|
auto Resample(InputStream&, OutputStream&) -> bool; |
||||||
|
auto ApplyDither(cpp::span<sample::Sample> samples, uint_fast8_t bits) |
||||||
|
-> void; |
||||||
|
auto Downscale(cpp::span<sample::Sample>, cpp::span<int16_t>) -> void; |
||||||
|
|
||||||
|
enum class Command { |
||||||
|
kReadBytes, |
||||||
|
kSetSourceFormat, |
||||||
|
kSetTargetFormat, |
||||||
|
}; |
||||||
|
|
||||||
|
struct Args { |
||||||
|
Command cmd; |
||||||
|
StreamInfo::Pcm format; |
||||||
|
}; |
||||||
|
|
||||||
|
QueueHandle_t commands_; |
||||||
|
SemaphoreHandle_t is_idle_; |
||||||
|
|
||||||
|
std::unique_ptr<Resampler> resampler_; |
||||||
|
|
||||||
|
std::unique_ptr<RawStream> input_stream_; |
||||||
|
std::unique_ptr<RawStream> resampled_stream_; |
||||||
|
|
||||||
|
StreamInfo::Pcm target_format_; |
||||||
|
StreamBufferHandle_t source_; |
||||||
|
StreamBufferHandle_t sink_; |
||||||
|
}; |
||||||
|
|
||||||
|
} // namespace audio
|
@ -0,0 +1,205 @@ |
|||||||
|
/*
|
||||||
|
* Copyright 2023 jacqueline <me@jacqueline.id.au> |
||||||
|
* |
||||||
|
* SPDX-License-Identifier: GPL-3.0-only |
||||||
|
*/ |
||||||
|
#include "resample.hpp" |
||||||
|
/*
|
||||||
|
* This file contains the implementation for a 32-bit floating point resampler. |
||||||
|
* It is largely based on David Bryant's ART resampler, which is BSD-licensed, |
||||||
|
* and available at https://github.com/dbry/audio-resampler/.
|
||||||
|
* |
||||||
|
* This resampler uses windowed sinc interpolation filters, with an additional |
||||||
|
* lowpass filter to reduce aliasing. |
||||||
|
*/ |
||||||
|
|
||||||
|
#include <stdint.h> |
||||||
|
#include <stdlib.h> |
||||||
|
#include <string.h> |
||||||
|
#include <algorithm> |
||||||
|
#include <cmath> |
||||||
|
#include <numeric> |
||||||
|
|
||||||
|
#include "esp_log.h" |
||||||
|
|
||||||
|
#include "sample.hpp" |
||||||
|
#include "stream_info.hpp" |
||||||
|
|
||||||
|
namespace audio { |
||||||
|
|
||||||
|
static constexpr double kLowPassRatio = 0.5; |
||||||
|
static constexpr size_t kNumFilters = 64; |
||||||
|
static constexpr size_t kFilterSize = 16; |
||||||
|
|
||||||
|
typedef std::array<float, kFilterSize> Filter; |
||||||
|
static std::array<Filter, kNumFilters + 1> sFilters{}; |
||||||
|
static bool sFiltersInitialised = false; |
||||||
|
|
||||||
|
auto InitFilter(int index) -> void; |
||||||
|
|
||||||
|
Resampler::Resampler(uint32_t source_sample_rate, |
||||||
|
uint32_t target_sample_rate, |
||||||
|
uint8_t num_channels) |
||||||
|
: source_sample_rate_(source_sample_rate), |
||||||
|
target_sample_rate_(target_sample_rate), |
||||||
|
factor_(static_cast<double>(target_sample_rate) / |
||||||
|
static_cast<double>(source_sample_rate)), |
||||||
|
num_channels_(num_channels) { |
||||||
|
channel_buffers_.resize(num_channels); |
||||||
|
channel_buffer_size_ = kFilterSize * 16; |
||||||
|
|
||||||
|
for (int i = 0; i < num_channels; i++) { |
||||||
|
channel_buffers_[i] = |
||||||
|
static_cast<float*>(calloc(sizeof(float), channel_buffer_size_)); |
||||||
|
} |
||||||
|
|
||||||
|
output_offset_ = kFilterSize / 2.0f; |
||||||
|
input_index_ = kFilterSize; |
||||||
|
|
||||||
|
if (!sFiltersInitialised) { |
||||||
|
sFiltersInitialised = true; |
||||||
|
for (int i = 0; i < kNumFilters + 1; i++) { |
||||||
|
InitFilter(i); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
Resampler::~Resampler() {} |
||||||
|
|
||||||
|
auto Resampler::Process(cpp::span<const sample::Sample> input, |
||||||
|
cpp::span<sample::Sample> output, |
||||||
|
bool end_of_data) -> std::pair<size_t, size_t> { |
||||||
|
size_t samples_used = 0; |
||||||
|
size_t samples_produced = 0; |
||||||
|
|
||||||
|
size_t input_frames = input.size() / num_channels_; |
||||||
|
size_t output_frames = output.size() / num_channels_; |
||||||
|
|
||||||
|
int half_taps = kFilterSize / 2; |
||||||
|
while (output_frames > 0) { |
||||||
|
if (output_offset_ >= input_index_ - half_taps) { |
||||||
|
if (input_frames > 0) { |
||||||
|
// Check whether the channel buffers will overflow with the addition of
|
||||||
|
// this sample. If so, we need to move the remaining contents back to
|
||||||
|
// the beginning of the buffer.
|
||||||
|
if (input_index_ == channel_buffer_size_) { |
||||||
|
for (int i = 0; i < num_channels_; ++i) { |
||||||
|
memmove(channel_buffers_[i], |
||||||
|
channel_buffers_[i] + channel_buffer_size_ - kFilterSize, |
||||||
|
kFilterSize * sizeof(float)); |
||||||
|
} |
||||||
|
|
||||||
|
output_offset_ -= channel_buffer_size_ - kFilterSize; |
||||||
|
input_index_ -= channel_buffer_size_ - kFilterSize; |
||||||
|
} |
||||||
|
|
||||||
|
for (int i = 0; i < num_channels_; ++i) { |
||||||
|
channel_buffers_[i][input_index_] = |
||||||
|
sample::ToFloat(input[samples_used++]); |
||||||
|
} |
||||||
|
|
||||||
|
input_index_++; |
||||||
|
input_frames--; |
||||||
|
} else { |
||||||
|
break; |
||||||
|
} |
||||||
|
} else { |
||||||
|
for (int i = 0; i < num_channels_; i++) { |
||||||
|
output[samples_produced++] = sample::FromFloat(Subsample(i)); |
||||||
|
} |
||||||
|
|
||||||
|
// NOTE: floating point division here is potentially slow due to FPU
|
||||||
|
// limitations. Consider explicitly bunding the xtensa libgcc divsion via
|
||||||
|
// reciprocal implementation if we care about portability between
|
||||||
|
// compilers.
|
||||||
|
output_offset_ += 1.0f / factor_; |
||||||
|
output_frames--; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
return {samples_used, samples_produced}; |
||||||
|
} |
||||||
|
|
||||||
|
/*
|
||||||
|
* Constructs the filter in-place for the given index of sFilters. This only |
||||||
|
* needs to be done once, per-filter. 64-bit math is okay here, because filters |
||||||
|
* will not be initialised within a performance critical path. |
||||||
|
*/ |
||||||
|
auto InitFilter(int index) -> void { |
||||||
|
Filter& filter = sFilters[index]; |
||||||
|
std::array<double, kFilterSize> working_buffer{}; |
||||||
|
|
||||||
|
double fraction = index / static_cast<double>(kNumFilters); |
||||||
|
double filter_sum = 0.0; |
||||||
|
|
||||||
|
for (int i = 0; i < kFilterSize; ++i) { |
||||||
|
// "dist" is the absolute distance from the sinc maximum to the filter tap
|
||||||
|
// to be calculated, in radians.
|
||||||
|
double dist = fabs((kFilterSize / 2.0 - 1.0) + fraction - i) * M_PI; |
||||||
|
// "ratio" is that distance divided by half the tap count such that it
|
||||||
|
// reaches π at the window extremes
|
||||||
|
double ratio = dist / (kFilterSize / 2.0); |
||||||
|
|
||||||
|
double value; |
||||||
|
if (dist != 0.0) { |
||||||
|
value = sin(dist * kLowPassRatio) / (dist * kLowPassRatio); |
||||||
|
|
||||||
|
// Hann window. We could alternatively use a Blackman Harris window,
|
||||||
|
// however our unusually small filter size makes the Hann window's
|
||||||
|
// steeper cutoff more important.
|
||||||
|
value *= 0.5 * (1.0 + cos(ratio)); |
||||||
|
} else { |
||||||
|
value = 1.0; |
||||||
|
} |
||||||
|
|
||||||
|
working_buffer[i] = value; |
||||||
|
filter_sum += value; |
||||||
|
} |
||||||
|
|
||||||
|
// Filter should have unity DC gain
|
||||||
|
double scaler = 1.0 / filter_sum; |
||||||
|
double error = 0.0; |
||||||
|
|
||||||
|
for (int i = kFilterSize / 2; i < kFilterSize; |
||||||
|
i = kFilterSize - i - (i >= kFilterSize / 2)) { |
||||||
|
working_buffer[i] *= scaler; |
||||||
|
filter[i] = working_buffer[i] - error; |
||||||
|
error += static_cast<double>(filter[i]) - working_buffer[i]; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
/*
|
||||||
|
* Performs sub-sampling with interpolation for the given channel. Assumes that |
||||||
|
* the channel buffer has already been filled with samples. |
||||||
|
*/ |
||||||
|
auto Resampler::Subsample(int channel) -> float { |
||||||
|
cpp::span<float> source{channel_buffers_[channel], channel_buffer_size_}; |
||||||
|
|
||||||
|
int offset_integral = std::floor(output_offset_); |
||||||
|
source = source.subspan(offset_integral); |
||||||
|
float offset_fractional = output_offset_ - offset_integral; |
||||||
|
|
||||||
|
offset_fractional *= kNumFilters; |
||||||
|
int filter_index = std::floor(offset_fractional); |
||||||
|
|
||||||
|
float sum1 = ApplyFilter(sFilters[filter_index], |
||||||
|
{source.data() - kFilterSize / 2 + 1, kFilterSize}); |
||||||
|
|
||||||
|
offset_fractional -= filter_index; |
||||||
|
|
||||||
|
float sum2 = ApplyFilter(sFilters[filter_index + 1], |
||||||
|
{source.data() - kFilterSize / 2 + 1, kFilterSize}); |
||||||
|
|
||||||
|
return (sum2 * offset_fractional) + (sum1 * (1.0f - offset_fractional)); |
||||||
|
} |
||||||
|
|
||||||
|
auto Resampler::ApplyFilter(cpp::span<float> filter, cpp::span<float> input) |
||||||
|
-> float { |
||||||
|
float sum = 0.0; |
||||||
|
for (int i = 0; i < kFilterSize; i++) { |
||||||
|
sum += filter[i] * input[i]; |
||||||
|
} |
||||||
|
return sum; |
||||||
|
} |
||||||
|
|
||||||
|
} // namespace audio
|
@ -0,0 +1,224 @@ |
|||||||
|
/*
|
||||||
|
* Copyright 2023 jacqueline <me@jacqueline.id.au> |
||||||
|
* |
||||||
|
* SPDX-License-Identifier: GPL-3.0-only |
||||||
|
*/ |
||||||
|
|
||||||
|
#include "sink_mixer.hpp" |
||||||
|
|
||||||
|
#include <stdint.h> |
||||||
|
#include <cmath> |
||||||
|
|
||||||
|
#include "esp_heap_caps.h" |
||||||
|
#include "esp_log.h" |
||||||
|
#include "freertos/portmacro.h" |
||||||
|
#include "freertos/projdefs.h" |
||||||
|
#include "idf_additions.h" |
||||||
|
#include "resample.hpp" |
||||||
|
#include "sample.hpp" |
||||||
|
|
||||||
|
#include "stream_info.hpp" |
||||||
|
#include "tasks.hpp" |
||||||
|
|
||||||
|
static constexpr char kTag[] = "mixer"; |
||||||
|
|
||||||
|
static constexpr std::size_t kSourceBufferLength = 8 * 1024; |
||||||
|
static constexpr std::size_t kSampleBufferLength = 240 * 2 * sizeof(int32_t); |
||||||
|
|
||||||
|
namespace audio { |
||||||
|
|
||||||
|
SinkMixer::SinkMixer(StreamBufferHandle_t dest) |
||||||
|
: commands_(xQueueCreate(1, sizeof(Args))), |
||||||
|
is_idle_(xSemaphoreCreateBinary()), |
||||||
|
resampler_(nullptr), |
||||||
|
source_(xStreamBufferCreateWithCaps(kSourceBufferLength, |
||||||
|
1, |
||||||
|
MALLOC_CAP_SPIRAM)), |
||||||
|
sink_(dest) { |
||||||
|
input_stream_.reset(new RawStream(kSampleBufferLength)); |
||||||
|
resampled_stream_.reset(new RawStream(kSampleBufferLength)); |
||||||
|
|
||||||
|
// Pin to CORE0 because we need the FPU.
|
||||||
|
// FIXME: A fixed point implementation could run freely on either core,
|
||||||
|
// which should lead to a big performance increase.
|
||||||
|
tasks::StartPersistent<tasks::Type::kMixer>(0, [&]() { Main(); }); |
||||||
|
} |
||||||
|
|
||||||
|
SinkMixer::~SinkMixer() { |
||||||
|
vQueueDelete(commands_); |
||||||
|
vSemaphoreDelete(is_idle_); |
||||||
|
vStreamBufferDelete(source_); |
||||||
|
} |
||||||
|
|
||||||
|
auto SinkMixer::MixAndSend(InputStream& input, const StreamInfo::Pcm& target) |
||||||
|
-> std::size_t { |
||||||
|
if (input.info().format_as<StreamInfo::Pcm>() != |
||||||
|
input_stream_->info().format_as<StreamInfo::Pcm>()) { |
||||||
|
xSemaphoreTake(is_idle_, portMAX_DELAY); |
||||||
|
Args args{ |
||||||
|
.cmd = Command::kSetSourceFormat, |
||||||
|
.format = input.info().format_as<StreamInfo::Pcm>().value(), |
||||||
|
}; |
||||||
|
xQueueSend(commands_, &args, portMAX_DELAY); |
||||||
|
xSemaphoreGive(is_idle_); |
||||||
|
} |
||||||
|
if (target_format_ != target) { |
||||||
|
xSemaphoreTake(is_idle_, portMAX_DELAY); |
||||||
|
Args args{ |
||||||
|
.cmd = Command::kSetTargetFormat, |
||||||
|
.format = target, |
||||||
|
}; |
||||||
|
xQueueSend(commands_, &args, portMAX_DELAY); |
||||||
|
xSemaphoreGive(is_idle_); |
||||||
|
} |
||||||
|
|
||||||
|
Args args{ |
||||||
|
.cmd = Command::kReadBytes, |
||||||
|
.format = {}, |
||||||
|
}; |
||||||
|
xQueueSend(commands_, &args, portMAX_DELAY); |
||||||
|
|
||||||
|
auto buf = input.data(); |
||||||
|
std::size_t bytes_sent = |
||||||
|
xStreamBufferSend(source_, buf.data(), buf.size_bytes(), portMAX_DELAY); |
||||||
|
input.consume(bytes_sent); |
||||||
|
return bytes_sent; |
||||||
|
} |
||||||
|
|
||||||
|
auto SinkMixer::Main() -> void { |
||||||
|
OutputStream input_receiver{input_stream_.get()}; |
||||||
|
xSemaphoreGive(is_idle_); |
||||||
|
|
||||||
|
for (;;) { |
||||||
|
Args args; |
||||||
|
while (!xQueueReceive(commands_, &args, portMAX_DELAY)) { |
||||||
|
} |
||||||
|
switch (args.cmd) { |
||||||
|
case Command::kSetSourceFormat: |
||||||
|
ESP_LOGI(kTag, "setting source format"); |
||||||
|
input_receiver.prepare(args.format, {}); |
||||||
|
resampler_.reset(); |
||||||
|
break; |
||||||
|
case Command::kSetTargetFormat: |
||||||
|
ESP_LOGI(kTag, "setting target format"); |
||||||
|
target_format_ = args.format; |
||||||
|
resampler_.reset(); |
||||||
|
break; |
||||||
|
case Command::kReadBytes: |
||||||
|
xSemaphoreTake(is_idle_, 0); |
||||||
|
while (!xStreamBufferIsEmpty(source_)) { |
||||||
|
auto buf = input_receiver.data(); |
||||||
|
std::size_t bytes_received = xStreamBufferReceive( |
||||||
|
source_, buf.data(), buf.size_bytes(), portMAX_DELAY); |
||||||
|
input_receiver.add(bytes_received); |
||||||
|
HandleBytes(); |
||||||
|
} |
||||||
|
xSemaphoreGive(is_idle_); |
||||||
|
break; |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
auto SinkMixer::HandleBytes() -> void { |
||||||
|
InputStream input{input_stream_.get()}; |
||||||
|
auto pcm = input.info().format_as<StreamInfo::Pcm>(); |
||||||
|
if (!pcm) { |
||||||
|
ESP_LOGE(kTag, "mixer got unsupported data"); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
if (*pcm == target_format_) { |
||||||
|
// The happiest possible case: the input format matches the output
|
||||||
|
// format already. Streams like this should probably have bypassed the
|
||||||
|
// mixer.
|
||||||
|
// TODO(jacqueline): Make this an error; it's slow to use the mixer in this
|
||||||
|
// case, compared to just writing directly to the sink.
|
||||||
|
auto buf = input.data(); |
||||||
|
std::size_t bytes_sent = |
||||||
|
xStreamBufferSend(sink_, buf.data(), buf.size_bytes(), portMAX_DELAY); |
||||||
|
input.consume(bytes_sent); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
while (input_stream_->info().bytes_in_stream() >= sizeof(sample::Sample)) { |
||||||
|
RawStream* output_source; |
||||||
|
if (pcm->sample_rate != target_format_.sample_rate) { |
||||||
|
OutputStream resampled_writer{resampled_stream_.get()}; |
||||||
|
if (Resample(input, resampled_writer)) { |
||||||
|
// Zero samples used or written. We need more input.
|
||||||
|
break; |
||||||
|
} |
||||||
|
output_source = resampled_stream_.get(); |
||||||
|
} else { |
||||||
|
output_source = input_stream_.get(); |
||||||
|
} |
||||||
|
|
||||||
|
size_t bytes_consumed = output_source->info().bytes_in_stream(); |
||||||
|
size_t bytes_to_send = output_source->info().bytes_in_stream(); |
||||||
|
|
||||||
|
if (target_format_.bits_per_sample == 16) { |
||||||
|
// This is slightly scary; we're basically reaching into the internals of
|
||||||
|
// the stream buffer to do in-place conversion of samples. Saving an
|
||||||
|
// extra buffer + copy into that buffer is certainly worth it however.
|
||||||
|
cpp::span<sample::Sample> src = |
||||||
|
output_source->data_as<sample::Sample>().first( |
||||||
|
output_source->info().bytes_in_stream() / sizeof(sample::Sample)); |
||||||
|
cpp::span<int16_t> dest{reinterpret_cast<int16_t*>(src.data()), |
||||||
|
src.size()}; |
||||||
|
|
||||||
|
ApplyDither(src, 16); |
||||||
|
Downscale(src, dest); |
||||||
|
|
||||||
|
bytes_consumed = src.size_bytes(); |
||||||
|
bytes_to_send = src.size_bytes() / 2; |
||||||
|
} |
||||||
|
|
||||||
|
InputStream output{output_source}; |
||||||
|
cpp::span<const std::byte> buf = output.data(); |
||||||
|
|
||||||
|
size_t bytes_sent = 0; |
||||||
|
while (bytes_sent < bytes_to_send) { |
||||||
|
auto cropped = buf.subspan(bytes_sent, bytes_to_send - bytes_sent); |
||||||
|
bytes_sent += xStreamBufferSend(sink_, cropped.data(), |
||||||
|
cropped.size_bytes(), portMAX_DELAY); |
||||||
|
} |
||||||
|
output.consume(bytes_consumed); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
auto SinkMixer::Resample(InputStream& in, OutputStream& out) -> bool { |
||||||
|
if (resampler_ == nullptr) { |
||||||
|
ESP_LOGI(kTag, "creating new resampler"); |
||||||
|
auto format = in.info().format_as<StreamInfo::Pcm>(); |
||||||
|
resampler_.reset(new Resampler( |
||||||
|
format->sample_rate, target_format_.sample_rate, format->channels)); |
||||||
|
} |
||||||
|
|
||||||
|
auto res = resampler_->Process(in.data_as<sample::Sample>(), |
||||||
|
out.data_as<sample::Sample>(), false); |
||||||
|
|
||||||
|
in.consume(res.first * sizeof(sample::Sample)); |
||||||
|
out.add(res.second * sizeof(sample::Sample)); |
||||||
|
|
||||||
|
return res.first == 0 && res.second == 0; |
||||||
|
} |
||||||
|
|
||||||
|
auto SinkMixer::Downscale(cpp::span<sample::Sample> samples, |
||||||
|
cpp::span<int16_t> output) -> void { |
||||||
|
for (size_t i = 0; i < samples.size(); i++) { |
||||||
|
output[i] = sample::ToSigned16Bit(samples[i]); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
auto SinkMixer::ApplyDither(cpp::span<sample::Sample> samples, |
||||||
|
uint_fast8_t bits) -> void { |
||||||
|
static uint32_t prnd; |
||||||
|
for (auto& s : samples) { |
||||||
|
prnd = (prnd * 0x19660dL + 0x3c6ef35fL) & 0xffffffffL; |
||||||
|
s = sample::Clip( |
||||||
|
static_cast<int64_t>(s) + |
||||||
|
(static_cast<int>(prnd) >> (sizeof(sample::Sample) - bits))); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} // namespace audio
|
@ -0,0 +1,65 @@ |
|||||||
|
#pragma once |
||||||
|
|
||||||
|
#include <stdint.h> |
||||||
|
|
||||||
|
#include <algorithm> |
||||||
|
|
||||||
|
#include <mad.h> |
||||||
|
|
||||||
|
namespace sample { |
||||||
|
|
||||||
|
// A signed, 32-bit PCM sample.
|
||||||
|
typedef int32_t Sample; |
||||||
|
|
||||||
|
constexpr auto Clip(int64_t v) -> Sample { |
||||||
|
if (v > INT32_MAX) |
||||||
|
return INT32_MAX; |
||||||
|
if (v < INT32_MIN) |
||||||
|
return INT32_MIN; |
||||||
|
return v; |
||||||
|
} |
||||||
|
|
||||||
|
constexpr auto FromSigned(int32_t src, uint_fast8_t bits) -> Sample { |
||||||
|
// Left-align samples, effectively scaling them up to 32 bits.
|
||||||
|
return src << (sizeof(Sample) * 8 - bits); |
||||||
|
} |
||||||
|
|
||||||
|
constexpr auto FromUnsigned(uint32_t src, uint_fast8_t bits) -> Sample { |
||||||
|
// Left-align, then substract the max value / 2 to make the sample centred
|
||||||
|
// around zero.
|
||||||
|
return (src << (sizeof(uint32_t) * 8 - bits)) - (~0UL >> 1); |
||||||
|
} |
||||||
|
|
||||||
|
constexpr auto FromFloat(float src) -> Sample { |
||||||
|
return std::clamp<float>(src, -1.0f, 1.0f) * static_cast<float>(INT32_MAX); |
||||||
|
} |
||||||
|
|
||||||
|
constexpr auto FromDouble(double src) -> Sample { |
||||||
|
return std::clamp<double>(src, -1.0, 1.0) * static_cast<double>(INT32_MAX); |
||||||
|
} |
||||||
|
|
||||||
|
constexpr auto FromMad(mad_fixed_t src) -> Sample { |
||||||
|
// Round the bottom bits.
|
||||||
|
src += (1L << (MAD_F_FRACBITS - 24)); |
||||||
|
|
||||||
|
// Clip the leftover bits to within range.
|
||||||
|
if (src >= MAD_F_ONE) |
||||||
|
src = MAD_F_ONE - 1; |
||||||
|
else if (src < -MAD_F_ONE) |
||||||
|
src = -MAD_F_ONE; |
||||||
|
|
||||||
|
// Quantize.
|
||||||
|
return FromSigned(src >> (MAD_F_FRACBITS + 1 - 24), 24); |
||||||
|
} |
||||||
|
|
||||||
|
constexpr auto ToSigned16Bit(Sample src) -> int16_t { |
||||||
|
return src >> 16; |
||||||
|
} |
||||||
|
|
||||||
|
static constexpr float kFactor = 1.0f / static_cast<float>(INT32_MAX); |
||||||
|
|
||||||
|
constexpr auto ToFloat(Sample src) -> float { |
||||||
|
return src * kFactor; |
||||||
|
} |
||||||
|
|
||||||
|
} // namespace sample
|
@ -0,0 +1,253 @@ |
|||||||
|
#include "bluetooth.hpp" |
||||||
|
|
||||||
|
#include <stdint.h> |
||||||
|
|
||||||
|
#include <atomic> |
||||||
|
#include <ostream> |
||||||
|
#include <sstream> |
||||||
|
|
||||||
|
#include "esp_a2dp_api.h" |
||||||
|
#include "esp_avrc_api.h" |
||||||
|
#include "esp_bt.h" |
||||||
|
#include "esp_bt_device.h" |
||||||
|
#include "esp_bt_main.h" |
||||||
|
#include "esp_gap_bt_api.h" |
||||||
|
#include "esp_log.h" |
||||||
|
#include "esp_mac.h" |
||||||
|
#include "tinyfsm/include/tinyfsm.hpp" |
||||||
|
|
||||||
|
namespace drivers { |
||||||
|
|
||||||
|
static constexpr char kTag[] = "bluetooth"; |
||||||
|
|
||||||
|
static std::atomic<StreamBufferHandle_t> sStream; |
||||||
|
|
||||||
|
auto gap_cb(esp_bt_gap_cb_event_t event, esp_bt_gap_cb_param_t* param) -> void { |
||||||
|
tinyfsm::FsmList<bluetooth::BluetoothState>::dispatch( |
||||||
|
bluetooth::events::internal::Gap{.type = event, .param = param}); |
||||||
|
} |
||||||
|
|
||||||
|
auto avrcp_cb(esp_avrc_ct_cb_event_t event, esp_avrc_ct_cb_param_t* param) |
||||||
|
-> void { |
||||||
|
tinyfsm::FsmList<bluetooth::BluetoothState>::dispatch( |
||||||
|
bluetooth::events::internal::Avrc{.type = event, .param = param}); |
||||||
|
} |
||||||
|
|
||||||
|
auto a2dp_cb(esp_a2d_cb_event_t event, esp_a2d_cb_param_t* param) -> void { |
||||||
|
tinyfsm::FsmList<bluetooth::BluetoothState>::dispatch( |
||||||
|
bluetooth::events::internal::A2dp{.type = event, .param = param}); |
||||||
|
} |
||||||
|
|
||||||
|
auto a2dp_data_cb(uint8_t* buf, int32_t buf_size) -> int32_t { |
||||||
|
if (buf == nullptr || buf_size <= 0) { |
||||||
|
return 0; |
||||||
|
} |
||||||
|
StreamBufferHandle_t stream = sStream.load(); |
||||||
|
if (stream == nullptr) { |
||||||
|
return 0; |
||||||
|
} |
||||||
|
return xStreamBufferReceive(stream, buf, buf_size, 0); |
||||||
|
} |
||||||
|
|
||||||
|
Bluetooth::Bluetooth() { |
||||||
|
tinyfsm::FsmList<bluetooth::BluetoothState>::start(); |
||||||
|
} |
||||||
|
|
||||||
|
auto Bluetooth::Enable() -> bool { |
||||||
|
tinyfsm::FsmList<bluetooth::BluetoothState>::dispatch( |
||||||
|
bluetooth::events::Enable{}); |
||||||
|
|
||||||
|
return !bluetooth::BluetoothState::is_in_state<bluetooth::Disabled>(); |
||||||
|
} |
||||||
|
|
||||||
|
auto Bluetooth::Disable() -> void { |
||||||
|
tinyfsm::FsmList<bluetooth::BluetoothState>::dispatch( |
||||||
|
bluetooth::events::Disable{}); |
||||||
|
} |
||||||
|
|
||||||
|
auto DeviceName() -> std::string { |
||||||
|
uint8_t mac[8]{0}; |
||||||
|
esp_efuse_mac_get_default(mac); |
||||||
|
std::ostringstream name; |
||||||
|
name << "TANGARA " << std::hex << mac[0] << mac[1]; |
||||||
|
return name.str(); |
||||||
|
} |
||||||
|
|
||||||
|
namespace bluetooth { |
||||||
|
|
||||||
|
static bool sIsFirstEntry = true; |
||||||
|
|
||||||
|
void Disabled::entry() { |
||||||
|
if (sIsFirstEntry) { |
||||||
|
// We only use BT Classic, to claw back ~60KiB from the BLE firmware.
|
||||||
|
esp_bt_controller_mem_release(ESP_BT_MODE_BLE); |
||||||
|
sIsFirstEntry = false; |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
esp_bluedroid_disable(); |
||||||
|
esp_bluedroid_deinit(); |
||||||
|
esp_bt_controller_disable(); |
||||||
|
} |
||||||
|
|
||||||
|
void Disabled::react(const events::Enable&) { |
||||||
|
esp_bt_controller_config_t config = BT_CONTROLLER_INIT_CONFIG_DEFAULT(); |
||||||
|
if (esp_bt_controller_init(&config) != ESP_OK) { |
||||||
|
ESP_LOGE(kTag, "initialize controller failed"); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
if (esp_bt_controller_enable(ESP_BT_MODE_CLASSIC_BT) != ESP_OK) { |
||||||
|
ESP_LOGE(kTag, "enable controller failed"); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
if (esp_bluedroid_init() != ESP_OK) { |
||||||
|
ESP_LOGE(kTag, "initialize bluedroid failed"); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
if (esp_bluedroid_enable() != ESP_OK) { |
||||||
|
ESP_LOGE(kTag, "enable bluedroid failed"); |
||||||
|
return; |
||||||
|
} |
||||||
|
|
||||||
|
// Enable Secure Simple Pairing
|
||||||
|
esp_bt_sp_param_t param_type = ESP_BT_SP_IOCAP_MODE; |
||||||
|
esp_bt_io_cap_t iocap = ESP_BT_IO_CAP_IO; |
||||||
|
esp_bt_gap_set_security_param(param_type, &iocap, sizeof(uint8_t)); |
||||||
|
|
||||||
|
// Set a reasonable name for the device.
|
||||||
|
std::string name = DeviceName(); |
||||||
|
esp_bt_dev_set_device_name(name.c_str()); |
||||||
|
|
||||||
|
// Initialise GAP. This controls advertising our device, and scanning for
|
||||||
|
// other devices.
|
||||||
|
esp_bt_gap_register_callback(gap_cb); |
||||||
|
|
||||||
|
// Initialise AVRCP. This handles playback controls; play/pause/volume/etc.
|
||||||
|
// esp_avrc_ct_init();
|
||||||
|
// esp_avrc_ct_register_callback(avrcp_cb);
|
||||||
|
|
||||||
|
// Initialise A2DP. This handles streaming audio. Currently ESP-IDF's SBC
|
||||||
|
// encoder only supports 2 channels of interleaved 16 bit samples, at
|
||||||
|
// 44.1kHz, so there is no additional configuration to be done for the
|
||||||
|
// stream itself.
|
||||||
|
esp_a2d_source_init(); |
||||||
|
esp_a2d_register_callback(a2dp_cb); |
||||||
|
esp_a2d_source_register_data_callback(a2dp_data_cb); |
||||||
|
|
||||||
|
// Don't let anyone interact with us before we're ready.
|
||||||
|
esp_bt_gap_set_scan_mode(ESP_BT_NON_CONNECTABLE, ESP_BT_NON_DISCOVERABLE); |
||||||
|
|
||||||
|
transit<Scanning>(); |
||||||
|
} |
||||||
|
|
||||||
|
static constexpr uint8_t kDiscoveryTimeSeconds = 10; |
||||||
|
static constexpr uint8_t kDiscoveryMaxResults = 0; |
||||||
|
|
||||||
|
void Scanning::entry() { |
||||||
|
ESP_LOGI(kTag, "scanning for devices"); |
||||||
|
esp_bt_gap_start_discovery(ESP_BT_INQ_MODE_GENERAL_INQUIRY, |
||||||
|
kDiscoveryTimeSeconds, kDiscoveryMaxResults); |
||||||
|
} |
||||||
|
|
||||||
|
void Scanning::exit() { |
||||||
|
esp_bt_gap_cancel_discovery(); |
||||||
|
} |
||||||
|
|
||||||
|
auto OnDeviceDiscovered(esp_bt_gap_cb_param_t* param) -> void { |
||||||
|
ESP_LOGI(kTag, "device discovered"); |
||||||
|
} |
||||||
|
|
||||||
|
void Scanning::react(const events::internal::Gap& ev) { |
||||||
|
switch (ev.type) { |
||||||
|
case ESP_BT_GAP_DISC_RES_EVT: |
||||||
|
OnDeviceDiscovered(ev.param); |
||||||
|
break; |
||||||
|
case ESP_BT_GAP_DISC_STATE_CHANGED_EVT: |
||||||
|
if (ev.param->disc_st_chg.state == ESP_BT_GAP_DISCOVERY_STOPPED) { |
||||||
|
ESP_LOGI(kTag, "still scanning"); |
||||||
|
esp_bt_gap_start_discovery(ESP_BT_INQ_MODE_GENERAL_INQUIRY, |
||||||
|
kDiscoveryTimeSeconds, kDiscoveryMaxResults); |
||||||
|
} |
||||||
|
break; |
||||||
|
case ESP_BT_GAP_MODE_CHG_EVT: |
||||||
|
// todo: mode change. is this important?
|
||||||
|
ESP_LOGI(kTag, "GAP mode changed"); |
||||||
|
break; |
||||||
|
default: |
||||||
|
ESP_LOGW(kTag, "unhandled GAP event: %u", ev.type); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void Connecting::entry() { |
||||||
|
ESP_LOGI(kTag, "connecting to device"); |
||||||
|
esp_a2d_source_connect(nullptr); |
||||||
|
} |
||||||
|
|
||||||
|
void Connecting::exit() {} |
||||||
|
|
||||||
|
void Connecting::react(const events::internal::Gap& ev) { |
||||||
|
switch (ev.type) { |
||||||
|
case ESP_BT_GAP_AUTH_CMPL_EVT: |
||||||
|
// todo: auth completed. check if we succeeded.
|
||||||
|
break; |
||||||
|
case ESP_BT_GAP_PIN_REQ_EVT: |
||||||
|
// todo: device needs a pin to connect.
|
||||||
|
break; |
||||||
|
case ESP_BT_GAP_CFM_REQ_EVT: |
||||||
|
// todo: device needs user to click okay.
|
||||||
|
break; |
||||||
|
case ESP_BT_GAP_KEY_NOTIF_EVT: |
||||||
|
// todo: device is telling us a password?
|
||||||
|
break; |
||||||
|
case ESP_BT_GAP_KEY_REQ_EVT: |
||||||
|
// todo: device needs a password
|
||||||
|
break; |
||||||
|
case ESP_BT_GAP_MODE_CHG_EVT: |
||||||
|
// todo: mode change. is this important?
|
||||||
|
break; |
||||||
|
default: |
||||||
|
ESP_LOGW(kTag, "unhandled GAP event: %u", ev.type); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void Connecting::react(const events::internal::A2dp& ev) { |
||||||
|
switch (ev.type) { |
||||||
|
case ESP_A2D_CONNECTION_STATE_EVT: |
||||||
|
// todo: connection state changed. we might be connected!
|
||||||
|
break; |
||||||
|
default: |
||||||
|
ESP_LOGW(kTag, "unhandled A2DP event: %u", ev.type); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void Connected::react(const events::internal::A2dp& ev) { |
||||||
|
switch (ev.type) { |
||||||
|
case ESP_A2D_CONNECTION_STATE_EVT: |
||||||
|
// todo: connection state changed. we might have dropped
|
||||||
|
break; |
||||||
|
case ESP_A2D_AUDIO_STATE_EVT: |
||||||
|
// todo: audio state changed. who knows, dude.
|
||||||
|
break; |
||||||
|
default: |
||||||
|
ESP_LOGW(kTag, "unhandled A2DP event: %u", ev.type); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
void Connected::react(const events::internal::Avrc& ev) { |
||||||
|
switch (ev.type) { |
||||||
|
case ESP_AVRC_CT_CONNECTION_STATE_EVT: |
||||||
|
// todo: avrc connected. send our capabilities.
|
||||||
|
default: |
||||||
|
ESP_LOGW(kTag, "unhandled AVRC event: %u", ev.type); |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
} // namespace bluetooth
|
||||||
|
|
||||||
|
} // namespace drivers
|
||||||
|
|
||||||
|
FSM_INITIAL_STATE(drivers::bluetooth::BluetoothState, |
||||||
|
drivers::bluetooth::Disabled) |
@ -0,0 +1,108 @@ |
|||||||
|
|
||||||
|
#pragma once |
||||||
|
|
||||||
|
#include <string> |
||||||
|
#include <vector> |
||||||
|
|
||||||
|
#include <freertos/FreeRTOS.h> |
||||||
|
#include <freertos/stream_buffer.h> |
||||||
|
#include "esp_a2dp_api.h" |
||||||
|
#include "esp_avrc_api.h" |
||||||
|
#include "esp_gap_bt_api.h" |
||||||
|
#include "tinyfsm.hpp" |
||||||
|
#include "tinyfsm/include/tinyfsm.hpp" |
||||||
|
|
||||||
|
namespace drivers { |
||||||
|
|
||||||
|
/*
|
||||||
|
* A handle used to interact with the bluetooth state machine. |
||||||
|
*/ |
||||||
|
class Bluetooth { |
||||||
|
public: |
||||||
|
Bluetooth(); |
||||||
|
|
||||||
|
auto Enable() -> bool; |
||||||
|
auto Disable() -> void; |
||||||
|
|
||||||
|
auto SetSource(StreamBufferHandle_t) -> void; |
||||||
|
}; |
||||||
|
|
||||||
|
namespace bluetooth { |
||||||
|
|
||||||
|
namespace events { |
||||||
|
struct Enable : public tinyfsm::Event {}; |
||||||
|
struct Disable : public tinyfsm::Event {}; |
||||||
|
|
||||||
|
namespace internal { |
||||||
|
struct Gap : public tinyfsm::Event { |
||||||
|
esp_bt_gap_cb_event_t type; |
||||||
|
esp_bt_gap_cb_param_t* param; |
||||||
|
}; |
||||||
|
struct A2dp : public tinyfsm::Event { |
||||||
|
esp_a2d_cb_event_t type; |
||||||
|
esp_a2d_cb_param_t* param; |
||||||
|
}; |
||||||
|
struct Avrc : public tinyfsm::Event { |
||||||
|
esp_avrc_ct_cb_event_t type; |
||||||
|
esp_avrc_ct_cb_param_t* param; |
||||||
|
}; |
||||||
|
} // namespace internal
|
||||||
|
} // namespace events
|
||||||
|
|
||||||
|
class BluetoothState : public tinyfsm::Fsm<BluetoothState> { |
||||||
|
public: |
||||||
|
virtual ~BluetoothState(){}; |
||||||
|
|
||||||
|
virtual void entry() {} |
||||||
|
virtual void exit() {} |
||||||
|
|
||||||
|
virtual void react(const events::Enable& ev){}; |
||||||
|
virtual void react(const events::Disable& ev) = 0; |
||||||
|
|
||||||
|
virtual void react(const events::internal::Gap& ev) = 0; |
||||||
|
virtual void react(const events::internal::A2dp& ev) = 0; |
||||||
|
virtual void react(const events::internal::Avrc& ev){}; |
||||||
|
}; |
||||||
|
|
||||||
|
class Disabled : public BluetoothState { |
||||||
|
void entry() override; |
||||||
|
|
||||||
|
void react(const events::Enable& ev) override; |
||||||
|
void react(const events::Disable& ev) override{}; |
||||||
|
|
||||||
|
void react(const events::internal::Gap& ev) override {} |
||||||
|
void react(const events::internal::A2dp& ev) override {} |
||||||
|
}; |
||||||
|
|
||||||
|
class Scanning : public BluetoothState { |
||||||
|
void entry() override; |
||||||
|
void exit() override; |
||||||
|
|
||||||
|
void react(const events::Disable& ev) override; |
||||||
|
|
||||||
|
void react(const events::internal::Gap& ev) override; |
||||||
|
void react(const events::internal::A2dp& ev) override; |
||||||
|
}; |
||||||
|
|
||||||
|
class Connecting : public BluetoothState { |
||||||
|
void entry() override; |
||||||
|
void exit() override; |
||||||
|
|
||||||
|
void react(const events::Disable& ev) override; |
||||||
|
void react(const events::internal::Gap& ev) override; |
||||||
|
void react(const events::internal::A2dp& ev) override; |
||||||
|
}; |
||||||
|
|
||||||
|
class Connected : public BluetoothState { |
||||||
|
void entry() override; |
||||||
|
void exit() override; |
||||||
|
|
||||||
|
void react(const events::Disable& ev) override; |
||||||
|
void react(const events::internal::Gap& ev) override; |
||||||
|
void react(const events::internal::A2dp& ev) override; |
||||||
|
void react(const events::internal::Avrc& ev) override; |
||||||
|
}; |
||||||
|
|
||||||
|
} // namespace bluetooth
|
||||||
|
|
||||||
|
} // namespace drivers
|
@ -0,0 +1,27 @@ |
|||||||
|
/*
|
||||||
|
* Copyright 2023 jacqueline <me@jacqueline.id.au> |
||||||
|
* |
||||||
|
* SPDX-License-Identifier: GPL-3.0-only |
||||||
|
*/ |
||||||
|
|
||||||
|
#pragma once |
||||||
|
|
||||||
|
#include "esp_err.h" |
||||||
|
#include "nvs.h" |
||||||
|
|
||||||
|
namespace drivers { |
||||||
|
|
||||||
|
class NvsStorage { |
||||||
|
public: |
||||||
|
static auto Open() -> NvsStorage*; |
||||||
|
|
||||||
|
auto SchemaVersion() -> uint8_t; |
||||||
|
|
||||||
|
explicit NvsStorage(nvs_handle_t); |
||||||
|
~NvsStorage(); |
||||||
|
|
||||||
|
private: |
||||||
|
nvs_handle_t handle_; |
||||||
|
}; |
||||||
|
|
||||||
|
} // namespace drivers
|
@ -0,0 +1,73 @@ |
|||||||
|
/*
|
||||||
|
* Copyright 2023 jacqueline <me@jacqueline.id.au> |
||||||
|
* |
||||||
|
* SPDX-License-Identifier: GPL-3.0-only |
||||||
|
*/ |
||||||
|
|
||||||
|
#include "nvs.hpp" |
||||||
|
#include <stdint.h> |
||||||
|
|
||||||
|
#include <cstdint> |
||||||
|
#include <memory> |
||||||
|
|
||||||
|
#include "esp_log.h" |
||||||
|
#include "nvs.h" |
||||||
|
#include "nvs_flash.h" |
||||||
|
|
||||||
|
namespace drivers { |
||||||
|
|
||||||
|
static constexpr char kTag[] = "nvm"; |
||||||
|
static constexpr uint8_t kSchemaVersion = 1; |
||||||
|
|
||||||
|
static constexpr char kKeyVersion[] = "ver"; |
||||||
|
|
||||||
|
auto NvsStorage::Open() -> NvsStorage* { |
||||||
|
esp_err_t err = nvs_flash_init(); |
||||||
|
if (err == ESP_ERR_NVS_NO_FREE_PAGES) { |
||||||
|
ESP_LOGW(kTag, "partition needs initialisation"); |
||||||
|
nvs_flash_erase(); |
||||||
|
err = nvs_flash_init(); |
||||||
|
} |
||||||
|
if (err != ESP_OK) { |
||||||
|
ESP_LOGE(kTag, "failed to init nvm"); |
||||||
|
return nullptr; |
||||||
|
} |
||||||
|
|
||||||
|
nvs_handle_t handle; |
||||||
|
if ((err = nvs_open("tangara", NVS_READWRITE, &handle)) != ESP_OK) { |
||||||
|
ESP_LOGE(kTag, "failed to open nvs namespace"); |
||||||
|
return nullptr; |
||||||
|
} |
||||||
|
|
||||||
|
std::unique_ptr<NvsStorage> instance = std::make_unique<NvsStorage>(handle); |
||||||
|
if (instance->SchemaVersion() < kSchemaVersion) { |
||||||
|
ESP_LOGW(kTag, "namespace needs downgrading"); |
||||||
|
nvs_erase_all(handle); |
||||||
|
nvs_set_u8(handle, kKeyVersion, kSchemaVersion); |
||||||
|
err = nvs_commit(handle); |
||||||
|
if (err != ESP_OK) { |
||||||
|
ESP_LOGW(kTag, "failed to init namespace"); |
||||||
|
return nullptr; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
ESP_LOGI(kTag, "nvm storage initialised okay"); |
||||||
|
return instance.release(); |
||||||
|
} |
||||||
|
|
||||||
|
NvsStorage::NvsStorage(nvs_handle_t handle) : handle_(handle) {} |
||||||
|
|
||||||
|
NvsStorage::~NvsStorage() { |
||||||
|
nvs_close(handle_); |
||||||
|
nvs_flash_deinit(); |
||||||
|
} |
||||||
|
|
||||||
|
auto NvsStorage::SchemaVersion() -> uint8_t { |
||||||
|
uint8_t ret; |
||||||
|
if (nvs_get_u8(handle_, kKeyVersion, &ret) != ESP_OK) { |
||||||
|
return UINT8_MAX; |
||||||
|
} |
||||||
|
return ret; |
||||||
|
} |
||||||
|
|
||||||
|
} // namespace drivers
|
Loading…
Reference in new issue