This guide explains how to transform a fresh installation of Raspberry Pi OS into a stratum‑1 NTP server. The project leverages a GPIO‑based GPS module with PPS, the PTP hardware timestamping capability of the Ethernet interface, Chrony for NTP synchronization, and Telegraf with InfluxDB for monitoring.
1. Overview
In this project you’ll build a stratum‑1 NTP server using:
- Raspberry Pi 5 as the host.
- A GPS module (via the serial UART port) for an absolute time reference.
- A PPS (Pulse Per Second) signal from the same GPS for sub-microsecond timing accuracy.
- The PTP hardware timestamping on the Ethernet interface for improved precision on PTP packets.
- Chrony for NTP time synchronization.
- gpsd to interface with your GPS receiver.
- Telegraf for monitoring various metrics (including Chrony statistics) sent to an InfluxDB backend.
2. Prerequisites and Required Hardware
Before you begin, ensure you have the following:
- Raspberry Pi 5 running a fresh install of Raspberry Pi OS.
- A GPS module with serial (UART) output and PPS capability.
- An external GPIO connection for the PPS signal (typically connected to a dedicated GPIO pin, e.g. GPIO4).
- An Ethernet interface with PTP hardware timestamping support.
- A monitoring server running InfluxDB (or you can install InfluxDB locally if desired).
3. Software Packages to Install
For a fully functional stratum‑1 NTP server, you’ll need to install these packages:
- chrony – Provides the NTP daemon with support for hardware timestamping and refclock interfaces.
- gpsd – Daemon to interact with your GPS device.
- gpsd-clients – Utilities to query the GPS data.
- telegraf – For gathering system and application metrics and sending them to InfluxDB.
- ethtool – For inspecting and verifying hardware timestamping support on the Ethernet interface.
- (Optionally) pps-tools – To assist with troubleshooting PPS signals.
Install these packages using:
sudo apt update
sudo apt install chrony gpsd gpsd-clients telegraf ethtool pps-tools
4. Kernel and Firmware Configuration
To properly use the GPIO PPS and enable the UART for your GPS, update your boot configuration and load the required kernel modules.
4.1. Modify the Boot Configuration
Edit the /boot/firmware/config.txt
file to enable the UART and configure the PPS overlay (using GPIO4):
[all]
dtparam=uart0=on
dtoverlay=pps-gpio,gpiopin=4
dtoverlay=disable-wifi
4.2. Ensure Kernel Modules Load at Boot
Edit /etc/modules
to load the necessary kernel modules:
#/etc/modules: kernel modules to load at boot time.
pps-gpio
i2c-dev
This configuration ensures the PPS GPIO module and I2C support (if needed) are loaded during boot.
5. Configuring gpsd
Configure gpsd
to start at boot and use the correct devices. Edit /etc/default/gpsd
as follows:
# Devices gpsd should collect to at boot time.
DEVICES="/dev/ttyAMA0 /dev/pps0"
# Other options you want to pass to gpsd
GPSD_OPTIONS="-n"
# Automatically hot add/remove USB GPS devices via gpsdctl
USBAUTO="true"
This configuration tells gpsd
to open the UART device (for NMEA data from the GPS) and the PPS device for precise pulse timing.
6. Configuring Chrony
Chrony will serve as your NTP daemon. Its configuration utilizes the GPS and PPS refclocks and leverages PTP hardware timestamping on the Ethernet interface.
Edit /etc/chrony/chrony.conf
to include:
confdir /etc/chrony/conf.d
server 0.us.pool.ntp.org iburst
server 1.us.pool.ntp.org iburst
server 2.us.pool.ntp.org iburst
server 3.us.pool.ntp.org iburst
keyfile /etc/chrony/chrony.keys
driftfile /var/lib/chrony/chrony.drift
ntsdumpdir /var/lib/chrony
#log tracking measurements statistics
logdir /var/log/chrony
maxupdateskew 100.0
rtcsync
makestep 1 3
leapsectz right/UTC
refclock SOCK /run/chrony.ttyAMA0.sock refid GPS precision 1e-3 noselect
refclock SOCK /run/chrony.pps0.sock refid PPS precision 1e-7 lock GPS
allow 192.168.109.0/24
allow fd79:605:8025:a109::/64
ptpport 319
hwtimestamp *
Key Points:
- Refclock SOCK entries: These specify the Unix sockets where Chrony receives data from gpsd (for the serial GPS) and the PPS signal.
- PTP support: The
ptpport 319
andhwtimestamp *
lines enable PTP hardware timestamping, which improves timing accuracy. - Network access: The
allow
lines configure which subnets may query your NTP server.
7. Verifying PTP Hardware Timestamping
To verify that your Ethernet interface supports hardware timestamping, run:
ethtool -T eth0
The output should indicate support for hardware transmit/receive timestamps and include details similar to:
PTP Hardware Clock: 0
Hardware Transmit Timestamp Modes:
off
on
onestep-sync
Hardware Receive Filter Modes:
none
all
This confirms that your interface is correctly configured for precise timestamping.
8. Configuring Telegraf for Monitoring
Telegraf collects and sends system metrics to your InfluxDB instance. Edit /etc/telegraf/telegraf.conf
to monitor Chrony statistics, GPS data, and general system metrics:
## Minimal telegraf.conf for monitoring a Raspberry Pi with gpsd, chrony, and full system metrics
[agent]
interval = "10s"
round_interval = true
metric_batch_size = 1000
metric_buffer_limit = 10000
collection_jitter = "0s"
flush_interval = "10s"
flush_jitter = "0s"
debug = false
quiet = false
[[outputs.influxdb]]
## URL of your InfluxDB instance using HTTPS
urls = ["https://192.168.109.7:8086"]
## Ignore certificate chain and host verification (not recommended for production)
insecure_skip_verify = true
## The target database for metrics.
database = "ntpserver"
retention_policy = ""
write_consistency = "any"
timeout = "5s"
## Uncomment and set credentials if authentication is required.
username = "ntp"
password = "022510"
## Plugin for Chrony time synchronization data
[[inputs.chrony]]
server = "udp://localhost:323"
timeout = "5s"
metrics = ["tracking", "activity", "serverstats", "sources", "sourcestats"]
socket_group = "chrony"
socket_perms = "0660"
## CPU metrics: monitors CPU usage across cores and total CPU usage.
[[inputs.cpu]]
percpu = true
totalcpu = true
fielddrop = ["time_*"]
## Memory metrics: tracks usage, free memory, and swap.
[[inputs.mem]]
## Disk metrics: monitors disk usage; ignore temporary filesystems.
[[inputs.disk]]
ignore_fs = ["tmpfs", "devtmpfs"]
## Disk I/O metrics: measures read/write activity.
[[inputs.diskio]]
## Network metrics: captures network interface statistics.
[[inputs.net]]
## System metrics: collects system-level stats such as uptime and load.
[[inputs.system]]
## Optional: Raspberry Pi specific temperature metric.
[[inputs.exec]]
commands = ["vcgencmd measure_temp"]
name_override = "rpi_temp"
interval = "10s"
timeout = "5s"
data_format = "value"
data_type = "float"
9. Testing and Verification
After rebooting your Pi to apply all changes, verify that everything is working correctly.
9.1. Check Chrony Status
Run the following command to check Chrony’s synchronization status:
chronyc tracking
A sample output might look like:
Reference ID : 50505300 (PPS)
Stratum : 1
Ref time (UTC) : Sat Mar 29 05:35:12 2025
System time : 0.000000086 seconds fast of NTP time
Last offset : +0.000000175 seconds
RMS offset : 0.000000729 seconds
Frequency : 1.945 ppm slow
Residual freq : +0.000 ppm
Skew : 0.012 ppm
Root delay : 0.000000001 seconds
Root dispersion : 0.000019379 seconds
Update interval : 16.0 seconds
Leap status : Normal
This output indicates that Chrony is synchronizing correctly with your GPS/PPS source.
9.2. Verify NTP Sources
To view the status of the NTP sources, run:
chronyc sources -v
This command displays the current state of all NTP sources, including the GPS and PPS inputs along with fallback servers.
10. Configuring PTP as a Grandmaster Clock
To use your Raspberry Pi as a PTP grandmaster clock, you’ll need to run the PTP daemon (ptp4l) in master mode so it broadcasts precise timing over the network. Follow these steps:
10.1. Install Linux PTP
Make sure the Linux PTP package is installed:
sudo apt update
sudo apt install linuxptp
10.2. Create a PTP Configuration File
Create a configuration file for ptp4l (for example, /etc/linuxptp/ptp4l.conf
) with the following content:
[global]
masterOnly 1
priority1 128
priority2 128
logAnnounceInterval 1
logSyncInterval 0
logMinDelayReqInterval 0
time_stamping hardware
Key Points:
- masterOnly 1: Forces the PTP daemon to run in grandmaster mode.
- time_stamping hardware: Uses the Ethernet interface’s hardware timestamping for best accuracy.
- The priority settings help determine your device’s preference as a master if multiple grandmasters exist.
10.3. Start ptp4l in Grandmaster Mode
You can test the configuration by running:
sudo ptp4l -f /etc/linuxptp/ptp4l.conf -i eth0 -m
Replace eth0
with the name of your Ethernet interface if different.
10.4. (Optional) Create Systemd Service Files
To ensure ptp4l starts automatically at boot as a grandmaster, create a systemd service file.
Create the ptp4l Service File
Create /etc/systemd/system/ptp4l.service
with the following content:
[Unit]
Description=PTP Grandmaster Service
After=network.target
[Service]
Type=simple
ExecStart=/usr/sbin/ptp4l -f /etc/linuxptp/ptp4l.conf -i eth0 -m
Restart=always
RestartSec=5
[Install]
WantedBy=multi-user.target
Reload the systemd daemon and enable the service:
sudo systemctl daemon-reload
sudo systemctl enable ptp4l.service
sudo systemctl start ptp4l.service
11. Verifying the PTP Grandmaster Status
After configuring ptp4l in master mode, you can verify that your Raspberry Pi is functioning as the PTP grandmaster.
11.1. Use the PMC Tool
The pmc
utility helps check PTP status. Run:
sudo pmc -u -b 0 'GET CURRENT_DATA_SET'
A successful output might look similar to:
stepsRemoved 0
offsetFromMaster 0
meanPathDelay 2000.0
- stepsRemoved: Should be 0, indicating the device is not acting as a slave.
- offsetFromMaster: Near 0 when operating as a grandmaster.
- meanPathDelay: Shows the average network delay measured.
11.2. Monitor ptp4l Logs
Check the service logs for any errors:
sudo journalctl -u ptp4l.service -f
This helps ensure that ptp4l is correctly broadcasting time over your network.
12. Conclusion
In this guide, you learned how to build stratum‑1 NTP server by:
- Installing essential packages like chrony, gpsd, telegraf, and others.
- Configuring the kernel and boot settings to enable GPS and PPS support.
- Setting up
gpsd
for interfacing with the GPS module. - Configuring Chrony to use both GPS and PPS for time synchronization, with the additional benefit of PTP hardware timestamping.
- Using Telegraf to monitor system and synchronization metrics, sending data to an InfluxDB instance.
- NTP Service: Using Chrony to provide precise NTP time synchronization with GPS/PPS support.
- PTP Grandmaster: Running ptp4l in master mode to offer precise time distribution via PTP on the local network.
By following these steps, you can build a highly accurate time server that not only ensures precise network synchronization but also offers detailed monitoring of its performance.
Happy time syncing!