NRF51822 广播与扫描

分类: 蓝牙 |
from:http://blog.chinaunix.net/xmlrpc.php?r=blog/article&uid=28852942&id=5752396
解决以下几个问题:
1 SDK9 中的几种广播模型
2 广播超时如何进入睡眠
3
1
SDK9 中的几种广播模型
Nordci SDK对于广播方面有一个模块。这个模式定义了几种广播模式,当然这些模式并非规范中的广播模式,仅仅是sdk自己定义的一些模式而已。
如下图
http://blog.chinaunix.net/attachment/201609/26/28852942_1474850442GiVL.png广播与扫描" />
Direct模式及直连模式,利用的就是ble中的直连广播,该模式是为了快速重连上刚刚断开的设备,比如利用在快速重连上意外断开的设备,已达到无缝恢复的目的。(实践代码中还分成了)
Fast模式:就是普通的广播,不过连接间隔我们可以设置的快一点。
Slow模式:普通广播,连接间隔设置的慢一点
Idle模式:停止广播。
这四种模式是递进的,比如你设置了启动广播时选择Direct模式,但是如果你并未在初始化时设置Direct模式的相关参数,那么它就会回尝试Fast模式,如果初始化时Fast模式的相关信息也没设置,就会再尝试Slow模式,如果初始化时Slow模式相关信息也没设置最后就直接进入到Idle模式了。
同样的,广播超时后的超时处理就是选择下一个模式再进行广播,比如你Fast模式启动广播成功后,如果超时时间是3分钟,3分钟后,广播超时处理中就是选择尝试Slow模式广播。
其实模式的定义只是给出了一个可以直接利用的模块,比如Fast模式和Slow模式并没有定义所谓的快慢是多少,只是给以一个你可以直接使用的代码模块。比如你的使用场景是希望设备上电后以30ms快速广播20s,如果一直都没有被连接上,30s后切换成200ms的广播3分钟以达到减小功耗目的。
那么在广播初始化的时候就可以忽略Direct模式的相关设置,设置Fast模式使能,并且广播间隔为30ms,超时时间为20s,设置Slow模式使能,并且其广播间隔为200ms,超时时间设置为180s。之后启动Fast模式广播就可以了,当20s超时到期后,就会收到协议栈的BLE timeout事件,sdk中的处理就是切换成Slow模式继续广播。如果Slow模式180s也超时了,协议栈就会上抛Slow timeout事件,最终就停止广播了。
2广播超时后如何进入睡眠:
Main函数中调用了advertising_init来初始化广播参数
http://blog.chinaunix.net/attachment/201609/26/28852942_1474850442Ew2v.png广播与扫描" />
从上图可以看到,sdk中默认只初始化了Fast
Main中初始化广播后,启动FAST模式广播
http://blog.chinaunix.net/attachment/201609/26/28852942_1474850443q2y2.png广播与扫描" />如果一直没有其他设备来连接,当广播超时后,协议栈就会上抛一个TIMEOUT事件,广播模块的事件处理函数会处理这个事件
http://blog.chinaunix.net/attachment/201609/26/28852942_14748504434pP3.png广播与扫描" />
http://blog.chinaunix.net/attachment/201609/26/28852942_1474850443g2eO.png广播与扫描" />
http://blog.chinaunix.net/attachment/201609/26/28852942_14748504443KTe.png广播与扫描" />
可以看到对TIMEOUT的事件处理为先判断是不是广播的TIMEOUT事件。如果是就判断模式,因为是FAST模式启动,所以代码中就会再次启动SLOW模式的广播。
再看下启动广播函数,只看相关的几个代码段:
http://blog.chinaunix.net/attachment/201609/26/28852942_14748504446R4O.png广播与扫描" />下面这段代码就是根据启动广播的模式和初始化时设置的参数来决定真正的广播模式。
第一次启动时,因为是FAST模式启动的,并且初始化函数advertising_init中设置了FAST模式的相关参数,所以确定就是FAST模式,当FAST广播模式超后,上面的代码显示是又启动了SLOW模式,但是advertising_init函数中并未设置SLOW模式的相关参数,从下面的代码中看到最终设置成了IDLE模式。即并未按照要求启动了SLOW广播,因为初始化时没有设置SLOW模式的相关参数。
http://blog.chinaunix.net/attachment/201609/26/28852942_1474850445kIYR.png广播与扫描" />PS:上图的DIRECT_SLOW这个模式就是利用ble中的Low Duty Cycle Directed Advertising。4.0时规范只定义了DIRECT广播类型,就是定向广播,并且广播周期为3.75ms,并且只能持续1.28s。4.1开始分成了Low Duty Cycle Directed Advertising和High Duty Cycle Directed Advertising,High Duty Cycle Directed保持了之前的定义即3.5ms周期和最多持续1.28s,而Low Duty Cycle Directed类似普通广播,不过是定向的而已。
函数的最后有如下片段,因为最终设置成了IDLE模式,所以没有启动广播,于是调用了m_evt_handler函数。
http://blog.chinaunix.net/attachment/201609/26/28852942_1474850445rHJH.png广播与扫描" />
这个函数时在广播初始化里面设置的
http://blog.chinaunix.net/attachment/201609/26/28852942_1474850445sV2j.png广播与扫描" />
实现如下
sleep_mode_enter函数的实现就是设置唤醒按键,然后进入深度睡眠
唤醒按键设置了BUTTON0 和BUTTON1
这里就是将17 ,18连个引脚设置为唤醒引脚。(17引脚唤醒的同时会删除绑定信息)
http://blog.chinaunix.net/attachment/201609/26/28852942_1474850446ulyj.png广播与扫描" />http://blog.chinaunix.net/attachment/201609/26/28852942_1474850447114R.png广播与扫描" />
综上:广播初始化中设置了FAST模式广播的相关参数,然后按FAST模式启动广播。当广播超时后,超时时间处理中判断是FAST模式超时,于是再启动SLOW模式广播,但是因为SLOW模式广播的相关参数并没有设置,于是切换成IDLE模式,并且调用了初始化时设置的回调函数。回函数中会设置唤醒按键然后设置深度睡眠。
3如何取消广播超时睡眠使其可以无限广播
http://blog.chinaunix.net/attachment/201609/26/28852942_1474850447M5Z7.png广播与扫描" />
扫描
from:http://blog.chinaunix.net/uid-28852942-id-5748020.html
找到m_scan_param参数定义
http://s9/mw690/003hEzXrzy77QPBiQkMb8&690广播与扫描" TITLE="NRF51822
在实际中如果需要获得扫描响应,需要主机设置为主动扫描。如果仅仅是需要广播数据则设置为被动扫描。主动扫描和被动扫描的区别在于:主动扫描可以获得广播数据和扫描回应数据。而被动扫描只能获得广播数据不能获得扫描数据。
不返回扫描回应数据
Scan_rsp: This is 1 if the packet is a scan response. It is 0 if the packet is a normal advertisement packet.
将ble_gap_scan_params_t m_scan_params 结构体的成员active置0,不得到 扫描回应数据
SCAN_TIMEOUT也设置为0 设备一直扫描 不超时
1.
2.
3.
4.
5.
6.
7.
8.
9.
10. };
值的意义依次为:
0,表示只侦听广播,不主动发起scan_req请求。
0,表示侦听所有广播包,不做过滤。
NULL,没有白名单
扫描间隔,和扫描窗口。(比如1s的间隔和0.5s的窗口,那么每1s就会启动侦听,但是只侦听0.5s,只会休眠。等待下一个1s到来)
0,表示没有扫描超时,即设备会一值侦听除非主动关闭扫描
- Active: Perform active scanning, which is sending scan requests to all advertisers asking for their scan response packets in addition to the advertisement packet.
- Selective: Ignore devices not in our whitelist, which is a list of known devices.
- Whitelist: Our list of known devices.
- Interval: How often we start a scan window
- Scan window: How long we scan every scan interval
- Timeout: How long the scanner will run before stopping automatically.
启动广播之后,设备就会不断侦听其他设备的广播。
当收到广播数据后,协议栈会上抛一个BLE_GAP_EVT_ADV_REPORT事件给上层,所以实现一个简单的侦听器,我们直接在事件处理中添加一个
官方的例子里面已经有针对
http://s9/mw690/003hEzXrzy77QPDGtQs68&690广播与扫描" TITLE="NRF51822
The advertisement report is found inside the event structure:
p_gap_evt->params.adv_report. The type of this report
is
- peer_addr: The Bluetooth address of the device broadcasting the advertisement
-
发送广播设备的地址 - Rssi: Received Signal Strength Indication. The signal strength of the device, in dBm.
-
Scan_rsp: This is 1 if the packet is a scan response. It is 0 if the packet is a normal
-
advertisement packet.
-
1--扫描回应包
0--正常广播包 -
Advertisement type
(Connectable, direct/indirect etc…). -
Dlen: Length of the advertisement packet.
- Data: The advertisement packet as a byte array. (uint8_t *)
The data field is structured in the following way:
[LENGTH_0][TYPE_0][VALUE_0][LENGTH_1][TYPE_1][VALUE_1] … [LENGTH_n][TYPE_n][VALUE_n]
If it contains the name “Hello” as the first value of the packet, it will look like this:
[0x5][0x9][0x48,0x65,0x6c,0x6c ,0x6f] … [LENGTH_n][TYPE_n][VALUE_n]
这个是有问题的应该是[0x6][0x9][0x48,0x65,0x6c,0x6c,0x6f]
length的数据长度包含 type和value值总的长度
0x5 is the length of the name, and 0x9 is the type COMPLETE_LOCAL_NAME
.
The type codes can be foundhere
type
#define
BLE_GAP_AD_TYPE_FLAGS
#define
BLE_GAP_AD_TYPE_16BIT_SERVICE_UUID_MORE_AVAILABLE
#define
BLE_GAP_AD_TYPE_16BIT_SERVICE_UUID_COMPLETE
#define
BLE_GAP_AD_TYPE_32BIT_SERVICE_UUID_MORE_AVAILABLE
#define
BLE_GAP_AD_TYPE_32BIT_SERVICE_UUID_COMPLETE
#define
BLE_GAP_AD_TYPE_128BIT_SERVICE_UUID_MORE_AVAILABLE
#define
BLE_GAP_AD_TYPE_128BIT_SERVICE_UUID_COMPLETE
#define
BLE_GAP_AD_TYPE_SHORT_LOCAL_NAME
#define
BLE_GAP_AD_TYPE_COMPLETE_LOCAL_NAME
#define
BLE_GAP_AD_TYPE_TX_POWER_LEVEL
#define
BLE_GAP_AD_TYPE_CLASS_OF_DEVICE
#define
BLE_GAP_AD_TYPE_SIMPLE_PAIRING_HASH_C
#define
BLE_GAP_AD_TYPE_SIMPLE_PAIRING_RANDOMIZER_R
#define
BLE_GAP_AD_TYPE_SECURITY_MANAGER_TK_VALUE
#define
BLE_GAP_AD_TYPE_SECURITY_MANAGER_OOB_FLAGS
#define
BLE_GAP_AD_TYPE_SLAVE_CONNECTION_INTERVAL_RANGE
#define
BLE_GAP_AD_TYPE_SOLICITED_SERVICE_UUIDS_16BIT
#define
BLE_GAP_AD_TYPE_SOLICITED_SERVICE_UUIDS_128BIT
#define
BLE_GAP_AD_TYPE_SERVICE_DATA
#define
BLE_GAP_AD_TYPE_PUBLIC_TARGET_ADDRESS
#define
BLE_GAP_AD_TYPE_RANDOM_TARGET_ADDRESS
#define
BLE_GAP_AD_TYPE_APPEARANCE