Nordic-Thingy52-FW DFU

GATT(Generic Attribute Profile):

GATT是一个在蓝牙连接之上的发送和接收很短的数据段的通用规范,这些很短的数据段被称为属性(Attribute)。

GATT连接是独占的。也就是一个BLE外设同时只能被一个中心设备连接。一旦外设被连接,它就会马上停止广播,这样它就是一个BLE外设同时只能被一个中心设备连接。一旦外设被连接,它就会马上停止广播,这样它就对其他设备不可见了。当设备断开,它又开始广播。中心设备和外设需要双向通信的话,唯一的方式就是建立GATT连接。

GAP(Generic Access Profile):

它在用来控制设备和广播。GAP使你的设备被其他设备可见,并决定了你的设备是否可以或者怎样与合同设备进行交互。例如Beacon设备就只是向外广播,不支持连接,小米手环就等设备就可以与中心设备连接。

GAP给设备定义了若干角色,其中主要的两个是:外围设备(Peripheral)和中心设备(Central)。

外围设备:这一般就是非常小或者简单的低功耗设备,用来提供数据,并连接到一个更加相对强大的中心设备。例如小米手环。

中心设备:中心设备相对比较强大,用来连接其他外围设备。例如手机等。

在GAP中外围设备通过两种方式向外广播数据:Advertising Data Payload(广播数据)和Scan Response Data(扫描回复),这里包含一些设备额外的信息,例如设备的名字。

CCCD (Client Characteristic Configuration

Descriptor),这个描述符是给任何支持通知或指示功能的特性额外增加的。在CCCD中写入“1”使能通知功能,写入“2”使能指示功能,写入“0”同时禁止通知和指示功能。这个描述符是服务器来配置的,通过写事件来向其中写入值。

DFU(Device Firmware Update),设备固件升级,而OTA(Over The Air)是实现DFU的一种方式,OTA的全称应该是OTA DFU,即通过空中无线方式实现设备固件升级。为了方便起见,直接用OTA来指代固件空中升级。只要是通过无线通信方式实现DFU的,都可以叫OTA,比如2G/3G/4G/WiFi/蓝牙/NFC/Zigbee,他们都支持OTA。DFU除了可以通过无线方式(OTA)进行升级,也可以通过有线方式进行升级,比如通过UART,USB或者SPI通信接口来升级设备固件。不管采用OTA方式还是有线通信方式,DFU包括后台式(background)和非后台式两种模式。

后台式DFU,又称静默式DFU(Silent DFU),在升级的时候,新固件在后台悄悄下载,即新固件下载属于应用程序功能的一部分,在新固件下载过程中,应用可以正常使用,也就是说整个下载过程对用户来说是无感的,下载完成后,系统再跳到BootLoader模式,由BootLoader完成新固件覆盖老固件的操作,至此整个升级过程结束。

非后台式DFU,在升级的时候,系统需要先从应用模式跳入到BootLoader模式,由BootLoader进行新固件下载工作,下载完成后BootLoader继续完成新固件覆盖老固件的操作,至此升级结束。

双区DFU(dual bank)和单区DFU(single

bank),双区或者单区DFU是新固件和老固件覆盖的两种方式。后台式DFU必须采用双区模式进行升级,即老系统(老固件)和新系统(新固件)各占一块bank(存储区),假设老固件放在bank0中,新固件放在bank1中,升级的时候,应用程序先把新固件下载到bank1中,只有当新固件下载完成并校验成功后,系统才会跳入BootLoader模式,然后擦除老固件所在的bank0区,并把新固件拷贝到bank0中。非后台式DFU可以采用双区也可以采用单区模式,与后台式DFU相似,双区模式下新老固件各占一块bank(老固件为bank0,新固件为bank1),升级时,系统先跳入BootLoader模式,然后BootLoader程序把新固件下载到bank1中,只有新固件下载完成并校验成功后,才会去擦除老固件所在的bank0区,并把新固件拷贝到bank0区。单区模式的非后台式DFU只有一个bank0,老固件和新固件分享这一个bank0,升级的时候,进入bootloader模式后立马擦除老固件,然后直接把新固件下载到同一个bank中,下载完成后校验新固件的有效性,新固件有效升级完成,否则要求重来。跟非后台式DFU双区模式相比,单区模式节省了一个bank的Flash空间,在系统资源比较紧张的时候,单区模式是一个不错的选择。不管是双区模式还是单区模式,升级过程出现问题后,都可以进行二次升级,都不会出现“变砖”情况。不过双区模式有一个好处,如果升级过程中出现问题或者新固件有问题,它还可以选择之前的老固件老系统继续执行而不受其影响。而单区模式碰到这种情况就只能一直待在bootloader中,然后等待二次或者多次升级尝试,此时设备的正常功能已无法使用,从用户使用这个角度来说,你的确可以说此时设备已经“变砖”了。所以说,虽然双区模式牺牲了很多存储空间,但是换来了更好的升级体验。

如果采用双区模式DFU,那么Bank0放的是应用程序,即老固件,Bank1放的是新固件。平时,Bank1为空或者忽略,系统只跑Bank0里面的应用程序;升级的时候,先跳到BootLoader,然后接收新固件并把它放在Bank1中,最后把Bank1里面的固件拷贝到Bank0中。

如果采用单区模式,则没有Bank1这个区。平时,系统只跑Bank0里面的代码;升级的时候,跳到BootLoader,先擦除Bank0里面的老程序,并把新固件直接放在Bank0中。

根据升级时如何跳转到Bootloader,Nordic SDK又将DFU分为按键式DFU和非按键式(Buttonless)DFU:

按键式DFU:就是上电时长按某个按键以进入bootloader模式。

非按键式DFU:就是整个DFU过程中设备端无任何人工干预,通过BLE/UART/USB接口给应用程序发送一条指令,应用程序收到指令后再自动跳入bootloader模式。

不管是按键式DFU还是非按键式DFU,两者只是进入BootLoader的方式不一样,其余基本一样,尤其是BootLoader工作过程基本上是一模一样的。

程序跳到BootLoader后,根据BootLoader需不需要对新固件进行验签,Nordic SDK又把DFU分为开放式DFU和安全式DFU(又称签名DFU)。

Legacy DFU:BootLoader不做任何验证,直接把新固件接收下来。

Security DFU:BootLoader存有一把公钥,BootLoader会先用这把公钥验证新固件的签名,只有验签通过,才允许后续的工作:比如把新固件接收下来;如果验签失败,BootLoader将拒绝升级,重新跳回应用程序。

BootLoader可以通过不同的通信接口来接收新的固件,Nordic

SDK支持BLE,UART和USB三种接口。


nRF52的启动流程:

上电后,系统先执行softdevice,softdevice通过读取UICR一个寄存器的值,来判断目前系统是否有BootLoader,如果没有BootLoader,系统直接跳到application;如果有BootLoader,系统先跳到BootLoader,BootLoader再根据目前的情况来决定是进入升级模式还是跳往application,BootLoader主要判断如下几种情况:

按键是否按下

保持寄存器GPREGRET1是否为0xB1

上次DFU过程是否还在进行中

应用程序校验是否通过

如果按键没有按下,GPREGRET1不为0xB1,本次复位不是上次DFU的继续,并且应用程序校验通过,那么BootLoader就会直接跳到application,去执行application应用程序。那怎么去校验应用程序的有效性呢?为此BootLoader引入了一个放在Flash的结构体参数:m_dfu_settings_buffer(数据类型:nrf_dfu_settings_t),这个结构体参数虽然只有896字节,但由于Flash只能按页擦除,所以这个参数实际占用了一个Flash page,这个page称为settings page。

Bootloader Setting Page是Bootloader工程开辟的一段Flash空间,其中保存了固件镜像的信息和DFU进度信息,这些信息称为Bootloader Settings。nRF52832芯片的Flash大小为512 kB(0x0008 0000),Settings位于最顶端(0x0007 F000 – 0x0008 0000),大小为1 Page(4 kB)。在Settings的下方是MBR

Param Storage,它的长度也是4 kB,地址为:0x0007 E000 – 0x0007 F000。这块区域用于存放Settings的备份。Bootloader启动时候,会检查Settings中以下信息:

bank0_bank_code

bank0_img_crc

如果二者都正确,则执行跳转进入Application,否则驻留在Bootloader中执行DFU。

nRF52系列芯片,Settings的位置如图:

settings page放在Flash的最后一个页面,settings page目前有2个版本:版本1(SDK15.2及以前版本)和版本2(SDK15.3及以后版本),版本2可以兼容版本1,前面所述的896字节是指settings page版本2的大小。

Settings page包含的信息比较多,用得比较多的是:

各种版本信息

DFU升级过程信息

Application image的CRC值和大小

应用程序的bonding信息

Init command内容

application/softdevice的启动校验信息(版本2才有)

版本1的settings page只校验application image的CRC值,如果CRC匹配,则认为application有效。

版本2的settings page不仅可以校验application image的CRC值,还可以校验application/softdevice的CRC值或者hash值或者签名,你可以选择你自己想要的校验方式,只有CRC值或者hash值或者签名校验通过(三选其一),应用程序才算有效,这时BootLoader才会跳到application去执行。

无按键式BLE OTA的工作流程:

1)     正常启动后,系统运行在应用程序中,此时手机通过app发送一条开始DFU的指令给设备,设备收到指令后,将GPREGRET1赋值0xB1,并触发软复位

#defineBOOTLOADER_DFU_GPREGRET                    (0xB0)    

#defineBOOTLOADER_DFU_START_BIT_MASK          (0x01)    

#defineBOOTLOADER_DFU_START   (BOOTLOADER_DFU_GPREGRET | BOOTLOADER_DFU_START_BIT_MASK)

sd_power_gpregret_set(0,BOOTLOADER_DFU_START);

//手机Android端log

D/BluetoothGatt:setCharacteristicNotification() - uuid: 8ec90004-f315-4f60-9fb8-838830daea50enable: true

D/BluetoothGatt:writeDescriptor() - uuid: 00002902-0000-1000-8000-00805f9b34fb

D/BluetoothGatt:onDescriptorWrite() - Device=C7:77:73:**:**:** handle=17

I/DfuImpl: SendingEnter Bootloader (Op Code = 1)

2)     复位后,系统再次进入BootLoader,因为GPREGRET1等于0xB1,BootLoader进入DFU模式,等待新固件接收

static booldfu_enter_check(void)

{

    if (NRF_BL_DFU_ENTER_METHOD_PINRESET&&

       (NRF_POWER->RESETREAS &POWER_RESETREAS_RESETPIN_Msk))

    {

        NRF_LOG_DEBUG("DFU mode requestedvia pin-reset.");

        return true;

}

}

3)     手机先将init packet发送给设备,设备先做前期检验prevalidation,主要是版本校验以及签名验签,校验通过后,更新settings

 page并准备开始数据接收。

nrf_dfu_validation_prevalidate();

//手机Android端log

I/DfuImpl: Sending 141 bytes of init

packet...

I/DfuImpl: Sendinginit packet (Value =12-8A-01-0A-44-08-01-12-40-08-03-10-34-1A-02-CB-01-20-00-28-00-30-00-38-D4-87-05-42-24-08-03-12-20-A0-AA-67-6B-51-58-81-96-0A-D5-01-21-EB-6E-4F-0E-A1-B6-36-1F-0C-2E-87-DA-F4-17-81-D2-DE-3F-8F-45-48-00-52-04-08-01-12-00-10-00-1A-40-31-B2-5C-54-CF-41-FB-68-05-68-8E-FD-19-FB-45-AD-C0-F7-08-C9-25-84-A0-CC-E4-E0-4E-CE-AF-44-A7-EE-E7-36-C7-67-A1-DC-A7-F6-B5-82-B3-AC-17-56-82-FE-81-6E-BB-45-48-2B-F2-E8-03-A3-73-4A-E5-EA-9A-14)

D/BluetoothGatt:writeCharacteristic() - uuid: 8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

I/DfuImpl: Sending Calculate Checksumcommand (Op Code = 3)

D/BluetoothGatt:writeCharacteristic() - uuid: 8ec90001-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onNotify() - Device=C7:77:73:**:**:** handle=18

    onCharacteristicWrite() -Device=C7:77:73:**:**:** handle=18 Status=0

I/DfuImpl: Checksumreceived (Offset = 141, CRC = 03F169BE)

    Executing init packet (Op Code = 4)

D/BluetoothGatt:writeCharacteristic() - uuid: 8ec90001-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=18 Status=0

D/BluetoothGatt:onNotify() - Device=C7:77:73:**:**:** handle=18

4)     接收新固件。每接收4kB数据,回复一次CRC校验值,直至整个新固件image接收完毕,如果新固件校验通过(版本1校验CRC值,版本2校验hash值),就会去invalidate bank0里面的老固件,更新settings page,并再次触发软复位。

//手机Android端log

I/DfuImpl: Uploading firmware...

D/BluetoothGatt:writeCharacteristic() - uuid: 8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

D/BluetoothGatt:writeCharacteristic() - uuid: 8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

D/BluetoothGatt:writeCharacteristic() - uuid: 8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

D/BluetoothGatt:writeCharacteristic() - uuid: 8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid:8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

    writeCharacteristic() - uuid: 8ec90002-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onCharacteristicWrite() - Device=C7:77:73:**:**:** handle=16 Status=0

I/DfuImpl: Sending Calculate Checksumcommand (Op Code = 3)

D/BluetoothGatt:writeCharacteristic() - uuid: 8ec90001-f315-4f60-9fb8-838830daea50

D/BluetoothGatt:onNotify() - Device=C7:77:73:**:**:** handle=18

    onCharacteristicWrite() -Device=C7:77:73:**:**:** handle=18 Status=0

I/DfuImpl: Checksum received (Offset =4096, CRC = 825DA3DF)

5)     BootLoader启动后发现有新固件需要activate(激活),此时会去擦掉bank0里面的固件,并把bank1里面的固件拷贝到bank0,然后更新settings page,并再次触发软复位。

注:上面讲的是dual bank的流程,single bank与之相似,只不过在第3)步的时候就会去擦除老固件。

6)     BootLoader再次启动后,检查新image的有效性,校验通过后,跳到新的application去执行代码

从上面流程可以看出,DFU过程中,系统需要跑两段完全独立的代码:Application和BootLoader,Application和BootLoader都支持蓝牙功能,也就是说,两者都有自己的蓝牙广播和蓝牙连接。这里面有一个问题:当系统从Application跳到BootLoader后,手机怎么辨别两者为同一个设备?很多人会说,可以让BootLoader和Application两者的广播名字一样,根据广播名字的一致性,来判断二者来自同一个设备。这种方法存在两个问题:

一:大部分手机都支持GATT cache(缓存)功能,当application跟手机相连后,手机会把application的GATT数据缓存下来以加快下次连接的速度(这个现象在苹果手机最明显),之后如果系统跳到BootLoader,然后再跟手机相连,如果两者的蓝牙设备地址一样,手机会认为是同一个设备,从而跳过服务发现的过程而直接使用之前缓存下来的GATT数据,这样会导致BootLoader的服务无法被手机发现,从而出现升级失败。

二:如果多个设备同时在升级,而我们仅仅依靠广播名字来决定两者属不属于同一个设备,这会导致设备A application有可能跟设备B的BootLoader进行错配。

为了解决这个问题,Nordic提出了两套方案:

方案一:假设application的蓝牙设备地址为x,跳到BootLoader后蓝牙设备地址会变成x+1,这样手机就可以通过这种地址+1的方式来辨别两者属不属于同一个设备,由于application和BootLoader使用不同的蓝牙设备地址,前面的GATT缓存问题也就不存在。关于方案一,有一个问题需要特别注意:如果你想修改例子默认的蓝牙设备地址(比如使用IEEE的public蓝牙MAC地址),此时一定要记得同时更改application和BootLoader的蓝牙设备地址,使他们满足+1的条件,否则Nordic手机DFU库无法辨别两者是否属于同一个设备,以致于无法完成OTA过程。

方案二:application和BootLoader的蓝牙设备地址一模一样,但设备跟手机执行配对和bonding操作,设备跟手机bonding后,就可以支持service changed indicate操作,这样跳到BootLoader后可以让手机主动再执行一次服务发现过程,从而解决GATT缓存问题。

签名验签的工作原理:首先,你需要一对公私钥,其中私钥用来生成新固件的签名,公钥用来验证签名的有效性,大家可以用nrfutil来生成自己需要的公私钥对,公私钥制作成功后,私钥一定要妥善保管。公钥可以变成一个.c文件,并覆盖DFU工程下的同名文件:dfu_public_key.c。其次,BootLoader要支持签名验签密码算法,这个DFU代码已经有了,并且有四种后端可选:micro-ecc,cc310_bl,Oberon和mbedtls,四选其一即可(这4种后端,只有cc310是硬件实现,其余都是软件实现),nRF52840推荐选择cc310作为算法后端,其他nRF52芯片推荐选择micro-ecc作为算法后端。micro-ecc效率高,占用的代码空间最小,但它的版权是CPOL,只要你能接受CPOL,那么推荐使用micro-ecc;反之,如果接受不了CPOL版权,而且硬件又不支持cc310,那么推荐使用Oberon,不过Oberon占用的代码空间比micro-ecc要大一些。再次,手机端要生成新固件的签名,并把新固件的签名传给设备端。大家还是可以用nrfutil去生成新固件的签名。最后,BootLoader接收到新固件hash值和签名,并使用自己的公钥对该签名进行验签。由于nrfutil是PC端应用程序,所以它可以集成各种加密算法库,并完成上面提及的公私钥对,hash和签名的生成工作。

NRF_SDH_BLE_OBSERVER用来为本地文件注册一个BLE回调函数(此处为ble_evt_handler),NRF_SDH_BLE_OBSERVER这个宏执行成功后,所有的BLE事件都会被ble_evt_handler捕获。进入ble_evt_handler,你会发现BLE有上百个回调事件,你不需要每个都处理,你只需要处理你关心的事件即可,比如连接成功事件BLE_GAP_EVT_CONNECTED或者连接断开事件BLE_GAP_EVT_DISCONNECTED。NRF_SDH_BLE_OBSERVER有一个很大的好处:某个文件如果需要捕获BLE事件,那么它只需在本文件中某处(可以在函数内也可以在函数外)调用NRF_SDH_BLE_OBSERVER这个宏去注册一个回调函数即可,而不再需要在其它文件中去注册这个回调函数,将模块的耦合性降到最低,符合模块化编程思想。

//APP固件主函数

int main(void)

{   

#ifdefBUTTONLESS_ENABLED

    // Initialize the async SVCI interface tobootloader before any interrupts are enabled.

    ret_code_t err_code =ble_dfu_buttonless_async_svci_init();

    APP_ERROR_CHECK(err_code);

#endif

    // Initialize.

    system_init();

    scheduler_init();

    log_init(); //打印日志初始化

    timers_init(); //APP timer初始化

#ifdef DEV_BSP

    buttons_leds_init();

#endif

    power_management_init(); //低功耗管理初始化

    ble_stack_init();  //蓝牙协议栈初始化

    mac_address_get();

#ifdefBOND_ENABLE   

    peer_manager_init();

#endif

    gap_params_init(); //修改广播名字和连接间隔的

    gatt_init(); //用来修改底层数据包长度的

    advertising_init(); //用来修改广播包内容,广播间隔以及广播超时时间

    services_init(); //蓝牙服务,characteristic,NUS, buttonless,

    advertising_init(); //为啥调用两次?

    conn_params_init(); //用来请求更新连接间隔

    fs_init();

    application_timers_start();

    // Start execution.

    NRF_LOG_INFO("Debug logging for UARTover RTT started.");

    ctl_advertising(); //启动广播         

    twi_master_init(); //I2C初始化

    nfc_init(); //NFC初始化

    wdt_init(); //看门狗初始化

    // Enter main loop.

    for (;;)

    {

        main_loop();

        app_sched_execute();

        //先把需要打印的日志打印完,然后让系统进入idle状态(SystemON),

        //在idle状态下,蓝牙连接或者广播可以正常进行而不受影响

        idle_state_handle();

    }

}


#defineNUS_BASE_UUID{{0x9E, 0xCA, 0xDC, 0x24, 0x0E, 0xE5, 0xA9, 0xE0, 0x93, 0xF3, 0xA3, 0xB5, 0x00,0x00, 0x40, 0x6E}}

uint32_tble_nus_init(ble_nus_t * p_nus, ble_nus_init_t const * p_nus_init)

{

    //添加自定义UUID

    sd_ble_uuid_vs_add(&nus_base_uuid,&p_nus->uuid_type);

    //添加服务,UUID,返回服务的句柄

    sd_ble_gatts_service_add(BLE_GATTS_SRVC_TYPE_PRIMARY,&ble_uuid,&p_nus->service_handle);

    //添加RX特征值

characteristic_add(p_nus->service_handle,&add_char_params, &p_nus->rx_handles);

//添加TX特征值

characteristic_add(p_nus->service_handle,&add_char_params, &p_nus->tx_handles);

}

ble_nus_init同时注册了nus_data_handler回调函数,当设备收到手机发过来的数据时,就会触发nus_data_handler,用户可以在nus_data_handler中对接收到的数据进行处理。


蓝牙数据传输吞吐率一个连接间隔传输的数据连接间隔一个数据包长度包数连接间隔


一个数据包的长度取决于下面两个参数:

#defineNRF_SDH_BLE_GAP_DATA_LENGTH  251

#defineNRF_SDH_BLE_GATT_MAX_MTU_SIZE        247

NRF_SDH_BLE_GAP_DATA_LENGT≥(NRF_SDH_BLE_GATT_MAX_MTU_SIZE+4) ,

NRF_SDH_BLE_GAP_DATA_LENGTH最大值为251,MTU设为247,相当于把数据包应用数据最大长度设为247-3=244字节


连接间隔由下面两个参数建议:

#defineMIN_CONN_INTERVAL              MSEC_TO_UNITS(15, UNIT_1_25_MS)

#defineMAX_CONN_INTERVAL              MSEC_TO_UNITS(30, UNIT_1_25_MS)

终连接间隔设为多少只能由蓝牙主设备决定,蓝牙从设备只有建议权。所以上面就是对主机提出建议:希望主机把连接间隔设为30ms,主机可以接受也可以拒绝这个请求。


一个连接间隔可以发多个包由下面参数决定:

#defineNRF_SDH_BLE_GAP_EVENT_LENGTH 6

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,776评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,527评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,361评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,430评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,511评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,544评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,561评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,315评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,763评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,070评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,235评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,911评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,554评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,173评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,424评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,106评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,103评论 2 352

推荐阅读更多精彩内容