#[cfg(feature = "xb100")]
mod xb100;
#[cfg(feature = "xb200")]
pub mod xb200;
#[cfg(feature = "xb300")]
mod xb300;
use crate::bladerf1::board::RfLinkSession;
use crate::error::{Error, Result};
#[cfg(any(feature = "xb100", feature = "xb200", feature = "xb300"))]
use crate::nios_client::NiosCore;
#[derive(Clone, Copy, Debug, PartialEq, Eq)]
pub enum ExpansionBoard {
XbNone = 0,
#[cfg(feature = "xb100")]
Xb100,
#[cfg(feature = "xb200")]
Xb200,
#[cfg(feature = "xb300")]
Xb300,
}
#[cfg(any(feature = "xb100", feature = "xb200"))]
impl NiosCore {
pub(crate) fn detect_xb_board(&mut self, check_mask: u32) -> Result<bool> {
let gpio = self.nios_expansion_gpio_read()?;
if gpio == 0xffffffff {
return Ok(false);
}
Ok((gpio & check_mask) != 0)
}
}
#[cfg(feature = "xb300")]
impl NiosCore {
pub(crate) fn detect_xb_board_by_dir(&mut self, check_mask: u32) -> Result<bool> {
let gpio_dir = self.nios_expansion_gpio_dir_read()?;
Ok((gpio_dir & check_mask) != 0)
}
}
#[cfg(feature = "xb200")]
impl NiosCore {
pub(crate) fn xb200_is_enabled(&mut self) -> Result<bool> {
self.detect_xb_board(xb200::BLADERF_XB_RF_ON)
}
}
impl RfLinkSession<'_> {
pub fn expansion_gpio_read(&mut self) -> Result<u32> {
self.nios.nios_expansion_gpio_read()
}
pub fn expansion_gpio_write(&mut self, val: u32) -> Result<()> {
self.nios.nios_expansion_gpio_write(0xffffffff, val)
}
pub fn expansion_gpio_masked_write(&mut self, mask: u32, val: u32) -> Result<()> {
self.nios.nios_expansion_gpio_write(mask, val)
}
pub fn expansion_gpio_dir_read(&mut self) -> Result<u32> {
self.nios.nios_expansion_gpio_dir_read()
}
pub fn expansion_gpio_dir_write(&mut self, val: u32) -> Result<()> {
self.nios.nios_expansion_gpio_dir_write(0xffffffff, val)
}
pub fn expansion_gpio_dir_masked_write(&mut self, mask: u32, val: u32) -> Result<()> {
self.nios.nios_expansion_gpio_dir_write(mask, val)
}
pub fn expansion_get_attached(&mut self) -> Result<ExpansionBoard> {
self.require_initialized()?;
if self.nios.nios_expansion_gpio_read()? == 0xffffffff {
return Ok(ExpansionBoard::XbNone);
}
#[cfg(feature = "xb100")]
if self.nios.detect_xb_board(xb100::XB100_DETECT_MASK)? {
return Ok(ExpansionBoard::Xb100);
}
#[cfg(feature = "xb200")]
if self.nios.detect_xb_board(xb200::BLADERF_XB_RF_ON)? {
return Ok(ExpansionBoard::Xb200);
}
#[cfg(feature = "xb300")]
if self.nios.detect_xb_board_by_dir(xb300::XB300_DETECT_MASK)? {
return Ok(ExpansionBoard::Xb300);
}
Ok(ExpansionBoard::XbNone)
}
pub fn expansion_attach(&mut self, xb: ExpansionBoard) -> Result<()> {
self.require_initialized()?;
let attached = self.expansion_get_attached()?;
if xb != attached && attached != ExpansionBoard::XbNone {
log::error!("Switching XB types is not supported.");
return Err(Error::Unsupported("switching XB types"));
}
#[cfg(feature = "xb100")]
if xb == ExpansionBoard::Xb100 {
self.xb100_attach()?;
self.xb100_enable(true)?;
self.xb100_init()?;
return Ok(());
}
#[cfg(feature = "xb200")]
if xb == ExpansionBoard::Xb200 {
self.xb200_attach()?;
self.xb200_enable(true)?;
self.xb200_init()?;
return Ok(());
}
#[cfg(feature = "xb300")]
if xb == ExpansionBoard::Xb300 {
self.xb300_attach()?;
self.xb300_enable(true)?;
self.xb300_init()?;
return Ok(());
}
if xb == ExpansionBoard::XbNone {
log::error!("Disabling an attached XB is not supported.");
return Err(Error::Unsupported("disabling attached XB"));
}
log::error!("Unknown xb type: {xb:?}");
Err(Error::Unsupported("unknown XB type"))
}
}