RW007系列综合实战3柿饼派上怎样更新RW007固件和驱动?
Posted RT-Thread物联网操作系统
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了RW007系列综合实战3柿饼派上怎样更新RW007固件和驱动?相关的知识,希望对你有一定的参考价值。
前面已经分享过部分关于 RW007 模块如何更新的带 BLE 功能的方法,这里也再次分享一下在柿饼派上怎样更新 RW007 固件和更新 RW007 的驱动,并且测试 RW007 的 BLE 功能。
步骤一:柿饼派更新 RW007 的固件
在柿饼派上更新 RW007 的固件,可以参考官方提供的 RW007 通过 UDP_OTA 工具升级 RW007 固件的文档,进行操作。
1. RW007 模块需要和 PC 电脑连在同一个局域网
2. 打开升级工具配置升级
(1)双击运行 udp_ota.exe
软件
3. 升级完成
等待进度条完成,然后重启板子。
步骤二:柿饼派修改 SDK 更新 RW007 的驱动
由于需要使用新版本的 RW007 模块的 BLE 功能,所以需要对应更新主机端柿饼派的 RW007 驱动,主要是替换SDK
中project\\firmware\\packages
目录下的rw007-latest
目录先删除,然后拷贝提供的rw007-v2.0.1
文件夹,即可完成替换更新RW007 的驱动。
替换驱动文件后,通过 env
编译柿饼派的固件,然后通过persimmon mod
工具更新柿饼派的固件,即可。
步骤三:柿饼派测试 RW007 的 BLE 功能
当按照前面的步骤进行操作后,可以通过调试串口的 msh
命令行进行调试。因为在RW007
驱动包里面默认开启了相关的功能测试示例,可以在MSH
命令行中通过rw007_ble
查询。下面的内容来自于官方提供的 RW007
BLE 功能使用说明操作。
1msh />rw007_ble
2[rw007_ble command]
3
4rw007_ble help
5rw007_ble init central/peripheral Note: init ble mode
6rw007_ble get_addr Note: get ble address
7rw007_ble update_params Note: update connect parameters(no support)
8rw007_ble scan Note: scan ble slave
9rw007_ble stop_scan Note: stop scan
10rw007_ble connect xx:xx:xx:xx:xx:xx Note: use slave addr to connect
11rw007_ble disconnect [conn_handle] Note: disconnect slave
12rw007_ble get_server Note: discover all server(no support)
13rw007_ble get_char Note: discover all description(no support)
14rw007_ble get_disc Note: discover all characteristic(no support)
15rw007_ble mtu_exch Note: ble mtu exchange(no support)
16rw007_ble notify Note: enable ble notify(no support)
17rw007_ble write Note: ble write data(no support)
18rw007_ble read Note: ble read data(no support)
19rw007_ble notify_change [conn_handle] [char_value] [UUID] Note: ble notify configure by uuid
20rw007_ble write_uuid [conn_handle] [UUID] [data] Note: ble write data by uuid
21rw007_ble read_uuid Note: ble read data by uuid(no support)
BLE 功能初始化
BLE
功能初始化函数,使用 BLE
功能必须调用。
1.RW007
BLE 设备初始化为主机设备。
1rw007_ble init central
测试示例
1msh />rw007_ble init central
2122 - ble_cmd_init
3start ble central
4msh />
2. RW007
BLE 设备初始化为从机设备
注意:目前不支持使用从机模式。
1rw007_ble init peripheral
BLE 功能获取本机的BLE设备地址
由于BLE设备的地址分为public
公共地址和random
随机地址。
1rw007_ble get_addr
测试示例
1msh />rw007_ble get_addr
2273 - ble_cmd_get_addr
3msh />resp_type: 0, len: 12
4RW007_BLE_RSP_TYPE_ADDR_GET
5public_id_addr = 48:00:42:8c:47:c9
6random_id_addr = 00:04:5c:43:00:00
BLE 功能扫描设备
目前RW007
BLE 功能支持通过 scan
命令,扫描周围的BLE
设备。
1.执行扫描周围BLE
设备
1rw007_ble scan
测试示例:
1msh />rw007_ble scan
2176 - ble_cmd_scan
3msh />resp_type: 1, len: 65
4RW007_BLE_NTF_TYPE_DISCOVER
5received advertisement; event_type=0 rssi=-60 addr_type=0 addr=4c:ed:fb:00:04:b1
6resp_type: 1, len: 56
这里的 addr
地址内容可以为后面连接设备使用,由于目前没有直接显示对应 BLE 设备的名称,需要先人工确定 BLE 设备的地址。
2.停止扫描周围BLE
设备
该指令用于 RW007
模块正在执行scan
扫描操作过程中,马上停止扫描的动作,可以执行此命令。
1rw007_ble stop_scan
BLE 功能设备连接
目前 RW007
BLE 功能设备支持通过 BLE
设备的类mac
地址来连接设备,该命令仅支持主机模式去连接从机设备的类mac
地址,连接成功后会有对应的conn_handle
值,这个conn_handle
值在其他功能沿用。
1rw007_ble connect xx:xx:xx:xx:xx:xx
测试示例
1msh />rw007_ble connect 4c:ed:fb:00:04:b1
2198 - ble_cmd_connect
3str_addr: 4c:ed:fb:00:04:b1
4mac addr: b1: 4: 0:fb:ed:4c
5msh />resp_type: 1, len: 52
6RW007_BLE_NTF_TYPE_CONNECT
7type: 0, status: 0, conn_handle: 1
8
9resp_type: 1, len: 44
10RW007_BLE_NTF_TYPE_CONNECT_DESC
11our_id_addr: c9:47:8c:42:00:48
12peer_id_addr: b1:04:00:fb:ed:4c
BLE 功能设备断开连接
目前RW007
BLE 功能设备支持主动断开从设备的连接。
1rw007_ble disconnect [conn_handle]
其中 conn_handle
与 connect
连接时的需要一致。
1msh />rw007_ble disconnect 1
2341 - ble_cmd_disconnect
3msh />resp_type: 1, len: 52
4RW007_BLE_NTF_TYPE_DISCONN
BLE 功能写数据
目前RW007
BLE 功能写数据,支持通过指定UUID
来写对应的数据。
1rw007_ble write_uuid [conn_handle] [UUID] [data]
指令说明:[conn_handle]
为 RW007
连接设备时生成的对应值,[UUID]
为对应 BLE
设备的特征参数,[data]
为需要发送的数据。
测试示例:
1msh />rw007_ble write_uuid 1 ffe1 Hello,RT-Thread....
2402 - ble_cmd_gattc_write_by_uuid
3write conn_hanle:1 uuid:ffe1 data:Hello,RT-Thread....
4write rc:0
5msh />
注意:目前已经支持通过 128 位的 UUID 进行写数据操作
测试示例:
1msh />rw007_ble write_uuid 1 ebe0ccbe7a0a4b0c8a1a6ff2997da3a6 00
2data: 00
3write conn_hanle:1 uuid:ebe0ccbe7a0a4b0c8a1a6ff2997da3a6
4write rc:0
说明:
测试示例中的 [conn_handle]
为 1,[UUID]
为ebe0ccbe7a0a4b0c8a1a6ff2997da3a6
,[data]
为 0x00
(这里是使用 hex 值)
BLE 功能更新notify 参数
目前 RW007
BLE 功能支持修改接收 notify
参数。
1rw007_ble notify_change [conn_handle] [char_value] [UUID]
参数说明:
[conn_handle]
为 RW007
连接设备时生成的对应值
[char_value]
为配置参数,具体如下
10:disable indication¬ification
21:enable notification,disable indication
32:enable indication, disable notification
43:enable indication ¬ification
[UUID]
为对应 BLE
设备的特征参数值。
BLE 功能关于广播包类型的说明
目前 RW007
BLE 功能中可以关于广播包 event
的类型可以分为下面几种。
1/* Advertising report */
2#define BLE_HCI_ADV_RPT_EVTYPE_ADV_IND (0)
3#define BLE_HCI_ADV_RPT_EVTYPE_DIR_IND (1)
4#define BLE_HCI_ADV_RPT_EVTYPE_SCAN_IND (2)
5#define BLE_HCI_ADV_RPT_EVTYPE_NONCONN_IND (3)
6#define BLE_HCI_ADV_RPT_EVTYPE_SCAN_RSP (4)
广播包结构如下:
1广播包结构描述:
2struct ble_gap_disc_desc {
3 uint8_t event_type;
4 uint8_t length_data;
5 ble_addr_t addr;
6 int8_t rssi;
7 uint8_t *data;
8
9 /***
10 * LE direct advertising report fields; direct_addr is BLE_ADDR_ANY if
11 * direct address fields are not present.
12 */
13 ble_addr_t direct_addr;
14};
15
16通过event_type区分出不同的广播包类型,所有类型分为如下几种:
17/* Advertising report */
180 : 普通广播数据包
191 : 直接广播包
202 : 扫描请求包
213 : 不可连接广播指示
224 : 扫描响应数据包
因此,在设备的scan
扫描回应中可以通过对应的event_type
来判定是哪种广播包的类型。
在 RW007
的驱动代码中,可以通过下面的event_type
来判断是哪种类型的广播包,可以让用户自行实现对应功能。
1case RW007_BLE_NTF_TYPE_DISC:
2 {
3 rt_kprintf("RW007_BLE_NTF_TYPE_DISCOVER\\n");
4 ....
5 switch(event_type)
6 {
7 case BLE_HCI_ADV_RPT_EVTYPE_ADV_IND:
8
9 break;
10 case BLE_HCI_ADV_RPT_EVTYPE_DIR_IND:
11
12 break;
13 case BLE_HCI_ADV_RPT_EVTYPE_SCAN_IND:
14
15 break;
16 case BLE_HCI_ADV_RPT_EVTYPE_NONCONN_IND:
17
18 break;
19 case BLE_HCI_ADV_RPT_EVTYPE_SCAN_RSP:
20
21 break;
22 }
23 ....
24 }
测试示例:
1.初始化 RW007
BLE 功能为主机模式。
1msh /> rw007_ble init central
2263 - ble_cmd_init
3start ble central
4msh />
2.执行BLE
扫描指令
1msh />rw007_ble scan
2318 - ble_cmd_scan
3msh />resp_type: 1, len: 83
4RW007_BLE_NTF_TYPE_DISCOVER
5received advertisement; event_type=3 rssi=-56 addr_type=1 addr=41:ef:77:50:2b:29
6mfg_data = 1e ff 06 00 01 09 20 02 5b 5d cd 33 a4 8c c7 c5 36 ac de 12 ab 17 87 89 e1 84 e9 19 ec c4 ad
7
8resp_type: 1, len: 83
9RW007_BLE_NTF_TYPE_DISCOVER
10received advertisement; event_type=3 rssi=-65 addr_type=1 addr=3c:2e:33:ea:8d:aa
11mfg_data = 1e ff 06 00 01 09 20 02 28 4c cf 9b a7 f2 7d f6 c7 7e d9 54 ac dd 91 a6 ee 0d 22 43 e1 8f e3
12
13resp_type: 1, len: 83
14RW007_BLE_NTF_TYPE_DISCOVER
15received advertisement; event_type=3 rssi=-59 addr_type=1 addr=42:18:ab:80:5c:11
16mfg_data = 1e ff 06 00 01 09 20 02 eb bf cf 15 97 aa 50 7f a4 57 25 d9 21 35 71 eb 6f aa a1 56 5c d5 e2
3.观察event_type=
可以确定是哪种广播包类型。
1RW007_BLE_NTF_TYPE_DISCOVER
2received advertisement; event_type=0 rssi=-64 addr_type=1 addr=61:63:46:c3:49:99
3mfg_data = 02 01 1a 02 0a 0c 0b ff 4c 00 10 06 00 19 1d 6d 63 18
4
5resp_type: 1, len: 52
6RW007_BLE_NTF_TYPE_DISCOVER
7received advertisement; event_type=4 rssi=-64 addr_type=1 addr=61:63:46:c3:49:99
8mfg_data =
9
10resp_type: 1, len: 73
11RW007_BLE_NTF_TYPE_DISCOVER
12received advertisement; event_type=0 rssi=-73 addr_type=0 addr=91:21:52:00:4e:b8
13mfg_data = 14 ff 4c 00 07 0f 00 02 20 91 21 52 00 4e b8 f5 58 58 3c 39 00
BLE 功能读数据
目前 RW007
支持通过 UUID
读取数据的功能
1rw007_ble read_uuid [conn_handle] [UUID]
参数说明:
[conn_handle]
为 RW007
连接设备时生成的对应值
[UUID]
为对应 BLE
设备的特征参数。
测试示例:
1msh />rw007_ble read_uuid 1 2a24
2read conn_hanle:1 uuid:2a24
3read rc:0
4msh />ble data input packet resp_type: 0, len: 18
5RW007_BLE_RSP_TYPE_READ
6connect:1 attr_handle:14 uuid: 24 2a read data:
700000000: 4C 59 57 53 44 30 33 4D 4D 43 00 LYWSD03MMC.
说明:这里的 [conn_handle]
为 1,[UUID]
为 2a24
,读取到的数据为 BLE 设备的名称。
目前 RW007 模块已经支持读取 128 位的 UUID 的功能。
测试示例:
1msh />rw007_ble read_uuid 1 ebe0ccbe7a0a4b0c8a1a6ff2997da3a6
2read conn_hanle:1 uuid:ebe0ccbe7a0a4b0c8a1a6ff2997da3a6
3read rc:0
4msh />ble data input packet resp_type: 0, len: 22
5RW007_BLE_RSP_TYPE_READ
6connect:1 attr_handle:51 uuid: a6 a3 7d 99 f2 6f 1a 8a 0c 4b 0a 7a be cc e0 eb read data:
700000000: 00
说明:这里的 [conn_handle]
为 1,[UUID]
为128位的 ebe0ccbe7a0a4b0c8a1a6ff2997da3a6
,读取到的数据值为 00
步骤四:柿饼派读取米家蓝牙温湿度计数据显示在界面上
感谢能坚持看到这里的每一个你,下面将会通过添加编写部分功能代码和创建 UI 工程,实现柿饼派读取米家温湿度计数据显示在界面上的功能。接下来会贴一部分实现代码和代码说明,可能会有点乏味,但是尽可能把实现步骤描述得具体些,避免采坑。
1. 添加测试文件,参考代码
2. 编译固件,验证功能
3. 创建 UI 工程,简单布局
4. c-js 之间互通说明
5. UI 工程修改,验证功能
1.添加测试代码
在官方提供的 SDK
中的project\\firmware\\applications
目录下添加ble_example.c
文件,然后参考project\\firmware\\packages\\rw007-v2.0.1\\src\\ble_cmd_rw007.c
文件,主要拷贝几个对应的函数。通过前面的可以了解到,需要(1)RW007 初始化为 BLE 主机模式 (2)RW007 连接米家蓝牙温湿度计2代设备 (3)RW007 通过 UUID 读取米家蓝牙温湿度计2代设备的数据 ,所以需要参考 BLE功能初始化、连接、通过 UUID 读取数据的功能实现函数。
(1)在ble_example.c
中添加 RW007 初始化为BLE主机模式的函数
1static int ble_example_init(void)
2{
3 rt_kprintf("ble_example_init\\n");
4 rt_uint8_t roles = 0;
5 roles = RW007_BLE_INIT_ROLE_CENTRAL;
6
7 rw007_ble_init(roles);
8
9 rw007_ble_resp_handle_cb_reg(rw007_ble_resp_handle);
10 rw007_ble_ntf_handle_cb_reg(rw007_ble_ntf_handle);
11 return 0;
12
13}
由于在初始化函数中,需要注册两个回调函数rw007_ble_resp_handle
和 rw007_ble_ntf_handle
,所以也要从project\\firmware\\packages\\rw007-v2.0.1\\src\\ble_cmd_rw007.c
文件中对应拷贝过来。
1static void rw007_ble_resp_handle(rt_uint16_t resp_type, void *data, rt_uint16_t size)
2{
3 switch(resp_type)
4 {
5 case RW007_BLE_RSP_TYPE_INIT:
6 rt_kprintf("RW007_BLE_RSP_TYPE_INIT\\n");
7 break;
8 case RW007_BLE_RSP_TYPE_ADDR_GET:
9 rt_kprintf("RW007_BLE_RSP_TYPE_ADDR_GET\\n");
10 ble_get_addr *get_addr = (ble_get_addr *)data;
11 rt_kprintf("public_id_addr = %02x:%02x:%02x:%02x:%02x:%02x\\n", get_addr->public_id_addr[5], get_addr->public_id_addr[4],
12 get_addr->public_id_addr[3], get_addr->public_id_addr[2],
13 get_addr->public_id_addr[1], get_addr->public_id_addr[0]);
14 rt_kprintf("random_id_addr = %02x:%02x:%02x:%02x:%02x:%02x\\n", get_addr->random_id_addr[5], get_addr->random_id_addr[4],
15 get_addr->random_id_addr[3], get_addr->random_id_addr[2],
16 get_addr->random_id_addr[1], get_addr->random_id_addr[0]);
17 break;
18 case RW007_BLE_RSP_TYPE_CONN_INFO_GET:
19 rt_kprintf("RW007_BLE_RSP_TYPE_CONN_INFO_GET\\n");
20 break;
21 case RW007_BLE_RSP_TYPE_CONN_UPD_PARAMS:
22 rt_kprintf("RW007_BLE_RSP_TYPE_CONN_UPD_PARAMS\\n");
23 break;
24 case RW007_BLE_RSP_TYPE_SCAN:
25 rt_kprintf("RW007_BLE_RSP_TYPE_SCAN\\n");
26 break;
27 case RW007_BLE_RSP_TYPE_CONNECT:
28 rt_kprintf("RW007_BLE_RSP_TYPE_CONNECT\\n");
29 break;
30 case RW007_BLE_RSP_TYPE_DISCONN:
31 rt_kprintf("RW007_BLE_RSP_TYPE_DISCONN\\n");
32 break;
33 case RW007_BLE_RSP_TYPE_GATT_DIS_FULL:
34 rt_kprintf("RW007_BLE_RSP_TYPE_GATT_DIS_FULL\\n");
35 break;
36 case RW007_BLE_RSP_TYPE_GATTC_SRV:
37 rt_kprintf("RW007_BLE_RSP_TYPE_GATTC_SRV\\n");
38 break;
39 case RW007_BLE_RSP_TYPE_GATTC_CHR:
40 rt_kprintf("RW007_BLE_RSP_TYPE_GATTC_CHR\\n");
41 break;
42 case RW007_BLE_RSP_TYPE_GATTC_DSC:
43 rt_kprintf("RW007_BLE_RSP_TYPE_GATTC_DSC\\n");
44 break;
45 case RW007_BLE_RSP_TYPE_MTU_EXCHANGE:
46 rt_kprintf("RW007_BLE_RSP_TYPE_MTU_EXCHANGE\\n");
47 break;
48 case RW007_BLE_RSP_TYPE_NOTIFY:
49 rt_kprintf("RW007_BLE_RSP_TYPE_NOTIFY\\n");
50 break;
51 case RW007_BLE_RSP_TYPE_WRITE:
52 rt_kprintf("RW007_BLE_RSP_TYPE_WRITE\\n");
53 break;
54 case RW007_BLE_RSP_TYPE_READ:
55 rt_kprintf("RW007_BLE_RSP_TYPE_READ\\n");
56 break;
57 default:
58 rt_kprintf("error response\\n");
59 break;
60 }
61}
62// recv slave notity handle
63static void rw007_ble_ntf_handle(rt_uint16_t ntf_type, void *data, rt_uint16_t size)
64{
65 switch(ntf_type)
66 {
67 case RW007_BLE_NTF_TYPE_CONN_UPD:
68 rt_kprintf("RW007_BLE_NTF_TYPE_CONN_UPD\\n");
69 break;
70 case RW007_BLE_NTF_TYPE_CONN_UPD_PARAMS:
71 rt_kprintf("RW007_BLE_NTF_TYPE_CONN_UPD_PARAMS\\n");
72 break;
73 case RW007_BLE_NTF_TYPE_DISC:
74 {
75 struct rw007_ble_gap_event_discov *disc_desc = RT_NULL;
76 char *dicover_data = RT_NULL;
77
78 rt_kprintf("RW007_BLE_NTF_TYPE_DISCOVER\\n");
79
80 disc_desc = (struct rw007_ble_gap_event_discov *)data;
81 dicover_data = (char*)(disc_desc + 1);
82
83 switch(disc_desc->discov_type)
84 {
85 case BLE_HCI_ADV_RPT_EVTYPE_ADV_IND:
86
87 break;
88 case BLE_HCI_ADV_RPT_EVTYPE_DIR_IND:
89
90 break;
91 case BLE_HCI_ADV_RPT_EVTYPE_SCAN_IND:
92
93 break;
94 case BLE_HCI_ADV_RPT_EVTYPE_NONCONN_IND:
95
96 break;
97 case BLE_HCI_ADV_RPT_EVTYPE_SCAN_RSP:
98
99 break;
100 }
101
102 rt_kprintf("received advertisement; event_type=%d rssi=%d addr_type=%d addr=",
103 disc_desc->discov_type,
104 disc_desc->rssi,
105 disc_desc->addr_type);
106 rt_kprintf("%02x:%02x:%02x:%02x:%02x:%02x\\n", disc_desc->addr[5], disc_desc->addr[4],
107 disc_desc->addr[3], disc_desc->addr[2],
108 disc_desc->addr[1], disc_desc->addr[0]);
109 rt_kprintf("mfg_data = ");
110 for (int i = 0; i < disc_desc->length_data; i++)
111 {
112 rt_kprintf("%02x ", dicover_data[i]);
113 }
114 rt_kprintf("\\n\\n");
115 break;
116 }
117 case RW007_BLE_NTF_TYPE_CONNECT:
118 {
119 struct rw007_ble_gap_event_connect *connect_event;
120 struct rw007_ble_conn_desc *conn_desc;
121
122 rt_kprintf("RW007_BLE_NTF_TYPE_CONNECT\\n");
123
124 connect_event = (struct rw007_ble_gap_event_connect *)data;
125 conn_desc = (struct rw007_ble_conn_desc*)(connect_event + 1);
126
127 rt_kprintf("connect event status: %d, conn_handle: %d\\n", connect_event->status, connect_event->conn_handle);
128 rt_kprintf("our_id_type:%d our_id_addr: ", conn_desc->our_id_type);
129 _print_addr(conn_desc->our_id_addr);
130 rt_kprintf("\\npeer_id_type:%d peer_id_addr: ", conn_desc->peer_id_type);
131 _print_addr(conn_desc->peer_id_addr);
132 rt_kprintf("\\nconn_itv:%d, conn_latency:%d, conn_suptout:%d, role:%d\\n", conn_desc->conn_itvl,
133 conn_desc->conn_latency, conn_desc->supervision_timeout, conn_desc->role);
134 break;
135 }
136 case RW007_BLE_NTF_TYPE_DISCONN:
137 rt_kprintf("RW007_BLE_NTF_TYPE_DISCONN\\n");
138 break;
139 case RW007_BLE_NTF_TYPE_MTU_EXCHANGE:
140 rt_kprintf("RW007_BLE_NTF_TYPE_MTU_EXCHANGE\\n");
141 break;
142 case RW007_BLE_NTF_TYPE_DISC_COMPLETE:
143 rt_kprintf("RW007_BLE_NTF_TYPE_DISC_COMPLETE\\n");
144 break;
145 case RW007_BLE_NTF_TYPE_ADV_COMPLETE:
146 rt_kprintf("RW007_BLE_NTF_TYPE_ADV_COMPLETE\\n");
147 break;
148 case RW007_BLE_NTF_TYPE_SUBSCRIBE:
149 rt_kprintf("RW007_BLE_NTF_TYPE_SUBSCRIBE\\n");
150 break;
151 case RW007_BLE_NTF_TYPE_NOTIFY_RX:
152 {
153 struct rw007_ble_gap_event_notify_rx *notify_rx;
154
155 rt_kprintf("RW007_BLE_NTF_TYPE_NOTIFY_RX\\n");
156
157 notify_rx = (struct rw007_ble_gap_event_notify_rx *)data;
158 rt_kprintf("conn_handle:%d, attr_handle:%d, rcv (%s) length:%d, data:\\n",
159 notify_rx->conn_handle, notify_rx->attr_handle,
160 notify_rx->indication ? "indication" : "notification", notify_rx->length_data);
161 hex_dump((const rt_uint8_t *)(notify_rx + 1), notify_rx->length_data);
162 break;
163 }
164 default:
165 rt_kprintf("error notify\\n");
166 break;
167 }
168}
(2) 添加 RW007
连接米家蓝牙温湿度计2代设备的实现函数
1tatic int ble_example_connect(void)
2{
3 rt_kprintf("ble_example_connect\\n");
4 ble_addr_t addr;
5 addr.type = RW007_BLE_ADDR_PUBLIC;
6
7 rt_kprintf("str_addr: %s\\n", "A4:C1:38:35:52:94");
8
9 _hexstrtoaddr("A4:C1:38:35:52:94", addr.val);
10
11 rt_kprintf("mac addr: %2x:%2x:%2x:%2x:%2x:%2x\\n", addr.val[0], addr.val[1],
12 addr.val[2], addr.val[3],
13 addr.val[4], addr.val[5]);
14
15 rw007_ble_connect(&addr);
16 return 0;
17}
说明:前面已经通过 BLE 测试和手机的 BLE 调试助手可以知道米家蓝牙温湿度计2代设备的 mac 地址为A4:C1:38:35:52:94
,所以这里直接固定设备的mac
地址用于连接。
(3)RW007 通过 UUID
读取米家蓝牙温湿度计2代设备的数据
由于前面通过 BLE 测试连接上米家蓝牙温湿度计2代设备后,会主动收到来自米家蓝牙温湿度计2代设备的温湿度数据的notify
,并在rw007_ble_ntf_handle
函数中打印输出,现在需要解析接收到的温湿度数据,在rw007_ble_ntf_handle
的函数中添加解析数据的功能。
c case RW007_BLE_NTF_TYPE_NOTIFY_RX: { struct rw007_ble_gap_event_notify_rx *notify_rx; char cRes[50] = {0}; ¨K60K
(4) 添加通过命令启动函数,测试功能
1static int ble_example_start(void)
2{
3 ble_example_init();
4 rt_thread_delay(1000);
5 ble_example_connect();
6 rt_thread_delay(5000);
7 return 0;
8
9}
10MSH_CMD_EXPORT(ble_example_start,ble_example_start)
说明:这里加了一些延时是考虑到连接设备需要一些时间。
(5)编译固件,烧录验证功能
编译固件,给柿饼派升级固件后,通过在调试串口 msh
中输入ble_example_start
命令进行启动,启动后便会连接米家蓝牙温湿度计2代设备,并把获取到温湿度数据进行解析,打印出来。
1msh />ble_example_start
2ble_example_init
3ble_example_connect
4str_addr: A4:C1:38:35:52:94
5mac addr: 94:52:35:38:c1:a4
6ble data input packet resp_type: 1, len: 46
7RW007_BLE_NTF_TYPE_CONNECT
8connect event status: 0, conn_handle: 1
9our_id_type:0 our_id_addr: 48:00:42:8c:47:c9
10peer_id_type:0 peer_id_addr: a4:c1:38:35:52:94
11conn_itv:80, conn_latency:0, conn_suptout:256, role:0
12ble data input packet resp_type: 1, len: 13
13RW007_BLE_NTF_TYPE_NOTIFY_RX
14conn_handle:1, attr_handle:54, rcv (notification) length:5, data:
1500000000: 47 0B 3B B9 0A G.;..
16temp=2887 mC,hum=59%,v_bat=2745mV
17cres=temp=2887mC,hum=59%,v_bat=2745mV
添加把数据传递到界面的实现函数(C-To-JS)
通过前面的步骤,已经成功解析到了米家蓝牙温湿度计2代设备的温湿度数据,现在需要把这些数据显示到 LCD 屏上,需要在 SDK 中参考示例代码中的docs\\src\\PersimM3_JS_GUI_C_TransData\\js_message_test.c
进行数据的封装传到GUI 中。
(1)创建 module
并初始化
需要创建添加module
初始化便于 JS 中导入该模块。
1static js_object_t js_message_obj = JS_ECMA_VALUE_UNDEFINED;
2static void js_message_info_free(void *native)
3{
4 js_message_obj = JS_ECMA_VALUE_UNDEFINED;
5 rt_kprintf("==> js_message_obj = JS_ECMA_VALUE_UNDEFINED;\\n");
6}
7
8static const js_object_native_info_t js_message_info =
9{
10 .free_cb = js_message_info_free
11};
12
13
14js_object_t example_module_init(void)
15{
16 js_object_t obj = js_create_object();
17
18 if (js_resolve_error(obj))
19 return js_create_null();
20 rt_kprintf("==> module_init\\n");
21
22 js_message_obj = obj;
23 js_emitter(js_message_obj, js_create_undefined());
24 js_set_property_native_pointer(js_message_obj, "_free_cb", NULL, &js_message_info);
25
26 return obj;
27}
28JS_MODULE(example_module,example_module_init)
注意:这里的example_module
在 js 中
(2)封装数据传输函数
在 C代码层到JS脚本层主要是通过事件监听机制异步上报数据或触发JS逻辑主动到C层取数据;当前异步机制是通过GUI的消息队列实现,先往GUI的消息队列发送消息,然后GUI收到对应消息后触发监听回调,所以需要js_message_send_data
和js_callback_message
函数,可以直接从示例代码中拷贝过来使用。
1static rt_bool_t js_message_send_data(const char *name, js_object_t data)
2{
3 rt_bool_t ret = RT_FALSE;
4
5 if (js_context_lock() != RT_EOK)
6 return ret;
7 rt_kprintf("==> js_message_send_data start\\n");
8 if (js_object_is_object(js_message_obj))
9 {
10 js_object_t msg = js_create_object();
11 if (!js_resolve_error(msg))
12 {
13 js_set_string_property_value(msg, "name", name);
14 js_set_property_value(msg, "data", data);
15
16 ret = js_send_callback_func(js_callback_message, msg);
17 js_release_object(msg);
18 }
19 }
20 rt_kprintf("==> js_message_send_data end\\n");
21 js_context_unlock();
22
23 return ret;
24}
1static rt_bool_t js_callback_message(js_object_t args)
2{
3 if (js_context_lock() != RT_EOK)
4 return RT_FALSE;
5
6 rt_kprintf("==> js_callback_message start\\n");
7 if (js_object_is_object(js_message_obj))
8 {
9 js_object_t msg_name = js_get_property_value(args, "name");
10 js_object_t msg_data = js_get_property_value(args, "data");
11 char str_buf[JS_STRING_BUFFER_SIZE];
12 char *name = js_object_to_string(msg_name, str_buf, JS_STRING_BUFFER_SIZE);
13 if (name)
14 {
15 js_event_emit(js_message_obj, name, &msg_data, 1);
16 if (name != str_buf)
17 JS_FREE(name);
18 }
19 js_release_object(msg_data);
20 js_release_object(msg_name);
21 }
22 js_release_object(args);
23 rt_kprintf("==> js_callback_message end\\n");
24 js_context_unlock();
25
26 return RT_TRUE;
27}
(3)在解析数据后把数据传到界面显示
在rw007_ble_ntf_handle
的函数中添加把解析数据后把数据传到界面显示的功能,只需要使用js_message_send_data
函数进行传输数据即可。
1 case RW007_BLE_NTF_TYPE_NOTIFY_RX:
2 {
3 struct rw007_ble_gap_event_notify_rx *notify_rx;
4 char cRes[50] = {0};
5
6 rt_kprintf("RW007_BLE_NTF_TYPE_NOTIFY_RX\\n");
7
8 notify_rx = (struct rw007_ble_gap_event_notify_rx *)data;
9 rt_kprintf("conn_handle:%d, attr_handle:%d, rcv (%s) length:%d, data:\\n",
10 notify_rx->conn_handle, notify_rx->attr_handle,
11 notify_rx->indication ? "indication" : "notification", notify_rx->length_data);
12 hex_dump((const rt_uint8_t *)(notify_rx + 1), notify_rx->length_data);
13
14 int32_t temp = 0xffff;
15 int hum = 0xff;
16 int32_t v_bat = 0xffff;
17
18 rt_uint8_t *ptr = (const rt_uint8_t *)(notify_rx + 1);
19
20 //Here to get mi tempture data
21 temp = 0xffff & ptr[1];
22 temp = temp << 8;
23 temp = temp | ptr[0];
24 hum = ptr[2];
25 v_bat = 0xffff & ptr[4];
26 v_bat = v_bat << 8;
27 v_bat = v_bat | ptr[3];
28 rt_kprintf("temp=%d mC,hum=%d%,v_bat=%dmV\\n", temp, hum, v_bat);
29 memset(cRes,0,sizeof(cRes));
30 sprintf(cRes,"temp=%dmC,hum=%d%%,v_bat=%dmV", temp, hum, v_bat);
31 rt_kprintf("cres=%s \\n",cRes);
32
33 if (js_context_lock() == RT_EOK)
34 {
35 js_object_t value =string_to_js_object(cRes);
36 js_message_send_data("mi_data", value);
37 rt_kprintf("value:%s\\n", value);
38 js_release_object(value);
39 js_context_unlock();
40 }
41
42 break;
43 }
说明:
需要注意js_message_send_data("mi_data", value);
这里的mi_data
,后面UI界面中将通过这个属性来接收数据。
(4)编译固件,烧录程序。
根据前面的步骤操作,这里柿饼派的固件已经准备好了,下面需要创建 UI 工程进行界面显示米家蓝牙温湿度计2代设备的温湿度数据。
创建 UI 工程显示米家蓝牙温湿度计2代设备的温湿度数据
关于UI
界面工程的创建,这里不做具体说明,可以参考 SDK 目录下docs\\2-PersimM3_UI_Quick_Start.pdf
文档的操作,这里只说明核心部分的操作。
(1) UI 工程界面控件布局
由于这里只做数据显示,所以仅需要放置几个 Label
控件,为了美观些这里也特意放置了一些图标和设置背景图片。
(2) JS 代码的编写
主要通过require
导入example_module
,这里的example_module
是前面C
代码中初始化过的,然后通过this.example_module.on("mi_data",function(data){});
进行数据解析,并把数据更新到UI界面中。这里的mi_data
也是需要与C
代码中配对使用的。
1var page = {
2
3 /* 此方法在第一次显示窗体前发生 */
4 onLoad: function (event) {
5 this.example_module = require("example_module");
6 console.dir(this.example_module);
7 var that = this;
8 function insertStr(soure, start, newStr) {
9 return soure.slice(0, start) + newStr + soure.slice(start);
10 }
11
12 this.example_module.on("mi_data",function(data){
13
14 //temp=3205mC,hum=54%,v_bat=2766mV
15 //data_length:32
16 s_l1 = data.indexOf(',')
17 temp_data =data.substring(0,s_l1)
18 temp_value_l = temp_data.indexOf('=')
19 temp_value1 = temp_data.substring(temp_value_l+1,temp_data.length -2 )
20 temp_value=insertStr(temp_value1,2,".");
21 s_l2 = data.lastIndexOf(',')
22 hum_value = data.substring(s_l1+5,s_l2)
23 vbat_data = data.substring(s_l2+7,data.length -2)
24 vbat_value=insertStr(vbat_data,1,".");
25 that.setData({temp_value : temp_value+'℃'});
26 that.setData({hum_value : hum_value});
27 that.setData({bat_value : vbat_value+'V'});
28 });
29 },
30
31 /* 此方法展示窗体后发生 */
32 onResume: function (event) {
33
34 },
35
36 /* 当前页状态变化为显示时触发 */
37 onShow: function (event) {
38
39 },
40
41 /* 当前页状态变化为隐藏时触发 */
42 onHide: function (event) {
43
44 },
45
46 /* 此方法关闭窗体前发生 */
47 onExit: function (event) {
48
49 },
50};
51
52Page(page);
53
54page = 0;
(3) 下载 UI 工程到柿饼派,体验效果
UI 工程下载完成后,在 调试串口的msh
命令行中输入ble_example_start
,然后观察UI界面的效果。
1msh />ble_example_start
2ble_example_init
3ble_example_connect
4str_addr: A4:C1:38:35:52:94
5mac addr: 94:52:35:38:c1:a4
6ble data input packet resp_type: 1, len: 46
7RW007_BLE_NTF_TYPE_CONNECT
8connect event status: 0, conn_handle: 1
9our_id_type:0 our_id_addr: 48:00:42:8c:47:c9
10peer_id_type:0 peer_id_addr: a4:c1:38:35:52:94
11conn_itv:80, conn_latency:0, conn_suptout:256, role:0
12msh />ble data input packet resp_type: 1, len: 13
13RW007_BLE_NTF_TYPE_NOTIFY_RX
14conn_handle:1, attr_handle:54, rcv (notification) length:5, data:
1500000000: 28 0B 42 A9 0A (.B..
16temp=2856 mC,hum=66%,v_bat=2729mV
17cres=temp=2856mC,hum=66%,v_bat=2729mV
18==> js_message_send_data start
19==> js_message_send_data end
20value:
21==> js_callback_message start
22==> js_callback_message end
此时,柿饼派的界面就显示米家蓝牙温湿度计2代设备的温湿度和电量的数据。
总结
经过这次使用柿饼派上的 RW007 WIFI 模块学习 BLE 功能的使用和读取米家温湿度计数据显示在界面上的过程中,学会 RW007 上如何使用 BLE 功能和巩固了柿饼派JS与C 之间的数据交互的操作,在这次仅是使用了BLE 的初始化和连接功能,希望在后面使用更多的功能制作更多有趣的作品与大家分享。
????RW007资料链接:
奶牛网盘 https://realthread.cowtransfer.com/s/274a38fde4b543
????RW007购买链接:
https://item.taobao.com/item.htm?&id=587364858221
口令(复制至X宝打开):5嘻6bLxXVYVZEv嘻 https://m.tb.cn/h.4rai8N8?sm=35e672 RW007 高速WIFI模块 SPI 物联网 透传模块 无线模块
你可以添加微信17775982065为好友,注明:公司+姓名,拉进 RT-Thread 官方微信交流群!
???????????? 点击阅读原文进入官网
以上是关于RW007系列综合实战3柿饼派上怎样更新RW007固件和驱动?的主要内容,如果未能解决你的问题,请参考以下文章
RW007系列综合实战2使用RW007模块连接小米蓝牙温湿度计2代