【Android】【Bluetooth Stack】蓝牙电话本协议分析(超详细)

2024-03-20 10:12

本文主要是介绍【Android】【Bluetooth Stack】蓝牙电话本协议分析(超详细),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1. 精讲蓝牙协议栈(Bluetooth Stack):SPP/A2DP/AVRCP/HFP/PBAP/IAP2/HID/MAP/OPP/PAN/GATTC/GATTS/HOGP等协议理论
2. 欢迎大家关注和订阅,【蓝牙协议栈】专栏会持续更新中.....敬请期待!

目录

1. 协议简述

1.1 PBAP

1.2 OBEX

2. PBAP协议栈

2.1 vCard格式

2.2 PBAP 角色介绍

2.3 PSE & PCE 功能

2.3.1 Download

2.3.1.1 PullPhonebook Data Format

2.3.1.2 Application Parameters Header

2.3.2 Browsing

2.3.2.1 SetPhonebook Data Format

2.3.2.2 PullvCardListing Data Format

2.3.2.3 PullvCardEntry Data Format

2.3.3 vcf文件


1. 协议简述

        蓝牙电话应用不但需要HFP协议来支持打电话的功能,同时在很多车载蓝牙应用中,都支持查看通讯录和通话记录等信息,而这一部分的所涉及到的协议为PBAP.

 1.1 PBAP

        PBAP(Phone Book Access Profile):电话本访问协议 ,是一种基于OBEX的上层协议,该协议可以同步手机这些具有电话本功能设备上的通讯录和通话记录等信息,用于访问电话本对象(通过 Vcard形式),是基于客户端/服务器的模型,一般是 client从 server端下载电话本。这个协议是为 HFP/SIM协议设计. 

1.2 OBEX

        Object Exchange,对象交换协议,来源与红外通讯协议,但又不局限与具体的传输方式,后来被蓝牙组织SIG吸纳其中部分并进行优化处理作为蓝牙协议中的OBEX层用于蓝牙设备间的文件数据传输,如蓝牙传输文件(OPP)、同步电话簿(PBAP)和同步短信(MAP)等场景下都是以OBEX协议组织相关数据进行传输的;

        OBEX协议有两种角色:Server和Client,通过request-response(请求-响应)形式进行交互,即客户端Client进行请求,服务端Server响应客户端请求的方式传输数据对象;应用于PBAP协议中,Client只能进行数据的读取操作,不能对源数据进行修改,保证了源数据的安全性;

2. PBAP协议栈

 

        PBAP应用层协议处于最上层,之后就是数据格式处理方式,由于通讯录在手机中都是以vCard的格式存储的,所以这边为vCard的数据处理格式。在往下就是通过OBEX协议层联通蓝牙协议栈中的RFCOMM,最后通过统一的数据传输通道L2CAP链路发送数据;

2.1 vCard格式

BEGIN:VCARD\r\n
VERSION:3.0\r\n
N:;胡x;;;\r\n
FN:胡x\r\n
TEL;TYPE=CELL:610xxx\r\n
END:VCARD\r\n

  • BEGIN:VCARD:一组联系人信息开始标志
  • END:VCARD:结束标志
  • VERSION:版本
  • FN:姓名
  • TEL;TYPE=CELL:联系人联系方式

上述的参数为必要的,有些参数是可选项,例如:住址、邮件等信息;

当前vCard的版本有vCard 2.1 和 vCard 3.0 两个版本,所以PSE需要两种数据格式都支持,同步数据时根据PCE请求的哪种格式就以哪种格式封装数据进行传输。但是无论是哪种格式,vCard属性内容字符集使用唯一的字符编码utf-8格式进行编码转换;

2.2 PBAP 角色介绍

协议栈中定义了两种角色:

  • PSE:Phone Book Server Equipment,拥有电话本源数据的设备,作为Server,比如手机;
  • PCE:Phone Book Client Equipment,向PSE端请求电话本源数据的设备,作为Client,例如车机;

因为PBAP协议是基于OBEX协议实现的,那PBAP协议获取数据的方式也是通过request-response形式传输的;

2.3 PSE & PCE 功能

FeatureSupport by the PCESupport by the PSE
DownloadC1M
BrowsingC1M
Database IdentifierC3M
Folder Version CountersOM
vCard SelectingOM
Enhanced Missed CallsOO
X-BT-UCI vCard PropertyOO
X-BT-UID vCard PropertyOO
Referencing ContactsC2C2
Contact Image Default FormatXM
  • C1:至少支持其中一种
  • C2:如果支持' X-BT-UID vCard Property ',则可选,否则不支持不可选
  • C3:如果支持“Folder Version Counters” 或 “X-BT-UID vCard Property”,则必选,否则可选
  • O:可选
  • M:必选,必须支持

我们常使用到的功能为:Download和Browsing;我们分析一下;

2.3.1 Download

        Download功能可以将电话簿对象的全部内容同步到PCE,从而PCE端获取到数据后完全可以通过蓝牙电话等应用程序将数据显示到界面,一样可以达到滚动浏览电话簿信息的目的;

        Download这个功能特别是用于PSE端存储的电话簿容量相对较大,PCE设备通常从PSE端下载这些大容量数据并在其本地存储整个电话簿的场景。

FeatureFunctionSupport by the PCESupport by the PSE
Phone Book Download featurePullPhonebookMM

协议层提供了PullPhonebook函数,这个函数就是用来下载自己感兴趣的电话簿对象;

2.3.1.1 PullPhonebook Data Format

        由于PBAP协议是基于OBEX的,所以PullPhonebook函数顾名思义也是采用request-response这种一问一答的形式传输数据;

请求格式如下:

Field / HeaderNameValueStatus
FieldOpcodeGET(0x03 or 0x83)M
FieldPacket LengthVariesM
HeaderConnection IDVariesM
HeaderSingle Response Mode0x01C1
HeaderSingle Response Mode Param0x01C2
HeaderNameObject name (*.vcf)M
HeaderType"x-bt/phonebook"M
HeaderApplication Parameters
- PropertySelectorVariesO
- FormatVariesO
- MaxListCountVariesO
- ListStartOffsetVariesO
- ResetNewMissedCallsVariesC3
- vCardSelectorVariesC4
- vCardSelectorOperatorVariesC5
  • Opcode:操作码,可以理解为标识码,和Type参数组合形成了唯一标识;
  • Connection ID:PBAP连接指令中PSE回复的连接ID 号;
  • Name:表明了需要同步哪种数据;
  • Type:对应了Function,基本上每一个Function对应一个Type,除了SetPhonebook Function;
  • Application Parameters:应用设置的参数,PSE的回复数据会根据这些参数来组装回复data数据,其中就包括了PropertySelector、Format、MaxListCount等参数信息;
  • MaxListCount:本次 GET 获取的最大List Count;
  • ListStartOffset:本次 GET 开始的List Offset;

响应格式如下:

Field / HeaderNameValueStatus
FieldResponse0x09 or 0xA0 or Error CodeM
FieldPacket LengthVariesM
HeaderSingle Response Mode0x01C1
HeaderSingle Response Mode Param0x01C2
HeaderApplication Parameters
- PhonebookSizeVariesC3
- NewMissedCallsVariesC4
- PrimaryFolderVersionVariesC5
- SecondaryFolderVersionVariesC5
- DatabaseIdentifierVariesC6
HeaderBody/End of BodyvCard object(s)C7
  • PhonebookSize:Name个数,当MaxListCount = 0时,返回Name的总个数;
  • NewMissedCalls:新增的未接电话
  • Body/End or Body:请求中MaxListCount != 0时,返回Name对应的数据,回复数据中的vCard对象只应包含使用属性选择器Attribute Selector参数指示的属性,并且应使用格式Format参数指示的格式组装数据;
2.3.1.2 Application Parameters Header

该数据是由一组不同的TAG组成的整体的Application Parameters;

ValueTag IDLengthPossible Values
Order0x011 byte0x00 = indexed 0x01 = alphanumeric 0x02 = phonetic
Search Value0x02variableText
SearchProperty0x031 byte0x00 = Name 0x01 = Number 0x02 = Sound
MaxListCount0x042 bytes0x0000 to 0xFFFF
ListStartOffset0x052 bytes0x0000 to 0xFFFF
PropertySelector0x068 bytes64 bits mask
Format0x071 byte0x00 = 2.1 0x01 = 3.0
PhonebookSize0x082 bytes0x0000 to 0xFFFF
NewMissedCalls0x091 byte0x00 to 0xFF
PrimaryVersionCounter0x0A16 bytes0 to (2 128 – 1)
SecondaryVersionCounter0x0B16 bytes0 to (2 128 – 1)
vCardSelector0x0C8 bytes64 bits mask
DatabaseIdentifier0x0D16 bytes0 to (2 128 – 1)
vCardSelectorOperator0x0E1 byte0x00 = OR 0x01 = AND
ResetNewMissedCalls0x0F1 byte0x01 = Reset
PbapSupportedFeatures0x104 bytesBit 0 = Download Bit 1 = Browsing Bit 2 = Database Identifier Bit 3 = Folder Version Counters Bit 4 = vCard Selecting Bit 5 = Enhanced Missed Calls Bit 6 = X-BT-UCI vCard Property Bit 7 = X-BT-UID vCard Property Bit 8 = Contact Referencing Bit 9 = Default Contact Image Format Bit 10 ~ 31 Reserved 1
  • PropertySelector:用于指示请求的vCard object中应该包含的属性,PSE根据这些属性来组织恢复的Body/End of Body Header中包含的数据,PCE只能使用此Header接收所请求的vCard所需要内容,PSE不得回复任何其他性能数据,除非PCE有其他要求;

        PropertySelector的值是由一个64位的数据组成,所以每一位都代表了一种属性,如果PCE请求的电话簿需要包含对应的数据,就将该数据对应在PropertySelector的二进制位设置为true(1)。具体每一位的含义见下图:

流程图

        这里有一个需要注意的,PCE和PSE的服务连接不是一直保持的,只有在同步Phone Book的时候,服务保持连接,同步完成之后,就会断开;

2.3.2 Browsing

FeatureFunctionSupport by the PCESupport by the PSE
Phone Book Browsing FeatureSetPhonebookMM
PullvCardListingMM
PullvCardEntryMM

  • SetPhonebook:选择感兴趣的Phone Object
  • PullvCardListing:client使用该Function获取感兴趣的Phone Object 列表
  • PullvCardEntry:client使用该Function获取感兴趣的Phone Object(单个)
2.3.2.1 SetPhonebook Data Format

请求格式:

Field / HeaderNameValueStatus
FieldOpcodeSETPATH (0x05)M
FieldPacket LengthVariesM
FieldFlagsUp / Down / RootM
FieldConstantReserved (0)M
HeaderConnection IDVariesM
HeaderNameName of the folderO

响应格式:

Field / HeaderNameValueStatus
FieldResponse Code0xA0 or Error CodeM
FieldPacket Length3M
2.3.2.2 PullvCardListing Data Format

请求格式:

Field / HeaderNameValueStatus
FieldOpcodeGET(0x03 or 0x83)M
FieldPacket LengthVariesM
HeaderConnection IDVariesM
HeaderSingle Response Mode0x01C1
HeaderSingle Response Mode Param0x01C2
HeaderNameName of the folderM
HeaderType"x-bt/vcard-listing"M
HeaderApplication Parameters
- OrderVariesO
- SearchValueVariesO
- SearchPropertyVariesC3
- MaxListCountVariesO
- ListStartOffsetVariesO
- ResetNewMissedCallsVariesC4
- vCardSelectorVariesC5
- vCardSelectorOperatorVariesC6

响应格式:

Field / HeaderNameValueStatus
FieldResponse Code0x09 or 0xA0 or Error CodeM
FieldPacket LengthVariesM
HeaderSingle Response Mode0x01C1
HeaderSingle Response Mode Param0x01C2
HeaderApplication Parameters
- PhonebookSizeVariesC3
- NewMissedCallsVariesC4
- PrimaryFolderVersionVariesC5
- SecondaryFolderVersionVariesC5
- DatabaseIdentifierVariesC6
HeaderBody/End of BodyvCard object(s)C7
2.3.2.3 PullvCardEntry Data Format

请求格式:

Field / HeaderNameValueStatus
FieldOpcodeGET(0x03 or 0x83)M
FieldPacket LengthVariesM
HeaderConnection IDVariesM
HeaderSingle Response Mode0x01C1
HeaderSingle Response Mode Param0x01C2
HeaderNameObject name (*.vcf) or X-BT-UID (X-BT-UID)M
HeaderType"x-bt/vcard"M
HeaderApplication Parameters
- PropertySelectorVariesO
- FormatVariesO

响应格式:

Field / HeaderNameValueStatus
FieldResponse Code0x09 or 0xA0 or Error CodeM
FieldPacket LengthVariesM
HeaderSingle Response Mode0x01C1
HeaderSingle Response Mode Param0x01C2
HeaderApplication Parameters
- DatabaseIdentifierVariesC3
HeaderBody/End of BodyvCard objectC4

 流程图

2.3.3 vcf文件

        Download和Browsing功能描述完成之后,我们需要知道,所有的Phone Object信息都是来自于PSE端,而在PSE端保存Phone Object的方式或者是路径可能有很多;

  • 存储设备

    • 手机:telecom/xxx.vcf
    • SIM卡:SIM1/telecom/xxx.vcf

        无论是哪种存储方式,其对应都有相同的数据存储类型,例如通讯录、通讯记录,而在通讯记录中,又可以分为:来电、去电、未接来电、所有通讯记录。同时电话簿还提供了两个功能:快速拨号和收藏联系人;

上述描述的分类方式,对应了PSE端存储文件格式;

Phone ObjectASFormatDesc
Phone Book Objectpbpb.vcf通讯录
Incoming Calls History Objectichich.vcf来电通话记录
Outgoing Calls History Objectochoch.vcf去电通话记录
Missed Calls History Objectmchmch.vcf未接来电通话记录
Commbined Calls History Objectcchcch.vcf所有的通话记录
Speed-Dial Objectspdspd.vcf快速拨号
Favorite Contacts Objectfavfav.vcf收藏通讯录

这篇关于【Android】【Bluetooth Stack】蓝牙电话本协议分析(超详细)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/829129

相关文章

将Mybatis升级为Mybatis-Plus的详细过程

《将Mybatis升级为Mybatis-Plus的详细过程》本文详细介绍了在若依管理系统(v3.8.8)中将MyBatis升级为MyBatis-Plus的过程,旨在提升开发效率,通过本文,开发者可实现... 目录说明流程增加依赖修改配置文件注释掉MyBATisConfig里面的Bean代码生成使用IDEA生

Go标准库常见错误分析和解决办法

《Go标准库常见错误分析和解决办法》Go语言的标准库为开发者提供了丰富且高效的工具,涵盖了从网络编程到文件操作等各个方面,然而,标准库虽好,使用不当却可能适得其反,正所谓工欲善其事,必先利其器,本文将... 目录1. 使用了错误的time.Duration2. time.After导致的内存泄漏3. jsO

Linux系统配置NAT网络模式的详细步骤(附图文)

《Linux系统配置NAT网络模式的详细步骤(附图文)》本文详细指导如何在VMware环境下配置NAT网络模式,包括设置主机和虚拟机的IP地址、网关,以及针对Linux和Windows系统的具体步骤,... 目录一、配置NAT网络模式二、设置虚拟机交换机网关2.1 打开虚拟机2.2 管理员授权2.3 设置子

Linux系统中卸载与安装JDK的详细教程

《Linux系统中卸载与安装JDK的详细教程》本文详细介绍了如何在Linux系统中通过Xshell和Xftp工具连接与传输文件,然后进行JDK的安装与卸载,安装步骤包括连接Linux、传输JDK安装包... 目录1、卸载1.1 linux删除自带的JDK1.2 Linux上卸载自己安装的JDK2、安装2.1

Android中Dialog的使用详解

《Android中Dialog的使用详解》Dialog(对话框)是Android中常用的UI组件,用于临时显示重要信息或获取用户输入,本文给大家介绍Android中Dialog的使用,感兴趣的朋友一起... 目录android中Dialog的使用详解1. 基本Dialog类型1.1 AlertDialog(

Java使用Curator进行ZooKeeper操作的详细教程

《Java使用Curator进行ZooKeeper操作的详细教程》ApacheCurator是一个基于ZooKeeper的Java客户端库,它极大地简化了使用ZooKeeper的开发工作,在分布式系统... 目录1、简述2、核心功能2.1 CuratorFramework2.2 Recipes3、示例实践3

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

找不到Anaconda prompt终端的原因分析及解决方案

《找不到Anacondaprompt终端的原因分析及解决方案》因为anaconda还没有初始化,在安装anaconda的过程中,有一行是否要添加anaconda到菜单目录中,由于没有勾选,导致没有菜... 目录问题原因问http://www.chinasem.cn题解决安装了 Anaconda 却找不到 An

Spring定时任务只执行一次的原因分析与解决方案

《Spring定时任务只执行一次的原因分析与解决方案》在使用Spring的@Scheduled定时任务时,你是否遇到过任务只执行一次,后续不再触发的情况?这种情况可能由多种原因导致,如未启用调度、线程... 目录1. 问题背景2. Spring定时任务的基本用法3. 为什么定时任务只执行一次?3.1 未启用

Android Kotlin 高阶函数详解及其在协程中的应用小结

《AndroidKotlin高阶函数详解及其在协程中的应用小结》高阶函数是Kotlin中的一个重要特性,它能够将函数作为一等公民(First-ClassCitizen),使得代码更加简洁、灵活和可... 目录1. 引言2. 什么是高阶函数?3. 高阶函数的基础用法3.1 传递函数作为参数3.2 Lambda