forked from EnviroDIY/Arduino-SDI-12
-
Notifications
You must be signed in to change notification settings - Fork 0
/
SDI12.cpp
804 lines (611 loc) · 30.4 KB
/
SDI12.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
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
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
/* ======================== Arduino SDI-12 =================================
Arduino library for SDI-12 communications to a wide variety of environmental
sensors. This library provides a general software solution, without requiring
any additional hardware.
======================== Attribution & License =============================
Copyright (C) 2013 Stroud Water Research Center
Available at https://github.com/StroudCenter/Arduino-SDI-12
Authored initially in August 2013 by:
Kevin M. Smith (http://ethosengineering.org)
Inquiries: [email protected]
based on the SoftwareSerial library (formerly NewSoftSerial), authored by:
ladyada (http://ladyada.net)
Mikal Hart (http://www.arduiniana.org)
Paul Stoffregen (http://www.pjrc.com)
Garrett Mace (http://www.macetech.com)
Brett Hagman (http://www.roguerobotics.com/)
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
==================== Notes on SDI-12, Specification v1.3 ======================
Overview:
SDI-12 is a communications protocol that uses a single data wire to
communicate with up to 62 uniquely addressed sensors. So long as each
sensor supports SDI-12, mixed sensor types can appear on the same data
bus. Each address is a single character. The valid ranges are 0-9, a-z,
and A-Z. Only the datalogger can initiate communications on the data
bus.
It does so by pulling the data line into a 5v state for at least 12
milliseconds to wake up all the sensors, before returning the line into
a 0v state for 8 milliseconds announce an outgoing command. The command
contains both the action to be taken, and the address of the device who
should respond. If there is a sensor on the bus with that address, it is
responsible for responding to the command. Sensors should ignore
commands that were not issued to them, and should return to a sleep
state until the datalogger again issues the wakeup sequence.
Physical Connections: 1 data line (0v - 5.5v)
1 12v power line (9.6v - 16v)
1 ground line
Baud Rate: 1200 bits per second
Data Frame Format: 10 bits per data frame
1 start bit
7 data bits (least significant bit first)
1 even parity bit
1 stop bit
Data Line: SDI-12 communication uses a single
bi-directional data line
with three-state, inverse logic.
LINE CONDITION | BINARY STATE | VOLTAGE RANGE
-----------------------------------------------
marking 1 -0.5 to 1.0 volts
spacing 0 3.5 to 5.5 volts
transition undefined 1.0 to 3.5 volts
_____ _____ _____ _____ _____ spacing
5v | | | | | | | | | |
| 0 | 1 | 0 | 1 | 0 | 1 | 0 | 1 | 0 | transition
Ov___| |_____| |_____| |_____| |_____| |___ marking
For more information, and for a list of commands and responses, please see
SDI-12.org, official site of the SDI-12 Support Group.
==================== Code Organization ======================
0. Includes, Defines, & Variable Declarations
1. Buffer Setup
2. Data Line States, Overview of Interrupts
3. Constructor, Destructor, SDI12.begin(), and SDI12.end()
4. Waking up, and talking to, the sensors.
5. Reading from the SDI-12 object. available(), peek(), read(), flush()
6. Using more than one SDI-12 object, isActive() and setActive().
7. Interrupt Service Routine (getting the data into the buffer)
=========== 0. Includes, Defines, & Variable Declarations =============
0.1 - Include the header file for this library.
0.2 - defines the size of the buffer
0.3 - defines value for DISABLED state (see section 2)
0.4 - defines value for ENABLED state (not used, reserved for future)
0.5 - defines value for DISABLED state (see section 2)
0.6 - defines value for TRANSMITTING state (see section 2)
0.7 - defines value for LISTENING state
0.8 - defines value for the spacing of bits.
1200 bits per second implies 833 microseconds per bit.
830 seems to be a reliable value given the overhead of the call.
0.9 - a static pointer to the active object. See section 6.
0.10 - a reference to the data pin, used throughout the library
0.11 - holds the buffer overflow status
*/
#include <SDI12.h> // 0.1 header file for this library
#define _BUFFER_SIZE 64 // 0.2 max RX buffer size
#define DISABLED 0 // 0.3 value for DISABLED state
#define ENABLED 1 // 0.4 value for ENABLED state
#define HOLDING 2 // 0.5 value for DISABLED state
#define TRANSMITTING 3 // 0.6 value for TRANSMITTING state
#define LISTENING 4 // 0.7 value for LISTENING state
#define SPACING 830 // 0.8 bit timing in microseconds
SDI12 *SDI12::_activeObject = NULL; // 0.9 pointer to active SDI12 object
uint8_t _dataPin; // 0.10 reference to the data pin
bool _bufferOverflow; // 0.11 buffer overflow status
/* =========== 1. Buffer Setup ============================================
The buffer is used to store characters from the SDI-12 data line.
Characters are read into the buffer when an interrupt is received on the
data line. The buffer uses a circular implementation with pointers to
both the head and the tail. There is one buffer per instance of the
SDI-12 object. This will consume a good deal of RAM, so be prudent in
running multiple instances.
For more information on circular buffers:
http://en.wikipedia.org/wiki/Circular_buffer
1.1 - Define a maximum buffer size (in number of characters). Increasing
the buffer size will use more RAM. If you exceed 256 characters, be sure
to change the data type of the index to support the larger range of
addresses.
1.2 - Create a character array of the specified size. 1.3 - Index to
buffer head. (unsigned 8-bit integer, can map from 0-255) 1.4 - Index to
buffer tail. (unsigned 8-bit integer, can map from 0-255)
*/
// See section 0 above. // 1.1 - max buffer size
char _rxBuffer[_BUFFER_SIZE]; // 1.2 - buff for incoming
uint8_t _rxBufferHead = 0; // 1.3 - index of buff head
uint8_t _rxBufferTail = 0; // 1.4 - index of buff tail
/* =========== 2. Data Line States ===============================
The Arduino is responsible for managing communication with the sensors.
Since all the data transfer happens on the same line, the state of the
data line is very important.
When the pin is in the HOLDING state, it is holding the line LOW so that
interference does not unintentionally wake the sensors up. The interrupt
is disabled for the dataPin, because we are not expecting any SDI-12
traffic. In the TRANSMITTING state, we would like exclusive control of
the Arduino, so we shut off all interrupts, and vary the voltage of the
dataPin in order to wake up and send commands to the sensor. In the
LISTENING state, we are waiting for a sensor to respond, so we drop the
voltage level to LOW and relinquish control (INPUT). If we would like to
disable all SDI-12 functionality, then we set the system to the DISABLED
state, removing the interrupt associated with the dataPin. For
predictability, we set the pin to a LOW level high impedance state
(INPUT).
State Interrupts Pin Mode Pin Level
HOLDING Pin Disable OUTPUT LOW
TRANSMITTING All Disable OUTPUT VARYING
LISTENING All Enable INPUT LOW
DISABLED Pin Disable INPUT LOW
------------------------------| Sequencing |------------------------------
Generally, this is acceptable.
HOLDING --> TRANSMITTING --> LISTENING --> TRANSMITTING --> LISTENING -->
If you have interference, you should force a hold, using forceHold();
HOLDING --> TRANSMITTING --> LISTENING --> done reading, forceHold(); HOLDING
-------------------------| Function Descriptions |-------------------------
2.1 - Sets the proper state. This is a private function, and only used
internally. It uses #define values of HOLDING, TRANSMITTING, LISTENING,
and DISABLED to determine which state should be set. The grid above
defines the settings applied in changing to each state.
2.2 - A public function which forces the line into a "holding" state.
This is generally unneeded, but for deployments where interference is an
issue, it should be used after all expected bytes have been returned
from the sensor.
------------------------| Overview of Interrupts |-------------------------
Some vocabulary:
Registers: PCMSK0, PCMSK1, PCMSK2 :registers that enable or disable
pin-change interrupts on individual pins
PCICR : a register where the three least significant bits enable or
disable pin change interrupts on a range of pins, i.e.
{0,0,0,0,0,PCIE2,PCIE1,PCIE0}, where PCIE2 maps to PCMSK2, PCIE1 maps to
PCMSK1, and PCIE0 maps to PCMSK0.
noInterrupts() globally disables interrupts (of all types), while interrupts()
globally enables interrupts (of all types), but they will only occur if
the requisite registers are set (e.g. PCMSK and PCICR).
--------------| What is an interrupt?
An interrupt is a signal that causes the microcontroller to halt
execution of the program, and perform a subroutine known as an interrupt
handler or Interrupt Service Routine (ISR). After the ISR, program
execution continues. This allows the microcontroller to efficiently
handle a time-sensitive function such as receiving a burst of data on
one of its pins, by not forcing the microcontroller to wait for the
data. It can perform other tasks until the interrupt is called.
Physically, interrupts take the form of signals on the pins of the
microcontroller. To make the library generic, the dataPin responds to
pin change interrupts which are available on all pins of the Arduino
Uno, and a majority of pins on other variants. Pin change interrupts
trigger an ISR on any change of state detected for a specified pin.
Obviously, interrupts are not always desirable, so they can be
controlled by manipulating registers. Registers are small 1-byte (8-bit)
stores of memory directly accessible by processor. There is one register
PCICR, which can enable and disable pin change interrupts for a range of
pins at a single time.
There are also three registers that store the state (enabled/disabled) of the
pin change interrupts: PCMSK0, PCMSK1, and PCMSK2. Each bit stores a 1
(enabled) or 0 (disabled).
For example, PCMSK0 holds an 8 bits which represent:
PCMSK0 {PCINT7, PCINT6, PCINT5, PCINT4, PCINT3, PCINT2, PCINT1, PCINT0}
On an Arduino Uno, these map to:
PCMSK0 {XTAL2, XTAL1, Pin 13, Pin 12, Pin 11, Pin 10, Pin 9, Pin 8}
==================================================| Enabling an interrupt.
Initially, no interrupts are enabled, so PCMSK0 looks like: {00000000}.
If we were to use pin 9 as the data pin, we would set the bit in the pin
9 position to 1, like so: {00000010}.
To accomplish this, we can make use of some predefined macros.
One macro "digitalPinToPCMSK" is defined in "pins_arduino.h" which
allows us to quickly get the proper register (PCMSK0, PCMSK1, or PCMSK2)
given the number of the Arduino pin. So when we write:
digitalPinToPCMSK(9), the address of PCMSK0 is returned. We can use the
dereferencing operator '*' to get the value of the address.
That is, *digitalPinToPCMSK(9) returns: {00000000}.
Another macro , "digitalPinToPCMSKbit" is also defined in
"pins_arduino.h" and returns the position of the bit of interest within
the PCMSK of interest.
So when we write: digitalPinToPCMSKbit(9), the value returned is 1,
because pin 9 is represented by the "1st bit" within PCMSK0, which has a
zero-based index. That is: PCMSK { 7th bit, 6th bit, 5th bit, 4th bit,
3rd bit, 2nd bit, 1st bit, 0th bit }
The leftward bit shift operator "<<" can then used to create a "mask".
Masks are data that are used during bitwise operations to manipulate
(enable / disable) one or more individual bits simultaneously.
The syntax for the operator is (variable << number_of_bits).
Some examples:
byte a = 5; // binary: a = 00000101
byte b = a << 4; // binary: b = 01010000
So when we write: (1<<digitalPinToPCMSKbit(9)), we get: {00000010}.
Or equivalently: (1<<1), we get: {00000010}.
To use the mask to set the bit of interest we use the bitwise or
operator '|'. We will use the compact '|=' notation which does the
operation and then stores the result back into the left hand side.
So the operation:
*digitalPinToPCMSK(_dataPin) |= (1<<digitalPinToPCMSKbit(9));
Accomplishes:
(1<<digitalPinToPCMSKbit(9)) {00000010}
PCMSK0 | {00000000}
-------------
{00000010}
We must also enable the global control for the interrupt. This is done
in a similar fashion.
*digitalPinToPCICR(_dataPin) |= (1<<digitalPinToPCICRbit(_dataPin));
Now let's assume that part of your Arduino sketch outside of SDI-12 had
set a pin change interrupt on pin 13. Pin 9 and pin 13 are on the same
PCMSK in the case of the Arduino Uno.
This time before we set the bit for pin nine,
*digitalPinToPCMSK(9) returns: {00100000}.
So now:
(1<<digitalPinToPCMSKbit(9)) {00000010}
PCMSK0 | {00100000}
-------------
{00100010}
By using a bitmask and bitwise operation, we have successfully enabled
pin 9 without effecting the state of pin 13.
================================================= Disabling an interrupt.
When the we would like to put the SDI-12 object in the DISABLED state,
(e.g. the destructor is called), we need to make sure the bit
corresponding to the data pin is unset.
Let us consider again the case of where an interrupt has been enabled on
pin 13: {00100010}. We want to be sure not to disturb this interrupt
when disabling the interrupt on pin 9.
We will make use of similar macros, but this time we will use an
inverted bit mask and the AND operation. The "&=" operator is equivalent
to a bitwise AND operation between the PCMSK register of interest and
the previous result, which is then stored back into the PCMSK of
interest. The "*" is the dereferencing operator, which is equivalently
translated to "value pointed by", and allows us to store the result back
into the proper PCMSK.
Again 1<<digitalPinToPCMSKbit(9) returns {00000010}
The inversion symbol ~ modifies the result to {11111101}
So to finish our example:
~(1<<digitalPinToPCMSKbit(9)) {11111101}
PCMSK0 & {00100010}
-------------
{00100000}
So only the interrupt on pin 13 remains set. As a matter of book
keeping, if we unset the last bit in the PCMSK, we ought to also unset
the respective bit in the PCICR.
!(*digitalPinToPCMSK(9)
will evaluate TRUE if PCMSK {00000000}
will evaluate FALSE if PCMSK != {00000000}
In this case, pin 13 is set, so the expression would be FALSE. If we go
back to the original case without pin 13, the expression after disabling
pin 9 would evaluate to TRUE.
Therefore if we evaluate to TRUE, we should tidy up:
if(!*digitalPinToPCMSK(_dataPin)){
*digitalPinToPCICR(_dataPin) &= ~(1<<digitalPinToPCICRbit(_dataPin));
}
*/
// 2.1 - sets the state of the SDI-12 object.
void SDI12::setState(uint8_t state){
if(state == HOLDING){
pinMode(_dataPin,OUTPUT);
digitalWrite(_dataPin,LOW);
*digitalPinToPCMSK(_dataPin) &= ~(1<<digitalPinToPCMSKbit(_dataPin));
return;
}
if(state == TRANSMITTING){
pinMode(_dataPin,OUTPUT);
noInterrupts(); // supplied by Arduino.h, same as cli()
return;
}
if(state == LISTENING) {
digitalWrite(_dataPin,LOW);
pinMode(_dataPin,INPUT);
interrupts(); // supplied by Arduino.h, same as sei()
*digitalPinToPCICR(_dataPin) |= (1<<digitalPinToPCICRbit(_dataPin));
*digitalPinToPCMSK(_dataPin) |= (1<<digitalPinToPCMSKbit(_dataPin));
} else { // implies state==DISABLED
digitalWrite(_dataPin,LOW);
pinMode(_dataPin,INPUT);
*digitalPinToPCMSK(_dataPin) &= ~(1<<digitalPinToPCMSKbit(_dataPin));
if(!*digitalPinToPCMSK(_dataPin)){
*digitalPinToPCICR(_dataPin) &= ~(1<<digitalPinToPCICRbit(_dataPin));
}
}
}
// 2.2 - forces a HOLDING state.
void SDI12::forceHold(){
setState(HOLDING);
}
/* ======= 3. Constructor, Destructor, SDI12.begin(), and SDI12.end() =======
3.1 - The constructor requires a single parameter: the pin to be used
for the data line. When the constructor is called it resets the buffer
overflow status to FALSE and assigns the pin number "dataPin" to the
private variable "_dataPin".
3.2 - When the destructor is called, it's main task is to disable any
interrupts that had been previously assigned to the pin, so that the pin
will behave as expected when used for other purposes. This is achieved
by putting the SDI-12 object in the DISABLED state.
3.3 - This is called to begin the functionality of the SDI-12 object. It
has no parameters as the SDI-12 protocol is fully specified (e.g. the
baud rate is set). It sets the object as the active object (if multiple
SDI-12 instances are being used simultaneously).
3.4 - This can be called to temporarily cease all functionality of the
SDI-12 object. It is not as harsh as destroying the object with the
destructor, as it will maintain the memory buffer.
*/
// 3.1 Constructor
SDI12::SDI12(uint8_t dataPin){ _bufferOverflow = false; _dataPin = dataPin; }
// 3.2 Destrutor
SDI12::~SDI12(){ setState(DISABLED); }
// 3.3 Begin
void SDI12::begin() { setState(HOLDING); setActive(); }
// 3.4 End
void SDI12::end() { setState(DISABLED); }
/* ============= 4. Waking up, and talking to, the sensors. ===================
4.1 - wakeSensors() literally wakes up all the sensors on the bus. The
SDI-12 protocol requires a pulse of HIGH voltage for at least 12
milliseconds followed immediately by a pulse of LOW voltage for at least
8.3 milliseconds. The values here are close to those values, but provide
100 extra microseconds of wiggle room. Setting the SDI-12 object into
the TRANSMITTING allows us to assert control of the line without
triggering any interrupts.
4.2 - This function writes a character out to the data line. SDI-12
specifies the general transmission format of a single character as:
10 bits per data frame
1 start bit
7 data bits (least significant bit first)
1 even parity bit
1 stop bit
We also recall that we are using inverse logic, so HIGH represents 0,
and LOW represents a 1. If you are unclear on any of these terms, I
would recommend that you look them up before proceeding. They will be
better explained elsewhere.
The transmission takes several steps.
The variable name for the outgoing character is "out".
+ 4.2.1 - Determine the proper even parity bit (will an additional 1 or 0
make the final number of 1's even?)
First we grab the bit using an optimized macro from parity.h
parity_even_bit(out)
Then we bit shift it into the proper place, which is the most
significant bit position, since the characters we are using are only
7-bits.
(parity_even_bit(out)<<7);
Then we use the '|=' operator to set the bit if necessary.
+ 4.2.2 - Send the start bit. The start bit is always a '0', so we simply
write the dataPin HIGH for SPACING microseconds.
+ 4.2.3 - Send the payload (the 7 character bits and the parity bit) least
significant bit first. This is accomplished bitwise AND operations on a
moving mask (00000001) --> (00000010) --> (00000100)... and so on. This
functionality makes use of the '<<=' operator which stores the result of
the bit-shift back into the left hand side.
If the result of (out & mask) determines whether a 1 or 0 should be sent.
Again, here inverse logic may lead to easy confusion.
if(out & mask){
digitalWrite(_dataPin, LOW);
}
else{
digitalWrite(_dataPin, HIGH);
}
+ 4.2.4 - Send the stop bit. The stop bit is always a '1', so we simply
write the dataPin LOW for SPACING microseconds.
4.3 - sendCommand(String cmd) is a publicly accessible function that
sends out a String byte by byte the command line.
*/
// 4.1 - this function wakes up the entire sensor bus
void SDI12::wakeSensors(){
setState(TRANSMITTING);
digitalWrite(_dataPin, HIGH);
delayMicroseconds(12100);
digitalWrite(_dataPin, LOW);
delayMicroseconds(8400);
}
// 4.2 - this function writes a character out on the data line
void SDI12::writeChar(uint8_t out)
{
out |= (parity_even_bit(out)<<7); // 4.2.1 - parity bit
digitalWrite(_dataPin, HIGH); // 4.2.2 - start bit
delayMicroseconds(SPACING);
for (byte mask = 0x01; mask; mask<<=1){ // 4.2.3 - send payload
if(out & mask){
digitalWrite(_dataPin, LOW);
}
else{
digitalWrite(_dataPin, HIGH);
}
delayMicroseconds(SPACING);
}
digitalWrite(_dataPin, LOW); // 4.2.4 - stop bit
delayMicroseconds(SPACING);
}
// 4.3 - this function sends out the characters of the String cmd, one by one
void SDI12::sendCommand(String cmd){
wakeSensors(); // wake up sensors
for (int i = 0; i < cmd.length(); i++){
writeChar(cmd[i]); // write each characters
}
setState(LISTENING); // listen for reply
}
/* ============= 5. Reading from the SDI-12 object. ===================
5.1 - available() is a public function that returns the number of
characters available in the buffer.
To understand how:
_rxBufferTail + _BUFFER_SIZE - _rxBufferHead) % _BUFFER_SIZE;
accomplishes this task, we will use a few examples.
To start take the buffer below that has _BUFFER_SIZE = 10. The
message "abc" has been wrapped around (circular buffer).
_rxBufferTail = 1 // points to the '-' after c
_rxBufferHead = 8 // points to 'a'
[ c ] [ - ] [ - ] [ - ] [ - ] [ - ] [ - ] [ - ] [ a ] [ b ]
The number of available characters is (1 + 10 - 8) % 10 = 3
The '%' or modulo operator finds the remainder of division of one number
by another. In integer arithmetic 3 / 10 = 0, but has a remainder of 3.
We can only get the remainder by using the the modulo '%'. 3 % 10 = 3.
This next case demonstrates more clearly why the modulo is used.
_rxBufferTail = 4 // points to the '-' after c
_rxBufferHead = 1 // points to 'a'
[ a ] [ b ] [ c ] [ - ] [ - ] [ - ] [ - ] [ - ] [ - ] [ - ]
The number of available characters is (4 + 10 - 1) % 10 = 3
If we did not use the modulo we would get either ( 4 + 10 - 1 ) = 13
characters or ( 4 + 10 - 1 ) / 10 = 1 character. Obviously neither is
correct.
If there has been a buffer overflow, available() will return -1.
5.2 - peek() is a public function that allows the user to look at the
character that is at the head of the buffer. Unlike read() it does not
consume the character (i.e. the index addressed by _rxBufferHead is not
changed). peek() returns -1 if there are no characters to show.
5.3 - flush() is a public function that clears the buffers contents by
setting the index for both head and tail back to zero.
5.4 - read() returns the character at the current head in the buffer
after incrementing the index of the buffer head. This action 'consumes'
the character, meaning it can not be read from the buffer again. If you
would rather see the character, but leave the index to head intact, you
should use peek();
*/
// 5.1 - reveals the number of characters available in the buffer
int SDI12::available()
{
if(_bufferOverflow) return -1;
return (_rxBufferTail + _BUFFER_SIZE - _rxBufferHead) % _BUFFER_SIZE;
}
// 5.2 - reveals the next character in the buffer without consuming
int SDI12::peek()
{
if (_rxBufferHead == _rxBufferTail) return -1; // Empty buffer? If yes, -1
return _rxBuffer[_rxBufferHead]; // Otherwise, read from "head"
}
// 5.3 - a public function that clears the buffer contents and
// resets the status of the buffer overflow.
void SDI12::flush()
{
_rxBufferHead = _rxBufferTail = 0;
_bufferOverflow = false;
}
// 5.4 - reads in the next character from the buffer (and moves the index ahead)
int SDI12::read()
{
_bufferOverflow = false; //reading makes room in the buffer
if (_rxBufferHead == _rxBufferTail) return -1; // Empty buffer? If yes, -1
uint8_t nextChar = _rxBuffer[_rxBufferHead]; // Otherwise, grab char at head
_rxBufferHead = (_rxBufferHead + 1) % _BUFFER_SIZE; // increment head
return nextChar; // return the char
}
/* ============= 6. Using more than one SDI-12 object. ===================
This library is allows for multiple instances of itself running on
different pins, however, you should use care. The instances DO NOT share
a buffer, as they do in the multi-instance case for SoftwareSerial, so
it will consume more RAM than you might expect.
SDI-12 can support up to 62 sensors on a single pin/bus, so it is not
necessary to use an instance for each sensor.
Because we are using pin change interrupts there can only be one active
object at a time (since this is the only reliable way to determine which
pin the interrupt occurred on). The active object is the only object
that will respond properly to interrupts. However promoting another
instance to Active status does not automatically remove the interrupts
on the other pin. For proper behavior it is recommended to use this
pattern:
mySDI12.forceHold();
myOtherSDI12.setActive();
Other notes: Promoting an object into the Active state will set it as
HOLDING. See 6.1 for more information.
Calling mySDI12.begin() will assert mySDI12 as the new active object,
until another instance calls myOtherSDI12.begin() or
myOtherSDI12.setActive().
Calling mySDI12.end() does NOT hand-off active status to another SDI-12
instance.
You can check on the active object by calling mySDI12.isActive(), which
will return a boolean value TRUE if active or FALSE if inactive.
6.1 - a method for setting the current object as the active object.
returns TRUE if the object was not formerly the active object and now
is. returns
Promoting an inactive to the active instance will start it in the
HOLDING state and return TRUE.
Otherwise, if the object is currently the active instance, it will
remain unchanged and return FALSE.
6.2 - a method for checking if the object is the active object. Returns
true if the object is currently the active object, false otherwise.
*/
// 6.1 - a method for setting the current object as the active object
bool SDI12::setActive()
{
if (_activeObject != this)
{
setState(HOLDING);
_activeObject = this;
return true;
}
return false;
}
// 6.2 - a method for checking if this object is the active object
bool SDI12::isActive() { return this == _activeObject; }
/* ============== 6. Interrupt Service Routine ===================
We have received an interrupt signal, what should we do?
6.1 - Passes off responsibility for the interrupt to the active object.
6.2 - This function quickly reads a new character from the data line in
to the buffer. It takes place over a series of key steps.
+ 6.2.1 - Check for the start bit. If it is not there, interrupt may be
from interference or an interrupt we are not interested in, so return.
+ 6.2.2 - Make space in memory for the new character "newChar".
+ 6.2.3 - Wait half of a SPACING to help center on the next bit. It will
not actually be centered, or even approximately so until
delayMicroseconds(SPACING) is called again.
+ 6.2.4 - For each of the 8 bits in the payload, read wether or not the
line state is HIGH or LOW. We use a moving mask here, as was previously
demonstrated in the writeByte() function.
The loop runs from i=0x1 (hexadecimal notation for 00000001) to i<0x80
(hexadecimal notation for 10000000). So the loop effectively uses the
masks following masks: 00000001
00000010
00000100
00001000
00010000
00100000
01000000 and their inverses.
Here we use an if / else structure that helps to balance the time it
takes to either a HIGH vs a LOW, and helps maintain a constant timing.
+ 6.2.5 - Skip the parity bit. There is no error checking.
+ 6.2.6 - Skip the stop bit.
+ 6.2.7 - Check for an overflow. We do this by checking if advancing the
tail would make it have the same index as the head (in a circular
fashion).
+ 6.2.8 - Save the byte into the buffer if there has not been an
overflow, and then advance the tail index.
6.3 - Check if the various interrupt vectors are defined. If they are
the ISR is instructed to call _handleInterrupt() when they trigger. */
// 6.1 - Passes off responsibility for the interrupt to the active object.
inline void SDI12::handleInterrupt(){
if (_activeObject) _activeObject->receiveChar();
}
// 6.2 - Quickly reads a new character into the buffer.
void SDI12::receiveChar()
{
if (digitalRead(_dataPin)) // 6.2.1 - Start bit?
{
uint8_t newChar = 0; // 6.2.2 - Make room for char.
delayMicroseconds(SPACING/2); // 6.2.3 - Wait 1/2 SPACING
for (uint8_t i=0x1; i<0x80; i <<= 1) // 6.2.4 - read the 7 data bits
{
delayMicroseconds(SPACING);
uint8_t noti = ~i;
if (!digitalRead(_dataPin))
newChar |= i;
else
newChar &= noti;
}
delayMicroseconds(SPACING); // 6.2.5 - Skip the parity bit.
delayMicroseconds(SPACING); // 6.2.6 - Skip the stop bit.
// 6.2.7 - Overflow? If not, proceed.
if ((_rxBufferTail + 1) % _BUFFER_SIZE == _rxBufferHead)
{ _bufferOverflow = true;
} else { // 6.2.8 - Save char, advance tail.
_rxBuffer[_rxBufferTail] = newChar;
_rxBufferTail = (_rxBufferTail + 1) % _BUFFER_SIZE;
}
}
}
//6.3
#if defined(PCINT0_vect)
ISR(PCINT0_vect){ SDI12::handleInterrupt(); }
#endif
#if defined(PCINT1_vect)
ISR(PCINT1_vect){ SDI12::handleInterrupt(); }
#endif
#if defined(PCINT2_vect)
ISR(PCINT2_vect){ SDI12::handleInterrupt(); }
#endif
#if defined(PCINT3_vect)
ISR(PCINT3_vect){ SDI12::handleInterrupt(); }
#endif