Reflecta is a communications library for Arduino that enables remote calling of functions, sending & receiving frames of byte data or string messages, and receiving high speed 'heartbeats' of sensor data.
![Gitter](https://badges.gitter.im/Join Chat.svg)
To install, download the contents of the repository as an archive file and decompress into your 'sketchbook\libraries' folder (on Windows this is My Documents\arduino\libraries, on Linux this is ~\sketchbook\libraries). When you are done you should have a file like My Documents\arduino\libraries\Reflecta\Reflecta.h. Now you can restart the Arduino IDE and use Sketch -> Import Library.
Examples
StandardReflecta -- opens a Reflecta listener and exposes the Arduino Core (ardu1) functions to be called. This is the 'Arduino side' of the conversation.
node-reflecta -- a NodeJS library that will talk to StandardReflecta and call Arduino functions such as digitalWrite. See simple.js sample as an example.
Arduino easily connects to a PC over USB as a Virtual COM port but turning this serial data stream into a reliable communications channel takes considerable work. Firmata is a comparable library but it has some limitations:
-
Firmata is based on MIDI and uses a '7 bit data' design. The 8th bit is reserved for 'commands' so all raw data sent over the wire has to be converted to/from 7 bits.
-
Firmata is difficult to extend with new methods due to its SYSEX design and 7-bitness.
-
Firmata doesn't detect corrupt or lost data. Testing using USB Virtual COM with two-chip (USB chip -> UART -> MCU chip) boards like the UNO detected frequent data corruption at speeds over 9600 baud. RF communications like Bluetooth or Zigbee need to be able to detect data loss and corruption in order to modify their sending behavior based on changing physical conditions such as antenna blockage or distance.
The goal is to develop a protocol that fixes these issues, specifically:
-
Use 'escaping' to delineate frames so we don't need to convert payload data to 7 bits.
-
Add Sequence and Checksum to detect data loss and corruption.
-
Make it easy to extend the protocol with new functions (bind) and discover what functions are implemented (queryinterface). Minimize the amount of work needed for a library to be exposed for remote calling.
-
Design for CPU and communications efficiency in order to take best advantage of limited microcontroller resources.
Reflecta is four Arduino libraries and one NodeJS client:
-
ReflectaFramesSerial packages byte[] data into frames over a stream, adds Sequence to detect lost frames, and adds Checksum to detect data corruption. Uses the Arduino Serial library for communications, a future Raw Hid implementation is planned.
-
ReflectaFunctions is a remote function calling protocol that builds on top of ReflectaFrames. Arduino functions are registered by calling bind(interfaceId, function pointer).
ReflectaFunctions uses a stack-based approach to calling functions. push() parameters on the stack, invoke function(s), functions pop() their parameters off the stack and push() their return values back on.
ReflectaFunctions exposes bind() and queryInterface() to determine which interfaces (e.g. function groups) are on the Arduino.
-
ReflectaArduinoCore is a binding of the Arduino core library functions such as pinMode, digitalRead, analogWrite to the 'ARDU1' interface.
-
ReflectaHeartbeat is a library for reading digital and analog pins very efficienctly, calling functions and gathering their results into a data packet, and sending the results at a fixed frequency to the host PC. ReflectaHeartbeat is optimized to quickly gather data off the Arduino while effectively sharing the CPU by using asynchronous polling inside loop() rather than delay().
-
node-reflecta is a NodeJSnpm library built on top of node-serialport. node-reflecta uses ReflectaFunctions to queryInterface and dynamically load javascript objects for the libraries installed on the Arduino using npm.
Design Stability: High
After reviewing existing technologies, the approach settled on is:
- Start with the STK500 protocol from Atmel which has MESSAGE_START, SEQUENCE_NUMBER, and a simple 8 bit xor CHECKSUM.
- STK500 doesn't escape the data, so substitute SLIP framing for the STK500 message start/message size design. SLIP is very simple to understand and code.
SLIP (Serial Line IP) is an IETF standard developed to send network packets over a serial line. SLIP defines two special characters, END (0xC0) and ESCAPE (0xDB), that must be escaped if they are found in the payload data. The pseudocode for SLIP encoding is:
for each byte in payload[]
if byte == ESCAPE (0xDB) write ESCAPE (0xDB) + ESCAPED_ESCAPE (0xDD)
else if byte == END (0xC0) write ESCAPE (0xDB) + ESCAPED_END (0xDC)
else write byte
write END (0xC0)
Over the wire, a frame of data looks like:
Sequence Payload[] Checksum END
Sequence is a byte that increments on each frame sent and rolls over from 255 back to 0.
Payload[] is the data bytes you want to transfer.
Checksum is calculated using XOR ( ^= ) on each unescaped byte of Sequence and Payload[]. Checksum is validated by calling XOR on each byte of the incoming frame such that when the END character is reached, the current value of the calculated checksum should be zero because the frame's checksum just XORed with itself.
This compares well to STK500 and Firmata in efficiency (only 3 byte overhead per message) and ease of calculation. The length of payload is inferred from the length between END characters rather than encoded into the frame.
- 0.3.x: Beta release of Reflecta protocol
RocketBot is a robot built around a Parallax Eddie chassis that adds a pneumatic straw rocket launcher and blinky lights. It was created for Maker Faire 2012 in San Mateo CA and Seattle WA.
See this Trello Board for planned work.
Jay Beavers' blog on Reflecta and RocketBot.