forked from ferdinandkeil/ADS129X
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathADS129X.cpp
354 lines (329 loc) · 11 KB
/
ADS129X.cpp
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
/**
* ADS129X.cpp
*
* Arduino library for the TI ADS129X series of analog-front-ends for
* biopotential measurements (EMG/EKG/EEG).
*
* This library offers two modes of operation: polling and interrupt.
* Polling mode should only be used in situations where multiple devices
* share the SPI bus. Interrupt mode is much faster (8kSPS on a Teensy 3.1),
* but starts receiving immediately when DRDY goes high.
* The API is the same for both modes. To activate polling mode add
* #define ADS129X_POLLING
* as first line to your sketch.
*
* Based on code by Conor Russomanno (https://github.com/conorrussomanno/ADS1299)
* Modified by Ferdinand Keil
*/
// Compatibility with the Arduino 1.0 library standard
#if defined(ARDUINO) && ARDUINO >= 100
#include "Arduino.h"
#else
#include "WProgram.h"
#endif
#include "ADS129X.h"
#include <SPI.h>
#ifndef ADS129X_POLLING
int ADS129X_CS;
long ADS129X_data[9];
boolean ADS129X_newData;
void ADS129X_dataReadyISR();
#endif
/**
* Initializes ADS129x library.
* @param _DRDY data ready pin
* @param _CS chip-select pin
*/
ADS129X::ADS129X(int _DRDY, int _CS) {
// SPI Setup
SPI.begin();
// // set clock divider
// SPI.setClockDivider(SPI_CLOCK_DIV2);
// // set data mode
// SPI.setDataMode(SPI_MODE1); //clock polarity = 0; clock phase = 1 (pg. 8)
// // set bit order
// SPI.setBitOrder(MSBFIRST); //SPI data format is MSB (pg. 25)
// initialize the data ready and chip select pins:
DRDY = _DRDY;
CS = _CS;
pinMode(DRDY, INPUT_PULLUP);
pinMode(CS, OUTPUT);
digitalWrite(CS, HIGH);
#ifndef ADS129X_POLLING
ADS129X_CS = _CS;
ADS129X_newData = false;
#endif
}
//System Commands
/**
* Exit Standby Mode.
*/
void ADS129X::WAKEUP() {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW); //Low to communicate
SPI.transfer(ADS129X_CMD_WAKEUP);
delayMicroseconds(2);
digitalWrite(CS, HIGH); //High to end communication
delayMicroseconds(2); //must way at least 4 tCLK cycles before sending another command (Datasheet, pg. 38)
SPI.endTransaction();
}
/**
* Enter Standby Mode.
*/
void ADS129X::STANDBY() {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW);
SPI.transfer(ADS129X_CMD_STANDBY);
delayMicroseconds(2);
digitalWrite(CS, HIGH);
SPI.endTransaction();
}
/**
* Reset Registers to Default Values.
*/
void ADS129X::RESET() {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW);
SPI.transfer(ADS129X_CMD_RESET);
delayMicroseconds(2);
digitalWrite(CS, HIGH);
delay(10); //must wait 18 tCLK cycles to execute this command (Datasheet, pg. 38)
SPI.endTransaction();
}
/**
* Start/restart (synchronize) conversions.
*/
void ADS129X::START() {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW);
SPI.transfer(ADS129X_CMD_START);
delayMicroseconds(2);
digitalWrite(CS, HIGH);
SPI.endTransaction();
#ifndef ADS129X_POLLING
attachInterrupt(DRDY, ADS129X_dataReadyISR, FALLING);
#endif
}
/**
* Stop conversion.
*/
void ADS129X::STOP() {
#ifndef ADS129X_POLLING
detachInterrupt(DRDY);
#endif
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW);
SPI.transfer(ADS129X_CMD_STOP);
delayMicroseconds(2);
digitalWrite(CS, HIGH);
SPI.endTransaction();
}
/**
* Enable Read Data Continuous mode (default).
*/
void ADS129X::RDATAC() {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW);
SPI.transfer(ADS129X_CMD_RDATAC);
delayMicroseconds(2);
digitalWrite(CS, HIGH);
delayMicroseconds(2); //must way at least 4 tCLK cycles before sending another command (Datasheet, pg. 39)
SPI.endTransaction();
}
/**
* Stop Read Data Continuously mode.
*/
void ADS129X::SDATAC() {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW);
SPI.transfer(ADS129X_CMD_SDATAC); //SDATAC
delayMicroseconds(2);
digitalWrite(CS, HIGH);
SPI.endTransaction();
}
/**
* Read data by command; supports multiple read back.
*/
void ADS129X::RDATA() {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW);
SPI.transfer(ADS129X_CMD_RDATA);
delayMicroseconds(2);
digitalWrite(CS, HIGH);
SPI.endTransaction();
}
/**
* Read register at address _address.
* @param _address register address
* @return value of register
*/
byte ADS129X::RREG(byte _address) {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
byte opcode1 = ADS129X_CMD_RREG | (_address & 0x1F); //001rrrrr; _RREG = 00100000 and _address = rrrrr
digitalWrite(CS, LOW); //Low to communicate
SPI.transfer(opcode1); //RREG
SPI.transfer(0x00); //opcode2
delayMicroseconds(1);
byte data = SPI.transfer(0x00); // returned byte should match default of register map unless edited manually (Datasheet, pg.39)
delayMicroseconds(2);
digitalWrite(CS, HIGH); //High to end communication
SPI.endTransaction();
return data;
}
/**
* Read _numRegisters register starting at address _address.
* @param _address start address
* @param _numRegisters number of registers
* @param data pointer to data array
*/
void ADS129X::RREG(byte _address, byte _numRegisters, byte *_data) {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
byte opcode1 = ADS129X_CMD_RREG | (_address & 0x1F); //001rrrrr; _RREG = 00100000 and _address = rrrrr
digitalWrite(CS, LOW); //Low to communicated
SPI.transfer(ADS129X_CMD_SDATAC); //SDATAC
SPI.transfer(opcode1); //RREG
SPI.transfer(_numRegisters-1); //opcode2
for(byte i = 0; i < _numRegisters; i++){
*(_data+i) = SPI.transfer(0x00); // returned byte should match default of register map unless previously edited manually (Datasheet, pg.39)
}
delayMicroseconds(2);
digitalWrite(CS, HIGH); //High to end communication
SPI.endTransaction();
}
/**
* Write register at address _address.
* @param _address register address
* @param _value register value
*/
void ADS129X::WREG(byte _address, byte _value) {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
byte opcode1 = ADS129X_CMD_WREG | (_address & 0x1F); //001rrrrr; _RREG = 00100000 and _address = rrrrr
digitalWrite(CS, LOW); //Low to communicate
SPI.transfer(opcode1);
SPI.transfer(0x00); // opcode2; only write one register
SPI.transfer(_value);
delayMicroseconds(2);
digitalWrite(CS, HIGH); //Low to communicate
SPI.endTransaction();
}
/**
* Read device ID.
* @return device ID
* 7:5 100 = ADS129x device family, 110 = ADS129xR device family
* 4:3 10
* 2:0 000 = 4-channel ADS1294 or ADS1294R, 001 = 6-channel ADS1296 or ADS1296R, 010 = 8-channel ADS1298 or ADS1298R
*/
byte ADS129X::getDeviceId() {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW); //Low to communicate
SPI.transfer(ADS129X_CMD_RREG); // RREG | 0
SPI.transfer(0x00); //Asking for 1 byte
byte data = SPI.transfer(0x00); // byte to read
delayMicroseconds(2);
digitalWrite(CS, HIGH); //Low to communicate
SPI.endTransaction();
return data;
}
#ifndef ADS129X_POLLING
/**
* Interrupt that gets called when DRDY goes HIGH.
* Transfers data and sets a flag.
*/
void ADS129X_dataReadyISR() {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(ADS129X_CS, LOW);
// status
((char*) ADS129X_data)[0*4+3] = 0;
((char*) ADS129X_data)[0*4+2] = SPI.transfer(0x00);
((char*) ADS129X_data)[0*4+1] = SPI.transfer(0x00);
((char*) ADS129X_data)[0*4+0] = SPI.transfer(0x00);
// channel 1
((char*) ADS129X_data)[1*4+3] = 0;
((char*) ADS129X_data)[1*4+2] = SPI.transfer(0x00);
((char*) ADS129X_data)[1*4+1] = SPI.transfer(0x00);
((char*) ADS129X_data)[1*4+0] = SPI.transfer(0x00);
// channel 2
((char*) ADS129X_data)[2*4+3] = 0;
((char*) ADS129X_data)[2*4+2] = SPI.transfer(0x00);
((char*) ADS129X_data)[2*4+1] = SPI.transfer(0x00);
((char*) ADS129X_data)[2*4+0] = SPI.transfer(0x00);
// channel 3
((char*) ADS129X_data)[3*4+3] = 0;
((char*) ADS129X_data)[3*4+2] = SPI.transfer(0x00);
((char*) ADS129X_data)[3*4+1] = SPI.transfer(0x00);
((char*) ADS129X_data)[3*4+0] = SPI.transfer(0x00);
// channel 4
((char*) ADS129X_data)[4*4+3] = 0;
((char*) ADS129X_data)[4*4+2] = SPI.transfer(0x00);
((char*) ADS129X_data)[4*4+1] = SPI.transfer(0x00);
((char*) ADS129X_data)[4*4+0] = SPI.transfer(0x00);
// channel 5
((char*) ADS129X_data)[5*4+3] = 0;
((char*) ADS129X_data)[5*4+2] = SPI.transfer(0x00);
((char*) ADS129X_data)[5*4+1] = SPI.transfer(0x00);
((char*) ADS129X_data)[5*4+0] = SPI.transfer(0x00);
// channel 6
((char*) ADS129X_data)[6*4+3] = 0;
((char*) ADS129X_data)[6*4+2] = SPI.transfer(0x00);
((char*) ADS129X_data)[6*4+1] = SPI.transfer(0x00);
((char*) ADS129X_data)[6*4+0] = SPI.transfer(0x00);
// channel 7
((char*) ADS129X_data)[7*4+3] = 0;
((char*) ADS129X_data)[7*4+2] = SPI.transfer(0x00);
((char*) ADS129X_data)[7*4+1] = SPI.transfer(0x00);
((char*) ADS129X_data)[7*4+0] = SPI.transfer(0x00);
// channel 8
((char*) ADS129X_data)[8*4+3] = 0;
((char*) ADS129X_data)[8*4+2] = SPI.transfer(0x00);
((char*) ADS129X_data)[8*4+1] = SPI.transfer(0x00);
((char*) ADS129X_data)[8*4+0] = SPI.transfer(0x00);
digitalWrite(ADS129X_CS, HIGH);
SPI.endTransaction();
ADS129X_newData = true;
}
#endif
/**
* Receive data when in continuous read mode.
* @param buffer buffer for received data
* @return true when received data
*/
boolean ADS129X::getData(long *buffer) {
#ifndef ADS129X_POLLING
if (ADS129X_newData) {
ADS129X_newData = false;
for (int i = 0; i < 9; i++) {
buffer[i] = ADS129X_data[i];
}
return true;
}
return false;
#else
if (digitalRead(DRDY) == LOW) {
SPI.beginTransaction(SPISettings(4000000, MSBFIRST, SPI_MODE1));
digitalWrite(CS, LOW);
for(int i = 0; i<9; i++){
long dataPacket = 0;
for(int j = 0; j<3; j++){
byte dataByte = SPI.transfer(0x00);
dataPacket = (dataPacket<<8) | dataByte;
}
buffer[i] = dataPacket;
}
digitalWrite(CS, HIGH);
SPI.endTransaction();
return true;
}
return false;
#endif
}
/**
* Configure channel _channel.
* @param _channel channel (1-8)
* @param _powerDown power down (true, false)
* @param _gain gain setting
* @param _mux mux setting
*/
void ADS129X::configChannel(byte _channel, boolean _powerDown, byte _gain, byte _mux) {
byte value = ((_powerDown & 1)<<7) | ((_gain & 7)<<4) | (_mux & 7);
WREG(ADS129X_REG_CH1SET + (_channel-1), value);
}