use ak8963::{self, AK8963};
use hal::blocking::delay::DelayMs;
use hal::blocking::i2c;
use hal::blocking::spi;
use hal::digital::v2::OutputPin;
use Register;
const MPU_I2C_ADDR: u8 = 0x68;
pub trait Releasable {
type Released;
fn release(self) -> Self::Released;
}
pub trait Device: Releasable {
type Error;
fn read_many(&mut self,
reg: Register,
buffer: &mut [u8])
-> Result<(), Self::Error>;
fn write(&mut self, reg: Register, val: u8) -> Result<(), Self::Error>;
fn read(&mut self, reg: Register) -> Result<u8, Self::Error> {
let buffer = &mut [0; 2];
self.read_many(reg, buffer)?;
Ok(buffer[1])
}
fn modify<F>(&mut self, reg: Register, f: F) -> Result<(), Self::Error>
where F: FnOnce(u8) -> u8
{
let r = self.read(reg)?;
self.write(reg, f(r))?;
Ok(())
}
}
pub struct SpiDevice<SPI, GPIO> {
spi: SPI,
ncs: GPIO,
}
impl<SPI, NCS, E> SpiDevice<SPI, NCS>
where SPI: spi::Write<u8, Error = E> + spi::Transfer<u8, Error = E>,
NCS: OutputPin
{
pub fn new(spi: SPI, ncs: NCS) -> Self {
SpiDevice { spi,
ncs }
}
}
impl<SPI, NCS, E> Releasable for SpiDevice<SPI, NCS>
where SPI: spi::Write<u8, Error = E> + spi::Transfer<u8, Error = E>,
NCS: OutputPin
{
type Released = (SPI, NCS);
fn release(self) -> (SPI, NCS) {
(self.spi, self.ncs)
}
}
#[derive(Debug, Copy, Clone)]
pub enum SpiError<E, E2> {
BusError(E),
NCSError(E2),
}
impl<E, E2> core::convert::From<E> for SpiError<E, E2> {
fn from(error: E) -> Self {
SpiError::BusError(error)
}
}
impl<SPI, NCS, E, EO> Device for SpiDevice<SPI, NCS>
where SPI: spi::Write<u8, Error = E> + spi::Transfer<u8, Error = E>,
NCS: OutputPin<Error = EO>
{
type Error = SpiError<E, EO>;
fn read_many(&mut self,
reg: Register,
buffer: &mut [u8])
-> Result<(), Self::Error> {
buffer[0] = reg.read_address();
self.ncs.set_low().map_err(|a| SpiError::NCSError(a))?;
self.spi.transfer(buffer)?;
self.ncs.set_high().map_err(|a| SpiError::NCSError(a))?;
Ok(())
}
#[inline(never)]
fn write(&mut self, reg: Register, val: u8) -> Result<(), Self::Error> {
self.ncs.set_low().map_err(|a| SpiError::NCSError(a))?;
self.spi.write(&[reg.write_address(), val])?;
self.ncs.set_high().map_err(|a| SpiError::NCSError(a))?;
Ok(())
}
}
impl<SPI, NCS, E, EO> AK8963 for SpiDevice<SPI, NCS>
where SPI: spi::Write<u8, Error = E> + spi::Transfer<u8, Error = E>,
NCS: OutputPin<Error = EO>
{
type Error = SpiError<E, EO>;
fn init<D: DelayMs<u8>>(&mut self,
delay: &mut D)
-> Result<(), Self::Error> {
Device::write(self, Register::USER_CTRL, 0x32)?;
delay.delay_ms(10);
Ok(())
}
fn finalize<D: DelayMs<u8>>(&mut self,
delay: &mut D)
-> Result<(), Self::Error> {
Device::write(self, Register::I2C_MST_CTRL, 0x0d)?;
delay.delay_ms(10);
Device::write(self,
Register::I2C_SLV0_ADDR,
ak8963::I2C_ADDRESS | ak8963::R)?;
Device::write(self,
Register::I2C_SLV0_REG,
ak8963::Register::XOUT_L.addr())?;
Device::write(self, Register::I2C_SLV0_CTRL, 0x87)?;
delay.delay_ms(10);
Ok(())
}
fn read(&mut self, reg: ak8963::Register) -> Result<u8, Self::Error> {
Device::write(self,
Register::I2C_SLV4_ADDR,
ak8963::I2C_ADDRESS | ak8963::R)?;
Device::write(self, Register::I2C_SLV4_REG, reg.addr())?;
Device::write(self, Register::I2C_SLV4_CTRL, 0x80)?;
while Device::read(self, Register::I2C_MST_STATUS)? & (1 << 6) == 0 {}
Device::read(self, Register::I2C_SLV4_DI)
}
fn write(&mut self,
reg: ak8963::Register,
value: u8)
-> Result<(), Self::Error> {
Device::write(self,
Register::I2C_SLV4_ADDR,
ak8963::I2C_ADDRESS | ak8963::W)?;
Device::write(self, Register::I2C_SLV4_REG, reg.addr())?;
Device::write(self, Register::I2C_SLV4_DO, value)?;
Device::write(self, Register::I2C_SLV4_CTRL, 0x80)?;
while Device::read(self, Register::I2C_MST_STATUS)? & (1 << 6) == 0 {}
Ok(())
}
fn read_xyz(&mut self, buffer: &mut [u8; 7]) -> Result<(), Self::Error> {
Device::read_many(self, Register::EXT_SENS_DATA_00, buffer)?;
Ok(())
}
}
pub struct I2cDevice<I2C> {
i2c: I2C,
}
impl<E, I2C> I2cDevice<I2C>
where I2C: i2c::Read<Error = E>
+ i2c::Write<Error = E>
+ i2c::WriteRead<Error = E>
{
pub fn new(i2c: I2C) -> Self {
I2cDevice { i2c }
}
}
impl<E, I2C> Releasable for I2cDevice<I2C>
where I2C: i2c::Read<Error = E>
+ i2c::Write<Error = E>
+ i2c::WriteRead<Error = E>
{
type Released = I2C;
fn release(self) -> I2C {
self.i2c
}
}
impl<E, I2C> Device for I2cDevice<I2C>
where I2C: i2c::Read<Error = E>
+ i2c::Write<Error = E>
+ i2c::WriteRead<Error = E>
{
type Error = E;
fn read_many(&mut self,
reg: Register,
buffer: &mut [u8])
-> Result<(), Self::Error> {
self.i2c.write_read(MPU_I2C_ADDR, &[reg as u8], &mut buffer[1..])?;
Ok(())
}
fn write(&mut self, reg: Register, val: u8) -> Result<(), Self::Error> {
let buff: [u8; 2] = [reg as u8, val];
self.i2c.write(MPU_I2C_ADDR, &buff)
}
}
impl<I2C, E> AK8963 for I2cDevice<I2C>
where I2C: i2c::Read<Error = E>
+ i2c::Write<Error = E>
+ i2c::WriteRead<Error = E>
{
type Error = E;
fn init<D: DelayMs<u8>>(&mut self,
delay: &mut D)
-> Result<(), Self::Error> {
Device::write(self, Register::USER_CTRL, 0)?;
delay.delay_ms(10);
const LATCH_INT_EN: u8 = 1 << 5;
const INT_ANYRD_CLEAR: u8 = 1 << 4;
const ACTL_ACTIVE_LOW: u8 = 1 << 7;
const BYPASS_EN: u8 = 1 << 1;
Device::write(self,
Register::INT_PIN_CFG,
LATCH_INT_EN
| INT_ANYRD_CLEAR
| ACTL_ACTIVE_LOW
| BYPASS_EN)?;
delay.delay_ms(10);
Ok(())
}
fn read(&mut self, reg: ak8963::Register) -> Result<u8, Self::Error> {
let mut buffer = [0; 1];
self.i2c.write_read(ak8963::I2C_ADDRESS, &[reg.addr()], &mut buffer)?;
Ok(buffer[0])
}
fn write(&mut self,
reg: ak8963::Register,
value: u8)
-> Result<(), Self::Error> {
let buff: [u8; 2] = [reg.addr(), value];
self.i2c.write(ak8963::I2C_ADDRESS, &buff)
}
fn read_xyz(&mut self, buffer: &mut [u8; 7]) -> Result<(), Self::Error> {
self.i2c.write_read(ak8963::I2C_ADDRESS,
&[ak8963::Register::XOUT_L.addr()],
buffer)?;
buffer[..].rotate_right(1);
buffer[0] = 0;
Ok(())
}
}
pub trait NineDOFDevice:
Device + AK8963<Error = <Self as Device>::Error>
{
fn read_9dof(&mut self,
reg: Register,
buffer: &mut [u8; 21])
-> Result<(), <Self as Device>::Error>;
}
impl<SPI, NCS, E, EO> NineDOFDevice for SpiDevice<SPI, NCS>
where SPI: spi::Write<u8, Error = E> + spi::Transfer<u8, Error = E>,
NCS: OutputPin<Error = EO>
{
fn read_9dof(&mut self,
reg: Register,
buffer: &mut [u8; 21])
-> Result<(), <Self as Device>::Error> {
self.read_many(reg, buffer)?;
Ok(())
}
}
impl<I2C, E> NineDOFDevice for I2cDevice<I2C>
where I2C: i2c::Read<Error = E>
+ i2c::Write<Error = E>
+ i2c::WriteRead<Error = E>
{
fn read_9dof(&mut self,
reg: Register,
buffer: &mut [u8; 21])
-> Result<(), <Self as Device>::Error> {
Device::read_many(self, reg, &mut buffer[0..15])?;
let xyz_buffer = &mut [0; 7];
AK8963::read_xyz(self, xyz_buffer)?;
buffer[15..21].copy_from_slice(&xyz_buffer[1..7]);
Ok(())
}
}