embassy/tests/stm32/src/bin/tl_mbox.rs

252 lines
8.0 KiB
Rust
Raw Normal View History

2023-05-30 00:10:36 +02:00
// required-features: ble
2023-05-04 00:36:31 +02:00
#![no_std]
#![no_main]
#![feature(type_alias_impl_trait)]
2023-05-30 00:10:36 +02:00
#[path = "../common.rs"]
mod common;
2023-05-04 00:36:31 +02:00
2023-06-23 04:05:51 +02:00
use core::time::Duration;
2023-05-30 00:10:36 +02:00
use common::*;
2023-05-04 00:36:31 +02:00
use embassy_executor::Spawner;
2023-06-12 15:44:30 +02:00
use embassy_stm32::bind_interrupts;
2023-06-17 19:00:33 +02:00
use embassy_stm32::ipcc::{Config, ReceiveInterruptHandler, TransmitInterruptHandler};
2023-06-24 00:54:06 +02:00
use embassy_stm32_wpan::hci::host::uart::UartHci;
use embassy_stm32_wpan::hci::host::{AdvertisingFilterPolicy, EncryptionKey, HostHci, OwnAddressType};
use embassy_stm32_wpan::hci::types::AdvertisingType;
use embassy_stm32_wpan::hci::vendor::stm32wb::command::gap::{
2023-06-23 04:05:51 +02:00
AdvertisingDataType, DiscoverableParameters, GapCommands, Role,
};
2023-06-24 00:54:06 +02:00
use embassy_stm32_wpan::hci::vendor::stm32wb::command::gatt::GattCommands;
use embassy_stm32_wpan::hci::vendor::stm32wb::command::hal::{ConfigData, HalCommands, PowerLevel};
use embassy_stm32_wpan::hci::BdAddr;
2023-06-23 04:05:51 +02:00
use embassy_stm32_wpan::lhci::LhciC1DeviceInformationCcrp;
2023-06-24 00:54:06 +02:00
use embassy_stm32_wpan::sub::mm;
use embassy_stm32_wpan::TlMbox;
2023-06-23 04:05:51 +02:00
use {defmt_rtt as _, panic_probe as _};
2023-05-04 00:36:31 +02:00
2023-05-26 11:03:01 +02:00
bind_interrupts!(struct Irqs{
2023-06-17 19:00:33 +02:00
IPCC_C1_RX => ReceiveInterruptHandler;
IPCC_C1_TX => TransmitInterruptHandler;
2023-05-26 11:03:01 +02:00
});
2023-06-23 04:05:51 +02:00
const BLE_GAP_DEVICE_NAME_LENGTH: u8 = 7;
2023-06-17 19:00:33 +02:00
#[embassy_executor::task]
2023-06-17 22:37:34 +02:00
async fn run_mm_queue(memory_manager: mm::MemoryManager) {
memory_manager.run_queue().await;
2023-06-17 19:00:33 +02:00
}
2023-05-04 00:36:31 +02:00
#[embassy_executor::main]
2023-06-23 04:05:51 +02:00
async fn main(_spawner: Spawner) {
let p = embassy_stm32::init(config());
2023-05-04 00:36:31 +02:00
info!("Hello World!");
let config = Config::default();
2023-06-23 04:05:51 +02:00
let mut mbox = TlMbox::init(p.IPCC, Irqs, config);
2023-06-17 19:00:33 +02:00
2023-06-23 04:05:51 +02:00
// spawner.spawn(run_mm_queue(mbox.mm_subsystem)).unwrap();
2023-06-23 04:05:51 +02:00
let sys_event = mbox.sys_subsystem.read().await;
info!("sys event: {}", sys_event.payload());
2023-06-17 19:00:33 +02:00
2023-06-17 22:37:34 +02:00
let fw_info = mbox.sys_subsystem.wireless_fw_info().unwrap();
let version_major = fw_info.version_major();
let version_minor = fw_info.version_minor();
let subversion = fw_info.subversion();
2023-05-04 00:36:31 +02:00
2023-06-17 22:37:34 +02:00
let sram2a_size = fw_info.sram2a_size();
let sram2b_size = fw_info.sram2b_size();
2023-05-04 00:36:31 +02:00
2023-06-17 22:37:34 +02:00
info!(
"version {}.{}.{} - SRAM2a {} - SRAM2b {}",
version_major, version_minor, subversion, sram2a_size, sram2b_size
);
2023-05-04 00:36:31 +02:00
2023-06-23 04:05:51 +02:00
mbox.sys_subsystem.shci_c2_ble_init(Default::default()).await;
info!("resetting BLE...");
mbox.ble_subsystem.reset().await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("config public address...");
mbox.ble_subsystem
.write_config_data(&ConfigData::public_address(get_bd_addr()).build())
.await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("config random address...");
mbox.ble_subsystem
.write_config_data(&ConfigData::random_address(get_random_addr()).build())
.await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("config identity root...");
mbox.ble_subsystem
.write_config_data(&ConfigData::identity_root(&get_irk()).build())
.await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("config encryption root...");
mbox.ble_subsystem
.write_config_data(&ConfigData::encryption_root(&get_erk()).build())
.await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("config tx power level...");
mbox.ble_subsystem.set_tx_power_level(PowerLevel::ZerodBm).await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("GATT init...");
mbox.ble_subsystem.init_gatt().await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("GAP init...");
mbox.ble_subsystem
.init_gap(Role::PERIPHERAL, false, BLE_GAP_DEVICE_NAME_LENGTH)
.await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
// info!("set scan response...");
// mbox.ble_subsystem.le_set_scan_response_data(&[]).await.unwrap();
// let response = mbox.ble_subsystem.read().await.unwrap();
// info!("{}", response);
info!("set discoverable...");
mbox.ble_subsystem
.set_discoverable(&DiscoverableParameters {
advertising_type: AdvertisingType::NonConnectableUndirected,
advertising_interval: Some((Duration::from_millis(250), Duration::from_millis(250))),
address_type: OwnAddressType::Public,
filter_policy: AdvertisingFilterPolicy::AllowConnectionAndScan,
local_name: None,
advertising_data: &[],
conn_interval: (None, None),
})
.await
.unwrap();
let response = mbox.ble_subsystem.read().await;
info!("{}", response);
// remove some advertisement to decrease the packet size
info!("delete tx power ad type...");
mbox.ble_subsystem
.delete_ad_type(AdvertisingDataType::TxPowerLevel)
.await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("delete conn interval ad type...");
mbox.ble_subsystem
.delete_ad_type(AdvertisingDataType::PeripheralConnectionInterval)
.await;
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("update advertising data...");
mbox.ble_subsystem
.update_advertising_data(&eddystone_advertising_data())
.await
.unwrap();
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("update advertising data type...");
mbox.ble_subsystem
.update_advertising_data(&[3, AdvertisingDataType::UuidCompleteList16 as u8, 0xaa, 0xfe])
.await
.unwrap();
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("update advertising data flags...");
mbox.ble_subsystem
.update_advertising_data(&[
2,
AdvertisingDataType::Flags as u8,
(0x02 | 0x04) as u8, // BLE general discoverable, without BR/EDR support
])
.await
.unwrap();
let response = mbox.ble_subsystem.read().await.unwrap();
info!("{}", response);
info!("Test OK");
cortex_m::asm::bkpt();
}
fn get_bd_addr() -> BdAddr {
let mut bytes = [0u8; 6];
2023-05-04 00:36:31 +02:00
2023-06-23 04:05:51 +02:00
let lhci_info = LhciC1DeviceInformationCcrp::new();
bytes[0] = (lhci_info.uid64 & 0xff) as u8;
bytes[1] = ((lhci_info.uid64 >> 8) & 0xff) as u8;
bytes[2] = ((lhci_info.uid64 >> 16) & 0xff) as u8;
bytes[3] = lhci_info.device_type_id;
bytes[4] = (lhci_info.st_company_id & 0xff) as u8;
bytes[5] = (lhci_info.st_company_id >> 8 & 0xff) as u8;
2023-06-14 00:12:34 +02:00
2023-06-23 04:05:51 +02:00
BdAddr(bytes)
}
2023-06-16 04:02:10 +02:00
2023-06-23 04:05:51 +02:00
fn get_random_addr() -> BdAddr {
let mut bytes = [0u8; 6];
2023-06-23 04:05:51 +02:00
let lhci_info = LhciC1DeviceInformationCcrp::new();
bytes[0] = (lhci_info.uid64 & 0xff) as u8;
bytes[1] = ((lhci_info.uid64 >> 8) & 0xff) as u8;
bytes[2] = ((lhci_info.uid64 >> 16) & 0xff) as u8;
bytes[3] = 0;
bytes[4] = 0x6E;
bytes[5] = 0xED;
2023-06-23 04:05:51 +02:00
BdAddr(bytes)
}
const BLE_CFG_IRK: [u8; 16] = [
0x12, 0x34, 0x56, 0x78, 0x9a, 0xbc, 0xde, 0xf0, 0x12, 0x34, 0x56, 0x78, 0x9a, 0xbc, 0xde, 0xf0,
];
const BLE_CFG_ERK: [u8; 16] = [
0xfe, 0xdc, 0xba, 0x09, 0x87, 0x65, 0x43, 0x21, 0xfe, 0xdc, 0xba, 0x09, 0x87, 0x65, 0x43, 0x21,
];
fn get_irk() -> EncryptionKey {
EncryptionKey(BLE_CFG_IRK)
}
fn get_erk() -> EncryptionKey {
EncryptionKey(BLE_CFG_ERK)
}
fn eddystone_advertising_data() -> [u8; 24] {
const EDDYSTONE_URL: &[u8] = b"www.rust-lang.com";
let mut service_data = [0u8; 24];
let url_len = EDDYSTONE_URL.len();
service_data[0] = 6 + url_len as u8;
service_data[1] = AdvertisingDataType::ServiceData as u8;
// 16-bit eddystone uuid
service_data[2] = 0xaa;
service_data[3] = 0xFE;
service_data[4] = 0x10; // URL frame type
service_data[5] = 22_i8 as u8; // calibrated TX power at 0m
service_data[6] = 0x03; // eddystone url prefix = https
service_data[7..(7 + url_len)].copy_from_slice(EDDYSTONE_URL);
service_data
2023-05-04 00:36:31 +02:00
}