Visuelle Lichtkommunikation

Sending text messages with visible light communication or LiFi – Pi to Arduino

I like the idea of sending data via visible light. In general, Li-Fi (Light Fidelity) and VLC (Visible Light Communication) are heavily discussed at the moment. VLC is a visible light communication, which means it is primarily a cable replacement, while LiFi is a similarity to WiFi. A LiFi device would be a kind of Internet-enabled router that transmits data via light signals instead of radio frequency signals. Currently, however, no company has developed or is selling a Li-Fi router. VLC technology is used prominently at the moment by Elon Musk’s StarLink project. This allows satellites to exchange data using laser beams. I have replicated that in a simplified way in this blog article.

At first, I only want to transmit individual text messages in one direction. In a next step this can then be extended to transmit whole files bidirectionally. To receive the messages, I chose an Ardunio, because unlike the Raspberry Pi, it has analog inputs and can thus be easily connected to analog sensors. For sending, a Raspberry Pi is suitable, since only digital pins are needed here and a Pi can generally transmit data more efficiently and faster. This has already been tested in the previous posts about defining a clock frequency for the Arduino and the Raspberry Pi. All scripts to run the VLC communication can be downloaded at the end of the article. The following three blogposts are directly related to this project:


How to program the clock frequency of the Arduino: Use timer interrupts as clock frequency.

How to program the clock frequency of the Pi: Precise timer function in C for the Pi.

How to prevent errors in data transmission: Using CRC to detect corrupted packets.

List of components

  • Arduino Uno
  • Raspberry Pi
  • 5V Solarcell
  • 5V Laserdiode
  • Jumper cables

Wiring for the visible light communication

Textnachrichten VLC Fritzing Verkabelung für Visuelle Lichtkommunikation

The wiring is rather simple. The 5V solar cell is very well suited to detect light signals. Therefore the solar cell is connected to the receiver, which is the Arduino. The ground of the solar cell is connected to a ground of the Arduino. The positive line of the solar cell is connected to an analog pin of the Arduino. Here, for example, A0 is suitable.

From the Raspberry Pi the messages are transmitted to the Arduino via light signals of the laser. For this, the ground of a 5V laser diode is connected to a ground pin of the Pi (see graphic). Then the positive line of the laser diode is connected to one of the digital pins of the Raspberry Pi. Here I chose the GPIO17 pin, which corresponds to pin 0 in the “wiringPi” library (see graphic).

VLC Textnachrichten visuelle Lichtkommunikation wiringPi

As an alternative to the laser and the solar cell combination, an LED and a photoresistor can also be used. However, a laser is better suited for faster and more precise data transmission. In addition, a solar cell offers a large area to detect the laser beams.

Modulationtechnique On-Off-Keying

The text messages are transmitted in binary code as “1” or “0”. There are different ways to modulate this data. One of the simplest possibilities is to set an identical clock frequency for the transmitter and receiver. Then either a “1” or a “0” is transmitted in each clock pulse.

The procedure is called aplitude shift keying. If the laser diode illuminates the solar plate particularly strongly, this is detected by the receiver as a binary “1”. If the laser diode shines only weakly, it will be detected as binary “0”.

In fact it makes sense not to let the laser diode shine at all for a binary “0”. This is called On-Off keying. This is a simplification of amplitude shift keying. The following graphic illustrates the difference between amplitude shift keying and on-off keying.

Modulationstechnik Visuelle Lichtkommunikation

That means, whenever the solar cell detects light, a corresponding voltage value is passed to the analog pin of the Arduino. If this voltage value exceeds a predefined value, the Arduino registers it as a binary “1”, otherwise as a binary “0”. It makes sense to adjust the predefined value to the daylight. Possibly with the help of an additional light sensor.

Arduino code to receive messages

//This is the "real" loop function
  switch (state)
    case 0:
      //looking for synchronization sequence

      if (synchro_Done== true)
    case 1:
      //receive Data
      receiveData_Done =false;

      if (receiveData_Done==true)

The software of the Arduino is basically built as a state machine. There are two states. One state for synchronization and one for reading the text message. In the state synchronization the receiver waits for a fixed bit sequence (preamble) for example “101010101111111111”. This sequence means that the receiver must listen now because a text message follows. The complete Arduino code can be downloaded at the end of the article. The following graphic shows the exact structure of the data packets.

Aufbau Textnachrichten VLC

As soon as the preamble is recognized, the receiver automatically switches to the second state and receives the actual message. However, the first 16 bits of the text message correspond to a decimal number, which tells the receiver how many characters the incoming text message contains. When this number is reached, the text is printed in the Serial Monitor of the Arduino and the state changes back again. Now the synchronization sequence is awaited again. The complete software code can be downloaded at the end of the blog article. To improve the transmission quality, a cyclic redundancy check (CRC) can be performed as shown in the following graphic.

CRC Datenpakete Aufbau VLC

Raspberry Pi code to send messages

//Read message
        char msg[3000]; 
        int len, k, length;
        printf("n Enter the Message: ");
        int2bin(len*8, 16); //len*8, because 8 bits are one byte

The program code of the Raspberry Pi must be written in C. Python would be too slow and would not achieve a stable clock frequency for sending the data. The complete program code can be downloaded at the end of the article. In order for the Pi to control the laser diode the “wiringPi.h” library is needed. With “digitalWrite(0, HIGH)” the diode can be switched on and with “digitalWrite(0, LOW)” it can be switched off. Important: Please don’t forget the “wiringPi” library when compiling! The command is: “gcc -o simpleLaser simpleLaser.c -lwiringPi”.

At the beginning the program asks for a text message using the printf function. This is read in and stored by a scanf function. Then each letter is converted to binary code and stored in an array. The conversion is shown in the following graphic. At the end, the binary code is transmitted from the array in a fixed clock pulse. The laser diode is switched on for a binary “1” and off again for a binary “0”.

Character to binary conversion

Conclusion on the Visible Light Communication for Arduino

The transmission of text messages by means of visual light communication works excellently. Thanks to the cyclic redundancy check, it is also absolutely error-free. However, the benefit of one-dimensional message communication is rather low. It would be better if any kind of files could be transmitted and that in both directions. An update will probably follow soon.

Set Up VLC Arduino Pi Visuelle Lichtkommunikation

Download files


9 thoughts on “Sending text messages with visible light communication or LiFi – Pi to Arduino

  1. Hi, I am trying to create the same thing but with my own code but that code is working with LDR Sensor and not with 5V solar cell please would you tell me which solar cell you used in this project because my solar cell is not able to detect laser but it is detecting flashlight (I have to put it too close to solar cell to detect it)

      1. There is nothing mention as polycrystalline or monocrystalline. I bought it from local shop so please would you suggest me which one should i use? and please give me any reference if possible.

  2. When I run the program for the simple laser, i keep getting undefined reference to ‘wiringpisetup’. Although i have already installed the wiring pi package on the raspberry.
    Any Help?

      1. Can you guide me how to do it?

        What have i done is that i went to Build -> Set Build Commands->Under the build command i wrote “gcc -o simpleLaser simpleLaser.c -lwiringPi”, and in working directory i wrote -lwiringPi

        I keep getting “Failed to change to the Working Directory”

        Any help?

Leave a Reply

Your email address will not be published. Required fields are marked *

Cookie Consent with Real Cookie Banner