wpan: add BLE HCI
This commit is contained in:
@ -1,6 +1,7 @@
|
||||
use core::marker::PhantomData;
|
||||
|
||||
use embassy_stm32::ipcc::Ipcc;
|
||||
use hci::Opcode;
|
||||
|
||||
use crate::channels;
|
||||
use crate::cmd::CmdPacket;
|
||||
@ -29,7 +30,7 @@ impl Ble {
|
||||
Self { phantom: PhantomData }
|
||||
}
|
||||
/// `HW_IPCC_BLE_EvtNot`
|
||||
pub async fn read(&self) -> EvtBox {
|
||||
pub async fn tl_read(&self) -> EvtBox {
|
||||
Ipcc::receive(channels::cpu2::IPCC_BLE_EVENT_CHANNEL, || unsafe {
|
||||
if let Some(node_ptr) = LinkedListNode::remove_head(EVT_QUEUE.as_mut_ptr()) {
|
||||
Some(EvtBox::new(node_ptr.cast()))
|
||||
@ -41,7 +42,7 @@ impl Ble {
|
||||
}
|
||||
|
||||
/// `TL_BLE_SendCmd`
|
||||
pub async fn write(&self, opcode: u16, payload: &[u8]) {
|
||||
pub async fn tl_write(&self, opcode: u16, payload: &[u8]) {
|
||||
Ipcc::send(channels::cpu1::IPCC_BLE_CMD_CHANNEL, || unsafe {
|
||||
CmdPacket::write_into(BLE_CMD_BUFFER.as_mut_ptr(), TlPacketType::BleCmd, opcode, payload);
|
||||
})
|
||||
@ -61,3 +62,17 @@ impl Ble {
|
||||
.await;
|
||||
}
|
||||
}
|
||||
|
||||
pub extern crate bluetooth_hci_async as hci;
|
||||
|
||||
impl hci::Controller for Ble {
|
||||
async fn controller_write(&mut self, opcode: Opcode, payload: &[u8]) {
|
||||
self.tl_write(opcode.0, payload).await;
|
||||
}
|
||||
|
||||
async fn controller_read(&self) -> &[u8] {
|
||||
let evt_box = self.tl_read().await;
|
||||
|
||||
evt_box.serial()
|
||||
}
|
||||
}
|
||||
|
@ -52,7 +52,7 @@ impl CmdPacket {
|
||||
p_cmd_serial,
|
||||
CmdSerialStub {
|
||||
ty: packet_type as u8,
|
||||
cmd_code: cmd_code,
|
||||
cmd_code,
|
||||
payload_len: payload.len() as u8,
|
||||
},
|
||||
);
|
||||
|
@ -1,6 +1,7 @@
|
||||
use core::{ptr, slice};
|
||||
|
||||
use super::PacketHeader;
|
||||
use crate::consts::TL_EVT_HEADER_SIZE;
|
||||
|
||||
/**
|
||||
* The payload of `Evt` for a command status event
|
||||
@ -105,6 +106,14 @@ impl EvtBox {
|
||||
Self { ptr }
|
||||
}
|
||||
|
||||
pub fn evt<'a>(&self) -> &'a [u8] {
|
||||
unsafe {
|
||||
let evt_packet = &(*self.ptr);
|
||||
|
||||
core::slice::from_raw_parts(evt_packet as *const _ as *const u8, core::mem::size_of::<EvtPacket>())
|
||||
}
|
||||
}
|
||||
|
||||
/// Returns information about the event
|
||||
pub fn stub(&self) -> EvtStub {
|
||||
unsafe {
|
||||
@ -124,6 +133,20 @@ impl EvtBox {
|
||||
slice::from_raw_parts(p_payload, payload_len as usize)
|
||||
}
|
||||
}
|
||||
|
||||
/// writes an underlying [`EvtPacket`] into the provided buffer.
|
||||
/// Returns the number of bytes that were written.
|
||||
/// Returns an error if event kind is unknown or if provided buffer size is not enough.
|
||||
pub fn serial<'a>(&self) -> &'a [u8] {
|
||||
unsafe {
|
||||
let evt_serial: *const EvtSerial = &(*self.ptr).evt_serial;
|
||||
let evt_serial_buf: *const u8 = evt_serial.cast();
|
||||
|
||||
let len = (*evt_serial).evt.payload_len as usize + TL_EVT_HEADER_SIZE;
|
||||
|
||||
slice::from_raw_parts(evt_serial_buf, len)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl Drop for EvtBox {
|
||||
|
111
embassy-stm32-wpan/src/lhci.rs
Normal file
111
embassy-stm32-wpan/src/lhci.rs
Normal file
@ -0,0 +1,111 @@
|
||||
use crate::cmd::CmdPacket;
|
||||
use crate::consts::{TlPacketType, TL_EVT_HEADER_SIZE};
|
||||
use crate::evt::{CcEvt, EvtPacket, EvtSerial};
|
||||
use crate::tables::{DeviceInfoTable, RssInfoTable, SafeBootInfoTable, WirelessFwInfoTable};
|
||||
use crate::TL_REF_TABLE;
|
||||
|
||||
const TL_BLEEVT_CC_OPCODE: u8 = 0x0e;
|
||||
const LHCI_OPCODE_C1_DEVICE_INF: u16 = 0xfd62;
|
||||
|
||||
const PACKAGE_DATA_PTR: *const u8 = 0x1FFF_7500 as _;
|
||||
const UID64_PTR: *const u32 = 0x1FFF_7580 as _;
|
||||
|
||||
#[derive(Debug, Copy, Clone)]
|
||||
#[repr(C, packed)]
|
||||
pub struct LhciC1DeviceInformationCcrp {
|
||||
pub status: u8,
|
||||
pub rev_id: u16,
|
||||
pub dev_code_id: u16,
|
||||
pub package_type: u8,
|
||||
pub device_type_id: u8,
|
||||
pub st_company_id: u32,
|
||||
pub uid64: u32,
|
||||
|
||||
pub uid96_0: u32,
|
||||
pub uid96_1: u32,
|
||||
pub uid96_2: u32,
|
||||
|
||||
pub safe_boot_info_table: SafeBootInfoTable,
|
||||
pub rss_info_table: RssInfoTable,
|
||||
pub wireless_fw_info_table: WirelessFwInfoTable,
|
||||
|
||||
pub app_fw_inf: u32,
|
||||
}
|
||||
|
||||
impl Default for LhciC1DeviceInformationCcrp {
|
||||
fn default() -> Self {
|
||||
let DeviceInfoTable {
|
||||
safe_boot_info_table,
|
||||
rss_info_table,
|
||||
wireless_fw_info_table,
|
||||
} = unsafe { &*(*TL_REF_TABLE.as_ptr()).device_info_table }.clone();
|
||||
|
||||
let device_id = stm32_device_signature::device_id();
|
||||
let uid96_0 = (device_id[3] as u32) << 24
|
||||
| (device_id[2] as u32) << 16
|
||||
| (device_id[1] as u32) << 8
|
||||
| device_id[0] as u32;
|
||||
let uid96_1 = (device_id[7] as u32) << 24
|
||||
| (device_id[6] as u32) << 16
|
||||
| (device_id[5] as u32) << 8
|
||||
| device_id[4] as u32;
|
||||
let uid96_2 = (device_id[11] as u32) << 24
|
||||
| (device_id[10] as u32) << 16
|
||||
| (device_id[9] as u32) << 8
|
||||
| device_id[8] as u32;
|
||||
|
||||
let package_type = unsafe { *PACKAGE_DATA_PTR };
|
||||
let uid64 = unsafe { *UID64_PTR };
|
||||
let st_company_id = unsafe { *UID64_PTR.offset(1) } >> 8 & 0x00FF_FFFF;
|
||||
let device_type_id = (unsafe { *UID64_PTR.offset(1) } & 0x000000FF) as u8;
|
||||
|
||||
LhciC1DeviceInformationCcrp {
|
||||
status: 0,
|
||||
rev_id: 0,
|
||||
dev_code_id: 0,
|
||||
package_type,
|
||||
device_type_id,
|
||||
st_company_id,
|
||||
uid64,
|
||||
uid96_0,
|
||||
uid96_1,
|
||||
uid96_2,
|
||||
safe_boot_info_table,
|
||||
rss_info_table,
|
||||
wireless_fw_info_table,
|
||||
app_fw_inf: (1 << 8), // 0.0.1
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
impl LhciC1DeviceInformationCcrp {
|
||||
pub fn new() -> Self {
|
||||
Self::default()
|
||||
}
|
||||
|
||||
pub fn write(&self, cmd_packet: &mut CmdPacket) {
|
||||
let self_size = core::mem::size_of::<LhciC1DeviceInformationCcrp>();
|
||||
|
||||
unsafe {
|
||||
let cmd_packet_ptr: *mut CmdPacket = cmd_packet;
|
||||
let evet_packet_ptr: *mut EvtPacket = cmd_packet_ptr.cast();
|
||||
|
||||
let evt_serial: *mut EvtSerial = &mut (*evet_packet_ptr).evt_serial;
|
||||
let evt_payload = (*evt_serial).evt.payload.as_mut_ptr();
|
||||
let evt_cc: *mut CcEvt = evt_payload.cast();
|
||||
let evt_cc_payload_buf = (*evt_cc).payload.as_mut_ptr();
|
||||
|
||||
(*evt_serial).kind = TlPacketType::LocRsp as u8;
|
||||
(*evt_serial).evt.evt_code = TL_BLEEVT_CC_OPCODE;
|
||||
(*evt_serial).evt.payload_len = TL_EVT_HEADER_SIZE as u8 + self_size as u8;
|
||||
|
||||
(*evt_cc).cmd_code = LHCI_OPCODE_C1_DEVICE_INF;
|
||||
(*evt_cc).num_cmd = 1;
|
||||
|
||||
let self_ptr: *const LhciC1DeviceInformationCcrp = self;
|
||||
let self_buf = self_ptr.cast();
|
||||
|
||||
core::ptr::copy(self_buf, evt_cc_payload_buf, self_size);
|
||||
}
|
||||
}
|
||||
}
|
@ -1,4 +1,5 @@
|
||||
#![no_std]
|
||||
#![cfg_attr(feature = "ble", feature(async_fn_in_trait))]
|
||||
|
||||
// This must go FIRST so that all the other modules see its macros.
|
||||
pub mod fmt;
|
||||
@ -21,6 +22,7 @@ pub mod channels;
|
||||
pub mod cmd;
|
||||
pub mod consts;
|
||||
pub mod evt;
|
||||
pub mod lhci;
|
||||
#[cfg(feature = "mac")]
|
||||
pub mod mac;
|
||||
pub mod mm;
|
||||
|
Reference in New Issue
Block a user