Writing embedded drivers in Rust isn't that hard. Part 2
Now that we implemented a small program that is able to read the ID field of the AT42QT1070, we will first write a small library that generalizes this implementation to every HAL that provides an I2C implementation and then expand upon that implementation with more and more functionality.
A simple driver structure
First off we define a simple Driver
struct which just contains a generic I2C object as well as
an Error
enum that, for now, only has the I2cError
variant which in turn contains a generic type,
this is because different HALs do of course return different I2C errors and we want to be prepared for all of them.
#![no_std]
use embedded_hal::blocking::i2c::WriteRead;
#[derive(Clone, Copy, Debug)]
pub enum Error<I2cError> {
I2cError(I2cError),
}
pub struct Driver<I2C> {
i2c: I2C,
}
As our I2C struct is purely generic, as of now our implementation will of course have to constrain it
to implementing the WriteRead
trait, this is quite simply done with:
impl<I2C, I2cError> Driver<I2C>
where
I2C: WriteRead<Error = I2cError>,
{
}
This declaration now also allows us to have a Result containing the I2C error from the HAL as we
have now bound this type to our generic parameter I2cError
, hence we can define a new
function
in order to (surprise) create a new instance of our struct.
pub fn new(i2c: I2C) -> Result<Driver<I2C>, Error<I2cError>> {
let mut driver = Driver {
i2c: i2c,
};
}
Integrating the ID getter
However when calling new
we of course do want to verify that the chip is actually on the I2C bus we
just got passed, this is quite simply done by trying to read the ID from our chip. If the chip is not
attached to the bus it will not respond which should cause the HAL to throw an error, if a chip with
the same address from another manufacturer (remember, I2C has per default only got 7 bits of address
space, collisions will happen) is on the bus it (hopefully) respond with something that is not equal
to the ID we expect. We already implemented the logic behind this last time so now we just got to
wrap it into our generic:
fn get_id(&mut self) -> Result<u8, Error<I2cError>> {
let mut buffer = [0u8; 1];
self.i2c.write_read(chip::I2C, &[chip::ID_ADDR], &mut buffer)?;
Ok(buffer[0])
}
// This will just contain all our addresses and constants related to the chip
mod chip {
pub const I2C: u8 = 0x1B << 1;
pub const ID: u8 = 0x2E;
pub const ID_ADDR: u8 = 0;
}
However this piece of code will actually not work:
error[E0277]: the trait bound `Error<I2cError>: core::convert::From<I2cError>` is not satisfied
--> src/lib.rs:38:70
|
38 | self.i2c.write_read(Chip::I2C, &[Chip::ID_ADDR], &mut buffer)?;
| ^ the trait `core::convert::From<I2cError>` is not implemented for `Error<I2cError>`
|
= note: required by `core::convert::From::from`
In order to solve this we basically have two options, we could either do a map_err()
every time we
try to write something on the I2C bus or we simply implement the From trait for our Error
as follows:
impl<E> From<E> for Error<E> {
fn from(error: E) -> Self {
Error::I2cError(error)
}
}
Of course now this piece of code will convert every error that we try to return as our Error
enum
into an I2cError
variant, that means we have to be careful if we should include more third party
libraries later as these errors too will be converted into I2cError
, however as long as we don't do
that this solution is actually really convenient.
Now where we got the get_id()
stuff down we can rewrite our new()
in order to do the ID check:
pub fn new(i2c: I2C) -> Result<Driver<I2C>, Error<I2cError>> {
let mut driver = Driver {
i2c: i2c,
};
let id = driver.get_id()?;
if id != chip::ID {
return Err(Error::IdMismatch(id));
}
Ok(driver)
}
// And of course extend the Error enum
#[derive(Clone, Copy, Debug)]
pub enum Error<I2cError> {
I2cError(I2cError),
IdMismatch(u8)
}
Initializing the chip properly
In order to make the chip work properly we have to configure it, the datasheet notes this can be done by writing a non zero value to address 56 (see Address 56: Calibrate), which will set the calibrate bit in the status register at address 2, once the bit is celared the calibration is done. The status information is represented as an 8 bit long value with 3 relevant and 5 reserved bits (see 5.4 Address 2: Detection Status). In order to represent this bit field in rust we will use the very commonly used bitfield (TODO crates.io link) crate which gives us a really nice macro in order to automatically generate such bitfield structs, so lets just include it in our Cargo.toml quickly
[dependencies]
embedded-hal = "0.2.3"
bitfield = "0.13.2"
And stick to the docs in order to denote the information we got from the datasheet in rust, using said bitfield macro:
use bitfield::bitfield;
bitfield!{
pub struct Status(u8);
impl Debug;
pub calibrate, _: 7;
pub overflow, _: 6;
pub touch, _: 0;
}
Now where we got our struct down we can write another simple method that fetches the status for us, and puts it into the struct so we can use it to, for example, find out wether the pads on the PCB have been touched yet using the touch bit.
pub fn get_status(&mut self) -> Result<Status, Error<I2cError>> {
let mut buffer = [0u8; 1];
self.i2c.write_read(chip::I2C, &[chip::STATUS_ADDR], &mut buffer)?
Ok(Status(buffer[0]))
}
// And as always, extend our chip mod
mod chip {
pub const I2C: u8 = 0x1B << 1;
pub const ID: u8 = 0x2E;
pub const ID_ADDR: u8 = 0;
pub const STATUS_ADDR: u8 = 2;
}
So all that is missing now is a routine that performs the calibration write before polling the status register until the calibration is done and we can start reading actual values!
Before we write this implementation though, we have to add another constraint on the generic I2C parameter as we are only supposed to write a non zero value into the calibrate register but not actually read a response, hence our implementation looks as follows:
use embedded_hal::blocking::i2c::{Write, WriteRead};
impl<I2C, I2cError> Driver<I2C>
where
I2C: WriteRead<Error = I2cError> + Write<Error = I2cError>,
{
pub fn calibrate(&mut self) -> Result<(), Error<I2cError>> {
self.i2c.write(chip::I2C, &[chip::CALIBRATE_ADDR, 0xFF])?;
loop {
let status = self.get_status()?;
if !status.calibrate() {
break;
}
}
Ok(())
}
}
// More constants!
mod chip {
pub const I2C: u8 = 0x1B << 1;
pub const ID: u8 = 0x2E;
pub const ID_ADDR: u8 = 0;
pub const STATUS_ADDR: u8 = 2;
pub const CALIBRATE_ADDR: u8 = 56;
}
Read the touch values!
This is actually surprisingly simple and similar to what we just did before, we just define another bitfield struct according to the specifications in chapter 5.5 Address 3: Key Status, read that one just like we read the previous status register and we are done
bitfield! {
pub struct KeyStatus(u8);
impl Debug;
pub key6, _: 6;
pub key5, _: 5;
pub key4, _: 4;
pub key3, _: 3;
pub key2, _: 2;
pub key1, _: 1;
pub key0, _: 0;
}
pub fn get_key_status(&mut self) -> Result<KeyStatus, Error<I2cError>> {
let mut buffer = [0u8; 1];
self.i2c.write_read(chip::I2C, &[chip::KEY_STATUS_ADDR], &mut buffer)?;
Ok(KeyStatus(buffer[0]))
}
mod chip {
pub const I2C: u8 = 0x1B << 1;
pub const ID: u8 = 0x2E;
pub const ID_ADDR: u8 = 0;
pub const STATUS_ADDR: u8 = 2;
pub const KEY_STATUS_ADDR: u8 = 3;
pub const CALIBRATE_ADDR: u8 = 56;
}
And...we are already done, quite simple once you got it down for the first time, right?
A small example on the stm32l4
In order to verify our blindly written implementation works, we can come up with a simple example for the stm32l4 chip from last time. First of all, we will, of course, take over the big blob that initializes our chip properly from last time:
#![no_main]
#![no_std]
extern crate panic_semihosting;
use cortex_m_rt::entry;
use stm32l4xx_hal::i2c::I2c;
use stm32l4xx_hal::prelude::*;
use at42qt1070::Driver;
#[entry]
fn main() -> ! {
let dp = stm32l4xx_hal::stm32::Peripherals::take().unwrap();
let mut flash = dp.FLASH.constrain();
let mut rcc = dp.RCC.constrain();
let clocks = rcc.cfgr.hclk(8.mhz()).freeze(&mut flash.acr);
let mut gpiob = dp.GPIOB.split(&mut rcc.ahb2);
let scl = gpiob
.pb10
.into_open_drain_output(&mut gpiob.moder, &mut gpiob.otyper);
let scl = scl.into_af4(&mut gpiob.moder, &mut gpiob.afrh);
let sda = gpiob
.pb11
.into_open_drain_output(&mut gpiob.moder, &mut gpiob.otyper);
let sda = sda.into_af4(&mut gpiob.moder, &mut gpiob.afrh);
let i2c = I2c::i2c2(dp.I2C2, (scl, sda), 400.khz(), clocks, &mut rcc.apb1r1);
}
On top of that we can now just use our driver:
let mut driver = Driver::new(i2c).unwrap();
driver.calibrate().unwrap();
And now write a simple routine that checks wether the touch bit (which indicates that one of the pads has been touched) from the status register has been set yet and if it has read the key status register in order to find out which pad:
loop {
let status = driver.get_status().unwrap();
if status.touch() {
break;
}
}
let key_status = driver.get_key_status().unwrap();
let all_pads = [
key_status.key0(),
key_status.key1(),
key_status.key2(),
key_status.key3(),
key_status.key4(),
key_status.key5(),
key_status.key6(),
];
// We have to loop in the end again so our main function never returns
loop {}
And if we quickly build and flash this onto the microcontroller, set a breakpoint at the proper spot,
run the example (while of course touching one of the pads) and print out the all_pads
variable
we will be greeted with:
(gdb) p all_pads
$1 = [false, true, false, false, false, false, false]
Which, according to the schematic, is the exact key we touched \o/.
Conclusion
The chip does have a few more settings such as the AKS groups and a few measurement sensitivity ones, however they will just require a few more register writes and thus don't add anything new conceptually to what we learned in this 2 part series, hence I'll just implement them in private and publish the crate once I'm done (consider it as an exercise for the reader wink) . I hope you learned something reading this little series. If you have any feedback etc. for me you can just send it to the address at the bottom of the web page.