1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
|
#![no_std]
#![no_main]
#![feature(type_alias_impl_trait)]
#[path = "../example_common.rs"]
mod example_common;
use example_common::*;
use core::mem;
use cortex_m_rt::entry;
use defmt::info;
use embassy::executor::{task, Executor};
use embassy::util::Forever;
use nrf_softdevice::ble::{central, gatt_client, Address, Connection, Uuid};
use nrf_softdevice::raw;
use nrf_softdevice::Softdevice;
static EXECUTOR: Forever<Executor> = Forever::new();
#[task]
async fn softdevice_task(sd: &'static Softdevice) {
sd.run().await;
}
#[nrf_softdevice::gatt_client(uuid = "180f")]
struct BatteryServiceClient {
#[characteristic(uuid = "2a19", read, write, notify)]
battery_level: u8,
}
#[task]
async fn ble_central_task(sd: &'static Softdevice, config: central::Config) {
let addrs = &[Address::new_random_static([
0x06, 0x6b, 0x71, 0x2c, 0xf5, 0xc0,
])];
let conn = central::connect(sd, addrs, config)
.await
.dexpect(intern!("connect"));
info!("connected");
let client: BatteryServiceClient = gatt_client::discover(&conn)
.await
.dexpect(intern!("discover"));
// Read
let val = client.battery_level_read().await.dexpect(intern!("read"));
info!("read battery level: {:u8}", val);
// Write, set it to 42
client
.battery_level_write(42)
.await
.dexpect(intern!("write"));
info!("Wrote battery level!");
// Read to check it's changed
let val = client.battery_level_read().await.dexpect(intern!("read"));
info!("read battery level: {:u8}", val);
}
#[entry]
fn main() -> ! {
info!("Hello World!");
let config = nrf_softdevice::Config {
clock: Some(raw::nrf_clock_lf_cfg_t {
source: raw::NRF_CLOCK_LF_SRC_XTAL as u8,
rc_ctiv: 0,
rc_temp_ctiv: 0,
accuracy: 7,
}),
conn_gap: Some(raw::ble_gap_conn_cfg_t {
conn_count: 6,
event_length: 6,
}),
conn_gatt: Some(raw::ble_gatt_conn_cfg_t { att_mtu: 128 }),
gatts_attr_tab_size: Some(raw::ble_gatts_cfg_attr_tab_size_t {
attr_tab_size: 32768,
}),
gap_role_count: Some(raw::ble_gap_cfg_role_count_t {
adv_set_count: 1,
periph_role_count: 3,
central_role_count: 3,
central_sec_count: 0,
_bitfield_1: raw::ble_gap_cfg_role_count_t::new_bitfield_1(0),
}),
gap_device_name: Some(raw::ble_gap_cfg_device_name_t {
p_value: b"HelloRust" as *const u8 as _,
current_len: 9,
max_len: 9,
write_perm: unsafe { mem::zeroed() },
_bitfield_1: raw::ble_gap_cfg_device_name_t::new_bitfield_1(
raw::BLE_GATTS_VLOC_STACK as u8,
),
}),
..Default::default()
};
let (sdp, p) = take_peripherals();
let sd = Softdevice::enable(sdp, &config);
let executor = EXECUTOR.put(Executor::new(cortex_m::asm::sev));
executor.spawn(softdevice_task(sd)).dewrap();
executor
.spawn(ble_central_task(sd, central::Config::default()))
.dewrap();
loop {
executor.run();
cortex_m::asm::wfe();
}
}
|