stm32wb: add gatt server example
This commit is contained in:
parent
5b076cb0dd
commit
283ec756a9
@ -38,3 +38,7 @@ required-features = ["mac"]
|
|||||||
[[bin]]
|
[[bin]]
|
||||||
name = "eddystone_beacon"
|
name = "eddystone_beacon"
|
||||||
required-features = ["ble"]
|
required-features = ["ble"]
|
||||||
|
|
||||||
|
[[bin]]
|
||||||
|
name = "gatt_server"
|
||||||
|
required-features = ["ble"]
|
397
examples/stm32wb/src/bin/gatt_server.rs
Normal file
397
examples/stm32wb/src/bin/gatt_server.rs
Normal file
@ -0,0 +1,397 @@
|
|||||||
|
#![no_std]
|
||||||
|
#![no_main]
|
||||||
|
#![feature(type_alias_impl_trait)]
|
||||||
|
|
||||||
|
use core::time::Duration;
|
||||||
|
|
||||||
|
use defmt::*;
|
||||||
|
use embassy_executor::Spawner;
|
||||||
|
use embassy_stm32::bind_interrupts;
|
||||||
|
use embassy_stm32::ipcc::{Config, ReceiveInterruptHandler, TransmitInterruptHandler};
|
||||||
|
use embassy_stm32_wpan::hci::event::command::{CommandComplete, ReturnParameters};
|
||||||
|
use embassy_stm32_wpan::hci::host::uart::{Packet, 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::{
|
||||||
|
AddressType, AuthenticationRequirements, DiscoverableParameters, GapCommands, IoCapability, LocalName, Pin, Role,
|
||||||
|
SecureConnectionSupport,
|
||||||
|
};
|
||||||
|
use embassy_stm32_wpan::hci::vendor::stm32wb::command::gatt::{
|
||||||
|
AddCharacteristicParameters, AddServiceParameters, CharacteristicEvent, CharacteristicPermission,
|
||||||
|
CharacteristicProperty, EncryptionKeySize, GattCommands, ServiceType, UpdateCharacteristicValueParameters, Uuid,
|
||||||
|
WriteResponseParameters,
|
||||||
|
};
|
||||||
|
use embassy_stm32_wpan::hci::vendor::stm32wb::command::hal::{ConfigData, HalCommands, PowerLevel};
|
||||||
|
use embassy_stm32_wpan::hci::vendor::stm32wb::event::{self, AttributeHandle, Stm32Wb5xEvent};
|
||||||
|
use embassy_stm32_wpan::hci::{BdAddr, Event};
|
||||||
|
use embassy_stm32_wpan::lhci::LhciC1DeviceInformationCcrp;
|
||||||
|
use embassy_stm32_wpan::sub::ble::Ble;
|
||||||
|
use embassy_stm32_wpan::TlMbox;
|
||||||
|
use {defmt_rtt as _, panic_probe as _};
|
||||||
|
|
||||||
|
bind_interrupts!(struct Irqs{
|
||||||
|
IPCC_C1_RX => ReceiveInterruptHandler;
|
||||||
|
IPCC_C1_TX => TransmitInterruptHandler;
|
||||||
|
});
|
||||||
|
|
||||||
|
const BLE_GAP_DEVICE_NAME_LENGTH: u8 = 7;
|
||||||
|
|
||||||
|
#[embassy_executor::main]
|
||||||
|
async fn main(_spawner: Spawner) {
|
||||||
|
/*
|
||||||
|
How to make this work:
|
||||||
|
|
||||||
|
- Obtain a NUCLEO-STM32WB55 from your preferred supplier.
|
||||||
|
- Download and Install STM32CubeProgrammer.
|
||||||
|
- Download stm32wb5x_FUS_fw.bin, stm32wb5x_BLE_Stack_full_fw.bin, and Release_Notes.html from
|
||||||
|
gh:STMicroelectronics/STM32CubeWB@2234d97/Projects/STM32WB_Copro_Wireless_Binaries/STM32WB5x
|
||||||
|
- Open STM32CubeProgrammer
|
||||||
|
- On the right-hand pane, click "firmware upgrade" to upgrade the st-link firmware.
|
||||||
|
- Once complete, click connect to connect to the device.
|
||||||
|
- On the left hand pane, click the RSS signal icon to open "Firmware Upgrade Services".
|
||||||
|
- In the Release_Notes.html, find the memory address that corresponds to your device for the stm32wb5x_FUS_fw.bin file
|
||||||
|
- Select that file, the memory address, "verify download", and then "Firmware Upgrade".
|
||||||
|
- Once complete, in the Release_Notes.html, find the memory address that corresponds to your device for the
|
||||||
|
stm32wb5x_BLE_Stack_full_fw.bin file. It should not be the same memory address.
|
||||||
|
- Select that file, the memory address, "verify download", and then "Firmware Upgrade".
|
||||||
|
- Select "Start Wireless Stack".
|
||||||
|
- Disconnect from the device.
|
||||||
|
- In the examples folder for stm32wb, modify the memory.x file to match your target device.
|
||||||
|
- Run this example.
|
||||||
|
|
||||||
|
Note: extended stack versions are not supported at this time. Do not attempt to install a stack with "extended" in the name.
|
||||||
|
*/
|
||||||
|
|
||||||
|
let p = embassy_stm32::init(Default::default());
|
||||||
|
info!("Hello World!");
|
||||||
|
|
||||||
|
let config = Config::default();
|
||||||
|
let mut mbox = TlMbox::init(p.IPCC, Irqs, config);
|
||||||
|
|
||||||
|
let sys_event = mbox.sys_subsystem.read().await;
|
||||||
|
info!("sys event: {}", sys_event.payload());
|
||||||
|
|
||||||
|
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;
|
||||||
|
defmt::debug!("{}", 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;
|
||||||
|
defmt::debug!("{}", 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;
|
||||||
|
defmt::debug!("{}", 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;
|
||||||
|
defmt::debug!("{}", 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;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
info!("config tx power level...");
|
||||||
|
mbox.ble_subsystem.set_tx_power_level(PowerLevel::ZerodBm).await;
|
||||||
|
let response = mbox.ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
info!("GATT init...");
|
||||||
|
mbox.ble_subsystem.init_gatt().await;
|
||||||
|
let response = mbox.ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", 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;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
info!("set IO capabilities...");
|
||||||
|
mbox.ble_subsystem.set_io_capability(IoCapability::DisplayConfirm).await;
|
||||||
|
let response = mbox.ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
info!("set authentication requirements...");
|
||||||
|
mbox.ble_subsystem
|
||||||
|
.set_authentication_requirement(&AuthenticationRequirements {
|
||||||
|
bonding_required: false,
|
||||||
|
keypress_notification_support: false,
|
||||||
|
mitm_protection_required: false,
|
||||||
|
encryption_key_size_range: (8, 16),
|
||||||
|
fixed_pin: Pin::Requested,
|
||||||
|
identity_address_type: AddressType::Public,
|
||||||
|
secure_connection_support: SecureConnectionSupport::Optional,
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
let response = mbox.ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
info!("set scan response data...");
|
||||||
|
mbox.ble_subsystem.le_set_scan_response_data(b"TXTX").await.unwrap();
|
||||||
|
let response = mbox.ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
info!("set scan response data...");
|
||||||
|
mbox.ble_subsystem.le_set_scan_response_data(b"TXTX").await.unwrap();
|
||||||
|
let response = mbox.ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
defmt::info!("initializing services and characteristics...");
|
||||||
|
let mut ble_context = init_gatt_services(&mut mbox.ble_subsystem).await.unwrap();
|
||||||
|
defmt::info!("{}", ble_context);
|
||||||
|
|
||||||
|
let discovery_params = DiscoverableParameters {
|
||||||
|
advertising_type: AdvertisingType::ConnectableUndirected,
|
||||||
|
advertising_interval: Some((Duration::from_millis(100), Duration::from_millis(100))),
|
||||||
|
address_type: OwnAddressType::Public,
|
||||||
|
filter_policy: AdvertisingFilterPolicy::AllowConnectionAndScan,
|
||||||
|
local_name: Some(LocalName::Complete(b"TXTX")),
|
||||||
|
advertising_data: &[],
|
||||||
|
conn_interval: (None, None),
|
||||||
|
};
|
||||||
|
|
||||||
|
info!("set discoverable...");
|
||||||
|
mbox.ble_subsystem.set_discoverable(&discovery_params).await.unwrap();
|
||||||
|
let response = mbox.ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let response = mbox.ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
if let Ok(Packet::Event(event)) = response {
|
||||||
|
match event {
|
||||||
|
Event::LeConnectionComplete(_) => {
|
||||||
|
defmt::info!("connected");
|
||||||
|
}
|
||||||
|
Event::DisconnectionComplete(_) => {
|
||||||
|
defmt::info!("disconnected");
|
||||||
|
ble_context.is_subscribed = false;
|
||||||
|
mbox.ble_subsystem.set_discoverable(&discovery_params).await.unwrap();
|
||||||
|
}
|
||||||
|
Event::Vendor(vendor_event) => match vendor_event {
|
||||||
|
Stm32Wb5xEvent::AttReadPermitRequest(read_req) => {
|
||||||
|
defmt::info!("read request received {}, allowing", read_req);
|
||||||
|
mbox.ble_subsystem.allow_read(read_req.conn_handle).await
|
||||||
|
}
|
||||||
|
Stm32Wb5xEvent::AttWritePermitRequest(write_req) => {
|
||||||
|
defmt::info!("write request received {}, allowing", write_req);
|
||||||
|
mbox.ble_subsystem
|
||||||
|
.write_response(&WriteResponseParameters {
|
||||||
|
conn_handle: write_req.conn_handle,
|
||||||
|
attribute_handle: write_req.attribute_handle,
|
||||||
|
status: Ok(()),
|
||||||
|
value: write_req.value(),
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
.unwrap()
|
||||||
|
}
|
||||||
|
Stm32Wb5xEvent::GattAttributeModified(attribute) => {
|
||||||
|
defmt::info!("{}", ble_context);
|
||||||
|
if attribute.attr_handle.0 == ble_context.chars.notify.0 + 2 {
|
||||||
|
if attribute.data()[0] == 0x01 {
|
||||||
|
defmt::info!("subscribed");
|
||||||
|
ble_context.is_subscribed = true;
|
||||||
|
} else {
|
||||||
|
defmt::info!("unsubscribed");
|
||||||
|
ble_context.is_subscribed = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
},
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_bd_addr() -> BdAddr {
|
||||||
|
let mut bytes = [0u8; 6];
|
||||||
|
|
||||||
|
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;
|
||||||
|
|
||||||
|
BdAddr(bytes)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn get_random_addr() -> BdAddr {
|
||||||
|
let mut bytes = [0u8; 6];
|
||||||
|
|
||||||
|
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;
|
||||||
|
|
||||||
|
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)
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(defmt::Format)]
|
||||||
|
pub struct BleContext {
|
||||||
|
pub service_handle: AttributeHandle,
|
||||||
|
pub chars: CharHandles,
|
||||||
|
pub is_subscribed: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(defmt::Format)]
|
||||||
|
pub struct CharHandles {
|
||||||
|
pub read: AttributeHandle,
|
||||||
|
pub write: AttributeHandle,
|
||||||
|
pub notify: AttributeHandle,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn init_gatt_services(ble_subsystem: &mut Ble) -> Result<BleContext, ()> {
|
||||||
|
let service_handle = gatt_add_service(ble_subsystem, Uuid::Uuid16(0x500)).await?;
|
||||||
|
|
||||||
|
let read = gatt_add_char(
|
||||||
|
ble_subsystem,
|
||||||
|
service_handle,
|
||||||
|
Uuid::Uuid16(0x501),
|
||||||
|
CharacteristicProperty::READ,
|
||||||
|
Some(b"Hello from embassy!"),
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let write = gatt_add_char(
|
||||||
|
ble_subsystem,
|
||||||
|
service_handle,
|
||||||
|
Uuid::Uuid16(0x502),
|
||||||
|
CharacteristicProperty::WRITE_WITHOUT_RESPONSE | CharacteristicProperty::WRITE | CharacteristicProperty::READ,
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
let notify = gatt_add_char(
|
||||||
|
ble_subsystem,
|
||||||
|
service_handle,
|
||||||
|
Uuid::Uuid16(0x503),
|
||||||
|
CharacteristicProperty::NOTIFY | CharacteristicProperty::READ,
|
||||||
|
None,
|
||||||
|
)
|
||||||
|
.await?;
|
||||||
|
|
||||||
|
Ok(BleContext {
|
||||||
|
service_handle,
|
||||||
|
is_subscribed: false,
|
||||||
|
chars: CharHandles { read, write, notify },
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn gatt_add_service(ble_subsystem: &mut Ble, uuid: Uuid) -> Result<AttributeHandle, ()> {
|
||||||
|
ble_subsystem
|
||||||
|
.add_service(&AddServiceParameters {
|
||||||
|
uuid,
|
||||||
|
service_type: ServiceType::Primary,
|
||||||
|
max_attribute_records: 8,
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
let response = ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
if let Ok(Packet::Event(Event::CommandComplete(CommandComplete {
|
||||||
|
return_params:
|
||||||
|
ReturnParameters::Vendor(event::command::ReturnParameters::GattAddService(event::command::GattService {
|
||||||
|
service_handle,
|
||||||
|
..
|
||||||
|
})),
|
||||||
|
..
|
||||||
|
}))) = response
|
||||||
|
{
|
||||||
|
Ok(service_handle)
|
||||||
|
} else {
|
||||||
|
Err(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn gatt_add_char(
|
||||||
|
ble_subsystem: &mut Ble,
|
||||||
|
service_handle: AttributeHandle,
|
||||||
|
characteristic_uuid: Uuid,
|
||||||
|
characteristic_properties: CharacteristicProperty,
|
||||||
|
default_value: Option<&[u8]>,
|
||||||
|
) -> Result<AttributeHandle, ()> {
|
||||||
|
ble_subsystem
|
||||||
|
.add_characteristic(&AddCharacteristicParameters {
|
||||||
|
service_handle,
|
||||||
|
characteristic_uuid,
|
||||||
|
characteristic_properties,
|
||||||
|
characteristic_value_len: 32,
|
||||||
|
security_permissions: CharacteristicPermission::empty(),
|
||||||
|
gatt_event_mask: CharacteristicEvent::all(),
|
||||||
|
encryption_key_size: EncryptionKeySize::with_value(7).unwrap(),
|
||||||
|
is_variable: true,
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
let response = ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
|
||||||
|
if let Ok(Packet::Event(Event::CommandComplete(CommandComplete {
|
||||||
|
return_params:
|
||||||
|
ReturnParameters::Vendor(event::command::ReturnParameters::GattAddCharacteristic(
|
||||||
|
event::command::GattCharacteristic {
|
||||||
|
characteristic_handle, ..
|
||||||
|
},
|
||||||
|
)),
|
||||||
|
..
|
||||||
|
}))) = response
|
||||||
|
{
|
||||||
|
if let Some(value) = default_value {
|
||||||
|
ble_subsystem
|
||||||
|
.update_characteristic_value(&UpdateCharacteristicValueParameters {
|
||||||
|
service_handle,
|
||||||
|
characteristic_handle,
|
||||||
|
offset: 0,
|
||||||
|
value,
|
||||||
|
})
|
||||||
|
.await
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
|
let response = ble_subsystem.read().await;
|
||||||
|
defmt::debug!("{}", response);
|
||||||
|
}
|
||||||
|
Ok(characteristic_handle)
|
||||||
|
} else {
|
||||||
|
Err(())
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user