I was curious how well I could make the Seeeduino Xiao DAC work well.
Parts:
Hookup is pretty simple:
Xiao SAMD21 Pin1 (DAC/A0) | LM386 “In” Pin |
Xiao SAMD21 (5V) | LM386 “VCC” pin |
Xiao SAMD21 (GND) | LM386 “GND” post (either) |
Then attach your speaker to the screw-terminal. Red to + Black to -.
Given the small cheap speaker I’m not expecting much. The sound is pretty rough and the inexpensive construction makes it act essentially like a low-pass filter.
float x = 0; // Value to take the sin of
float increment = 0.04; // Value to increment x by each time
void setup()
{
analogWriteResolution(10); // Set analog out resolution to max, 10-bits
Serial.begin(9600);
}
void loop()
{
// Generate a voltage value between 0 and 1023.
// Let's scale a sin wave between those values:
// Offset by 511.5, then multiply sin by 511.5.
int dacVoltage = (int)(511.5 + 511.5 * sin(x));
x += increment; // Increase value of x
// Generate a voltage between 0 and 3.3V.
// 0= 0V, 1023=3.3V, 512=1.65V, etc.
analogWrite(DAC_PIN, dacVoltage);
delay(1);
}
Kinda disappointing… No timing, 1000 Hz means my highest frequency will be very low. 2*pi/0.04= 157, so I won’t get more than a few Hertz out of this (I had to bump that increment up to around 20 to even hear anything).
Here are the issues I see:
- Computing sin(x) on the fly takes non-zero time. Esp loading a floating point library on what I presume is an integer processor.
- Obviously, I want my samples to be applied at exactly regular intervals. delay(1) will be delay(1ms)+time-it-takes-to-process.
So maybe I can set up a buffer with a timer?
First let’s check pointer access:
[Datasheet] page 74


char mypointer[10];
analogWrite(DAC_PIN, dacVoltage);
Serial.println(dacVoltage,DEC);
unsigned char * p1 = (unsigned char *) 0x42004808;
unsigned char * p2 = (unsigned char *) 0x42004809;
short p = *p1 + (short) 256* (*p2);
siprintf(mypointer,"%u",p);
Serial.write(mypointer);
Serial.write("\n");
Gives the expected values:

Actually, maybe I can get away with just a timer.
https://github.com/adafruit/Adafruit_ZeroTimer/blob/master/examples/timer_callback/timer_callback.ino Shows a timer example which seems to work.
So now I can build my notes and play them, right? Not so fast… only have 32K of sram (per Datasheet).

10-bit samples, I want to play “notes” from middle-C to C5. A5 is 440 Hz and I’d like to play at least 4 samples per wavelength (Nyquists’s theorem requires >2, the more the better but this is a cheap speaker anyway). So let’s aim for 10. That’s about 5000 samples/second. Unfortunately with only 32K of sram and 10-bit samples this complicates things. A few options:
- Dynamically compute the value of the sine wave.
- Dynamically change the timer count, so the same ‘sinewave’ gets pushed in faster/slower as the frequency changes.
- Packing the bytes or dropping to 8-bits.
- Lookup Tables in Program memory.
Let’s try the lookup tables. A quick python script should generate me the samples I need. Since there’s only 32K of SRAM, I want my arrays to be stored in program memory. There should be a linker directive for that:
#include <avr/pgmspace.h>
typedef char * NoteWaveform_t;
NoteWaveform_t Note_C4 PROGMEM = { ... };
At least that’s what I found online. But it didn’t work. My SRAM still filled up. Turns out the PROGMEM Macro was blank. I ended up digging into the linker and doing by hand:
#define TEXTSECTION __attribute((section(".text")))
NoteWaveform_t Note_C4 TEXTSECTION= {
.... };
seems to work. Be sure to use const so that the compiler knows it shouldn’t try to (accidentally) update the waveforms.
Quick first-week-of-Music-Theory-101: A440 (what you hear orchestras tune to right before the conductor comes out) is the A5 key on your piano. C4 is the C before it and is “middle C”. Each key, black or white, is twelfth-root-of-two (or about 1.0595) times (to the right) or divided by (to the left) from the adjacent key’s frequency. So twelve keys is an octave (the A below middle C is 220 Hz). You can build up frequency tables pretty easily from that information.
I updated analogWriteResolution to 8 bits. Middle-C is 3 half-steps above A220, so 220*(1.0595)^3 = ~261.13, I’m sampling at 5000 samples/second, and my samples should go from 128 up to 255 then down to 0 and back up.
for i in range (fs):
128 + 127*sin(2*3.1415*i*ft/fs)
This will give me one-second worth of middle-C. fs is my sample frequency (5000) and ft is my desired tone frequency (261.3).
Happy Birthday (looks like it’s in public Domain now) is a pretty useful test song. The data structure is the note and the number of beats (yeah yeah I know).
