This is an arduino library which enables you to use a simple laser or LED to transmit characters between two arduinos (transmitter arduino with the laser to receiver arduino with a photodiode). Each byte of data is encoded (but not encrypted) to add some robustness to noise during the transmission and then modulated. The pin connection setup is explained in the example skethes included in the library. I originally wrote this as part of a netduino robotics project for uni, but I've since modified it to work with an arduino. It took a lot of work but I'd like to share this valuable resource with you, if you would like to support my work please consider donating (https://www.paypal.me/HobbyTransform) and purchasing the material for your project through the ebay affiliate links to help fund other similar projects.
The materials can be found on eBay,
- 2 x arduinos with USB cables
- 2 x computers, one per arduino
- male to male and male to female jumper leads
- 1 x KY-008 laser module or a visible light LED
- 1 x laser sensor module, non-modulated
- 1 x mini breadboard for photo-receiver / photodiode
- download this library and add it to your libraries folder in the arduino folders on your computers
- open the arduino application
- open File -> Examples -> HT_LumenWire and select either the photoreceiver or transmitter examples
- plug in the arduino and select the appropriate COM port, then upload the transmitter / receiver sketches to their respective arduinos
- connect the laser as follows: laser S pin to pin 6, - pin to ground
- connect the photodiode as follows: photodiode out pin to pin 7, power pins as normal
- open the serial window (spyglass symbol in the IDE) on both computers
- point the laser or LED to the photodiode
- enter a character or phrase in the serial window on the computer doing the transmitting and press enter to send
- you should receive the message on the receiver serial window automatically
- each byte of data (in ASCII: D7 D6 D5...D1 D0, e.g. D3 is 3rd digit of the byte) typed into the serial window is split into its most significant and least significant 4 bits, AKA nibbles: D7...D4 and D3...D0
- each nibble is encoded via hamming encoding, producing an unsigned 8 bit integer. The extra 4 bits contain information about the original nibble to help detect errors and recover the original nibble in case some of the data was scrambled due to noise in the transmission. The encoding / decoding algorithm has its limits on how much unscrambling it can do but it is certainly more efficient than sending multiple times to improve accuracy.
- the encoding is as follows: y = x G, where x = (D0 D1 D2 D3) and G is a Hamming(7,4) matrix, producing y = (H0, H1, H2, D0, D1, D2, D3) and then we add an additional bit of even parity at the front y' = (P0, H0, H1, H2, D0, D1, D2, D3) where the parity bit P0 is just the XOR sum of the other bits.
- the two encoded nibbles (now 8 bits each) are concatenated, least significant byte first to product a 16-bit integer
- the 16-bit integer is then split into is least and most significant bytes
- each byte is modulated via manchester modulation: 2 start bits to let the receiver know it is about to receive actual data and not just stray signals, then the byte followed by 1 stop bit, producing 11 bits. Each of those 11 bits is represented with two half-bits: first half-bit is the original XOR'ed with 1 and the second is XOR'ed with 0 instead, this adds a clock pulse into the signal. This produces 22 bits for each byte.
- the pair of 22 bits are joined end on end, least significant first, producing 44 bits of data
The 44-bit data is sent via the laser or LED, 1's are on and 0's are off
The 44-bit signal is received by the photo-detector and manchester-demodulated (reverse algorithm of modulation) to reconstruct the 16-bit encoded signal
The 16-bit demodulated signal is then hamming decoded to reproduce the original byte
The byte is then printed on the serial window
- For a full list of included functions in the library, please refer to the .cpp and .h files.
- You can also set the bitrate via the .set_speed(...) function. It must be a minimum of 500 bits/sec but no more than the laser slew rate. You will need to set the speed manually into both the transmitter and receiver codes.
- I plan to modify the code so that you only need to set the speed on the transmitter end and the receiver will automatically calculate the bitrate based on the interval between received bits.
- I plan to make this project work on a micro-controller with a higher clock rate than the arduino, like a raspberry pi