草庐IT

简析开源鸿蒙蓝牙能力

王石 2023-03-28 原文

想了解更多关于开源的内容,请访问:

​51CTO 开源基础软件社区​

​https://ost.51cto.com​

蓝牙功能是无线短距的重要能力,在工作、生活中有很多蓝牙设备,比如车载蓝牙设备,蓝牙耳机,蓝牙键盘。1994年由电信商爱立信发展出这个技术,最初蓝牙的设计是系统创建出一个RS-232数据线的无线通信技术替代版,能够链接多个设备并克服同步问题。目前蓝牙技术由蓝牙技术联盟(SIG Special Interest Group)来负责维护其技术标准,IEEE曾经将蓝牙技术标准化为IEEE 802.15.1,但是这个标准已经不再继续使用。接下来我们就深入分析下开源鸿蒙的蓝牙结构和各层作用及工作内容。

概述

在开源鸿蒙的源码里和系统功能相关的部分大多都放在foundation这个文件夹里,而通讯相关的部分则是在communication这个文件夹内。我们本篇要分析的蓝牙功能就是在foundation/communication/bluetooth路径下,具体目录结构如下:

.
├── bundle.json
├── frameworks //框架层
├── inner
└── js
├── hisysevent.yaml
├── interfaces //接口层
├── inner_api
└── kits
├── LICENSE
├── README.md
├── README_zh.md
├── sa_profile //系统能力配置
├── 1130.xml
└── BUILD.gn
├── services //服务层
├── bluetooth
└── bluetooth_lite
└── test 测试代码
├── example
├── fuzztest
├── moduletest
└── unittest

架构简析

  • 根据目录结构和内部文件及编译框架总结架构如下:
  • 接口层:对外提供js接口,采用d.ts定义,蓝牙文件夹内位置interfaces/kits/js具体存放路径在interface/sdk-js/api/@ohos.bluetooth.d.ts;对内提供c接口,可以供softbus,netmanager,audioframework三个子系统调用,蓝牙文件夹内位置interfaces/inner_api/include。
  • 框架层:分两个子层,NAPI框架实现层,蓝牙文件夹内位置foundation/communication/bluetooth/frameworks/js/napi,实现所有js层代码接口适配并调用innerapi的实现;蓝牙接口实现层,使用IPC架构同蓝牙服务层通信,将从NAPI收到的命令,或者别的子系统收到的命令经由IPC架构发送给服务层,并注册服务层observer,收听由服务层上报的事件。
  • 服务层:分三个子层,系统能力层,蓝牙文件夹内位置foundation/communication/bluetooth/services/bluetooth/server,实现蓝牙系统能力服务,接收从框架层经由IPC发送下来的命令,并收听蓝牙service传上来的消息并回传给蓝牙框架层;蓝牙服务层,蓝牙文件夹内位置foundation/communication/bluetooth/services/bluetooth/service,实现不同蓝牙能力的adapter(包括classic_adapter和ble_adapter),通过调用蓝牙协议栈实现蓝牙业务的逻辑能力;蓝牙协议栈,蓝牙文件夹内位置foundation/communication/bluetooth/services/bluetooth/stack,实现蓝牙协议包解析,打包以及蓝牙协议流程的处理,实现蓝牙host,蓝牙profile以及蓝牙链路和hci接口,最后通过使用蓝牙硬件驱动收发蓝牙原始数据。
  • 硬件驱动:预编译包,通过dlopen提供hci的底层接口,注册回调以及发送数据。

功能简析

  1. NAPI功能
  • HAP应用层通过@ohos.bluetooth.d.ts引用调用蓝牙接口,蓝牙接口包括bluetooth接口,BLE接口,profile接口。bluetooth接口包括经典蓝牙接口,BLE接口接口,和profile接口;目前支持的profile有A2dpSourceProfile,HandsFreeAudioGatewayProfile,HidHostProfile和PanProfile。
  • NAPI框架接口均通过DECLARE_NAPI_FUNCTION,js的回调也通过此接口经由传入回调函数指针实现。
  • NAPI层通过调用蓝牙框架层的BluetoothHost实现功能,通过将g_bluetoothHostObserver注册到框架层内部实现来获得消息回调;同时NAPI层提供g_Observer存储js层的回调函数,在收到框架层的回调后在g_Observer里查找对应的回调函数来实现应用通知;此对象因为是map结构体,所以一个应用只能注册一个并在应用内部自己处理。
  1. 蓝牙框架功能
  • 蓝牙框架提供cadapter接口和蓝牙框架具体实现(蓝牙NAPI层调用的BluetoothHost即在具体实现层)。
  • cadapter实现c接口,通过BluetoothHost实现功能,其他外部模块均通过cadapter提供的c接口调用蓝牙能力。
  • 蓝牙框架内部实现层提供接口一一对应cadapter和NAPI接口。BluettoothHost内部通过SystemAbilityManagerClient::GetInstance().GetSystemAbilityManager()接口得到samgr,然后通过BLUETOOTH_HOST_SYS_ABILITY_ID标识获取蓝牙服务对象,然后通过蓝牙服务对象通过IPC接口调用蓝牙服务功能。
  • 蓝牙框架内部提供RegisterObserver方式,允许其他层应用(比如软总线层通过调用cadapter接口GapRegisterCallbacks收听蓝牙的回调事件)收听蓝牙回调事件。
  • 蓝牙框架因为是动态库实现,所以如果有多处依赖则会生成多份实例,如NAPI和软总线都依赖蓝牙框架库,所以在系统里会有两份实例同时若产生蓝牙通知也会有两个进程同时收到消息。
  1. 蓝牙系统能力
  • 蓝牙系统能力通过继承SystemAbility,BluetoothHostStub两个类实现蓝牙服务能力并实现IPC进行通信;IPC通信部分的代码在蓝牙文件夹foundation/communication/bluetooth/services/bluetooth/ipc内,提供proxy和stub实现,即btipc_static.a静态库。
  • 蓝牙系统能力可视作蓝牙服务进程,提供包括蓝牙状态机,蓝牙adapter(classic,ble),蓝牙profile服务实例。
  • 蓝牙系统配置在foundation/communication/bluetooth/services/bluetooth/etc/init目录内,此部分内容会在制作镜像包时打包到系统文件内去;在蓝牙服务启动的时候会读取蓝牙配置(AdapterConfig::GetInstance()->Load(),ProfileConfig::GetInstance()->Load())并确定蓝牙的snooplog(协议栈输出文件)。
  • 蓝牙系统能力内部使用Dispatcher模型执行任务事务,在Dispatcher对象内部它通过线程加任务队列的方式处理多条任务命令,目前实现方法为但线程依次处理队列内的事务。
  • 蓝牙系统能力层通过permission_utils提供应用权限保护能力,服务调用功能时首先调用PermissionUtils::VerifyXXXBluetoothPermission(XXX包括,Use,Discover,Manage,Location),然后调用AuthCenter对应功能,最后调用PermissionHelper对应接口确认权限;在AuthCenter里内部定义了g_permissionAlwaysGrant全局变量,可以方便调试时拥有所有权限。
  1. 蓝牙协议栈
  • 蓝牙协议栈通过c接口对蓝牙服务层提供接口。
  • 蓝牙协议栈内部他提供任务队列方式处理蓝牙命令(GapRunTaskBlockProcess, GapRunTaskUnBlockProcess),分别对于蓝牙协议栈同步接口和异步接口。
  • 蓝牙协议栈内部也有层级,对上的是gap,中间是btm,对下的是hci,中间部分和其他蓝牙协议站定义的内容一致,如:att,avctp,avdtp,l2cap,rfcomm,sdp,smp。
  • HCI层它通过g_hdiLib加载蓝牙驱动库libbluetooth_hdi_adapter.so,并通过加载蓝牙驱动接口(HdiInit,HdiSendHciPacket,HdiClose)实现蓝牙数据发送,蓝牙数据接收(g_hdiCallacks)。

总结

  1. 蓝牙服务的结构也是所有开源鸿蒙的结构能力,大致流程均为:NAPI框架,系统服务框架,IPC框架,Observer框架,原生driver框架(其实可以采用HDF框架)。
  2. 目前开源鸿蒙用的蓝牙协议站不是android,bluez,btstack这三种主流蓝牙协议栈,根据注释可得其参考《BLUETOOTH SPECIFICATION Version 5.0》实现;目前官方最新的协议栈是v5.3,参考链接如下:https://www.bluetooth.com/specifications/specs/core-specification-5-3/。
想了解更多关于开源的内容,请访问:

​51CTO 开源基础软件社区​

​https://ost.51cto.com​

有关简析开源鸿蒙蓝牙能力的更多相关文章

  1. 程序员如何提高代码能力? - 2

    前言作为一名程序员,自己的本质工作就是做程序开发,那么程序开发的时候最直接的体现就是代码,检验一个程序员技术水平的一个核心环节就是开发时候的代码能力。众所周知,程序开发的水平提升是一个循序渐进的过程,每一位程序员都是从“菜鸟”变成“大神”的,所以程序员在程序开发过程中的代码能力也是根据平时开发中的业务实践来积累和提升的。提高代码能力核心要素程序员要想提高自身代码能力,尤其是新晋程序员的代码能力有很大的提升空间的时候,需要针对性的去提高自己的代码能力。提高代码能力其实有几个比较关键的点,只要把握住这些方面,就能很好的、快速的提高自己的一部分代码能力。1、多去阅读开源项目,如有机会可以亲自参与开源

  2. 【鸿蒙应用开发系列】- 获取系统设备信息以及版本API兼容调用方式 - 2

    在应用开发中,有时候我们需要获取系统的设备信息,用于数据上报和行为分析。那在鸿蒙系统中,我们应该怎么去获取设备的系统信息呢,比如说获取手机的系统版本号、手机的制造商、手机型号等数据。1、获取方式这里分为两种情况,一种是设备信息的获取,一种是系统信息的获取。1.1、获取设备信息获取设备信息,鸿蒙的SDK包为我们提供了DeviceInfo类,通过该类的一些静态方法,可以获取设备信息,DeviceInfo类的包路径为:ohos.system.DeviceInfo.具体的方法如下:ModifierandTypeMethodDescriptionstatic StringgetAbiList​()Obt

  3. python - 开源 Twitter 克隆(在 Ruby/Python 中) - 2

    关闭。这个问题不符合StackOverflowguidelines.它目前不接受答案。我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。关闭6年前。Improvethisquestion是否有任何用Ruby或Python编写的生产就绪的开源Twitter克隆?我对功能丰富的实现更感兴趣,而不仅仅是简单的Twitter消息(例如:API、FBconnect、通知等)谢谢!

  4. 千耘农机导航的“星地一体”能力究竟是什么? - 2

    伴随农业机械化和智能化的发展,越来越多的人开始使用农机自动驾驶系统助力耕作,千耘农机导航的“星地一体”能力可有效解决信号受限的问题,实现作业提效。究竟什么是“星地一体”,又是如何解决智能化农机作业的痛点的?下面为大家揭秘。农机效率通常受限于通信网络目前虽然我国通讯网络的人口覆盖率达到99%,但地面移动通讯网络覆盖率仍小于国土面积的40%,而很多农田所在区域恰是山区、戈壁滩等偏远地区。两省交界地也会出现通信信号不稳定的状况;而国内大部分农机自动驾驶系统非常依赖通信网络,当通信网络弱的时候会出现系统掉线的现象,必须得携带小基站才能正常使用,极为繁琐。Q:什么是千耘农机导航“星地一体”能力?A:是星

  5. Gradle 自动化构建开源工具 - 2

    文章目录写在前面1、下载与安装(windows)1.1、idea中配置gradle2、基础知识(Gradle6.9为例)2.1、Gradle脚本语法2.1.1、dependsOn2.1.2、创建动态任务2.1.3、增加任务行为2.1.4、参数2.1.5、Ant任务2.1.6、方法2.1.7、默认任务2.1.6、依赖任务的不同输出3、java项目中使用3.1、在已有项目中构建gradle3.2、在新建项目时构建gradle(idea)3.3、gradle项目目录结构3.4、build.gradle3.4.1、plugins3.4.2、repositories3.4.3、dependencies3

  6. 停车系统源码-基于springboot+uniapp开源项目 - 2

    Iparking停车收费管理系统-可商用介绍Iparking是一款基于springBoot的停车收费管理系统,支持封闭车场和路边车场,支持微信支付宝多种支付渠道,支持多种硬件,涵盖了停车场管理系统的所有基础功能。技术栈Springboot,MybatisPlus,Beetl,Mysql,Redis,RabbitMQ,UniApp功能云端功能序号模块功能描述1系统管理菜单管理配置系统菜单2系统管理组织管理管理组织机构3系统管理角色管理配置系统角色,包含数据权限和功能权限配置4系统管理用户管理管理后台用户5系统管理租户管理多租户管理6系统管理公众号配置租户公众号配置7系统管理操作日志审计日志8系统

  7. ruby-on-rails - 有没有很好的引用(开源)Rails NoSQL应用程序? - 2

    我有兴趣了解使用nosql将如何影响rails应用程序的架构/设计/代码。有人知道使用nosql持久性的开源rails应用程序的一个好例子吗?谢谢 最佳答案 看看这些项目:卡桑德拉用法atDigg。卡桑德拉用法atTwitter。Friendly用法atFetLife(nsfw)。最后,MyNoSQL是一个提供nosql相关信息的好网站。 关于ruby-on-rails-有没有很好的引用(开源)RailsNoSQL应用程序?,我们在StackOverflow上找到一个类似的问题:

  8. ruby-on-rails - 请向我推荐一些需要编写文档/测试的 rails/ruby 开源代码 - 2

    关闭。这个问题不符合StackOverflowguidelines。它目前不接受答案。要求我们推荐或查找书籍、工具、软件库、教程或其他场外资源的问题对于StackOverflow来说是偏离主题的,因为它们往往会吸引自以为是的答案和垃圾邮件。相反,请描述问题以及迄今为止为解决该问题所做的工作。关闭8年前。Improvethisquestion在过去的4个月左右的时间里,我一直在使用ruby​​onrails,我真的很喜欢开源的整个概念。我知道它不是ruby​​/rails独有的,但来自Windows编程,这是我第一次真正接触它。我想尽我所能“回馈”,但我觉得我无法贡献任何有值(value

  9. ruby-on-rails - Rails CanCan gem 重构能力类 - 2

    我的Rails应用程序中有大约13个模型,我对所有模型都使用了ability。我的能力等级变得很大。我对不同的CRUD操作有不同的能力条件,这使得它很难管理。有人可以指导我如何重构这个……吗?比如,使用模块或类来使我的能力类看起来整洁。 最佳答案 Simpleescenario:如果您可以将权限拆分为几个互斥的集合,那么您应该查看来自CanCan创建者@ryanb的这个提案,他将权限拆分为几个不同的类别,然后覆盖ApplicationController中的current_ability方法Howyoucanbreakuplarge

  10. ruby-on-rails - 我如何找到可以贡献的开源项目(Ruby、Rails) - 2

    关闭。这个问题不符合StackOverflowguidelines.它目前不接受答案。我们不允许提问寻求书籍、工具、软件库等的推荐。您可以编辑问题,以便用事实和引用来回答。关闭7年前。Improvethisquestion我是一名RubyonRails开发人员,手头有一些时间。我想利用这段时间通过为开源项目做贡献来回馈和学习。我不是一流的程序员,想从小做起。在哪里可以找到Ruby或Rails中的小型开源项目?我该如何贡献?亚历克斯

随机推荐