One-shot reconfiguration and live monitoring of SocketCAN / CAN-FD interfaces
Project description
canconf / canmon / cantalk
Three small, single-purpose tools for people who spend their day staring at CAN buses on Linux:
canconf— reconfigure every SocketCAN / CAN-FD interface in one terse command, introspect the hardware, inspect the live configuration.canmon— passive, read-only health monitor: state transitions, configuration changes, auto-restarts, and bit-error bursts — printed only when something actually happens.cantalk— tiny REPL for talking to an ECU: set an arbitration pair, type hex bytes, see the reply. ISOTP by default (kernel handles flow control and reassembly);--rawfor plain CAN frames.
All three are pure Python 3.9+ stdlib, zero runtime dependencies, installable
as a single pipx install canconf and immediately useful.
Why this exists
Automotive CAN work — diagnostics, ECU flashing, gateway reverse-engineering, bench-testing on the sofa — spends an embarrassing amount of time on the same fiddly chores:
- Changing bitrates. Different ECUs, different generations, different
vehicle platforms all want different line speeds. Switching from 500k to 250k
to 1M to FD-at-2M happens several times a day. The stock
ip linkCLI wants half a dozen sudo-flavoured incantations for each iface, in the right order, with the right keywords — an error-prone tax on iteration speed. - Keeping two interfaces in lock-step. Dual-channel USB adapters and bench rigs with T-tap into the same bus mean every parameter change has to be mirrored across both ifaces. Forget one and your capture is misleading.
- Knowing the bus is healthy. When an ECU flash spirals into bus-off, or a faulty transceiver starts cooking bit errors, you want to see it — not realise hours later that half your log is garbage.
- Knowing what the hardware can actually do. Is this adapter FD-capable?
What bitrates will its clock + timing constants support? Which kernel driver
backs it? The answers are all there in
/sysand netlink, but nobody typesjq '.linkinfo.info_data.bittiming_const'for fun.
canconf and canmon compress those chores into commands short enough to
actually type during a debug loop. If you use mcangen, candump, python-can,
or any of the canmatrix family to do real work on the bus, these are the tools
that keep the plumbing out of your way.
Installation
pipx — recommended
Installs the binaries into an isolated venv and puts canconf, canmon, and
cantalk on your PATH:
pipx install canconf
pip
pip install canconf
From source
git clone https://github.com/mickeyl/canconf
cd canconf
pipx install . # or: pip install -e .
No runtime deps. Needs only iproute2 (you already have it) and sudo
(canconf self-elevates; canmon stays non-privileged; cantalk needs
CAP_NET_RAW or root). For ISOTP, cantalk relies on the in-tree
can-isotp kernel module (mainline since 5.10).
canconf
Replaces:
sudo ip link set can0 down
sudo ip link set can1 down
sudo ip link set can0 type can bitrate 500000 dbitrate 2000000 sample-point 0.875 dsample-point 0.75 fd on
sudo ip link set can1 type can bitrate 500000 dbitrate 2000000 sample-point 0.875 dsample-point 0.75 fd on
sudo ip link set can0 txqueuelen 10000
sudo ip link set can1 txqueuelen 10000
sudo ip link set can0 up
sudo ip link set can1 up
with:
canconf 500k/2M@0.875/0.75
Grammar
canconf show status of all can* interfaces
canconf 500k classic CAN @ 500 kbit/s, all interfaces, up
canconf 500k/2M CAN-FD: nominal 500k, data 2M
canconf 500k/2M@0.875/0.75 same, with nominal & data sample points
canconf off | down bring all interfaces down
canconf up bring all interfaces up (no reconfigure)
canconf bitrates show achievable bitrates per interface
Positional spec: BITRATE[/DBITRATE][@SP[/DSP]]. Bitrates accept the suffixes
k/M (125k, 500k, 1M, 2M, 5M, 8M).
Features
- Discovers all CAN interfaces by ARPHRD type (so
can*,vcan*,slcan*are all picked up — no name matching). - One compact spec for classic CAN and CAN-FD alike.
canconf bitratesreports the driver, reference clock, the achievable bitrate envelope (derived frombittiming_const), which standard bitrates fall within it, and whether the hardware supports CAN-FD at all. No brute-forcing, all introspection.- Sets
txqueuelen 10000by default (the kernel default of 10 is painfully low for anything other than idle buses). - Prints the live post-apply state — including
qlenand the underlying driver — so you can verify the driver actually accepted what you asked for (drivers silently round bitrates to what the clock can produce). - ANSI colour with sensible auto-detection,
NO_COLOR/FORCE_COLOR, and a--no-colorflag. - Self-elevates to root via
sudo;--dry-runis completely non-privileged.
Options
| Flag | Description |
|---|---|
-i, --ifaces a,b,c |
Restrict to these interfaces (default: all CAN interfaces) |
-r, --restart-ms N |
Auto-restart on bus-off after N ms |
--listen-only |
Listen-only mode |
--loopback |
Loopback mode |
--one-shot |
One-shot mode |
--berr |
Enable bus error reporting |
--term OHM |
Set termination resistor (if the hardware supports it) |
--txqueuelen N |
Override the tx queue length (default: 10000) |
-n, --dry-run |
Print the ip commands that would run, do nothing |
-v, --verbose |
Print each ip command as it runs |
-q, --quiet |
Suppress the post-apply status dump |
--no-color |
Disable ANSI colour |
-V, --version |
Print version and exit |
-h, --help |
Show help |
Example session
❯ canconf
can0 UP CAN 500k sp 0.875 qlen 10000 drv gs_usb
can1 UP CAN 500k sp 0.875 qlen 10000 drv gs_usb
❯ canconf 500k/2M@0.875/0.75
[sudo] password for mickey:
can0 UP CAN-FD 500k/2M sp 0.875/0.750 qlen 10000 drv gs_usb
can1 UP CAN-FD 500k/2M sp 0.875/0.750 qlen 10000 drv gs_usb
❯ canconf bitrates
=== can0 ===
driver: gs_usb
clock: 40 MHz
nominal: 202 .. 13333333
standard: 10k, 20k, 50k, 100k, 125k, 250k, 500k, 800k, 1M
FD data: 25510 .. 13333333
FD std: 1M, 2M, 4M, 5M, 8M
=== can1 ===
driver: gs_usb
clock: 48 MHz
nominal: 1875 .. 16M
standard: 10k, 20k, 50k, 100k, 125k, 250k, 500k, 800k, 1M
FD: not supported
❯ canconf -n 1M -i can0
+ ip link set can0 down
+ ip link set can0 type can bitrate 1000000
+ ip link set can0 txqueuelen 10000
+ ip link set can0 up
canmon
canmon tails every CAN interface at 1 Hz (tunable). It prints the current
state once at startup, keeps discovering CAN interfaces that appear later, and
then stays silent, emitting a new row only when something actually changes:
a state transition, a bittiming change, an auto-restart, or a tick in which the
CAN controller bit-error rate exceeds a threshold. It needs no root and reads
only from ip -j -details -s link show plus /sys/class/net — no CAN traffic
is injected, intercepted, or generated.
Between ticks with no events, nothing is printed. Pass -v to force a row
every tick.
Options
| Flag | Description |
|---|---|
-i, --ifaces a,b,c |
Restrict to these interfaces |
-r, --rate SECONDS |
Tick interval (default: 1.0) |
-t, --err-rate N |
Threshold for the Δbus/s flag (default: 1) |
-o, --once |
Print initial snapshot and exit |
-v, --verbose |
Emit a row every tick, not just on change |
--no-color |
Disable ANSI colour |
-V, --version |
Print version and exit |
-h, --help |
Show help |
Columns
- Δerr/s — frame-level
rx+txerror delta per second (from driverstats64). - Δbus/s — CAN controller bit-error delta per second (from
info_xstats.bus_error). The number you probably care about most during ECU flashes, cable swaps, or suspect wiring. - restarts — running total of driver-initiated auto-restarts after bus-off (requires
canconf … -r MSto be non-zero at configure time). - notes — event tags:
STATE a → b,CONFIG a → b,RESTART #N,BIT-ERRORS N/s > T/s. Initial rows also include static details such assp,qlen, anddrvwhen the driver exposes them. In a colour-capable terminal each tag is coloured by severity, and state tokens are rendered in the same colour as the state column so bus-off transitions jump out.
In default auto-discovery mode, interfaces that appear after startup are added
to the watch list. If a watched interface disappears between ticks, canmon
reports state MISSING and bitrate —. This usually means the netdev vanished
from ip link show for that poll, as can happen during USB adapter
re-enumeration or suspend/resume. If the netdev is present but ip does not
expose the CAN-specific state block, canmon reports NO-CAN-DATA instead.
cantalk
A minimal interactive CAN terminal. Set an arbitration pair, type hex,
see the reply. The interaction model deliberately mirrors the
ecuconnect-tool term REPL from Swift-CANyonero. On a TTY the prompt is
anchored to the bottom of the terminal in a fixed three-line frame
(top info rule, input line, bottom hint rule) and the request/response
log scrolls independently above it:
· iface=can0 mode=isotp tx=7E0 rx=7E8 timeout=2s pad=0xAA
→ 7E0 10 03
← 7E8 50 03 00 32 01 F4
-> P..2..
ℹ︎ ✓ positive response to Diagnostic Session Control
→ 7E0 22 F1 90
← 7E8 62 F1 90 57 56 57 5A 5A 5A 31 4B 5A 31 47 30 30 30 30 30 31
-> b..WVWZZZ1KZ1G000001
ℹ︎ ✓ positive response to Read Data By Identifier
── can0 · isotp · 7E0 → 7E8 · pad=0xAA · timeout=2s ──────────────────
❯ _
── :help · ↑↓ history · Ctrl-C exit ──────────────────────────────────
The arbitration pair is remembered per interface across runs (saved to
$XDG_STATE_HOME/cantalk/state.json); next time you cantalk can0 the
last-used tx/rx are restored automatically.
Pipe stdin or pass --plain for a one-prompt-per-line REPL without
the scroll region (e.g. for scripted use).
Modes
- ISOTP (default). Kernel
can-isotpmodule handles segmentation, flow control (FC), and reassembly. Payloads up to 4095 bytes; you only ever see complete messages. --raw. A bareAF_CANraw socket. Each typed payload becomes one classic CAN frame (≤ 8 bytes). The reply window collects every matching frame seen until 250 ms of silence — useful for snooping or talking to ECUs that don't speak ISOTP.
REPL grammar
| Input | What it does |
|---|---|
:7DF |
Set TX. RX is auto-derived (TX+8 for 11-bit IDs; J1939-style source/target swap for 18DA<target><source>). |
:7DF,7E8 |
Set TX and RX explicitly. |
:18DA10F1,18DAF110 |
29-bit extended addressing. |
:info / :help |
Show state / show commands. |
0902, 09 02, 09:02 |
Send hex bytes. Whitespace, :, , are separators. |
quit / Ctrl-C / Ctrl-D |
Exit. |
Options
| Flag | Description |
|---|---|
--raw |
Use raw CAN sockets instead of ISOTP. |
-t, --timeout SEC |
Response timeout (default: 2.0). Raw mode keeps listening for 250 ms after each received frame. |
-p, --pad HEX |
Pad every CAN frame to 8 bytes (default: AA; none to disable). Applies in both ISOTP and raw modes. |
--plain |
Disable the bottom-anchored TUI; one prompt per line. |
--no-color |
Disable ANSI colour. |
-V, --version |
Print version. |
-h, --help |
Show help. |
The interface must already be up — that's canconf's job.
Roadmap
Things I intend to build as soon as the underlying itch scratches hard enough. Contributions welcome; open an issue first if it's non-trivial.
Short-term
- JSON output (
canconf --json,canmon --json) for scripting and for wiring the tools into dashboards or CI. - Bus-load metrics in
canmon— frames/s, bytes/s, and estimated line utilisation (%) derived from the driver's rx/tx counters and the active bitrate. All the inputs are already read each tick. - Config profiles —
~/.config/canconf/profiles.tomlso you can name frequently-used configurations and apply them withcanconf @passive-500korcanconf @my-ecu. Profile files track sample-point, FD, termination, listen-only, restart-ms, and per-iface overrides. canconf save/canconf apply— dump the current configuration of all interfaces to a profile file, and re-apply it later. Handy for sharing bus setups between workstations.
Medium-term
canscan— sibling tool. Auto-detect the bitrate of an unknown bus by entering listen-only mode and trying each standard rate until frames arrive cleanly. Useful when probing an unknown vehicle network.canfind— mapcanXinterfaces back to the physical adapter: USB VID:PID, serial number, USB port path, product string. Indispensable when four identical USB-CAN dongles are plugged in.cansync— one-shot synchronised down → configure → up across interfaces, with an optional--at <time>for lab rigs that need deterministic startup.- Daemon mode for
canmon— long-running process that emits to syslog/journald (for ops use) and exposes a Prometheus text-format endpoint (for lab dashboards and long-term bus health recording).
Longer-term / speculative
- python-can integration — provide a small import so
canconf.discover()can feed bus factories, and so python-can users get the same topology awareness without re-implementing it. - Test-mode —
canconf testsends a handful of frames between discovered interfaces (pair-wise or via loopback) and verifies the wiring and bitrate match end-to-end. Catches bench setup errors before you chase them in protocol logs. - Termination auto-probe — some CAN transceivers can report whether the
bus looks correctly terminated. If the driver exposes it,
canmonshould surface it alongside the state column.
License
MIT — see LICENSE.
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
Built Distribution
Filter files by name, interpreter, ABI, and platform.
If you're not sure about the file name format, learn more about wheel file names.
Copy a direct link to the current filters
File details
Details for the file canconf-0.4.1.tar.gz.
File metadata
- Download URL: canconf-0.4.1.tar.gz
- Upload date:
- Size: 192.2 kB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
ec8f9ddd798b89d0e7c4d16365a17c9e46225d55be125749d75809258da5c825
|
|
| MD5 |
ee6b45ab6dd3d3b522718cf55288fdaa
|
|
| BLAKE2b-256 |
c5e604a411ee4ff6fd231424b8fe10c81d7686d74837d66cc807ed8c5275cfe1
|
File details
Details for the file canconf-0.4.1-py3-none-any.whl.
File metadata
- Download URL: canconf-0.4.1-py3-none-any.whl
- Upload date:
- Size: 29.1 kB
- Tags: Python 3
- Uploaded using Trusted Publishing? No
- Uploaded via: twine/6.2.0 CPython/3.12.3
File hashes
| Algorithm | Hash digest | |
|---|---|---|
| SHA256 |
85a9c42dc2f7c994aaa8a82b60da48894a84f4cfff756bc2029cdc0d5784acef
|
|
| MD5 |
2079e066ca2e388a06bcfdb2d98bfebf
|
|
| BLAKE2b-256 |
e149f496a83ab7ab3676029842ecaaa80880d1edb9d5c6dbb1b4c6bc32e294ba
|