Skip to content

Commit 08d37de

Browse files
Add PWMAudio for DAC-free audio playback (#1076)
Use the PWM hardware to generate a signal suitable for filtering and amplifying 16bps audio output. Refactor the AudioBufferManager to allow sharing with I2S Add example
1 parent b8906e0 commit 08d37de

File tree

11 files changed

+15443
-25
lines changed

11 files changed

+15443
-25
lines changed
Lines changed: 10 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
name=AudioBufferManager
2+
version=1.0.0
3+
author=Earle F. Philhower, III <[email protected]>
4+
maintainer=Earle F. Philhower, III <[email protected]>
5+
sentence=Manages DMA buffers for audio output
6+
paragraph=Manages DMA buffers for audio output
7+
category=Device Control
8+
url=https://github.com/earlephilhower/arduino-pico
9+
architectures=rp2040
10+
dot_a_linkage=true

libraries/I2S/src/AudioRingBuffer.cpp renamed to libraries/AudioBufferManager/src/AudioBufferManager.cpp

Lines changed: 18 additions & 17 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/*
2-
AudioRingBuffer for Raspnerry Pi Pico RP2040
3-
Implements a ring buffer for PIO DMA for I2S read or write
2+
AudioBufferManager for Raspnerry Pi Pico RP2040
3+
Implements a DMA controlled linked-list series of buffers
44
55
Copyright (c) 2022 Earle F. Philhower, III <[email protected]>
66
@@ -22,12 +22,12 @@
2222
#include <Arduino.h>
2323
#include "hardware/dma.h"
2424
#include "hardware/irq.h"
25-
#include "AudioRingBuffer.h"
25+
#include "AudioBufferManager.h"
2626

27-
static int __channelCount = 0; // # of channels left. When we hit 0, then remove our handler
28-
static AudioRingBuffer* __channelMap[12]; // Lets the IRQ handler figure out where to dispatch to
27+
static int __channelCount = 0; // # of channels left. When we hit 0, then remove our handler
28+
static AudioBufferManager* __channelMap[12]; // Lets the IRQ handler figure out where to dispatch to
2929

30-
AudioRingBuffer::AudioRingBuffer(size_t bufferCount, size_t bufferWords, int32_t silenceSample, PinMode direction) {
30+
AudioBufferManager::AudioBufferManager(size_t bufferCount, size_t bufferWords, int32_t silenceSample, PinMode direction, enum dma_channel_transfer_size dmaSize) {
3131
_running = false;
3232

3333
// Need at least 2 DMA buffers and 1 user or this isn't going to work at all
@@ -38,6 +38,7 @@ AudioRingBuffer::AudioRingBuffer(size_t bufferCount, size_t bufferWords, int32_t
3838
_bufferCount = bufferCount;
3939
_wordsPerBuffer = bufferWords;
4040
_isOutput = direction == OUTPUT;
41+
_dmaSize = dmaSize;
4142
_overunderflow = false;
4243
_callback = nullptr;
4344
_userOff = 0;
@@ -66,7 +67,7 @@ AudioRingBuffer::AudioRingBuffer(size_t bufferCount, size_t bufferWords, int32_t
6667
_active[1] = _silence;
6768
}
6869

69-
AudioRingBuffer::~AudioRingBuffer() {
70+
AudioBufferManager::~AudioBufferManager() {
7071
if (_running) {
7172
for (auto i = 0; i < 2; i++) {
7273
dma_channel_set_irq0_enabled(_channelDMA[i], false);
@@ -98,11 +99,11 @@ AudioRingBuffer::~AudioRingBuffer() {
9899
_deleteAudioBuffer(_silence);
99100
}
100101

101-
void AudioRingBuffer::setCallback(void (*fn)()) {
102+
void AudioBufferManager::setCallback(void (*fn)()) {
102103
_callback = fn;
103104
}
104105

105-
bool AudioRingBuffer::begin(int dreq, volatile void *pioFIFOAddr) {
106+
bool AudioBufferManager::begin(int dreq, volatile void *pioFIFOAddr) {
106107
_running = true;
107108

108109
// Get ping and pong DMA channels
@@ -119,7 +120,7 @@ bool AudioRingBuffer::begin(int dreq, volatile void *pioFIFOAddr) {
119120
// Need to know both channels to set up ping-pong, so do in 2 stages
120121
for (auto i = 0; i < 2; i++) {
121122
dma_channel_config c = dma_channel_get_default_config(_channelDMA[i]);
122-
channel_config_set_transfer_data_size(&c, DMA_SIZE_32); // 32b transfers into PIO FIFO
123+
channel_config_set_transfer_data_size(&c, _dmaSize); // 16b/32b transfers into PIO FIFO
123124
if (_isOutput) {
124125
channel_config_set_read_increment(&c, true); // Reading incrementing addresses
125126
channel_config_set_write_increment(&c, false); // Writing to the same FIFO address
@@ -155,7 +156,7 @@ bool AudioRingBuffer::begin(int dreq, volatile void *pioFIFOAddr) {
155156
// cause GCC to keep re-reading from memory and not use cached value read
156157
// on the first pass.
157158

158-
bool AudioRingBuffer::write(uint32_t v, bool sync) {
159+
bool AudioBufferManager::write(uint32_t v, bool sync) {
159160
if (!_running || !_isOutput) {
160161
return false;
161162
}
@@ -177,7 +178,7 @@ bool AudioRingBuffer::write(uint32_t v, bool sync) {
177178
return true;
178179
}
179180

180-
bool AudioRingBuffer::read(uint32_t *v, bool sync) {
181+
bool AudioBufferManager::read(uint32_t *v, bool sync) {
181182
if (!_running || _isOutput) {
182183
return false;
183184
}
@@ -201,13 +202,13 @@ bool AudioRingBuffer::read(uint32_t *v, bool sync) {
201202
return true;
202203
}
203204

204-
bool AudioRingBuffer::getOverUnderflow() {
205+
bool AudioBufferManager::getOverUnderflow() {
205206
bool hold = _overunderflow;
206207
_overunderflow = false;
207208
return hold;
208209
}
209210

210-
int AudioRingBuffer::available() {
211+
int AudioBufferManager::available() {
211212
AudioBuffer *p = _isOutput ? _empty : _filled;
212213

213214
if (!_running || !p) {
@@ -226,7 +227,7 @@ int AudioRingBuffer::available() {
226227
return avail;
227228
}
228229

229-
void AudioRingBuffer::flush() {
230+
void AudioBufferManager::flush() {
230231
AudioBuffer ** volatile a = (AudioBuffer ** volatile)&_active[0];
231232
AudioBuffer ** volatile b = (AudioBuffer ** volatile)&_active[1];
232233
AudioBuffer ** volatile c = (AudioBuffer ** volatile)&_filled;
@@ -235,7 +236,7 @@ void AudioRingBuffer::flush() {
235236
}
236237
}
237238

238-
void __not_in_flash_func(AudioRingBuffer::_dmaIRQ)(int channel) {
239+
void __not_in_flash_func(AudioBufferManager::_dmaIRQ)(int channel) {
239240
if (_isOutput) {
240241
if (_active[0] != _silence) {
241242
_addToList(&_empty, _active[0]);
@@ -265,7 +266,7 @@ void __not_in_flash_func(AudioRingBuffer::_dmaIRQ)(int channel) {
265266
}
266267
}
267268

268-
void __not_in_flash_func(AudioRingBuffer::_irq)() {
269+
void __not_in_flash_func(AudioBufferManager::_irq)() {
269270
for (size_t i = 0; i < sizeof(__channelMap); i++) {
270271
if (dma_channel_get_irq0_status(i) && __channelMap[i]) {
271272
__channelMap[i]->_dmaIRQ(i);

libraries/I2S/src/AudioRingBuffer.h renamed to libraries/AudioBufferManager/src/AudioBufferManager.h

Lines changed: 7 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/*
2-
AudioRingBuffer for Rasperry Pi Pico
3-
Implements a ring buffer for PIO DMA for I2S read or write
2+
AudioBufferManager for Rasperry Pi Pico
3+
Implements a DMA controlled linked-list series of buffers
44
55
Copyright (c) 2022 Earle F. Philhower, III <[email protected]>
66
@@ -21,11 +21,12 @@
2121

2222
#pragma once
2323
#include <Arduino.h>
24+
#include "hardware/dma.h"
2425

25-
class AudioRingBuffer {
26+
class AudioBufferManager {
2627
public:
27-
AudioRingBuffer(size_t bufferCount, size_t bufferWords, int32_t silenceSample, PinMode direction = OUTPUT);
28-
~AudioRingBuffer();
28+
AudioBufferManager(size_t bufferCount, size_t bufferWords, int32_t silenceSample, PinMode direction = OUTPUT, enum dma_channel_transfer_size dmaSize = DMA_SIZE_32);
29+
~AudioBufferManager();
2930

3031
void setCallback(void (*fn)());
3132

@@ -88,6 +89,7 @@ class AudioRingBuffer {
8889
int _bitsPerSample;
8990
size_t _wordsPerBuffer;
9091
size_t _bufferCount;
92+
enum dma_channel_transfer_size _dmaSize;
9193
bool _isOutput;
9294

9395
int _channelDMA[2];

libraries/I2S/src/I2S.cpp

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -139,7 +139,7 @@ bool I2S::begin() {
139139
if (!_bufferWords) {
140140
_bufferWords = 16 * (_bps == 32 ? 2 : 1);
141141
}
142-
_arb = new AudioRingBuffer(_buffers, _bufferWords, _silenceSample, _isOutput ? OUTPUT : INPUT);
142+
_arb = new AudioBufferManager(_buffers, _bufferWords, _silenceSample, _isOutput ? OUTPUT : INPUT);
143143
_arb->begin(pio_get_dreq(_pio, _sm, _isOutput), _isOutput ? &_pio->txf[_sm] : (volatile void*)&_pio->rxf[_sm]);
144144
_arb->setCallback(_cb);
145145
pio_sm_set_enabled(_pio, _sm, true);

libraries/I2S/src/I2S.h

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,7 @@
2121

2222
#pragma once
2323
#include <Arduino.h>
24-
#include "AudioRingBuffer.h"
24+
#include "AudioBufferManager.h"
2525

2626
class I2S : public Stream {
2727
public:
@@ -120,7 +120,7 @@ class I2S : public Stream {
120120

121121
void (*_cb)();
122122

123-
AudioRingBuffer *_arb;
123+
AudioBufferManager *_arb;
124124
PIOProgram *_i2s;
125125
PIO _pio;
126126
int _sm;
Lines changed: 37 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
/*
2+
This example plays a raw, headerless, mono 16b, 44.1 sample using the PWMAudio library on GPIO 1.
3+
4+
Released to the public domain by Earle F. Philhower, III <[email protected]>
5+
*/
6+
7+
#include <PWMAudio.h>
8+
#include "wav.h"
9+
10+
// The sample pointers
11+
const int16_t *start = (const int16_t *)out_raw;
12+
const int16_t *p = start;
13+
14+
// Create the PWM audio device on GPIO 1. Hook amp/speaker between GPIO1 and convenient GND.
15+
PWMAudio pwm(1);
16+
17+
unsigned int count = 0;
18+
19+
void cb() {
20+
while (pwm.availableForWrite()) {
21+
pwm.write(*p++);
22+
count += 2;
23+
if (count >= sizeof(out_raw)) {
24+
count = 0;
25+
p = start;
26+
}
27+
}
28+
}
29+
30+
void setup() {
31+
pwm.onTransmit(cb);
32+
pwm.begin(44100);
33+
}
34+
35+
void loop() {
36+
/* noop, everything is done in the CB */
37+
}

0 commit comments

Comments
 (0)