forked from kevinwlu/iot
-
Notifications
You must be signed in to change notification settings - Fork 0
/
mic_Saved_OnSDcard.ino
183 lines (160 loc) · 5.03 KB
/
mic_Saved_OnSDcard.ino
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
#if !defined(ARDUINO_ARCH_NRF52840)
#error "This demo targets XIAO BLE only at the moment"
#endif
#include <mic.h>
#include <SD.h>
// Settings
#define DEBUG 1 // Enable pin pulse during ISR
#define SAMPLES 16000*5
mic_config_t mic_config{
.channel_cnt = 1,
.sampling_rate = 16000,
.buf_size = 512,
.debug_pin = LED_BUILTIN // Toggles each DAC ISR (if DEBUG is set to 1)
};
NRF52840_ADC_Class Mic(&mic_config);
short sampleBuffer[256];
short sampleBuffer1[SAMPLES] = {0};
// Number of audio samples read
volatile int samplesRead = 0;
void finalize_template(File &sFile);
void record();
void create_template(File &sFile);
void setup() {
Serial.begin(9600);
while (!Serial);
// Configure the data receive callback
Mic.set_callback(audio_rec_callback);
// Optionally set the gain
// Defaults to 20 on the BLE Sense and 24 on the Portenta Vision Shield
// PDM.setGain(30);
if (!SD.begin()) {
Serial.println("initialization failed!");
while (1);
}
else{
Serial.println("SD initialization success!");
}
// Initialize PDM with:
// - one channel (mono mode)
// - a 16 kHz sample rate for the Arduino Nano 33 BLE Sense
// - a 32 kHz or 64 kHz sample rate for the Arduino Portenta Vision Shield
if (!Mic.begin()) {
Serial.println("Failed to start MIC!");
while (1);
}
}
int sample_cnt = 0;
void loop() {
// Wait for samples to be read
if (samplesRead) {
// Print samples to the serial monitor or plotter
for (int i = 0; i < samplesRead; i++) {
sampleBuffer1[sample_cnt++] = sampleBuffer[i];
if(sample_cnt >= SAMPLES)
{
Serial.println("End of PDM");
record();
Mic.end();
samplesRead = 0;
return;
}
}
// Clear the read count
samplesRead = 0;
}
}
File myFile;
void record(){
Serial.println("Finished sampling");
static char print_buf[128] = {0};
int r = sprintf(print_buf, "test%d.wav", millis());
File sFile = SD.open(print_buf, FILE_WRITE);
if (sFile) {
Serial.print("Writing to "); // txt
Serial.println(print_buf);
} else {
// if the file didn't open, print an error:
Serial.print("error opening ");
Serial.println(print_buf);
}
create_template(sFile);
for (int i = 0; i < SAMPLES; i++) {
//int16_t sample = filter.step(recording_buf[i]);
short sample = sampleBuffer1[i];
sFile.write(sample & 0xFF);
sFile.write((sample >> 8) & 0xFF);
//Serial.println(sample);
}
Serial.println("Finished writing");
finalize_template(sFile);
}
void create_template(File &sFile)
{
struct soundhdr {
char riff[4]; /* "RIFF" */
long flength; /* file length in bytes */
char wave[4]; /* "WAVE" */
char fmt[4]; /* "fmt " */
long chunk_size; /* size of FMT chunk in bytes (usually 16) */
short format_tag; /* 1=PCM, 257=Mu-Law, 258=A-Law, 259=ADPCM */
short num_chans; /* 1=mono, 2=stereo */
long srate; /* Sampling rate in samples per second */
long bytes_per_sec; /* bytes per second = srate*bytes_per_samp */
short bytes_per_samp; /* 2=16-bit mono, 4=16-bit stereo */
short bits_per_samp; /* Number of bits per sample */
char data[4]; /* "data" */
long dlength; /* data length in bytes (filelength - 44) */
} wavh;
// It's easy enough to initialize the strings
strncpy(wavh.riff,"RIFF", 4);
strncpy(wavh.wave,"WAVE", 4);
strncpy(wavh.fmt,"fmt ", 4);
strncpy(wavh.data,"data", 4);
// size of FMT chunk in bytes
wavh.chunk_size = 16;
wavh.format_tag = 1; // PCM
wavh.num_chans = 1; // mono
wavh.srate = 16000;
wavh.bytes_per_sec = (16000 * 1 * 16 * 1)/8;
wavh.bytes_per_samp = 2;
wavh.bits_per_samp = 16;
wavh.dlength = 16000 * 2 * 1 * 16/2;
sFile.seek(0);
sFile.write((byte *)&wavh, 44);
}
void finalize_template(File &sFile)
{
unsigned long fSize = sFile.size()-8;
sFile.seek(4);
byte data[4] = {lowByte(fSize), highByte(fSize), fSize >> 16, fSize >> 24};
sFile.write(data,4);
byte tmp;
sFile.seek(40);
fSize = fSize - 36;
data[0] = lowByte(fSize);
data[1]= highByte(fSize);
data[2]= fSize >> 16;
data[3]= fSize >> 24;
//sFile.write((byte*)data, 4);
sFile.close();
}
/**
* Callback function to process the data from the PDM microphone.
* NOTE: This callback is executed as part of an ISR.
* Therefore using `Serial` to print messages inside this function isn't supported.
* */
void audio_rec_callback(uint16_t *buf, uint32_t buf_len) {
static uint32_t idx = 0;
// Copy samples from DMA buffer to inference buffer
for (uint32_t i = 0; i < buf_len; i++) {
// Convert 12-bit unsigned ADC value to 16-bit PCM (signed) audio value
sampleBuffer[idx++] = buf[i];
if (idx >= mic_config.buf_size){
idx = 0;
//record_ready = true;
samplesRead = mic_config.buf_size;
break;
}
}
}