use crate::{
bindings,
error::{get_errno, Error},
rtos::DataSource,
};
pub struct AdiEncoder {
port: bindings::ext_adi_encoder_t,
}
impl AdiEncoder {
pub unsafe fn new(
top_port: u8,
bottom_port: u8,
reverse: bool,
extender_port: u8,
) -> Result<AdiEncoder, AdiEncoderError> {
match bindings::ext_adi_encoder_init(extender_port, top_port, bottom_port, reverse) {
bindings::PROS_ERR_ => Err(AdiEncoderError::from_errno()),
x => Ok(AdiEncoder { port: x }),
}
}
pub fn reset(&mut self) -> Result<(), AdiEncoderError> {
match unsafe { bindings::ext_adi_encoder_reset(self.port) } {
bindings::PROS_ERR_ => Err(AdiEncoderError::from_errno()),
_ => Ok(()),
}
}
pub fn get(&self) -> Result<i32, AdiEncoderError> {
match unsafe { bindings::ext_adi_encoder_get(self.port) } {
bindings::PROS_ERR_ => Err(AdiEncoderError::from_errno()),
x => Ok(x),
}
}
}
impl DataSource for AdiEncoder {
type Data = i32;
type Error = AdiEncoderError;
fn read(&self) -> Result<Self::Data, Self::Error> {
self.get()
}
}
impl Drop for AdiEncoder {
fn drop(&mut self) {
if let bindings::PROS_ERR_ = unsafe { bindings::ext_adi_encoder_shutdown(self.port) } {
panic!(
"failed to shutdown ADI encoder: {:?}",
AdiEncoderError::from_errno()
);
}
}
}
#[derive(Debug)]
pub enum AdiEncoderError {
PortsOutOfRange,
PortsNotAdiEncoder,
PortNonMatchingExtenders,
Unknown(i32),
}
impl AdiEncoderError {
fn from_errno() -> Self {
match get_errno() {
libc::ENXIO => Self::PortsOutOfRange,
libc::ENODEV => Self::PortsNotAdiEncoder,
x => Self::Unknown(x),
}
}
}
impl From<AdiEncoderError> for Error {
fn from(err: AdiEncoderError) -> Self {
match err {
AdiEncoderError::PortsOutOfRange => Error::Custom("ports out of range".into()),
AdiEncoderError::PortsNotAdiEncoder => Error::Custom("ports not an adi encoder".into()),
AdiEncoderError::PortNonMatchingExtenders => {
Error::Custom("ports from non-matching extenders".into())
}
AdiEncoderError::Unknown(n) => Error::System(n),
}
}
}