-
Notifications
You must be signed in to change notification settings - Fork 72
/
Copy pathRD117_ARDUINO.ino
executable file
·400 lines (368 loc) · 11.6 KB
/
RD117_ARDUINO.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
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
/********************************************************
*
* Project: MAXREFDES117#
* Filename: RD117_ARDUINO.ino
* Description: This module contains the Main application for the MAXREFDES117 example program.
*
* Revision History:
*\n 1-18-2016 Rev 01.00 GL Initial release.
*\n 12-22-2017 Rev 02.00 Significantlly modified by Robert Fraczkiewicz
*\n 08-22-2018 Rev 02.01 Added conditional compilation of the code related to ADALOGGER SD card operations
*
* --------------------------------------------------------------------
*
* This code follows the following naming conventions:
*
* char ch_pmod_value
* char (array) s_pmod_s_string[16]
* float f_pmod_value
* int32_t n_pmod_value
* int32_t (array) an_pmod_value[16]
* int16_t w_pmod_value
* int16_t (array) aw_pmod_value[16]
* uint16_t uw_pmod_value
* uint16_t (array) auw_pmod_value[16]
* uint8_t uch_pmod_value
* uint8_t (array) auch_pmod_buffer[16]
* uint32_t un_pmod_value
* int32_t * pn_pmod_value
*
* ------------------------------------------------------------------------- */
#include <Arduino.h>
#include <SPI.h>
#include "algorithm_by_RF.h"
#include "max30102.h"
//#define DEBUG // Uncomment for debug output to the Serial stream
//#define USE_ADALOGGER // Comment out if you don't have ADALOGGER itself but your MCU still can handle this code
//#define TEST_MAXIM_ALGORITHM // Uncomment if you want to include results returned by the original MAXIM algorithm
//#define SAVE_RAW_DATA // Uncomment if you want raw data coming out of the sensor saved to SD card. Red signal first, IR second.
#ifdef USE_ADALOGGER
#include <SD.h>
#endif
#ifdef TEST_MAXIM_ALGORITHM
#include "algorithm.h"
#endif
// Interrupt pin
const byte oxiInt = 10; // pin connected to MAX30102 INT
// ADALOGGER pins
#ifdef USE_ADALOGGER
File dataFile;
const byte chipSelect = 4;
const byte cardDetect = 7;
const byte batteryPin = 9;
const byte ledPin = 13; // Red LED on ADALOGGER
const byte sdIndicatorPin = 8; // Green LED on ADALOGGER
bool cardOK;
#endif
uint32_t elapsedTime,timeStart;
uint32_t aun_ir_buffer[BUFFER_SIZE]; //infrared LED sensor data
uint32_t aun_red_buffer[BUFFER_SIZE]; //red LED sensor data
float old_n_spo2; // Previous SPO2 value
uint8_t uch_dummy,k;
void setup() {
pinMode(oxiInt, INPUT); //pin D10 connects to the interrupt output pin of the MAX30102
#ifdef USE_ADALOGGER
pinMode(cardDetect,INPUT_PULLUP);
pinMode(batteryPin,INPUT);
pinMode(ledPin,OUTPUT);
digitalWrite(ledPin,LOW);
pinMode(sdIndicatorPin,OUTPUT);
digitalWrite(sdIndicatorPin,LOW);
#endif
#if defined(DEBUG) || !defined(USE_ADALOGGER)
// initialize serial communication at 115200 bits per second:
Serial.begin(115200);
#endif
maxim_max30102_init(); //initialize the MAX30102
old_n_spo2=0.0;
#ifdef USE_ADALOGGER
// Measure battery voltage
float measuredvbat = analogRead(batteryPin);
measuredvbat *= 2; // we divided by 2, so multiply back
measuredvbat *= 3.3; // Multiply by 3.3V, our reference voltage
measuredvbat /= 1024; // convert to voltage
char my_status[20];
if(HIGH==digitalRead(cardDetect)) {
// we'll use the initialization code from the utility libraries
// since we're just testing if the card is working!
if(!SD.begin(chipSelect)) {
cardOK=false;
strncpy(my_status,"CardInit!",9);
} else cardOK=true;
} else {
cardOK=false;
strncpy(my_status,"NoSDCard!",9);
}
if(cardOK) {
long count=0;
char fname[20];
do {
// if(useClock && now.month()<13 && now.day()<32) {
// sprintf(fname,"%d-%d_%d.txt",now.month(),now.day(),++count);
// } else {
sprintf(fname,"data_%d.txt",++count);
// }
} while(SD.exists(fname));
dataFile = SD.open(fname, FILE_WRITE);
strncpy(my_status,fname,19);
}
#ifdef DEBUG
while(Serial.available()==0) //wait until user presses a key
{
Serial.print(F("Vbatt=\t"));
Serial.println(measuredvbat);
Serial.println(my_status);
Serial.println(dataFile,HEX);
Serial.println(F("Press any key to start conversion"));
delay(1000);
}
uch_dummy=Serial.read();
#endif
blinkLED(ledPin,cardOK);
k=0;
dataFile.println(F("Vbatt=\t"));
dataFile.println(measuredvbat);
dataFile.println(my_status);
#ifdef TEST_MAXIM_ALGORITHM
dataFile.print(F("Time[s]\tSpO2\tHR\tSpO2_MX\tHR_MX\tClock\tRatio\tCorr\tTemp[C]”));
#else // TEST_MAXIM_ALGORITHM
dataFile.print(F("Time[s]\tSpO2\tHR\tClock\tRatio\tCorr\tTemp[C]"));
#endif // TEST_MAXIM_ALGORITHM
#ifdef SAVE_RAW_DATA
int32_t i;
// These are headers for the red signal
for(i=0;i<BUFFER_SIZE;++i) {
dataFile.print("\t");
dataFile.print(i);
}
// These are headers for the infrared signal
for(i=0;i<BUFFER_SIZE;++i) {
dataFile.print("\t");
dataFile.print(i);
}
#endif // SAVE_RAW_DATA
dataFile.println("");
#else // USE_ADALOGGER
while(Serial.available()==0) //wait until user presses a key
{
Serial.println(F("Press any key to start conversion"));
delay(1000);
}
uch_dummy=Serial.read();
#ifdef TEST_MAXIM_ALGORITHM
Serial.print(F("Time[s]\tSpO2\tHR\tSpO2_MX\tHR_MX\tClock\tRatio\tCorr\tTemp[C]"));
#else // TEST_MAXIM_ALGORITHM
Serial.print(F("Time[s]\tSpO2\tHR\tClock\tRatio\tCorr\tTemp[C]"));
#endif // TEST_MAXIM_ALGORITHM
#ifdef SAVE_RAW_DATA
int32_t i;
// These are headers for the red signal
for(i=0;i<BUFFER_SIZE;++i) {
Serial.print("\t");
Serial.print(i);
}
// These are headers for the infrared signal
for(i=0;i<BUFFER_SIZE;++i) {
Serial.print("\t");
Serial.print(i);
}
#endif // SAVE_RAW_DATA
Serial.println("");
#endif // USE_ADALOGGER
timeStart=millis();
}
//Continuously taking samples from MAX30102. Heart rate and SpO2 are calculated every ST seconds
void loop() {
float n_spo2,ratio,correl; //SPO2 value
int8_t ch_spo2_valid; //indicator to show if the SPO2 calculation is valid
int32_t n_heart_rate; //heart rate value
int8_t ch_hr_valid; //indicator to show if the heart rate calculation is valid
int32_t i;
char hr_str[10];
//buffer length of BUFFER_SIZE stores ST seconds of samples running at FS sps
//read BUFFER_SIZE samples, and determine the signal range
for(i=0;i<BUFFER_SIZE;i++)
{
while(digitalRead(oxiInt)==1); //wait until the interrupt pin asserts
maxim_max30102_read_fifo((aun_red_buffer+i), (aun_ir_buffer+i)); //read from MAX30102 FIFO
#ifdef DEBUG
Serial.print(i, DEC);
Serial.print(F("\t"));
Serial.print(aun_red_buffer[i], DEC);
Serial.print(F("\t"));
Serial.print(aun_ir_buffer[i], DEC);
Serial.println("");
#endif // DEBUG
}
//calculate heart rate and SpO2 after BUFFER_SIZE samples (ST seconds of samples) using Robert's method
rf_heart_rate_and_oxygen_saturation(aun_ir_buffer, BUFFER_SIZE, aun_red_buffer, &n_spo2, &ch_spo2_valid, &n_heart_rate, &ch_hr_valid, &ratio, &correl);
elapsedTime=millis()-timeStart;
millis_to_hours(elapsedTime,hr_str); // Time in hh:mm:ss format
elapsedTime/=1000; // Time in seconds
// Read the _chip_ temperature in degrees Celsius
int8_t integer_temperature;
uint8_t fractional_temperature;
maxim_max30102_read_temperature(&integer_temperature, &fractional_temperature);
float temperature = integer_temperature + ((float)fractional_temperature)/16.0;
#ifdef DEBUG
Serial.println("--RF--");
Serial.print(elapsedTime);
Serial.print("\t");
Serial.print(n_spo2);
Serial.print("\t");
Serial.print(n_heart_rate, DEC);
Serial.print("\t");
Serial.print(hr_str);
Serial.print("\t");
Serial.println(temperature);
Serial.println("------");
#endif // DEBUG
#ifdef TEST_MAXIM_ALGORITHM
//calculate heart rate and SpO2 after BUFFER_SIZE samples (ST seconds of samples) using MAXIM's method
float n_spo2_maxim; //SPO2 value
int8_t ch_spo2_valid_maxim; //indicator to show if the SPO2 calculation is valid
int32_t n_heart_rate_maxim; //heart rate value
int8_t ch_hr_valid_maxim; //indicator to show if the heart rate calculation is valid
maxim_heart_rate_and_oxygen_saturation(aun_ir_buffer, BUFFER_SIZE, aun_red_buffer, &n_spo2_maxim, &ch_spo2_valid_maxim, &n_heart_rate_maxim, &ch_hr_valid_maxim);
#ifdef DEBUG
Serial.println("--MX--");
Serial.print(elapsedTime);
Serial.print("\t");
Serial.print(n_spo2_maxim);
Serial.print("\t");
Serial.print(n_heart_rate_maxim, DEC);
Serial.print("\t");
Serial.println(hr_str);
Serial.println("------");
#endif // DEBUG
#endif // TEST_MAXIM_ALGORITHM
//save samples and calculation result to SD card
#ifdef TEST_MAXIM_ALGORITHM
if(ch_hr_valid && ch_spo2_valid || ch_hr_valid_maxim && ch_spo2_valid_maxim) {
#else // TEST_MAXIM_ALGORITHM
if(ch_hr_valid && ch_spo2_valid) {
#endif // TEST_MAXIM_ALGORITHM
#ifdef USE_ADALOGGER
++k;
dataFile.print(elapsedTime);
dataFile.print("\t");
dataFile.print(n_spo2);
dataFile.print("\t");
dataFile.print(n_heart_rate, DEC);
dataFile.print("\t");
#ifdef TEST_MAXIM_ALGORITHM
dataFile.print(n_spo2_maxim);
dataFile.print("\t");
dataFile.print(n_heart_rate_maxim, DEC);
dataFile.print("\t");
#endif // TEST_MAXIM_ALGORITHM
dataFile.print(hr_str);
dataFile.print("\t");
dataFile.print(ratio);
dataFile.print("\t");
dataFile.print(correl);
dataFile.print("\t");
dataFile.print(temperature);
#ifdef SAVE_RAW_DATA
// Save raw data for unusual O2 levels
for(i=0;i<BUFFER_SIZE;++i)
{
dataFile.print(F("\t"));
dataFile.print(aun_red_buffer[i], DEC);
}
for(i=0;i<BUFFER_SIZE;++i)
{
dataFile.print(F("\t"));
dataFile.print(aun_ir_buffer[i], DEC);
}
#endif // SAVE_RAW_DATA
dataFile.println("");
// Blink green LED to indicate save event
digitalWrite(sdIndicatorPin,HIGH);
delay(10);
digitalWrite(sdIndicatorPin,LOW);
// FLush SD buffer every 10 points
if(k>=10) {
dataFile.flush();
k=0;
}
#else // USE_ADALOGGER
Serial.print(elapsedTime);
Serial.print("\t");
Serial.print(n_spo2);
Serial.print("\t");
Serial.print(n_heart_rate, DEC);
Serial.print("\t");
#ifdef TEST_MAXIM_ALGORITHM
Serial.print(n_spo2_maxim);
Serial.print("\t");
Serial.print(n_heart_rate_maxim, DEC);
Serial.print("\t");
#endif //TEST_MAXIM_ALGORITHM
Serial.print(hr_str);
Serial.print("\t");
Serial.print(ratio);
Serial.print("\t");
Serial.print(correl);
Serial.print("\t");
Serial.print(temperature);
#ifdef SAVE_RAW_DATA
// Save raw data for unusual O2 levels
for(i=0;i<BUFFER_SIZE;++i)
{
Serial.print(F("\t"));
Serial.print(aun_red_buffer[i], DEC);
}
for(i=0;i<BUFFER_SIZE;++i)
{
Serial.print(F("\t"));
Serial.print(aun_ir_buffer[i], DEC);
}
#endif // SAVE_RAW_DATA
Serial.println("");
#endif // USE_ADALOGGER
old_n_spo2=n_spo2;
}
}
void millis_to_hours(uint32_t ms, char* hr_str)
{
char istr[6];
uint32_t secs,mins,hrs;
secs=ms/1000; // time in seconds
mins=secs/60; // time in minutes
secs-=60*mins; // leftover seconds
hrs=mins/60; // time in hours
mins-=60*hrs; // leftover minutes
itoa(hrs,hr_str,10);
strcat(hr_str,":");
itoa(mins,istr,10);
strcat(hr_str,istr);
strcat(hr_str,":");
itoa(secs,istr,10);
strcat(hr_str,istr);
}
#ifdef USE_ADALOGGER
// blink three times if isOK is true, otherwise blink continuously
void blinkLED(const byte led, bool isOK)
{
byte i;
if(isOK) {
for(i=0;i<3;++i) {
digitalWrite(led,HIGH);
delay(200);
digitalWrite(led,LOW);
delay(200);
}
} else {
while(1) {
for(i=0;i<2;++i) {
digitalWrite(led,HIGH);
delay(50);
digitalWrite(led,LOW);
delay(50);
}
delay(500);
}
}
}
#endif