Skip to content

Commit 9f5bf63

Browse files
authored
Implemented communications between Coral and Arduino (#36)
* Minimum working example of SPI communications between arduino and coral * Implemented communication with Arduino over SPI and created Arduino library to receive data * Started SPI comm protocol documentation * Fixed docs build issues * Documented SPI communications protocl * Changed travis file to only build on master branch
1 parent f7800f9 commit 9f5bf63

File tree

9 files changed

+596
-3
lines changed

9 files changed

+596
-3
lines changed

.travis.yml

+2-2
Original file line numberDiff line numberDiff line change
@@ -6,11 +6,11 @@ sudo: false
66
cache: pip
77

88
install:
9-
- "pip install sphinx sphinx_rtd_theme"
9+
- "pip install sphinx sphinx_rtd_theme recommonmark"
1010
- pip install -r requirements.txt
1111
- pip install pytest
1212

13-
script:
13+
script:
1414
- docs/build.sh
1515
- pytest
1616

+264
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,264 @@
1+
#include "AnythingSensor.h"
2+
3+
AnythingSensor* _instance = nullptr;
4+
5+
AnythingSensor::AnythingSensor()
6+
{
7+
_instance = this;
8+
}
9+
10+
bool AnythingSensor::begin()
11+
{
12+
// Create a temporary buffer with size 1 for reading config information
13+
AnythingSensor::buffer = new byte[4];
14+
AnythingSensor::bufferLength = 4;
15+
AnythingSensor::buffer[0] = 0;
16+
AnythingSensor::buffer[1] = 0;
17+
AnythingSensor::buffer[2] = 0;
18+
AnythingSensor::buffer[3] = 0;
19+
20+
21+
// Iniitliaze variables for ISR
22+
AnythingSensor::pos = 0;
23+
AnythingSensor::dataAvailable = false;
24+
25+
spiInit();
26+
27+
commInit();
28+
29+
dataInit();
30+
31+
return true;
32+
}
33+
34+
void AnythingSensor::spiInit()
35+
{
36+
// Set our commState to the INIT state
37+
commState = INIT;
38+
39+
// Set the signal pin to output mode
40+
pinMode(signalPin, OUTPUT);
41+
digitalWrite(signalPin, LOW);
42+
43+
// Enable SPI in slave mode
44+
// Set Master In Slave Out as output
45+
pinMode(MISO, OUTPUT);
46+
47+
48+
// Set SPDR to 0x00
49+
SPDR = 0x00;
50+
51+
// turn on SPI in slave mode
52+
SPCR |= _BV(SPE);
53+
54+
// turn on interrupt for
55+
SPCR |= _BV(SPIE);
56+
}
57+
58+
void AnythingSensor::commInit()
59+
{
60+
// Booleans to store the state of initialization
61+
bool initCompleted = false;
62+
bool startMessagedReceived = false;
63+
while(!initCompleted)
64+
{
65+
commState = INIT; // Ensure comm state is in init
66+
67+
// First step of initialization, receive 4 0xFF messages from the Anything Sensor
68+
startMessagedReceived = false;
69+
while(!startMessagedReceived)
70+
{
71+
if(AnythingSensor::dataAvailable)
72+
{
73+
// Check that all of the data in the buffer is 0xFF
74+
if(AnythingSensor::buffer[0] == 0xFF &&
75+
AnythingSensor::buffer[1] == 0xFF &&
76+
AnythingSensor::buffer[2] == 0xFF &&
77+
AnythingSensor::buffer[3] == 0xFF)
78+
{
79+
startMessagedReceived = true;
80+
dataAvailable = false;
81+
}
82+
else
83+
{
84+
// Reset buffer to wait for new group of start bits
85+
AnythingSensor::buffer[0] = 0;
86+
AnythingSensor::buffer[1] = 0;
87+
AnythingSensor::buffer[2] = 0;
88+
AnythingSensor::buffer[3] = 0;
89+
AnythingSensor::dataAvailable = false;
90+
}
91+
}
92+
}
93+
94+
Serial.println("Received starting message");
95+
96+
// Second step involves sending two things to the Anything Sensor
97+
// First, a digital HIGH is sent on the signal pin to signal receipt of message
98+
// The Anything Sensor will request data transfer. In that transfer, 0x01 must
99+
// be sent by the slave
100+
101+
Serial.println("Requesting sample data");
102+
// Write 0x01 to SPI output buffer to send to Anything Sensor
103+
requestData();
104+
105+
// Check that all of the data in the buffer is 0x00
106+
// If it is, the initialization has been completed successfully
107+
// Else, we return to the beginning of the init process to try again
108+
if(AnythingSensor::buffer[0] == 0x00 &&
109+
AnythingSensor::buffer[1] == 0x00 &&
110+
AnythingSensor::buffer[2] == 0x00 &&
111+
AnythingSensor::buffer[3] == 0x00)
112+
{
113+
AnythingSensor::dataAvailable = false;
114+
115+
initCompleted = true;
116+
Serial.println("Successfully Initialized");
117+
}
118+
}
119+
// Wait 1 second after init process
120+
delay(500);
121+
}
122+
123+
void AnythingSensor::requestData()
124+
{
125+
// Write high to signal pin to signal ready to receive data
126+
digitalWrite(signalPin, HIGH);
127+
128+
// Wait for data to be received from AnythingSensor
129+
while (!dataAvailable)
130+
{ }
131+
132+
dataAvailable = false;
133+
134+
digitalWrite(signalPin, LOW);
135+
}
136+
137+
void AnythingSensor::dataInit()
138+
{
139+
requestData();
140+
141+
Serial.print("Data initialized ");
142+
Serial.print(buffer[0]);
143+
Serial.print(" ");
144+
Serial.print(buffer[1]);
145+
Serial.print(" ");
146+
Serial.print(buffer[2]);
147+
Serial.print(" ");
148+
Serial.print(buffer[3]);
149+
Serial.print(" ");
150+
151+
// We have received the new data from the Anything Sensor
152+
// This data stores the data type and length of the data
153+
// buffer[0] - datatype: currently only float is supported
154+
// buffer[1] - length of the data
155+
// buffer[2] - Unused
156+
// buffer[3] - Unused
157+
158+
// Create a data array to store the data from the Anything Sensor
159+
dataLength = buffer[1];
160+
data = new float[dataLength];
161+
float test = 0.0;
162+
data[0] = 0;
163+
164+
// Disable SPI interrupt while we resize buffer
165+
SPCR &= ~(_BV(SPIE));
166+
167+
// Resize the buffer to the correct size for the streaming data
168+
delete[] buffer;
169+
bufferLength = 4*dataLength;
170+
buffer = new byte[bufferLength];
171+
172+
// Re-enable SPI interrupt
173+
SPCR |= _BV(SPIE);
174+
Serial.println(dataLength);
175+
}
176+
177+
bool AnythingSensor::read()
178+
{
179+
requestData();
180+
181+
memcpy(data, buffer, bufferLength);
182+
}
183+
184+
185+
void AnythingSensor::_isr()
186+
{
187+
switch(commState)
188+
{
189+
case WAITING_FOR_HEADER:
190+
{
191+
byte header = SPDR;
192+
193+
if(header == 0x10) // Heartbeat header, do nothing
194+
{ }
195+
else if(header == 0x20) // Message header, switch to receiving message
196+
{
197+
//Serial.println("Received Message header");
198+
commState = RECEIVING_MESSAGE;
199+
}
200+
else if(header == 0x30) // Config header, switch to receiving config state
201+
{
202+
// Reset heartbeat counter every time we receive config header
203+
// TODO This is a temporary, hacky solution
204+
// We are using this as a way to validate that both devices are in comms init
205+
// Currently, if the Arduino is requesting data while the Coral is in init mode,
206+
// the coral will interpret the data from the Arduino as the init sequence
207+
// Therefore, we need to have a different sequence that the Arduino sends as the heartbeat
208+
// than normal. Therefore, we make the heartbeat decrement instead of increment during the
209+
// init period
210+
// We need to add functionality such that the arduino recognizes the coral is in comm init
211+
// and resets its own comms
212+
commState = RECEIVING_CONFIG;
213+
}
214+
// else if(header == 0xFF) // Reset header, we need to reset comms with anything sensor
215+
// {
216+
// commState = INIT;
217+
// }
218+
219+
break;
220+
}
221+
default: // This is used for the INIT, RECEIVING_CONFIG, or RECEIVING_MESSAGE states
222+
{
223+
AnythingSensor::buffer[pos] = SPDR;
224+
225+
// Check if we've reached end of buffer
226+
// If at the end of buffer, we have received all of the data in this packet
227+
pos++;
228+
if(pos == bufferLength)
229+
{
230+
// State that new data available in the buffer
231+
dataAvailable = true;
232+
233+
// Reset the position in the buffer to the beginning
234+
pos = 0;
235+
236+
// We will always transition to the WAITING_FOR_HEADER STATE
237+
// after a message is received
238+
commState = WAITING_FOR_HEADER;
239+
}
240+
}
241+
}
242+
// Set the output register to the heartbeat counter and increment the heartbeat counter
243+
SPDR = heartbeatCounter;
244+
if(commState != INIT)
245+
heartbeatCounter++;
246+
else
247+
heartbeatCounter--;
248+
}
249+
250+
int AnythingSensor::length()
251+
{
252+
return dataLength;
253+
}
254+
255+
float AnythingSensor::get(int i)
256+
{
257+
return data[i];
258+
}
259+
260+
// SPI interrupt routine
261+
ISR (SPI_STC_vect)
262+
{
263+
_instance -> _isr();
264+
}
+65
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,65 @@
1+
#include "Arduino.h"
2+
//#include "AnythingSensorData.h"
3+
4+
enum SPIState
5+
{
6+
INIT,
7+
WAITING_FOR_HEADER,
8+
RECEIVING_CONFIG,
9+
RECEIVING_MESSAGE
10+
};
11+
12+
13+
class AnythingSensor
14+
{
15+
private:
16+
// Stores the sensor data collected from the Anything Sensor
17+
// The memory for this is dynamically allocated when the communication is established
18+
// because the Anything Sensor communicates the data size to the Arduino
19+
volatile byte* buffer;
20+
volatile int bufferLength;
21+
22+
// Position in the buffer the ISR is currently writing to
23+
volatile int pos;
24+
volatile bool dataAvailable;
25+
26+
volatile byte heartbeatCounter = 0xFF;
27+
28+
29+
30+
// Stores the data after it is converted to floating point
31+
float* data;
32+
33+
// Stores the length of the buffer
34+
int dataLength;
35+
36+
// This is the pin that the Arduino will send a signal to the
37+
// coral board over.
38+
int signalPin = 9;
39+
40+
volatile SPIState commState;
41+
42+
void spiInit();
43+
44+
void commInit();
45+
46+
void dataInit();
47+
48+
void requestData();
49+
50+
public:
51+
// Default constructor
52+
AnythingSensor();
53+
54+
bool begin();
55+
56+
bool end();
57+
58+
bool read();
59+
60+
int length();
61+
62+
void _isr();
63+
64+
float get(int i);
65+
};

arduino/test_spi.ino/test_spi.ino.ino

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
#include "pins_arduino.h"
2+
#include "AnythingSensor.h"
3+
4+
5+
AnythingSensor sensor = AnythingSensor();
6+
7+
void setup()
8+
{
9+
Serial.begin (9600); // debugging
10+
Serial.println("Beginning comms");
11+
12+
13+
sensor.begin();
14+
15+
}
16+
17+
// main loop - wait for flag set in interrupt routine
18+
void loop (void)
19+
{
20+
sensor.read();
21+
22+
for(int i = 0; i < sensor.length(); i++)
23+
{
24+
Serial.print(sensor.get(i));
25+
Serial.print(" ");
26+
}
27+
Serial.println(" ");
28+
delay(10);
29+
//
30+
} // end of loop

0 commit comments

Comments
 (0)