Contents
- Introduction
- Hardware
- Building the Containers
- Rebuilding the Container Images
- Setup Logging
- Running the Containers
- Running with Docker Compose and Feeding AIRFRAMES
- Validating Your Feeder
- Setting Up a Watchdog Timer (Optional)
- Setting up SNTP Time Synchronization (Optional)
- Troubleshooting
- References
- Disclaimers
This is a project to capture Aircraft Communications Addressing and Reporting System (ACARS) and VHF Data Link (VDL) messages using a Software Defined Radio (SDR) on a Raspberry Pi 4 using Docker containers and Docker Compose.
The ACARS messages are captured and decoded using the acarsdec project.
VDL message capture and decoding is using the vdlm2dec project.
Both processes use libacars for message decoding and librtlsdr for connecting to the SDR.
Each process will be compiled and a container will be created for each.
This project has been built and tested using the following equipment:
- Raspberry Pi 4 model B
- CanaKit setup for the Raspberry Pi 4 (Note: Some of these kits include the Raspberry Pi 4 model B board)
- Heatsinks
- Fan
- Power supply
- Case
- An rtl-SDR for ACARS including an antenna
- An rtl-SDR for VDL including an antenna
- 32GB or larger "high endurance" microSD card
For the rtl-SDR devices, I used the nooelec Nano 3 bundle as I had an extra one from my ADSB setup. It does not include the proper antenna for ACARS or VDL, but will work (with a shorter range). Other Software Defined Radios should work fine as long as they are rtl-SDR style.
Both the acarsdec and vdlm2dec directories contain a "build-me.sh" script and the required Dockerfile. All of the images and code required will be downloaded during the build process. Some components, such as the base image, will be shared between the two containers.
You will need to edit the Dockerfile for each, replacing the "<VDL SDR ID/serial>" and "<ACARS SDR ID/serial>" with the radio ID or serial number of each radio.
Note: The included Dockerfiles are configured to build code optimized for the Raspberry Pi 4 board. If you are using a different version of the Raspberry Pi, you may need to adjust or override the CFLAG settings being used. For example, to build for a Raspberry Pi 3, you would add the --build-arg option.
docker build --build-arg CFLAGS="-mcpu=cortex-a53+crypto -mtune=cortex-a53" -t johnsom/acarsdec-docker:0.01 .
From time to time you may want to update the containers with new versions of the software used in them. You can do this by running the build with --build-arg BUILD_DATE=<current_datetime_string> or by editing the Dockerfile and setting the BUILD_DATE value. Editing the Dockerfile will give you a record of the last time you updated the container.
docker build --build-arg BUILD_DATE=0424211030 -t johnsom/acarsdec-docker:0.01 .
If you want to rebuild the container images from scratch. You can do this by adding the "--no-cache" argument to the "docker build" command:
docker build --no-cache -t johnsom/acarsdec-docker:0.01 .
The containers will log startup information and the acarsdec container will log messages received. These messages are useful for monitoring your receiver. To limit the write wear on your microSD card, we will setup a 1GB tmpfs volume that will store the logs. This volume is stored in RAM which means it will be erased should the Raspberry Pi lose power or be rebooted. The provided install.sh script will configure the operating system to create the tmpfs volume, mount it at /var/log/containers, configure rsyslog to store the log messages in files in /var/log/containers, and enable daily log file rotation. To setup logging, run the following command:
sudo ./install.sh
Once the containers are running, the following log files will be created:
- ACARS - /var/log/containers/acars.log
- VLDM2 - /var/log/containers/vdlm2.log
The containers are setup to simply output received messages to the console. They are also configured for frequencies in the USA. You may need to change the frequencies the SDRs are listening on for your region. See the AIRFRAMES website for information on the appropriate frequencies for your area. Each can handle up to eight frequencies as long as they are within the same 2Mhz range.
To run the acarsdec container to test it out, you can run:
docker run -it --rm --privileged -v /dev/bus/usb:/dev/bus/usb johnsom/acarsdec-docker:0.01
To run the vdlm2dec container to test it out, you can run:
docker run -it --rm --privileged -v /dev/bus/usb:/dev/bus/usb johnsom/vdlm2dec-docker:0.01
When you are ready to start feeding AIRFRAMES you can use the included docker-compose.yml file to have Docker Compose manage running both containers.
Edit the docker-compose.yml file to configure your station identifier by replacing the "<your ACARS ID here>" and "<your VDL ID here>" fields. Each process should have a unique name, preferably by using the AIRFRAMES guidance. Next, configure each container to point to the appropriate SDR by replacing the "<VDL SDR ID/serial>" and "<ACARS SDR ID/serial>" with the radio ID or serial number of each radio. You may need to adjust the frequencies here as you did above for the individual containers.
The docker-compose file is already configured to start feeding AIRFRAMES.
From the directory that contains the docker-compose.yml file:
docker-compose up
Control-C can be used to exit the console.
From the directory that contains the docker-compose.yml file:
docker-compose up -d
This will also setup the containers to restart on a host reboot.
From the directory that contains the docker-compose.yml file:
docker-compose down
Once your feeder is up and running, you can visit the AIRFRAMES stations page to see how many messages have been received from each of your processes. Note: You might not receive any messages right away.
The Raspberry Pi includes a hardware watchdog device that can be used to reset the Raspberry Pi should the software freeze (such as a kernel panic).
Enable the watchdog hardware device:
- Edit the /boot/firmware/usercfg.txt
- Add "dtparam=watchdog=on". On reboot, this will enable the watchdog device.
- Edit the /boot/firmware/usercfg.txt
Install the watchdog system service:
sudo apt-get update sudo apt-get install watchdog
Configure the watchdog service:
- Edit the /etc/watchdog.conf file
- Add "watchdog-device = /dev/watchdog". This will set the location of the hardware watchdog device file.
- Add "watchdog-timeout = 15". This sets the time, in seconds, the hardware device will wait for an update before triggering a hardware reset.
- Add "max-load-1 = 24". This is the one-minute load average threshold at which the watchdog service will reboot the device. The one-minute load average is the first "load average" number when you run the "uptime" command. Twenty-four is a large number, approximately six times the load a four core Raspberry Pi can normally process.
- Add "ping = 192.0.2.1". This will cause the watchdog process to ping ip address 192.0.2.1 to make sure the networking is working correctly. I recommend using the subnget gateway address.
- Add "temperature-sensor = /sys/class/thermal/thermal_zone0/temp". This is the file where the Raspberry Pi core temperature is reported. Note, it is reported in thousandths of a degree Celsius.
- Add "max-temperature = 82". This sets the watchdog service temperature threshold to eighty-two degrees Celsius. This is the temperature the Raspberry Pi will start throttling the CPU.
- Add "min-memory = 3815". This sets a minimum available memory threshold for the watchdog process. A value of thirty-eight hundred fifteen will set a low memory threshold of two hundred fifty megabytes of available memory.
- Edit the /etc/watchdog.conf file
watchdog-device = /dev/watchdog
watchdog-timeout = 15
max-load-1 = 24
ping = 192.0.2.1
temperature-sensor = /sys/class/thermal/thermal_zone0/temp
max-temperature = 82
min-memory = 3815
Enable the watchdog service:
sudo systemctl enable watchdog
Reboot the Raspberry Pi to enable the watchdog device:
sudo reboot
Verify the watchdog service started successfully:
sudo systemctl status watchdog | less
The output should show that the service is active (running).
It is nice to have accurate timestamps for the messages received. Since the Raspberry Pi does not include a Real Time Clock (RTC), we would like to have a bootstrap timestamp to get the clock set early. To accomplish this you can use the systemd-timesyncd service. This will sync the clock using Simple Network Time Protocol (SNTP) and write out a time file that will be used on boot. As the clock stabilizes, it will increase the interval of the time synchronization and file update. The default maximum poll interval is thirty-four minutes, so the write wear from this bootstrap time file is low.
To configure systemd-timesyncd follow these steps:
Edit /etc/systemd/timesyncd.conf if you have preferred NTP server:
[Time] NTP=192.0.2.1
Make sure the service is enabled:
sudo systemctl enable systemd-timesyncd
Restart the servie:
sudo systemctl restart systemd-timesyncd
The systemd-timesyncd service can be monitored with the "timedatectl" command.
timedatectl timesync-status
If you need to get inside one of the containers, you can run the following commands:
docker ps
docker exec -it <container_id_or_name> bash
The "docker ps" command will list the containers running on your host. The "docker exec" command will launch a bash shell inside the container.
- Raspberry Pi is a trademark of the Raspberry Pi Foundation
- CanaKit is a registered trademark of CanaKit Corporation
- Nooelec is a registered trademark of Nooelec Inc.
- I did not get compensation from any of these companies for this project.
- This document comes without any warranty of any kind.
- Not intended for safety of life applications.
- The code provided in this repository is licensed under the GNU General Public License v3.0. See the included LICENSE for terms.
- This document is Copyright 2020 Michael Johnson
- This document is licensed under the Creative Commons Attribution-ShareAlike 4.0 International Public License
Capture ACARS and VDL messages using an SDR and Docker by Michael Johnson is licensed under a Creative Commons Attribution-ShareAlike 4.0 International License.