
0 前言6 G4 i5 e9 k5 T) }6 t# v3 k 自己的主要目的是学习BLE相关的知识,以及其具体的应用。因此不会像前一小节那样,组成那么长链路的系统。为最佳的学习BLE,就需要组成最小的闭环回路。# e% V @/ `/ g+ V7 W " N3 b; O; A/ C 1 最小闭环回路 最小闭环系统,就是STM32WB的开发板和笔记本组成的系统。现在基本的笔记本自带蓝牙。 1 ![]() $ s& u) J9 i5 B7 t9 s# V P 1 s, Y' H8 A0 b* [ 2 系统软件环境( a0 K: x5 q; P% U. [( ?6 k# o 在开发板中的软件使用的是:P-NUCLEO-WB55.Nucleo\Applications\BLE\BLE_HeartRate项目工程。 笔记本使用上位机是自己编写的python代码。 2 ![]() , R4 |, R. w/ e2 D% @5 m" A# U; V8 C 我们将开发板当作外围设备。将笔记本作为中心设备。( ]! y3 y2 X. ?4 r* W) N' a/ e) W4 j 7 T1 Z* T- R* b4 T 3 连接实验 3.1 发现设备 下载好单片机程序,给单片机上电。/ ]% ]2 ~3 W/ j6 c 在笔记本设置中打开蓝牙,并点击添加蓝牙。: n+ y9 l. o0 E( | 3 ![]() # [; Y8 ]- {$ q h7 X# l7 a# j% Q( k 可以看到我们的设备HRSTM。 接下来使用我编写好代码: import bleak.backends.dotnet.discovery as Discovery( G* ~* c$ X/ O- G; s& q import asyncio O5 V4 Z5 L: V7 P$ _ async def run():0 [) y3 @, t! `+ s/ `5 M8 v) ~* q devices = await Discovery.discover()! @. T: ^- m$ Q0 @+ {( `6 F/ |# p for d in devices:/ W) _0 A; F8 H! i9 j$ {* a# F print(d)' R; x( A! i7 E loop = asyncio.get_event_loop(), b/ D' I3 e8 }$ C5 B N6 [ loop.run_until_complete(run()) 在输出窗口可以看到可以被发现了的蓝牙:) s- m ~( ^$ e M$ t& | B8:7C:6F:47:81:BF: MiKettle 6B:96:A5:AF:06:A7: Apple, Inc. (b'\x10\x05Q\x1c\x82.\xec') 80:E1:26:00:68:7C: HRSTM 我们也可以看到我们的蓝牙应用HRSTM。其中,一个很重要的东西就是设备的ID,这个是我们连接的一个关键,就和电话号码一样,而HRSTM就是持有人的别名。. i9 Y3 N# G% A0 M' X4 { 8 x6 ~9 {/ v1 Y & z, U% ^( Q) t8 F 3.2 连接和发现服务 接下来,我们应用上面的地址,来连接和发现一下HRSTM有哪些服务:; S/ a* x- v2 _7 M' R async with BleakClient(address, loop=loop) as client:& f, K& U+ k- e$ L4 v4 ?# t4 f x = await client.is_connected() log.info("Connected: {0}".format(x)) log.info("[Service] {0}: {1}".format(service.uuid, service.description)) for char in service.characteristics:% h; t% ]0 X% F8 Q! ]0 | if "read" in char.properties:8 [' T. o' w8 B: s# y$ x try:. D3 Y8 v+ f# e# b1 T value = bytes(await client.read_gatt_char(char.uuid))% T! {1 U& c4 T3 R except Exception as e: value = str(e).encode() else:; v& q# P+ I: x! u5 H9 ]. { value = None log.info(! ]9 z d) C4 p/ H( Z "\t[Characteristic] {0}: ({1}) | Name: {2}, Value: {3} ".format(: e4 B) W3 i: s4 @ char.uuid, ",".join(char.properties), char.description, value )4 ]. \9 O7 K! E' m; P& e- s3 T( h ), }# w- {$ P8 s) W for descriptor in char.descriptors: value = await client.read_gatt_descriptor(descriptor.handle) c# R) y3 ^, q9 ~6 c9 ~8 w4 Q1 @8 f log.info(+ _* F: r# M- z% f( i/ h "\t\t[Descriptor] {0}: (Handle: {1}) | Value: {2} ".format( descriptor.uuid, descriptor.handle, bytes(value) )4 H# v. p2 s/ F |, F! G1 h )4 N: R* L' ~9 L% ]' L2 O7 J1 g# J$ t( m9 n : D/ E- b9 j3 ]& L7 t 输出的结果: Connected: True [Service] 00001801-0000-1000-8000-00805f9b34fb: Generic Attribute Profile [Characteristic] 00002a05-0000-1000-8000-00805f9b34fb: (indicate) | Name: , Value: None [Descriptor] 00002902-0000-1000-8000-00805f9b34fb: (Handle: 4) | Value: b'\x02\x00'1 z8 K: _, s- V3 D: ]8 J [Service] 00001800-0000-1000-8000-00805f9b34fb: Generic Access Profile8 a- ~7 z! \0 u! {& M8 | [Characteristic] 00002a00-0000-1000-8000-00805f9b34fb: (read,write-without-response,write,authenticated-signed-writes) | Name: , Value: b'STM32WB'- b; h. U- G; ` [Characteristic] 00002a01-0000-1000-8000-00805f9b34fb: (read,write-without-response,write,authenticated-signed-writes) | Name: , Value: b'@\x03'5 }4 [+ Y4 S4 G2 U% y" D5 x$ W+ D [Characteristic] 00002a04-0000-1000-8000-00805f9b34fb: (read) | Name: , Value: b'\xff\xff\xff\xff\x00\x00\xff\xff'; ?8 d) O) L& s/ Z2 g [Service] 0000180a-0000-1000-8000-00805f9b34fb: Device Information4 f3 o$ I2 J7 ?& e+ L9 J [Characteristic] 00002a29-0000-1000-8000-00805f9b34fb: (read) | Name: , Value: b'STM\x00' [Service] 0000180d-0000-1000-8000-00805f9b34fb: Heart Rate; ?/ C! ?. B6 p7 Z% [3 C9 r [Characteristic] 00002a37-0000-1000-8000-00805f9b34fb: (notify) | Name: , Value: None( [0 @/ W/ t* t) u [Descriptor] 00002902-0000-1000-8000-00805f9b34fb: (Handle: 18) | Value: b'\x00\x00' [Characteristic] 00002a38-0000-1000-8000-00805f9b34fb: (read) | Name: , Value: b'\x04'2 |, N- Y1 o3 J' ~" o) F# V [Characteristic] 00002a39-0000-1000-8000-00805f9b34fb: (write) | Name: , Value: None' ]0 ?1 O; J6 \ 观察,返回的信息,我们大致可以通过,服务、特性、描述等知道其性能和功能。其中,发现UUID都是单片机中定义好的,在UUID.h 文件目录下: /* UUIDs for Heart Rate Service */, s9 A6 z6 L9 J( H #define HEART_RATE_SERVICE_UUID (0x180D)& v6 n) r2 v& I- h2 `% l2 B3 I5 j #define CLIENT_CHARACTERISTIC_CONFIG_DESCRIPTOR_UUID (0x2902)' c* i- b1 @7 S R$ x #define HEART_RATE_MEASURMENT_UUID (0x2A37)' f0 j+ F( O& Q( D0 C #define SENSOR_LOCATION_UUID (0x2A38), p! Y) @& S: E+ v% c8 n5 h #define CONTROL_POINT_UUID (0x2A39) 9 @ Q! ?, M" j 3.3 通知和获取数据7 x8 u+ @/ j6 X8 n3 J( A 通过STM的蓝牙软件,发现软件是有数据传递的。 4 ![]() ! B# i* `3 @, U# A1 h& V 其采用的方式是通知的方式,我是用了一下程序: def notification_handler(sender, data): # strs=data.hex; H4 t. P. c2 @9 b print(f"{sender}: {data}") byarray = bytearray(data) print(byarray.hex()) 6 [1 T: I- ]7 j0 `7 F/ x: u( z async def run(address, loop, debug=False): async with BleakClient(address, loop=loop) as client:$ R- L3 l6 r/ \0 e0 P( y x = await client.is_connected()/ Y8 }. c- D, E; Q await client.start_notify(CHARACTERISTIC_UUID, notification_handler)) p2 e# p' [2 J. g await asyncio.sleep(5.0, loop=loop) await client.stop_notify(CHARACTERISTIC_UUID) if __name__ == "__main__":# P! O$ q, E1 ^, U$ C. j' k if __name__ == "__main__": address = ('80:E1:26:00:68:7C') #你的地址, S ~9 R/ y, y# x; I: Q loop = asyncio.get_event_loop() loop.run_until_complete(run(address, loop, True))' m2 e% ^9 C, g ` 9 o! o/ _3 g5 V7 j. J5 `' H7 p5 @ 返回了一下数据: 00002a37-0000-1000-8000-00805f9b34fb: bytearray(b'\x1f7\x00\x14\x00\x00\x04')' f1 v G& y3 M1 m" U 1f370014000004 00002a37-0000-1000-8000-00805f9b34fb: bytearray(b'\x1f6\x00\x1e\x00\x00\x04')2 g. g) \" q& s0 c 1f36001e000004 00002a37-0000-1000-8000-00805f9b34fb: bytearray(b'\x1f4\x00(\x00\x00\x04')2 f% T/ d- g& X4 r3 X2 L 1f340028000004: `: v8 z( {! k; K 其中00002a37是HEART_RATE_MEASURMENT_UUID ,数据是十六进制的1f36001e000004,其中0x36是心跳,1e是能量消耗,我将能量变化修改为,每次+10,从每次读回的值就可以看到。5 x" V( i: E1 Z6 n0 o% K9 ~. } HRSAPP_Context.MeasurementvalueChar.EnergyExpended += 10;% A/ G# p# u$ [; Y( ^6 @1 S; ^; e 6 y1 c( y q, U0 g, \ C' C: I 4 小结6 l2 C* X! F5 K ?1 Q 以上步骤,从发现蓝牙,连接蓝牙,获取蓝牙服务,以及获取数据。其实,实验中还有许多的小问题,比如指示灯的闪烁,为啥STM官方的APP可以直接连接,自己的却不行?6 x$ d0 s& G: D: |" H2 j 我相信,在后面一步一步的代码解析,就会解释这些现象。 % Y, u- M5 b0 M3 ]6 j 0 e. x9 ~9 u. Q+ p% r, U! ?7 |0 C# o : x$ x1 l' R% O* T ( b+ }# |9 D: S3 I9 N% {9 k |
Nucleo-STM32WB09KE测评-蓝牙调试并使用OLED显示数据
【NUCLEO-WB09KE评测】创建BLE工程和开启调试信息
【NUCLEO-WB09KE评测】+2.串口打印
【NUCLEO-WB09KE评测】+5.实现标准电量服务
【NUCLEO-WB09KE评测】BLE例程HeartRate及工具探索
【NUCLEO-WB09KE评测】+3.蓝牙广播
STM32WB系列MCU低功耗测试
Nucleo-STM32WB09KE测评--IIC传感器读取
【NUCLEO-WB09KE评测】+4.蓝牙控制LED
【NUCLEO-WB09KE评测】+1.点灯