tenover 0.0.2

Client library for the Garmin R10 launch monitor BLE/GFDI/protobuf protocol
Documentation

10over

CI crates.io docs.rs

Disclaimer

This project is not affiliated with or endorsed by Garmin Ltd. Garmin, Approach, and R10 are trademarks of their respective owner.

Description

Client library for the Garmin R10 launch monitor. Decodes the R10's BLE/GFDI/protobuf protocol and exposes shot data through a synchronous, poll-based API:

  • Client<T> — poll-based client over any Transport. Handles MultiLink registration, GFDI handshake, protobuf subscribe/wakeup, shot deduplication, and state tracking automatically.
  • BleTransport — platform-agnostic BLE transport (ble feature, enabled by default). Auto-selects BlueZ on Linux or btleplug on Windows/macOS.
  • FrpServerFlight Relay Protocol device server (frp feature). Bridges R10 shot data to any FRP controller over WebSocket (port 5880).

Legal Basis — DMCA Section 1201(f)

This project is an exercise of the interoperability exception under 17 U.S.C. § 1201(f):

(f) Reverse Engineering.—

(1) Notwithstanding the provisions of subsection (a)(1)(A), a person who has lawfully obtained the right to use a copy of a computer program may circumvent a technological measure that effectively controls access to a particular portion of that program for the sole purpose of identifying and analyzing those elements of the program that are necessary to achieve interoperability of an independently created computer program with other programs, and that have not previously been readily made available to the person engaging in the circumvention, to the extent any such acts of identification and analysis do not constitute infringement under this title.

(2) Notwithstanding the provisions of subsections (a)(2) and (b), a person may develop and employ technological means to circumvent a technological measure, or to circumvent protection afforded by a technological measure, in order to enable the identification and analysis described in paragraph (1), or for the purpose of enabling interoperability of an independently created computer program with other programs, if such means are necessary to achieve such interoperability, to the extent that doing so does not constitute infringement under this title.

The Garmin R10 uses a proprietary protocol over Bluetooth Low Energy to communicate shot data (ball speed, launch angle, spin, club data, etc.) to companion software. Garmin does not publish this protocol or provide an SDK for third-party integration. The protocol was reverse-engineered from the researcher's own lawfully purchased hardware, solely to enable interoperability with third-party golf simulation software.

No Garmin code is reproduced here. No access controls were circumvented — the device uses standard BLE bonding, and all protocol data was captured from the researcher's own device.

Acceptable Use

This project exists solely to enable interoperability between the Garmin R10 and third-party golf simulation software.

It must not be used to:

  • Circumvent licensing or subscription requirements on Garmin products
  • Unlock paid features without purchase
  • Bypass any access controls on Garmin software or services

Issues or discussions proposing circumvention of licensing will be closed and the user blocked.

License

Licensed under either of:

at your option.

Status

Alpha (0.0.x). The protocol is reverse-engineered and still being mapped — the API will change as new message types are decoded and existing ones are refined. This crate follows SemVer: while the version is 0.0.x, any release may contain breaking changes. Once the API stabilizes it will move to 0.1.0, after which breaking changes will bump the minor version until 1.0.

Protocol Documentation

Detailed specs live in docs/:

  • WIRE.md — Transport stack (BLE GATT, MultiLink, COBS, CRC-16, GFDI framing), protobuf message catalog, shot data fields and units.
  • SEQUENCE.md — Connection lifecycle, MultiLink registration, GFDI handshake, protobuf session setup, shot data flow.

Quick Start

The R10 must be paired at the OS level before connecting. This is a one-time operation:

  • Linux: bluetoothctl pair <address> then bluetoothctl trust <address>
  • Windows: Settings > Bluetooth & devices > Add device > Bluetooth
  • macOS: System Preferences > Bluetooth

The library has no pairing API — it connects to an already-paired device.

use tenover::ble::BleTransport;
use tenover::{Client, Event};

let transport = BleTransport::auto_connect()?;
let mut client = Client::new(transport, 20);
client.start()?;

loop {
    match client.poll()? {
        Some(Event::Ready) => println!("Waiting for shot..."),
        Some(Event::Shot(shot)) => {
            if let Some(b) = &shot.ball {
                println!("Ball: {:.1} m/s", b.ball_speed);
            }
            if let Some(c) = &shot.club {
                println!("Club: {:.1} m/s", c.club_head_speed);
            }
        }
        _ => {}
    }
}

See examples/r10.rs for a complete standalone example.

# Linux (auto-selects BlueZ)
cargo run --example r10

# Windows cross-compile
cargo build --example r10 --target x86_64-pc-windows-gnu --release

FRP device server

The frp feature adds an FRP device server that makes the R10 appear as a standard FlightRelay device. Any FRP controller (flighthook, dashboards, etc.) can connect over WebSocket and receive shot data without knowing anything about the R10 protocol.

A standalone binary is included:

# Discover and connect to R10, serve FRP on port 5880
cargo run --features frp --bin tenover-frp

# Custom FRP bind address
cargo run --features frp --bin tenover-frp -- 0.0.0.0:9000

Pre-built binaries for Linux and Windows are attached to each GitHub release.

Features

Feature Default Description
ble Yes Platform-agnostic BLE transport (BlueZ on Linux, btleplug on Windows/macOS)
frp No FRP WebSocket device server (FlightRelay)
serde No Serialize/deserialize shot data types
# Protocol-only (no BLE transport)
cargo build --no-default-features

# With FRP server
cargo build --features frp

Dependencies

Minimal by design:

  • prost — protobuf runtime
  • thiserror — error enum derives
  • zbus + libc (Linux, ble feature) — BlueZ D-Bus transport
  • btleplug + tokio (Windows/macOS, ble feature) — cross-platform BLE
  • flightrelay (optional, frp feature) — FRP WebSocket server
  • serde (optional, serde feature) — serialization support

No async runtime in the public API. The btleplug backend runs tokio internally in a background thread; the caller-facing API is fully synchronous.