BES2300YP - 千头万绪的各种入口: 线程,中断,定时器

2023-11-21 04:30

本文主要是介绍BES2300YP - 千头万绪的各种入口: 线程,中断,定时器,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

1.线程

2.app_mod_handler

3.中断

4.CMSIS RTOS 定时器

5.10s定时器

谁在管理10s定时器

a. 周期定时器

b.app_battery_measure.cb

c.电池管理线程

 结论


1.线程

        osThreadCreate这个函数会创建各种线程

        查找osThreadCreate关键字可以找到这些线程入口

2.app_mod_handler

        这实际上是app_thread中基于mod_id的回调接口, 
        app_thread就会在收到message时, 根据message里面的mod_id把消息交给对应的mod_handle来处理.

        每个mod的处理函数是通过app_set_threadhandle来设定的

        查找关键字app_set_threadhandle可以找到这些mode的处理入口

        以下是这些接口的列表,

enum APP_MODUAL_ID_T {APP_MODUAL_KEY = 0,APP_MODUAL_AUDIO,APP_MODUAL_BATTERY,APP_MODUAL_BT,APP_MODUAL_FM,APP_MODUAL_SD,APP_MODUAL_LINEIN,APP_MODUAL_USBHOST,APP_MODUAL_USBDEVICE,APP_MODUAL_WATCHDOG,APP_MODUAL_AUDIO_MANAGE,APP_MODUAL_ANC,APP_MODUAL_SMART_MIC,APP_MODUAL_CMD,APP_MODUAL_TILE,APP_MODUAL_MIC,APP_MODUAL_VOICE_DETECTOR,APP_MODUAL_IR,APP_MODUAL_OHTER,APP_MODUAL_NUM
};

我们注意到此列表中的第一个mod是用来处理按键消息的, 其mod_handle是app_key_handle_process

        在希望处理某种按键消息时,

        - 需要声明一个APP_KEY_HANDLE对象, 其中包含了按键事件和回调函数

        - 调用app_key_handle_registration注册该对象, 

        这样一来, app_thread在收到mod_id为APP_MODUAL_KEY时把消息转发到app_key_handle_process, 后者会根据按键键值和类型调用对应的函数.

        查找关键app_key_handle_registration可以找到按键类的处理函数的入口.

        其他mod的子入口关键字需要到各mod里面细细的找

3.中断

        中断函数的设定是通过IRQ_SetHandler来做到的, 我拿到的这版SDK中对它包了一层变成了宏NVIC_SetVector

       这里列出两个定义来感受下.

NVIC_SetVector(SYS_TICK_IRQn, (uint32_t)&SysTick_Handler);
NVIC_SetVector(TIMER01_IRQn, (uint32_t)hal_timer01_irq_handler);
NVIC_SetVector(GPIOAUX_IRQn, (uint32_t)_gpio_aux_irq_handler[bank]);

有意思的是, 

TIMER01_IRQn的处理函数hal_timer01_irq_handler,

- > 会去调用hal_timer_setup设定的hwtimer_handler

-> 后者会跟踪由hwtimer_alloc添加的一个hwtimer_list

而,按键处理除了直接使用按键相关中断外, 也使用了hwtimer_alloc来处理debounce事件

查找关键字NVIC_SetVector和hwtimer_alloc可以看到这两种入口

4.CMSIS RTOS 定时器

osTimerDef/osTimerCreate/osTimerStart是CMSIS中负责定时器管理的CMSIS-RTOS API

更详细的内容可以参考这篇 https://blog.csdn.net/ichamber/article/details/53240733

5.10s定时器

代码和调用关系很简洁

typedef struct
{uint8_t timer_id;uint8_t timer_en;uint8_t timer_count;uint8_t timer_period;APP_10_SECOND_TIMER_CB_T cb;
}APP_10_SECOND_TIMER_STRUCT;#define INIT_APP_TIMER(_id, _en, _count, _period, _cb) \{ \.timer_id = _id, \.timer_en = _en, \.timer_count = _count, \.timer_period = _period, \.cb = _cb, \}

其中start/stop/set都是用来控制定时器的, 

check是用来轮训检查各定时器并回调的.

谁在管理10s定时器

为啥是app_status_battery_report在维护10s定时器

原来这可以追溯电池状态管理进程, 是app_battery_open在其中穿针引线,

int app_battery_open(void) // 仅保留了关键代码
{// 1. 开启了一个周期定时器if (app_battery_timer == NULL)app_battery_timer = osTimerCreate (osTimer(APP_BATTERY), osTimerPeriodic, NULL);// 2.设置app_battery_measure.cbapp_battery_measure.cb = app_battery_event_process;// 3.创建电池管理线程app_set_threadhandle(APP_MODUAL_BATTERY, app_battery_handle_process);
}

如以上代码中的注释:

a. 周期定时器

通过app_battery_timer_handler来访问ADC调用app_battery_irqhandler,  后者会调用app_battery_measure.cb

static void app_battery_timer_handler(void const *param)
{hal_gpadc_open(HAL_GPADC_CHAN_BATTERY, HAL_GPADC_ATP_ONESHOT, app_battery_irqhandler);
}

b.app_battery_measure.cb

即app_battery_event_process, 会在app_battery_irqhandler中不断调用来发送消息到app_battery_handle_process

static void app_battery_event_process(enum APP_BATTERY_STATUS_T status, APP_BATTERY_MV_T volt)
{uint32_t app_battevt;APP_MESSAGE_BLOCK msg;APP_BATTERY_TRACE(3,"%s %d,%d",__func__, status, volt);msg.mod_id = APP_MODUAL_BATTERY; // 消息被发送到电池线程APP_BATTERY_SET_MESSAGE(app_battevt, status, volt);msg.msg_body.message_id = app_battevt;msg.msg_body.message_ptr = (uint32_t)NULL;app_mailbox_put(&msg);}

c.电池管理线程

会层层深入调用到

 结论

电池状态管理进程依赖于CMSIS RTOS定时器APP_BATTERY实现了10s定时器的管理

欢迎 BES专栏文章

这篇关于BES2300YP - 千头万绪的各种入口: 线程,中断,定时器的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单

《Springboot的ThreadPoolTaskScheduler线程池轻松搞定15分钟不操作自动取消订单》:本文主要介绍Springboot的ThreadPoolTaskScheduler线... 目录ThreadPoolTaskScheduler线程池实现15分钟不操作自动取消订单概要1,创建订单后

C语言线程池的常见实现方式详解

《C语言线程池的常见实现方式详解》本文介绍了如何使用C语言实现一个基本的线程池,线程池的实现包括工作线程、任务队列、任务调度、线程池的初始化、任务添加、销毁等步骤,感兴趣的朋友跟随小编一起看看吧... 目录1. 线程池的基本结构2. 线程池的实现步骤3. 线程池的核心数据结构4. 线程池的详细实现4.1 初

Java子线程无法获取Attributes的解决方法(最新推荐)

《Java子线程无法获取Attributes的解决方法(最新推荐)》在Java多线程编程中,子线程无法直接获取主线程设置的Attributes是一个常见问题,本文探讨了这一问题的原因,并提供了两种解决... 目录一、问题原因二、解决方案1. 直接传递数据2. 使用ThreadLocal(适用于线程独立数据)

第10章 中断和动态时钟显示

第10章 中断和动态时钟显示 从本章开始,按照书籍的划分,第10章开始就进入保护模式(Protected Mode)部分了,感觉从这里开始难度突然就增加了。 书中介绍了为什么有中断(Interrupt)的设计,中断的几种方式:外部硬件中断、内部中断和软中断。通过中断做了一个会走的时钟和屏幕上输入字符的程序。 我自己理解中断的一些作用: 为了更好的利用处理器的性能。协同快速和慢速设备一起工作

线程的四种操作

所属专栏:Java学习        1. 线程的开启 start和run的区别: run:描述了线程要执行的任务,也可以称为线程的入口 start:调用系统函数,真正的在系统内核中创建线程(创建PCB,加入到链表中),此处的start会根据不同的系统,分别调用不同的api,创建好之后的线程,再单独去执行run(所以说,start的本质是调用系统api,系统的api

java线程深度解析(六)——线程池技术

http://blog.csdn.net/Daybreak1209/article/details/51382604 一种最为简单的线程创建和回收的方法: [html]  view plain copy new Thread(new Runnable(){                @Override               public voi

java线程深度解析(五)——并发模型(生产者-消费者)

http://blog.csdn.net/Daybreak1209/article/details/51378055 三、生产者-消费者模式     在经典的多线程模式中,生产者-消费者为多线程间协作提供了良好的解决方案。基本原理是两类线程,即若干个生产者和若干个消费者,生产者负责提交用户请求任务(到内存缓冲区),消费者线程负责处理任务(从内存缓冲区中取任务进行处理),两类线程之

java线程深度解析(四)——并发模型(Master-Worker)

http://blog.csdn.net/daybreak1209/article/details/51372929 二、Master-worker ——分而治之      Master-worker常用的并行模式之一,核心思想是由两个进程协作工作,master负责接收和分配任务,worker负责处理任务,并把处理结果返回给Master进程,由Master进行汇总,返回给客

java线程深度解析(二)——线程互斥技术与线程间通信

http://blog.csdn.net/daybreak1209/article/details/51307679      在java多线程——线程同步问题中,对于多线程下程序启动时出现的线程安全问题的背景和初步解决方案已经有了详细的介绍。本文将再度深入解析对线程代码块和方法的同步控制和多线程间通信的实例。 一、再现多线程下安全问题 先看开启两条线程,分别按序打印字符串的

java线程深度解析(一)——java new 接口?匿名内部类给你答案

http://blog.csdn.net/daybreak1209/article/details/51305477 一、内部类 1、内部类初识 一般,一个类里主要包含类的方法和属性,但在Java中还提出在类中继续定义类(内部类)的概念。 内部类的定义:类的内部定义类 先来看一个实例 [html]  view plain copy pu