【BLE基础知识】--Slave latency设置流程及空中包解析

1、Slave latency基本概念

当BLE从设备对耗电量要求较高时,若需要节省耗电量,则可以通过设置Slave Latency参数来减少BLE从设备的耗电。

Slave Latency:允许Slave(从设备)在没有数据要发的情况下,跳过一定数目的连接事件,在这些连接事件中不必回复Master(主设备)的包

Slave在这些跳过的期间RX功能也不会开启(实测NXP KW38芯片),即不会收到Master的蓝牙数据帧故Slave在这些跳过的期间,既不接收,也不发送,就能更加省电

举例:

当Slave latency=0时,主设备每次发送数据,从设备都会响应,具体如下图:

当Slave latency=3时,表示主设备发送4次数据,从设备响应1次。具体如下图:

蓝牙5.1协议栈中规定Slave latency参数的取值范围是0x0000 ~ 0x01F3(即0 ~ 499),具体如下表第三个参数:

标准除了要求Slave latency的取值范围为0 ~ 499外,另外还要求Slave latency的取值范围为:

0 ~ ((connSupervisionTimeout / (connIntervalMax*2)) -1).

反之,connSupervisionTimeout要大于2 * Interval_Max * (Latency + 1)。

蓝牙5.1协议栈标准里有一句话,如下:

The connection slave latency used before the instant is known as connSlaveLatencyOLD.

The connection slave latency contained in the LL_CONNECTION_UPDATE_IND PDU and used at the instant and after, is known as connSlaveLatencyNEW.

从这句话可知, slave latency 生效的时刻是通过LL_CONNECTION_UPDATE_IND PDU的intstant参数来约定的。即在约定的Instant(EventCount)时刻及之后,采用新的Slave latency。

2、通信过程中,Slave latency切换流程

2.1 整体流程

根据蓝牙协议5.1.7章节Connection Parameters Request procedure流程,可进行蓝牙相关连接参数的更新,主要包含connInterval, connSlaveLatency and connSupervisionTimeout等几个参数的设置。通过LL_CONNECTION_PARAM_REQ、LL_CONNECTION_PARAM_RSP、LL_CONNECTION_UPDATE_IND等PDU进行参数设置。

大体流程为:

2.2 实际数据解析

整体流程结合数据流解析如下图:

2.2.1 步骤2

上图步骤2,Master通过LL_CONNECTION_PARAM_REQ请求Slave更改Slave latency为3(同时包含connInterval, connSupervisionTimeout等参数),具体数据流截图如下:

2.2.2 步骤3

上图步骤3,Slave通过LL_CONNECTION_PARAM_RSP响应Master返回的参数与LL_CONNECTION_PARAM_REQ命令相同,Slave latency为3(同时包含connInterval, connSupervisionTimeout等参数)。具体数据流截图如下:

2.2.3 步骤4

上图步骤4,Master通过LL_CONNECTION_UPDATE_IND请求Slave表示双方已完成Slave latency的设置协商,并通过该PUD中的Instant参数约定在某个EventCount之后,开始生效新的Slave latency具体数据流截图如下:

2.2.4 步骤5-10

等待新的slave latency生效,在这期间按照旧的slave latency进行通信

2.2.5 步骤11

在步骤4中,通过Instant参数,约定在该EventCount(如上图示例的9612)之后,开始生效新的Slave latency实际测试发现,在此EventCount的通信交互中,Slave一定会响应。

2.2.6 步骤12及之后

新的Slave latency=3生效,从设备4 次响应1次主设备的请求,以达到节电的效果。

3、总结

1) Slave Latency:允许Slave(从设备)在没有数据要发的情况下,跳过一定数目的连接事件,在这些连接事件中不必回复Master(主设备)的包假设Slave latency为3,则表示主设备发送4次请求,从设备才响应1次。

2但若从设备本身有数据要立即发送,则可以搭载在最近的连接事件中发送,无需等到Slave latency到达才发送。

3) 通过LL_CONNECTION_PARAM_REQLL_CONNECTION_PARAM_RSPLL_CONNECTION_UPDATE_IND等PDU来进行参数Slave latency设置。

4) 上面命令中,同时进行connInterval, connSlaveLatency and connSupervisionTimeout等三个参数的设置。

5) 大体设置流程如下图

  • 18
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是一个低功耗蓝牙广播的代码解析例子,以 Nordic nRF5 SDK 为例: ``` #include <stdbool.h> #include <stdint.h> #include "nrf.h" #include "nordic_common.h" #include "boards.h" #include "nrf_delay.h" #include "nrf_gpio.h" #include "nrf_drv_clock.h" #include "nrf_drv_power.h" #include "nrf_drv_rng.h" #include "nrf_drv_saadc.h" #include "nrfx_wdt.h" #include "app_error.h" #include "app_timer.h" #include "app_util_platform.h" #include "ble_advdata.h" #include "ble_gap.h" #include "ble_nus.h" #include "ble_hci.h" #include "ble_conn_params.h" #include "ble_conn_state.h" #include "ble_db_discovery.h" #include "ble_hci.h" #include "nrf_ble_gatt.h" #include "nrf_ble_qwr.h" #include "nrf_ble_scan.h" #include "nrf_ble_lesc.h" #include "nrf_ble_conn_params.h" #include "nrf_ble_gq.h" #include "nrf_ble_ancs_c.h" #include "nrf_ble_ans_c.h" #include "nrf_ble_bms_c.h" #include "nrf_ble_cscs_c.h" #include "nrf_ble_gattc.h" #include "nrf_ble_ias_c.h" #include "nrf_ble_lbs_c.h" #include "nrf_ble_rscs_c.h" #include "nrf_ble_tps_c.h" #include "nrf_ble_wscs_c.h" #include "bsp_btn_ble.h" #define DEVICE_NAME "LEDBlinker" /**< Name of device. Will be included in the advertising data. */ #define MANUFACTURER_NAME "NordicSemiconductor" /**< Manufacturer. Will be passed to Device Information Service. */ #define APP_BLE_CONN_CFG_TAG 1 /**< A tag identifying the SoftDevice BLE configuration. */ #define APP_ADV_INTERVAL 1600 /**< The advertising interval (in units of 0.625 ms. This value corresponds to 1 sec). */ #define APP_ADV_DURATION 0 /**< The advertising duration (180 seconds) in units of 10 milliseconds. */ #define APP_BLE_OBSERVER_PRIO 3 /**< Application's BLE observer priority. You shouldn't need to modify this value. */ #define APP_BLE_CONN_CFG_TAG 1 /**< A tag identifying the SoftDevice BLE configuration. */ #define APP_FEATURE_NOT_SUPPORTED BLE_GATT_STATUS_ATTERR_APP_BEGIN + 2 /**< Reply when unsupported features are requested. */ static ble_gap_adv_params_t m_adv_params; /**< Parameters to be passed to the stack when starting advertising. */ static uint8_t m_adv_handle = BLE_GAP_ADV_SET_HANDLE_NOT_SET; /**< Advertising handle used to identify an advertising set. */ static uint8_t m_enc_advdata[BLE_GAP_ADV_SET_DATA_SIZE_MAX]; /**< Buffer for storing an encoded advertising set. */ static uint8_t m_enc_scandata[BLE_GAP_ADV_SET_DATA_SIZE_MAX]; /**< Buffer for storing an encoded advertising set. */ static void advertising_start(void) { ret_code_t err_code; err_code = sd_ble_gap_adv_start(m_adv_handle, APP_BLE_CONN_CFG_TAG); APP_ERROR_CHECK(err_code); } static void advertising_init(void) { ret_code_t err_code; ble_advdata_t advdata; ble_advdata_manuf_data_t manuf_data; manuf_data.company_identifier = 0xFFFF; manuf_data.data.p_data = (uint8_t *) "Hello"; manuf_data.data.size = 5; memset(&advdata, 0, sizeof(advdata)); advdata.name_type = BLE_ADVDATA_FULL_NAME; advdata.include_appearance = true; advdata.flags = BLE_GAP_ADV_FLAGS_LE_ONLY_GENERAL_DISC_MODE; advdata.p_manuf_specific_data = &manuf_data; memset(m_enc_advdata, 0, sizeof(m_enc_advdata)); memset(m_enc_scandata, 0, sizeof(m_enc_scandata)); err_code = ble_advdata_encode(&advdata, m_enc_advdata, &m_adv_params.adv_data.len); APP_ERROR_CHECK(err_code); err_code = sd_ble_gap_adv_data_set(m_adv_handle, m_enc_advdata, m_adv_params.adv_data.len, m_enc_scandata, m_adv_params.scan_rsp_data.len); APP_ERROR_CHECK(err_code); } static void gap_params_init(void) { ret_code_t err_code; ble_gap_conn_params_t gap_conn_params; err_code = sd_ble_gap_tx_power_set(BLE_GAP_TX_POWER_ROLE_ADV, m_adv_handle, 4); APP_ERROR_CHECK(err_code); err_code = sd_ble_gap_conn_params_get(&gap_conn_params); APP_ERROR_CHECK(err_code); gap_conn_params.min_conn_interval = MSEC_TO_UNITS(20, UNIT_1_25_MS); gap_conn_params.max_conn_interval = MSEC_TO_UNITS(75, UNIT_1_25_MS); gap_conn_params.slave_latency = 0; gap_conn_params.conn_sup_timeout = MSEC_TO_UNITS(4000, UNIT_10_MS); err_code = sd_ble_gap_conn_params_set(&gap_conn_params); APP_ERROR_CHECK(err_code); } int main(void) { ret_code_t err_code; err_code = nrf_drv_clock_init(); APP_ERROR_CHECK(err_code); nrf_drv_clock_lfclk_request(NULL); err_code = nrf_drv_power_init(NULL); APP_ERROR_CHECK(err_code); err_code = nrf_drv_rng_init(NULL); APP_ERROR_CHECK(err_code); err_code = app_timer_init(); APP_ERROR_CHECK(err_code); err_code = bsp_init(BSP_INIT_LED, NULL); APP_ERROR_CHECK(err_code); err_code = sd_ble_gap_adv_set_configure(&m_adv_handle, &m_enc_advdata[0], &m_adv_params, &m_enc_scandata[0]); APP_ERROR_CHECK(err_code); gap_params_init(); advertising_init(); advertising_start(); while (1) { // Do nothing. } } ``` 该例子使用了 Nordic nRF5 SDK,实现了一个低功耗蓝牙广播的发送。在 `advertising_init()` 函数中,定义了广播的参数和内容,使用 `ble_advdata_encode()` 函数将广播编码成二进制数据,并通过 `sd_ble_gap_adv_data_set()` 函数设置广播数据。在 `advertising_start()` 函数中,启动了广播传输。 需要注意的是,该例子使用了 Nordic nRF5 SDK,如果您使用其他的嵌入式平台和开发工具,代码会有所不同。但是,低功耗蓝牙广播的实现原理是相同的。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值