蓝牙系列七:开源蓝牙协议栈BTStack数据处理(Wireshark抓包分析)

继续蓝牙系列的研究。

在上篇博客,通过阅读BTStack的源码,大体了解了其框架,对于任何一个BTStack的应用程序都有一个main函数,这个main函数是统一的。这个main函数做了某些初始化之后,最终会调用到应用程序提供的btstack_main,在btstack_main里面首先做一些初始化,然后调用hci_power_on函数去打开蓝牙模块。

一. 数据类型

运行BTStack程序时,会生成hci_dump.pklg文件,可以使用WireShark打开此文件,截图如下:

怎么理解上图中的数据呢?

BTStack中涉及的数据有2类:

1.从硬件上获得的数据、发给硬件的数据

2.为更新系统状态而虚构的数据

1. 跟硬件相关的数据有4类:

① 发送给蓝牙控制器的Command

② 从蓝牙控制器获得的Event,蓝牙控制器收到Command后会回复Event

③ ACL数据,这涉及收、发两个方向

④ SCO数据,这涉及收、发两个方向

注意:ACL、SCO数据的含义以后再讲。

这4种数据类型,用一个头部信息来表示,参考bluetooth.h:

#define HCI_COMMAND_DATA_PACKET 0x01

#define HCI_ACL_DATA_PACKET       0x02

#define HCI_SCO_DATA_PACKET       0x03

#define HCI_EVENT_PACKET           0x04

但是在程序中,单凭这4个数值无法分辨数据的流向,比如ACL数据的类型是0x03,我们单凭0x03无法知道这数据是发给硬件、还是从硬件读到。

为了便于调试,BTStack在打印Log信息时,把这些硬件数据类型转换为新数值:

参考函数: hci_dump_packetlogger_setup_header

1. Command :  0x00

2. Event:       0x01

3. ACL out     0x02

4. ACL in      0x03

5. SCO out    0x08

6. SCO in     0x09

7. Log Message 0xfc

我们可以使用WireShark打开Log文件hci_dump.pklg时,观察里面原始数据。

2. 为更新系统状态而虚构的数据:

有很多种虚构的数据,下面举几个例子:

① 提示状态发生了变化:

在BTStack中,可能有很多层对hci_stack->state感兴趣,所以当hci_stack->state发生变化时,可以使用hci_emit_state发送一个虚拟的Event数据包,这会导致这些层的处理函数被调用。

BTStack中使用下面函数发送state信息:

在WireShark中看到的原始数据为:01 60 01 xx,

第1个01表示Event,60表示BTSTACK_EVENT_STATE,第2个01表示数据长度为1, xx表示数据即state值。

② 当一个数据包已经成功发给硬件之后,我们要通知上层:你可以继续发送数据给硬件了。这通过hci_emit_transport_packet_sent函数来实现:

在WireShark中看到的原始数据为:01 6e 00,

第1个01表示Event,6e表示HCI_EVENT_TRANSPORT_PACKET_SENT,00表示后续数据长度为0。

二、状态机:

我们常说:初始化好蓝牙模块后,就可以使用它了。

仔细琢磨这句话,蓝牙模块至少有这2个状态:

1. 初始化状态:HCI_STATE_INITIALIZING

2. 工作状态:HCI_STATE_WORKING

当然,还有其他状态,在代码中如下表示(hci_cmd.h):

在HCI_STATE_INITIALIZING状态下,需要跟蓝牙模块多次交互,才可以完成蓝牙模块的初始化。使用“子状态”来表示这些多次交互,在代码中如下表示(hci.h):

 举个例子,子状态中有“HCI_INIT_SEND_RESET”和“HCI_INIT_W4_SEND_RESET”:

1.当子状态为HCI_INIT_SEND_RESET时:

我们发送复位命令给蓝牙模块,然后子状态变为HCI_INIT_W4_SEND_RESET,它的意思是“wait for”,等待收到复位命令的回复信息。

2.收到该回复信息后,子状态变为HCI_INIT_SEND_READ_LOCAL_VERSION_INFORMATION:

于是继续给蓝牙模块发送“read loacal version”命令,然后子状态变为HCI_INIT_W4_SEND_READ_LOCAL_VERSION_INFORMATION,表示等待回复信息

如此继续,直到子状态变为“HCI_INIT_DONE”,初始化才结束,蓝牙模块的“状态”才放为HCI_STATE_WORKING。

代码中有一个结构体:

static hci_stack_t * hci_stack

hci_stack->state表示“状态”,hci_stack->substate表示“子状态”。

BTStack的代码有函数hci_run,它就是根据hci_stack结构体中的这些状态、其他值来收发数据的。

注意:hci.c中的hci_run是核心函数,它根据hci_stack的状态进行不同的处理。

举例说明:

1.例子1:hci_power_control(HCI_POWER_ON);

hci_stack->state初始值为0,即HCI_STATE_OFF;

调用hci_power_transition_to_initializing后,各状态值如下:

// set up state machine

hci_stack->num_cmd_packets = 1; // assume that one cmd can be sent

hci_stack->hci_packet_buffer_reserved = 0;

hci_stack->state = HCI_STATE_INITIALIZING;

hci_stack->substate = HCI_INIT_SEND_RESET;

接着调用如下代码:

// trigger next/first action

hci_run();

hci_run函数中,在hci_stack->state等于HCI_STATE_INITIALIZING时,调用:hci_initializing_run();

hci_initializing_run()函数内部,会根据hci_stack->substate等于HCI_INIT_SEND_RESET而发出复位命令,并令substate等于HCI_INIT_W4_SEND_RESET,这表示等待收到该命令的回复信息。

在等待过程中,程序休眠。

当收到数据时,程序的主循环继续执行,根据上节内容,将会调用hci.c中的event_handler函数来处理

该函数有如下代码:

    // handle BT initialization

    if (hci_stack->state == HCI_STATE_INITIALIZING){

        hci_initializing_event_handler(packet, size);

}

……

hci_run( );

模块的当前状态仍为HCI_STATE_INITIALIZING,进而调用hci_initializing_event_handler(packet, size)。

hci_initializing_event_handler将调用hci_initializing_next_state(),把subsate设置为HCI_INIT_SEND_READ_LOCAL_VERSION_INFORMATION。

后续的hci_run会根据这个substate发出READ_LOCAL_VERSION_INFORMATION的命令。

  • 4
    点赞
  • 9
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 蓝牙协议栈BTstack是一种为蓝牙设备开发的基础框架。它旨在为使用低功耗蓝牙协议(BLE)和传统蓝牙协议(BR / EDR)的设备提供通用接口。 BTstack提供了一组API,使开发人员可以使用不同平台上的相同代码来处理蓝牙样式的应用程序。 BTstack提供的API包括协议栈初始化,L2CAP,RFCOMM,SDP,ATT / GATT等。 BTstack的另一个优点是它是一个开源项目,可以在许多平台上使用。它已被移植到多种硬件平台和操作系统中,例如ARM Cortex-M3 / M4(例如STM32),ATmega,Arduino,Raspberry Pi,Windows,Linux,macOS和iOS。 BTstack的应用广泛,包括智能手机,平板电脑,手表,耳机,音响,传感器和医疗设备。随着IoT设备的普及,BTstack将越来越受到重视。 总之,BTstack是一个功能强大、开源、跨平台的蓝牙协议栈,为开发人员提供了通用API,使他们可以在不同的平台上开发相同的应用程序。它是蓝牙设备开发者的重要工具之一。 ### 回答2: 蓝牙协议栈是指一套硬件和软件技术,用于使不同设备间的数据传输更加方便快捷。蓝牙协议栈的运行方式是通过多层协议交互实现的,向上层应用提供数据传输的支持,向下层硬件提供调用接口。 BTstack是一种开源蓝牙协议栈,提供了高效、可靠和易于使用的终端设备的蓝牙连接。BTstack支持各种蓝牙协议,包括BLE、HFP、A2DP、AVRCP和SPP等,可以在不同的操作系统和硬件平台上使用。BTstack的优点在于它是可移植的,可以快速适应各种不同的蓝牙设备。 BTstack还支持多个蓝牙连接,并提供了稳定的设备连接,具备智能重连和设备发现功能,同时还支持多个蓝牙配置文件。BTstack还支持单片机,特别是ARM Cortex-M处理器,以及一些嵌入式系统平台。 总之,BTstack是一种强大的、稳定的、高度可移植的蓝牙协议栈,可以帮助开发者快速开发出高效、可靠的蓝牙连接应用。 ### 回答3: 蓝牙协议栈是用来实现蓝牙通信协议的软件模块,因此需要进行一系列的层次化操作,这些层次可以分为物理层,链路层,传输层,应用层。而btstack则是其中一个基于C语言的开源协议栈btstack采用了模块化的设计,每个模块之间相对独立,且易于扩展。可以在移植到不同平台上时,很容易添加某些特定的硬件驱动和操作系统。同时,btstack也提供了丰富的API接口,便于快速实现各种蓝牙应用,例如数据传输、音频传输、打印机、键盘/鼠标等等。 除此之外,btstack还支持免费开源蓝牙协议栈,可供个人开源项目和商业开发者使用。同步提供了符合不同使用场景的两种许可证:BSD和GPLv2。且支持大部分操作系统,包括Windows、MacOS、Linux、Android、IOS 和Windows Phone等,在各个平台上表现出色。 总而言之,btstack是个高度模块化、可移植性高、及具有完整的特性和相同的性能的优秀蓝牙堆栈。它极大地推进了蓝牙技术的发展,且为蓝牙应用开发者提供了更多的灵活自由,是目前蓝牙通信领域应用很广泛的协议栈
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值