五、BLE(下)
1.1 GATT server Service
通过走读代码, GATT Server作为一个GATT service,我是没有发现其发挥了多大功能,其负责处理的消息GATT_SERVER_SERVICE_CHANGED_INDICATION_CFM,GATT_SERVER_READ_CLIENT_CONFIG_IND,GATT_SERVER_WRITE_CLIENT_CONFIG_IND都没有定义对应的handler。
1.2 GATT Service(Battery Service)
GATT Battery Servce是ADK4.0 sink例程中默认添加的一个比较简单和常用的服务,该服务只有一个设备电量(batteryLevel)的特征值。纵观GATT Battery Servce模块的接口可以大致看出该模块完成的主要功能有:
1.2.1 服务初始化
GattBatteryServerInit(GBASS *battery_server, Task app_task, init_params, start_h, end_h);
该接口主要用来初始化指定(#battery_server) GATT Battery Service实例,如lib_task和app_task。初始化实例之后,通过GattManagerRegisterServer()将其添加到GATT MGR模块。
1.2.2 回调函数
前面也提到过,这里涉及到两个回调函数,外部钩子函数主要用来接收application layer的数据,而内部钩子函数,则用来接收上级管理模块的信息。
内部钩子函数与外部钩子函数对应的函数名称以及处理的信号如下:
batteryServerMsgHandler(); /*内部钩子函数*/
--GATT_MANAGER_SERVER_ACCESS_IND
-- HANDLE_BATTERY_SERVICE
-- HANDLE_BATTERY_LEVEL à GATT_BATTERY_SERVER_READ_LEVEL_IND
-- HANDLE_BATTERY_LEVEL_CLIENT_CONFIG à GATT_BATTERY_SERVER_READ_CLIENT_CONFIG_IND / GATT_BATTERY_SERVER_WRITE_CLIENT_CONFIG_IND
-- HANDLE_BATTERY_LEVEL_PRESENTATIONà GATT_BATTERY_SERVER_READ_PRESENTATION_IND
-- GATT_MANAGER_REMOTE_CLIENT_NOTIFICATION_CFM
sinkGattBatteryServerMsgHandler(); /*外部钩子函数*/
-- GATT_BATTERY_SERVER_READ_LEVEL_IND
-- GATT_BATTERY_SERVER_READ_CLIENT_CONFIG_IND
-- GATT_BATTERY_SERVER_WRITE_CLIENT_CONFIG_IND
-- GATT_BATTERY_SERVER_READ_PRESENTATION_IND
从上面的分析可以看出,内部钩子函数在接收到的GATT请求,如果能够自己处理,则直接进行处理和应答,如HANDLE_BATTERY_SERVICE,其他大部分情况还需要application Layer的介入,因此将其映射为外部钩子函数对应的信号后转发给application Layer(通过app_task),application Layer通过直接调用外部钩子函数,进行请求转发,在外部钩子函数里面,实现大部分的GATT请求。
1.2.3 服务参数配置,特征值获取接口
实现内部钩子函数和外部钩子函数对GATT请求的处理接口。在Battery Service中主要有:
-- HANDLE_BATTERY_SERVICEà查询Battery Service,主要用于获取Battery Service句柄
-- HANDLE_BATTERY_LEVEL à获取电量特征值
--HANDLE_BATTERY_LEVEL_CLIENT_CONFIG à 获取电量特征值的客户端配置参数
-- HANDLE_BATTERY_LEVEL_PRESENTATIONà获取电量特征值的表示方式
1.3 初始化
The local database and the GATT Server libraries to manage the local GATT database must be registered. This section describes the procedure required to do this, a few points to note are:
§ The local GATT Database must be registered first.
§ Each required GATT Server library must be individually registered:
§ Only GATT Server Libraries that manage a section of the registered GATT database should be registered.
§ The handle range for each GATT Server library must not overlap.
§ When the local GATT database and all the required GATT Server libraries have been registered, the device can be registered with GATT (and the ATT firmware layer):
§ If a local GATT database is registered, but no GATT Server libraries have been registered, this request fails.

1.4 GATT Request Handle
当一个GATT客户设备与一个GATT服务设备建立连接后,就可以开始GATT操作。GATT MGR模块对每个到来的GATT请求,通过查询已注册的服务列表,经过一定的匹配算法寻找到合适GATT Service后将该请求分发给它。GATT Service模块对于到来的请求通常有两种不同的处理方式:
1.少数情况,GATT Service模块有能力可以立即对到来的GATT请求作出回应,例如,GATT请求时获取软件版本,确认某个服务等,这种情况无需Application Layer介入。
2.大多数情况下,GATT Service模块将到来的请求传递给Application Layer, Application Layer通过该模块的外部钩子函数API处理该请求,这也是为什么通常GATT Service模块需要提供外部钩子函数的原因(The GATT Server library should provide a downstream API for the application to respond to the request)。
下图分别展示上述两种不同的GATT请求的处理方式。上半部分展示了读取GATT Battery Service的电池电量的操作,在该GATT请求中,GATT Battery Service模块将该请求传递个Application Layer,后者通过GATT Battery Service模块提供的外部回调函数(downstream API) sinkGattBatteryServerMsgHandler()来处理请求。下半部分展示了一个简单的GATT请求,该请求无需Application Layer处理。

获取BatteryService的电量特征值的函数调用过程如下图:
gattMessageHandler(); (ATT_ACCESS_IND)->
gattHandleAttAccessInd((ATT_ACCESS_IND_T *) message);
MessageSend(gattGetCidMappedTask(ind->cid), GATT_ACCESS_IND, message);
gattManagerMessageHandler(); /* gatt_manager_data->gatt_manager_task.handler */
gattMessageHandler(); (GATT_ACCESS_IND)->
gattManagerServerAccessInd();
gattManagerDataResolveServerHandle(&discover);
serverAccessInd(ind, discover.task, discover.adjusted);
MessageSend(task, GATT_MANAGER_SERVER_ACCESS_IND, message);
batteryServerMsgHandler();->GATT_MANAGER_SERVER_ACCESS_IND
handleBatteryAccess()->HANDLE_BATTERY_LEVEL
batteryLevelAccess();->GATT_BATTERY_SERVER_READ_LEVEL_IND
MessageSend(battery_server->app_task, , message);-à
sinkBleMsgHandler()->GATT_BATTERY_SERVER_READ_LEVEL_IND
sinkGattBatteryServerMsgHandler()->
handleReadBatteryLevel();->
getBatteryLevelAsPercentage();
sendBatteryLevelAccessRsp(battery_server, cid, battery_level);->
sendBatteryLevelAccessRsp(); (HANDLE_BATTERY_LEVEL)
sendBatteryAccessRsp(&battery_server->lib_task, , , &battery_level);
GattAccessResponse(); (GATT_INTERNAL_ACCESS_RES)
MessageSend(&theGatt->task, GATT_INTERNAL_ACCESS_RES);
gattMessageHandler()->
gattHandleInternalAccessRes()->
VmSendAttPrim(prim);
1.5 Service Discovery
关于这一节,ADK提供的使用手册表诉的比较详细,这里就直接摘抄过来。
When a GATT Server device has been successfully connected, the Audio Sink application starts discovering which primary services are supported by the connected device. The procedure for discovering the GATT Primary Services of a remote GATT Server is:
§ The Audio Sink application discovers all primary services defined on the remote GATT Server:
§ Discovered primary services that are supported by the Audio Sink application is temporarily stored in a list.
§ Discovered primary services that are not supported by the Audio Sink application are ignored.
§ When all primary services have been discovered:
§ For each supported primary service that has been discovered, the Audio Sink application initialises the GATT client library used to manage the service. Library initialisation is done sequentially not simultaneously.
§ If no supported primary services have been discovered, the device is disconnected.

Description
§ The Audio Sink application discovers all primary services on the remote GATT Server.
§ The Audio Sink application discovers the remote GATT Server supports Service One and temporarily stores that information.
§ The Audio Sink application ignores the discovered primary service (Service Two) that is not supported.
§ When all primary services have been discovered, the Audio Sink application initialises the GATT Client library to manage Service One.
五、BLE(下)的更多相关文章
- 带你实现开发者头条APP(五)--RecyclerView下拉刷新上拉加载
title: 带你实现开发者头条APP(五)--RecyclerView下拉刷新上拉加载 tags: -RecyclerView,下拉刷新,上拉加载更多 grammar_cjkRuby: true - ...
- Fiddler (五) Mac下使用Fiddler
Fiddler是用C#开发的. 所以Fiddler不能在Mac系统中运行. 没办法直接用Fiddler来截获MAC系统中的HTTP/HTTPS, Mac 用户怎么办呢? Fiddler可 ...
- python学习笔记五 模块下(基础篇)
shevle 模块 扩展pickle模块... 1.潜在的陷进 >>> import shelve>>> s = shelve.open("nb" ...
- Python 第五篇(下):系统标准模块(shutil、logging、shelve、configparser、subprocess、xml、yaml、自定义模块)
目录: shutil logging模块 shelve configparser subprocess xml处理 yaml处理 自定义模块 一,系统标准模块: 1.shutil:是一种高层次的文件操 ...
- Linux学习之CentOS(五)--CentOS下VMware-Tools安装
已经进入到了Linux学习之CentOS的第六篇随笔了,所以这里就介绍一下VMware-Tools的安装. VMware-Tools的安装 VMware-Tools 主要的功能就是让用户在虚拟机和真实 ...
- 开源框架.netCore DncZeus学习(五)下拉树的实现
千里之行,始于足下,先从一个小功能研究起,在菜单管理页面有一个下拉树,先研究下它怎么实现的 1.先找到menu.vue页面 惯性思维先搜索请选择三个字,原来是动态生成的 再向上找DropDown组件, ...
- zeromq学习记录(五)vc下多线程
/************************************************************** 技术博客 http://www.cnblogs.com/itdef/ ...
- (十五)linux下gdb调试
一.gdb常用命令: 命令 描述 backtrace(或bt) 查看各级函数调用及参数 finish 连续运行到当前函数返回为止,然后停下来等待命令 frame(或f) 帧编号 选择栈帧 info(或 ...
- 《Javascript高级程序设计》阅读记录(四):第五章 下
这个系列,我会把阅读<Javascript高级程序设计>之后,感觉讲的比较深入,而且实际使用价值较大的内容记录下来,并且注释上我的一些想法.做这个一方面是提升了我的阅读效果以及方便我以后阅 ...
- SDUT-3332&3333_数据结构实验之栈与队列五:下一较大值
数据结构实验之栈与队列六:下一较大值 Time Limit: 150 ms Memory Limit: 8000 KiB Problem Description 对于包含n(1<=n<=1 ...
随机推荐
- 基于Selenium的自动化测试 C#版(1)
引子 我一直在思考,作为一个架构师,如何简化程序员的工作,减轻运维的压力,减低测试的要求.然后做了很多很多的尝试.最开始的公司培训文档,一键发布工具,Nuget版本管理,VS项目模板,SOA统一服务提 ...
- 即时通信系统中如何实现:聊天消息加密,让通信更安全? 【低调赠送:QQ高仿版GG 4.5 最新源码】
加密重要的通信消息,是一个常见的需求.在一些政府部门的即时通信软件中(如税务系统),对聊天消息进行加密是非常重要的一个功能,因为谈话中可能会涉及到机密的数据.我在最新的GG 4.5中,增加了对聊天消息 ...
- angular.js 例子
angular.js是一个前端的MVC框架,12年的时候曾近在一个portal平台的项目中使用过. 下面给出一个angular.js的典型例子,涵盖一些基础的知识点,用以复习备忘: <html ...
- 详解c#迭代器
迭代器模式是设计模式中行为模式(behavioral pattern)的一个例子,他是一种简化对象间通讯的模式,也是一种非常容易理解和使用的模式.简单来说,迭代器模式使得你能够获取到序列中的所有元素 ...
- [.net 面向对象程序设计进阶] (5) Lamda表达式(一) 创建委托
[.net 面向对象程序设计进阶] (5) Lamda表达式(一) 创建委托 本节导读: 通过学习Lambda表达式,学会创建委托和表达式目录树,深入了解Lambda的特性,让你的代码变的更加清晰. ...
- [.net 面向对象编程基础] (22) 事件
[.net 面向对象编程基础] (22) 事件 事件(Event)是学习.net面向对象编程很重要的一部分,在学习事件之前,我们实际上已经在很多地方使用了事件,比如控件的click事件等,这些都是. ...
- 页面动态加入<script>标签并执行代码
在页面中动态追加html片段的时候,有时候动态添加的代码会含有<script>标签,比如用了一些模板引擎,或者你的代码有些复杂的时候.然而我们用DOM提供的innerHTML方式来添加代码 ...
- js实现DOM结构
/* 编写一段js脚本生成下面的DOM结构.要求使用标准的DOM方法或属性 <div id='example'> <p class='slogan'>淘,你喜欢</p&g ...
- [ACM_动态规划] 数字三角形(数塔)_递推_记忆化搜索
1.直接用递归函数计算状态转移方程,效率十分低下,可以考虑用递推方法,其实就是“正着推导,逆着计算” #include<iostream> #include<algorithm> ...
- iOS开发——高级技术精选OC篇&Runtime之字典转模型实战
Runtime之字典转模型实战 如果您还不知道什么是runtime,那么请先看看这几篇文章: http://www.cnblogs.com/iCocos/p/4734687.html http://w ...