From 1e15831d56ccadd26938f5b92dba8413aad4d42a Mon Sep 17 00:00:00 2001 From: Samuel Tardieu <sam@rfc1149.net> Date: Mon, 15 Jul 2024 16:01:37 +0200 Subject: [PATCH] Implement FIRMWARE_VERSION command --- .gitlab-ci.yml | 9 ++++++ Cargo.lock | 9 +++++- README.org | 7 ++++- controller/Cargo.toml | 5 +++- controller/build.rs | 41 ++++++++++++++++++++++++++++ controller/python/controller.py | 40 ++++++++++++++++++++++++++- controller/python/controller_test.py | 23 ++++++++++++++++ controller/src/logic.rs | 8 ++++++ controller/src/main.rs | 7 ++++- 9 files changed, 144 insertions(+), 5 deletions(-) create mode 100644 controller/build.rs create mode 100644 controller/python/controller_test.py diff --git a/.gitlab-ci.yml b/.gitlab-ci.yml index 6552d4a..39be961 100644 --- a/.gitlab-ci.yml +++ b/.gitlab-ci.yml @@ -59,6 +59,15 @@ test-rust: - cd pid - cargo test --target x86_64-unknown-linux-gnu +test-python: + image: python + stage: test + script: + - cd controller/python + - python -munittest controller_test.py + cache: + paths: + deploy: image: alpine stage: deploy diff --git a/Cargo.lock b/Cargo.lock index d3cd2ff..409a64f 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -142,6 +142,7 @@ dependencies = [ "heapless", "i2c2-target", "panic-probe", + "semver 1.0.23", ] [[package]] @@ -615,7 +616,7 @@ version = "0.2.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "138e3e0acb6c9fb258b19b67cb8abd63c00679d2851805ea151465464fe9030a" dependencies = [ - "semver", + "semver 0.9.0", ] [[package]] @@ -633,6 +634,12 @@ dependencies = [ "semver-parser", ] +[[package]] +name = "semver" +version = "1.0.23" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "61697e0a1c7e512e84a621326239844a24d8207b4669b41bc18b32ea5cbf988b" + [[package]] name = "semver-parser" version = "0.7.0" diff --git a/README.org b/README.org index 81a7ec9..9f5f9c0 100644 --- a/README.org +++ b/README.org @@ -46,7 +46,12 @@ the TB6612FNG motor driver. * Command set -Values are exchanged in little endian format. +Multibyte values are exchanged in little endian format. + +** [IMPLEMENTED] 0x08 Firmware version (R) + +- Return three bytes containing the major, minor and patch version numbers + derived from =Cargo.toml=. ** [IMPLEMENTED] 0x0F Who am I? (R) diff --git a/controller/Cargo.toml b/controller/Cargo.toml index fa7a6d4..b6d96c3 100644 --- a/controller/Cargo.toml +++ b/controller/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "dc-motor-driver-hat" -version = "0.1.0" +version = "0.1.0" # Update in python/controller.py as well authors = ["Samuel Tardieu <sam@rfc1149.net>"] edition = "2021" @@ -19,3 +19,6 @@ panic-probe = { version = "0.3.2", features = ["print-defmt"] } [lints.clippy] pedantic = "deny" + +[build-dependencies] +semver = "1.0.23" diff --git a/controller/build.rs b/controller/build.rs new file mode 100644 index 0000000..566a915 --- /dev/null +++ b/controller/build.rs @@ -0,0 +1,41 @@ +use semver::Version; +use std::env; +use std::fs::{self, File}; +use std::io::Write; +use std::path::Path; +use std::str::FromStr; + +#[derive(Debug)] +enum Error { + InconsistentPythonLibraryFirmwareVersion, +} + +fn check_python_library_consistency(version: &Version) -> Result<(), Error> { + let haystack = fs::read_to_string("python/controller.py").unwrap(); + let needle = format!( + "REQUIRED_FIRMWARE_VERSION = ({}, {})", + version.major, version.minor + ); + if !haystack.contains(&needle) { + eprintln!("Cannot find the following string in Python library:\n {needle}"); + return Err(Error::InconsistentPythonLibraryFirmwareVersion); + } + Ok(()) +} + +fn main() -> Result<(), Error> { + let version_str = env!("CARGO_PKG_VERSION"); + let version = Version::from_str(version_str).unwrap(); + check_python_library_consistency(&version)?; + let mut out = + File::create(Path::new(&env::var("OUT_DIR").unwrap()).join("version.rs")).unwrap(); + writeln!( + out, + "pub const PKG_VERSION: [u8; 3] = [{}, {}, {}];", + u8::try_from(version.major).unwrap(), + u8::try_from(version.minor).unwrap(), + u8::try_from(version.patch).unwrap() + ) + .unwrap(); + Ok(()) +} diff --git a/controller/python/controller.py b/controller/python/controller.py index 05fe857..83b9a35 100644 --- a/controller/python/controller.py +++ b/controller/python/controller.py @@ -1,13 +1,20 @@ -import smbus import struct from numbers import Real from typing import Any, Optional +# Major and minor version of required firmware +_REQUIRED_FIRMWARE_VERSION = (0, 1) + + +class FirmwareVersionMismatch(Exception): + pass + class Controller: I2C_ADDR = 0x57 + FIRMWARE_VERSION = 0x08 WHO_AM_I = 0x0F MAX_MOTOR_PERCENTAGE = 0x11 MOTOR_SHUTDOWN_TIMEOUT = 0x28 @@ -16,9 +23,13 @@ class Controller: STATUS = 0x36 def __init__(self, i2cbus=8): + import smbus + self.I2C_BUS = i2cbus self.i2c = smbus.SMBus(self.I2C_BUS) + self.check_firmware_version() + def _read(self, command, n, unpack_spec) -> tuple: return struct.unpack( unpack_spec, @@ -108,3 +119,30 @@ class Controller: """Get the duration in seconds after which the motors will shut down if no valid command is received.""" return self._read(self.MOTOR_SHUTDOWN_TIMEOUT, 1, "B")[0] / 10 + + def get_firmware_version(self) -> tuple[int, int, int]: + """Get the firmware version (major, minor, patch).""" + return self._read(self.FIRMWARE_VERSION, 3, "BBB") + + def check_firmware_version(self): + """Check that the firmware uses a version compatible with this + library.""" + version = self.get_firmware_version() + Controller._check_firmware_version_consistency( + _REQUIRED_FIRMWARE_VERSION, version + ) + + def _check_firmware_version_consistency( + required: tuple[int, int], version: tuple[int, int, int] + ): + (MAJOR, MINOR) = required + (major, minor, patch) = version + error = None + if major != MAJOR or minor < MINOR: + version = f"{major}.{minor}.{patch}" + VERSION = f"{MAJOR}.{MINOR}.*" + error = ( + f"Hardware runs firmware version {version} which " + f"is not compatible with this library version ({VERSION})" + ) + raise FirmwareVersionMismatch(error) diff --git a/controller/python/controller_test.py b/controller/python/controller_test.py new file mode 100644 index 0000000..479e7ce --- /dev/null +++ b/controller/python/controller_test.py @@ -0,0 +1,23 @@ +from controller import Controller, FirmwareVersionMismatch +import unittest + + +class TestController(unittest.TestCase): + + def test_firmware_major_mismatch(self): + with self.assertRaises(FirmwareVersionMismatch): + Controller._check_firmware_version_consistency((2, 5), (3, 7, 4)) + + def test_firmware_minor_mismatch(self): + with self.assertRaises(FirmwareVersionMismatch): + Controller._check_firmware_version_consistency((2, 5), (2, 4, 3)) + + def test_firmware_perfect_match(self): + self.assertIsNone( + Controller._check_firmware_version_consistency((2, 4), (2, 4, 3)) + ) + + def test_firmware_subsequent_match(self): + self.assertIsNone( + Controller._check_firmware_version_consistency((2, 4), (2, 5, 3)) + ) diff --git a/controller/src/logic.rs b/controller/src/logic.rs index 21b3719..beeec99 100644 --- a/controller/src/logic.rs +++ b/controller/src/logic.rs @@ -131,6 +131,7 @@ fn i2c_callback(command: &[u8], response: &mut Deque<u8, MESSAGE_SIZE>) { } } +const FIRMWARE_VERSION: u8 = 0x08; const WHO_AM_I: u8 = 0x0f; const MAX_MOTOR_PERCENTAGE: u8 = 0x11; const MOTOR_SHUTDOWN_TIMEOUT: u8 = 0x28; @@ -138,6 +139,8 @@ const MOTOR_SPEED: u8 = 0x30; const ENCODER_TICKS: u8 = 0x32; const STATUS: u8 = 0x36; +include!(concat!(env!("OUT_DIR"), "/version.rs")); + fn process_command( state: &mut State, command: &[u8], @@ -145,6 +148,11 @@ fn process_command( ) -> bool { defmt::trace!("Processing command {:?}", command); match command { + [FIRMWARE_VERSION] => { + for b in PKG_VERSION { + response.push_back(b).unwrap(); + } + } [WHO_AM_I, ..] => { response.push_back(0x57).unwrap(); } diff --git a/controller/src/main.rs b/controller/src/main.rs index d97be89..d231548 100644 --- a/controller/src/main.rs +++ b/controller/src/main.rs @@ -38,7 +38,12 @@ async fn main(spawner: Spawner) { config.rcc.apb1_pre = APBPrescaler::DIV2; let p = embassy_stm32::init(config); - defmt::info!("Program starting"); + defmt::info!( + "Firmware {}.{}.{} starting", + logic::PKG_VERSION[0], + logic::PKG_VERSION[1], + logic::PKG_VERSION[2] + ); spawner.spawn(blink(p.PB15)).unwrap(); -- GitLab