How to interface a GPS/GNSS module with Ouster Lidar

How to interface a GPS/GNSS module with Ouster Lidar

This article provides instructions on setting up, wiring, and synchronizing with a GPS/GNSS device (NEO-M9N) and Ouster lidar sensor.

 

 


Required Items

Hardware Needed

  1. Ouster Sensor: OSx-[32, 64, 128] with included Interface Box (IBOX)

  2. GPS Board: u-center compatible GPS (below are the tested ones)

  3. PC: Ensure the PC has 1000Base-T (Gigabit Ethernet) capability (via ethernet port on PC or separate USB-Ethernet converter).

  4. Miscellaneous:

    • Required: Ethernet cable, USB-C cable, M3 torx screwdriver, jumper wires (male and female)

    • Optional: Breadboard (or plugblock), oscilloscope

Note: This specific SparkFun product requires an antenna (sold separately). Be sure to pick a suitable antenna with U.FL connector for this exercise.

 

Required Software

u-center

  • u-center GNSS evaluation software for Windows only [freeware]

The u-center version utilized for the creation of this article: u-center, v22.07

PyGPSClient

Compatible with MacOS, Linux, and Windows

https://github.com/semuconsulting/PyGPSClient

Installation Guide

  1. Create Virtual Environment
    python3 -m venv venv && source venv/bin/activate

  2. Install
    pip install --upgrade pygpsclient

  3. Run (a GUI window should pop up if successful)
    pygpsclient

[Optional] Desktop Launcher Installation: https://github.com/semuconsulting/PyGPSClient?tab=readme-ov-file#creating-a-desktop-application-launcher

Installation Troubleshooting

$ pygpsclient --help Traceback (most recent call last): File "/Users/dawheeler/ouster/venv/bin/pygpsclient", line 5, in <module> from pygpsclient.__main__ import main File "/Users/dawheeler/ouster/venv/lib/python3.13/site-packages/pygpsclient/__main__.py", line 15, in <module> from tkinter import Tk File "/usr/local/Cellar/python@3.13/3.13.1/Frameworks/Python.framework/Versions/3.13/lib/python3.13/tkinter/__init__.py", line 38, in <module> import _tkinter # If this fails your Python may not be configured for Tk ^^^^^^^^^^^^^^^ ModuleNotFoundError: No module named '_tkinter'

For MacOS, I installed Tk via brew to resolve it

brew install python-tk

Sensor and GPS/GNSS Configuration

Ouster Sensor Configuration via Web Interface

The Ouster sensor must be configured to accept externally generated, electrical signals as ‘inputs’ for both PPS and NMEA. There are many ways to configure the sensor (e.g. via TCP/HTTP api), however the Ouster Web Interface is probably the most simple and effective method.

See our documentation on Web Interface for further details.

Tip: to access the web interface, enter one of the following arguments into a web browser url bar.

  1. http://os-991234567890.local/

    1. where 991234567890 represents the unique serial number of your device.

      1. This can be found on the sensor label or GET /api/v1/sensor/metadata/sensor_info

  2. http://169.254.1.100/

    1. where 169.254.1.100 represents the set IPv4 address of the sensor

      1. Sensor IPv4: GET /api/v1/system/network/ipv4

From the Configuration tab, ensure the following items are configured as described. All other fields can remain as the default values.

Sensor configuration via Web UI

1

Timestamp Mode: TIME_FROM_SYNC_PULSE_IN

2

Multipurpose IO Mode: INPUT_NMEA_UART

3

NMEA In Polarity: ACTIVE_HIGH

4

NMEA Baud Rate: BAUD_9600

5

NMEA Leap Seconds: 0

6

Sync Pulse In Polarity: ACTIVE_HIGH

7

Click Apply Config (reinit)

8

Click Persist Active Config

Tip: If nmea_leap_seconds is set to 0 then the reported time is Unix Epoch time.

If nmea_leap_seconds is 37 then the reported time matches the TAI standard

 

GPS/GNSS Configuration with u-center

NEO-M9N Configuration Settings for use with Ouster sensor

It is important to ensure you have configured the GPS/GNSS device to be compatible with the Ouster sensor. For details on initial installation of https://www.u-blox.com/en/product/u-center, connecting the GPS receiver, and other introductory features, please refer to the user guide by visiting the manufacturer site.

Getting started:

  1. Open the Configuration View (Ctrl+F9)

After making a configuration change, write the change to the module by using the Send button at the bottom of the window.

List of configuration topics to be changed:

  1. Save (BBR/Flash)

  2. NMEA Protocol

  3. UART Out

  4. PPS Out

  5. NMEA Message Types

1. Save (BBR/Flash)

1

Highlight CFG (Configuration)

2

Select Save current configuration

3

Send the message to the receiver

CFG (Configuration)

 


2. Configure NMEA Protocol

1

Highlight NMEA (NMEA Protocol)

2

Select CFG-NMEA-DATA2

3

Number used for SV’s not supported by NMEA: 1-Extended (3 digit)

4

Main Talker ID: 1-GP (GPS)

5

Send the message to the receiver

NMEA (NMEA Protocol)

3. Configure UART Out

1

Highlight PRT (Ports)

2

Target: 1-UART1

3

Baudrate: 9600

4

Send the message to the receiver

PRT (Ports)

 


  1. Configure PPS Out

1

Highlight TP5 (Timepulse 5)

2

Timepulse Settings: 0-TIMEPULSE

3

Check the radio button: Frequency

4

Frequency: 1 [Hz]

5

Length: 10000 [us]

6

Length Locked: 10000 [us]

7

Select 0-UTC Time

8

Send the message to the receiver

TP5 (Timepulse)

 

  1. Configure NMEA Message Types

1

Highlight MSG (Messages)

2

Timepulse Settings: 0-TIMEPULSE

3

For each message type below, uncheck the box (OFF) for UART1 only.

F0-00 NMEA GxGGA

F0-01 NMEA GxGLL

F0-02 NMEA GxGSA

F0-03 NMEA GxGSV

F0-05 NMEA GxVTG

4

Ensure that UART1 checkbox is selected (ON) for the RMC message type:

F0-04 NMEA GxRMC

5

Send the message to the receiver after each step 3a, 3b, ….4a.

MSG (Messages)

Ouster sensors only accept NMEA sentence type: GPRMC

With NEO-M9N, there are several other NMEA sentence types enabled by default. And in this example, the interface to send these messages is over UART. Therefore, ensure that GxRMC sentences are enabled (ON) for interface UART1 and the others are disabled (OFF).

 

GPS/GNSS Configuration with PyGPS Client

Connect the device to your computer before launching the software

Connecting to the GPS device:

 

Configuring with a ubx Configuration File:

This file sets the sensor via ublox command to 1 hz message rate, disable all NMEA sentences, and enable RMC sentence

  1. Click UBX config and a new window should appear

    image-20250129-155049.png
  2. At the top left, click Load CFG Configuration dialog

    image-20250129-155410.png
  3. Select the desired UBX file

  4. Click the green arrow to send it to the GPS device

    image-20250129-155450.png

     

Configuring via the GUI

  1. Click UBX config and a new window should appear

    image-20250129-155049.png
  2. Set Solution Interval (ms) (under CFG-RATE Navigation Solution Rate Configuration) to 1000 and click the green arrow to send it to the GPS device

    image-20250129-155642.png
  3. Configure PPS

    1. Note: I couldn’t get my oscilloscope to get a reading but my sensor was able to achieve a sync pulse lock with the default GPS (Goouuu Tech GT-U7 GPS Module) settings. This should be the equivalent of the u-center PPS instructions

      image-20250129-174341.png

       

    2. Attribute | Value ------------------------ tpIdx | 0 version | 1 freqPeriod | 1 freqPeriodLock | 1 pulseLenRatio | 10000 pulseLenLock | 10000 active | 1 lockGnssFreq | 1 isFreq | 1 isLength | 1 alignToTow | 1 polarity | 1 gridUtcGnss | 0

      https://content.u-blox.com/sites/default/files/u-blox-F9-HPG-1.30_InterfaceDescription_UBX-21046737.pdf?utm_content=UBX-21046737
      fill out the any fields not listed here with 0

    3. Click the green arrow to send it to the GPS device

  4. Turn off all NMEA messages

    1. At the bottom right, find Preset UBX Configuration Command

      image-20250129-174614.png
    2. Select the appropriate ports (e.g. USB, UART-1)

    3. Select CFG-MSG - Turn OFF all NMEA msgs

    4. Click the green arrow to send it to the GPS device

  5. Enable RMC messages

    1. At the bottom left, find CFG-MSG Message Rate Configuration

      image-20250129-174657.png
    2. Scroll to find RMC

    3. Set the appropriate ports (e.g. USB, UART-1) value to 1

    4. Click the green arrow to send it to the GPS device

Tip: Besides configuring the PPS, each green arrow click, you should see the device’s output changes in PyGPSClient’s main dialog

image-20250129-174926.png
GPS Output behind the UBX Configuration Window

 

Hardware Interface and Connections

Component connections

  1. Connect the Ouster sensor to the Interface Box (provided in the box with the sensor). The IBOX comes in a kit which includes the IBOX unit, sensor cable, Ethernet cable ,and a AC-to-DC power supply.

  2. Plug the ethernet cable into the designated port on the IBOX. Connect the other end to your PC (or USB-Ethernet converter). Provide power to the IBOX via the DC jack.

  3. Power the SparkFun NEO-M9N board with a USB-C cable as shown and connect the other end to your PC.

  4. Attach the antenna with U.FL connector to the GPS board where indicated (not shown). Place the antenna module outdoors, or near a window for a sufficient view of the sky.

    Component connections

NEO-M9N + IBOX connections

  1. Remove the clear, plastic cover from the IBOX using the torx screwdriver.

  2. Using jumper wires, connect the PPS, GND and TX/MISO pin on the GPS board to the designated pins on the IBOX as shown in the table/figure below.

GPS Board pin

IBOX Pin

GPS Board pin

IBOX Pin

PPS

SYNC_PULSE_IN

TX/MISO

MULTIPURPOSE_IO

GND

GND

Note: The PPS and TX/MISO pins with this SparkFUN board do not require additional pull-up resistors or a voltage increase adapter. Connect them directly to the IBOX pins as described.

This may not be the case with other GPS/GNSS modules on the market. Please review the Ouster Hardware Manual for further details.

NEO-M9N + IBOX connections

Verification of Success

Once all prior steps are complete, follow the instructions below to check if the sensor is synchronized properly and reporting the expected values.

Note: Once you have configured your GPS, it is good practice to verify the signals using an oscilloscope. This will ensure you have the correct baud rate, polarity, voltage, and message type being output.

PPS Signal
NMEA UART Signal

 

Query the status via Ouster API

HTTP: GET /api/v1/time/sensor

The sensor can be queried and configured using HTTP GET requests. This can be done using several different tools such as HTTPie, cURL, Advanced REST Client, etc. You can execute these queries from a command line application (e.g. Windows CMD, macOS & Linux Terminal)

Here are two examples using a curl command:

  1. curl -i http://os-991234567890/api/v1/time/sensor

    1. where 991234567890 represents the unique serial number of your device. This can be found on the sensor label as a reference.

  2. curl -i http://169.254.1.100/api/v1/time/sensor

    1. where 169.254.1.100 represents the set IPv4 address of the sensor

Response from API Query

From the outputted response below:

{ "multipurpose_io": { "mode": "INPUT_NMEA_UART", "nmea": { "baud_rate": "BAUD_9600", "diagnostics": { "decoding": { "date_decoded_count": 384, "last_read_message": "GPRMC,164853.00,A,4102.23151,N,08141.59407,W,0.052,,080823,,,D,V*16", "not_valid_count": 0, "utc_decoded_count": 384 }, "io_checks": { "bit_count": 80332, "bit_count_unfiltered": 80331, "char_count": 27478, "start_char_count": 393 } }, "ignore_valid_char": 0, "leap_seconds": 0, "locked": 1, "polarity": "ACTIVE_HIGH" }, "sync_pulse_out": { "angle_deg": 360, "frequency_hz": 1, "polarity": "ACTIVE_HIGH", "pulse_width_ms": 10 } }, "sync_pulse_in": { "diagnostics": { "count": 401, "count_unfiltered": 363, "last_period_nsec": 0 }, "locked": 1, "polarity": "ACTIVE_HIGH" }, "timestamp": { "mode": "TIME_FROM_SYNC_PULSE_IN", "time": 1691513333.6113698, "time_options": { "internal_osc": 399, "ptp_1588": 1651168147, "sync_pulse_in": 1691513333 } } }

 

Verify the following:

  1. The sensor is locked on to the PPS signal

    1. "sync_pulse_in" (line 32)

      1. "locked": 1 (line 38).

“sync_pulse_in” is “locked”: 0 “locked”: 1

  1. The sensor is locked on the NMEA signal

    1. "nmea" (line 4)

      1. "locked": 1 (line 22)

“nmea” is “locked”: 0 “locked”: 1

  1. The last read message looks like a valid GPRMC sentence 

    1. "last_read_message" (line 9)

      1. "GPRMC,164853.00,A,4102.23151,N,08141.59407,W,0.052,,080823,,,D,V*16"

Field

Example Data

Description / Format

Field

Example Data

Description / Format

Message ID

GPRMC

RMC Protocol Header

Time (UTC)

164853.00

hhmmss.ss

Status

A

A - valid data

V - invalid data

Latitude

4102.23151

ddmm.mmmm

North/South indicator

N

N - North

S - South

Longitude

08141.59407

ddmm.mmmm

East/West indicator

W

E - East

W - West

Speed over ground

0.052

Knots

Course over ground (tracking)

““

Degrees (blank if slow or not moving)

Date

080823

ddmmyy

Magnetic variation

““

Degrees

Mode

D

A, D or E

Checksum

*16

hex

Sentence terminator

 

[CR][LF]

  1. The timestamp time has updated to a reasonable GPS time

    1. “timestamp” (line 41)

      1. "time": 1691513333.6113698 (line 43)

“timestamp_mode” is "TIME_FROM_SYNC_PULSE_IN"

“time”: Unix Epoch time [# of seconds since 00:00:00 January 1, 1970]

Tip: When Epoch time is converted to a human-readable time (using anhttps://www.epochconverter.com/#tools tool), the output should match both the date and time fields in the GPRMC sentence.

1691513333.6113698 = Tuesday, August 8, 2023 12:48:53 PM GMT-04:00 DST (16:48:53 GMT)

 Related articles