码迷,mamicode.com
首页 > 其他好文 > 详细

TI BLE协议栈软件框架分析

时间:2019-05-28 19:51:22      阅读:112      评论:0      收藏:0      [点我收藏+]

标签:span   适应   col   创建   好的   config   递增   怎样   pos   

看源代码的时候,一般都是从整个代码的入口处開始,TI? BLE 协议栈源代码也不例外。它的入口main()函数就是整个程序的入口,由系统上电时自己主动调用。

?

它主要做了以下几件事情:

(一)底层硬件初始化配置

(二)创建任务并初始化任务配置

(三)检測并运行有效的任务事件

?

Main() 函数源代码例如以下:

技术图片

一:底层硬件初始化设置

75行。设置系统时钟。使能内存缓冲功能。

78行。关中断,刚启动时。系统运行不稳定,通常会首先关中断。

81行。硬件相关的I/O 口配置。

84行。初始化mcu 内部的flash。

92行,开中断,当系统运行到这里的时候。状态已经非常稳定,能够将中断打开。

95行,I/O功能配置,及设置按键回调函数指针。

98行。配置省电模式。

?

二:创建任务并初始化任务配置

89行,最基本的功能就是给创建全部的任务。

?

三:检測并运行有效的任务事件

?

102行,此函数是整个程序运行的核心。一旦进入,就循环运行全部的任务,永远不会结束。

?

?

?

以下分别列出这3个部分的主要源代码,并以按键KEY的配置为线索进行分析:

===================================================================

一:底层硬件初始化设置

技术图片

a.????????HalDriverInit() 主要是硬件抽象层初始化。配置PIN脚的工作模式。

比方ADC,UART,KEY。LCD等。

147行。HalKeyInit()是按键I/O配置,以下以此举例说明:

技术图片

211-223行,将对应的PIN配置为GPIO,输入模式。

226行。初始化按键回调函数指针pHalKeyProcessFunction为NULL。

此回调函数在程序中的作用是,当driver层检測到按键中断后,仅仅须要调用回调函数指针就可以,至于函数运行什么功能则全然由用户层自己决定,这种做的优点是将用户层与driver层分离。提高代码的模块化及可操作特性。

?

b.????????InitBoard( OB_READY ),配置KEY? GPIO的中断功能

技术图片

126行。HalKeyConfig( OnboardKeyIntEnable, OnBoard_KeyCallback)此函数中配置中断使能,设置按键回调函数指针pHalKeyProcessFunction为OnBoard_KeyCallback。

OnBoard_KeyCallback()里面会继续调用函数OnBoard_SendKeys(),其功能是发送按键message给对应的任务。

?

硬件配置好后。仅仅能说明硬件具备完毕对应功能的条件,可是怎样让它的功能实现。那就须要创建对应的应用程序来让硬件工作起来,这个程序就叫task。task是怎样创建的呢?

?

以下接着分析,怎样创建task。

?

二:创建任务并初始化任务配置

?

前面讲过主函数main()89行osal_init_system(),主要功能是初始化系统设置,当中最重要的一个功能就是创建task,以下是创建task的源代码

?技术图片

122行,申请tasksEvents内存空间

123行。清零tasksEvents内存空间

126行,链路层task初始化

129行,硬件抽象层task初始化

132行,主机控制接口层task初始化

141行,逻辑链路控制及自适应协议层task初始化

144-156行,通用属性配置文件层task初始化

150行。安全管理层task初始化

159行,客户应用层task初始化

?

每一个task 初始化时都会分配一个taskID,并且是从0 递增。

以Hal_Init(taskID++ )为例,从上面代码能够看出来。硬件抽象层的taskID值是1

?技术图片

92行。将task_id形參赋值给Hal_TaskID,故初始化后Hal_TaskID等于1

因为Hal_TaskID是定义为一个全局变量。因此。整个程序中,仅仅要是与Hal_TaskID有关系的事情,都会交给Hal 层的task处理。

?

相同的道理。SimpleBLECentral_Init(taskID++)中会定义一个simpleBLETaskId,那么全部与simpleBLETaskId有关系的事情,也都会交给应用层任务函数处理。

?

?

那各层的任务函数是在哪里定义的呢?

在OSAL_simpleBLECentral.c文件里,定义了一个函数指针数组例如以下:

技术图片

87行,Hal_ProcessEvent即为Hal层的任务处理函数指针,它是数组的第2个元素tasksArr[1]。也就是说假设程序中要调用Hal层任务,直接写语句“tasksArr[1]();”

函数Hal_ProcessEvent()就会被运行了。

?

Task尽管被创建好了。可是task是要运行我们给它规定的功能的,那它是在哪里运行的呢?

接下来具体分析task 的运行的问题。

?

?

?

?

三:检測并运行有效的任务事件

?

系统中task 的运行。是由事件(evnet)来驱动的,程序会循环检測全部的task。假设发现某个task有新的event未被处理。那么这个task就会被调用。

?

osal_start_system()是整个程序的核心。里面是一个for死循环,不停调用函数osal_run_system(),它的实际功能就是不停检測是否有event产生。假设有event。就运行对应的task。请看源代码:

技术图片

1105行,定时器查询函数,它会检查全部的定时器。假设某个定时时间到达。就将对应的event 加入到tasksEvents[task_id]。这里task_id的值是加入定时器时设置好的,具体请看osal_set_event()函数代码。

1110-1115行,检查全部任务。是否有须要运行的event发生,并记录这个event的索引idx。

1117行。taskCnt 是系统加入的任务个数,也就是tasksArr[]数组中元素的个数。

1123行。保存当前任务的将要运行的全部event。

1124行,清除当前任务的全部event。

1127行,保存当前任务的idx,供系统自己使用。

1128行,依据当前任务索引idx。在指针数组tasksArr[]中寻址当前任务的函数指针。调用当前任务函数,处理当中一个event。处理完毕后,返回还未处理的event。

1132行,将未处理的event恢复给当前任务事件变量保存。等待下一次再处理。直至处理问全部的event。

?

?

?

总结:

?

本节仅仅是解说了协议栈的主体框架,协议栈仅仅是一个基础平台,在不同的方案中,就有不同的应用功能,对应的就必须为应用加入不同的task来实现实际的功能。

?

通过本节解说,加入一个task的基本过程例如以下:

1.????????在HAL层配置任务要用到的I/O 的属性(假设不涉及I/O操作,则可省略此步骤)

2.????????在数组tasksArr[]中加入任务(task)处理函数

3.????????在函数osalInitTasks()中初始化任务task

?

依据上面的步骤我们能够非常easy创建一个task。

可是,假设程序中没有产生task的事件(event),task永远都不会运行。

?

这就有一个新的问题:event在什么情况下产生?它又是怎样产生?

我们将在【事件和消息工作机制】一节中具体分解。

?

?

?

?

TI BLE协议栈软件框架分析

标签:span   适应   col   创建   好的   config   递增   怎样   pos   

原文地址:https://www.cnblogs.com/mqxnongmin/p/10939767.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
分享档案
周排行
mamicode.com排行更多图片
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!