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

相关文章

Java中实现线程的创建和启动的方法

《Java中实现线程的创建和启动的方法》在Java中,实现线程的创建和启动是两个不同但紧密相关的概念,理解为什么要启动线程(调用start()方法)而非直接调用run()方法,是掌握多线程编程的关键,... 目录1. 线程的生命周期2. start() vs run() 的本质区别3. 为什么必须通过 st

Linux实现线程同步的多种方式汇总

《Linux实现线程同步的多种方式汇总》本文详细介绍了Linux下线程同步的多种方法,包括互斥锁、自旋锁、信号量以及它们的使用示例,通过这些同步机制,可以解决线程安全问题,防止资源竞争导致的错误,示例... 目录什么是线程同步?一、互斥锁(单人洗手间规则)适用场景:特点:二、条件变量(咖啡厅取餐系统)工作流

Java中常见队列举例详解(非线程安全)

《Java中常见队列举例详解(非线程安全)》队列用于模拟队列这种数据结构,队列通常是指先进先出的容器,:本文主要介绍Java中常见队列(非线程安全)的相关资料,文中通过代码介绍的非常详细,需要的朋... 目录一.队列定义 二.常见接口 三.常见实现类3.1 ArrayDeque3.1.1 实现原理3.1.2

SpringBoot3中使用虚拟线程的完整步骤

《SpringBoot3中使用虚拟线程的完整步骤》在SpringBoot3中使用Java21+的虚拟线程(VirtualThreads)可以显著提升I/O密集型应用的并发能力,这篇文章为大家介绍了详细... 目录1. 环境准备2. 配置虚拟线程方式一:全局启用虚拟线程(Tomcat/Jetty)方式二:异步

如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socket read timed out的问题

《如何解决Druid线程池Cause:java.sql.SQLRecoverableException:IO错误:Socketreadtimedout的问题》:本文主要介绍解决Druid线程... 目录异常信息触发场景找到版本发布更新的说明从版本更新信息可以看到该默认逻辑已经去除总结异常信息触发场景复

JAVA保证HashMap线程安全的几种方式

《JAVA保证HashMap线程安全的几种方式》HashMap是线程不安全的,这意味着如果多个线程并发地访问和修改同一个HashMap实例,可能会导致数据不一致和其他线程安全问题,本文主要介绍了JAV... 目录1. 使用 Collections.synchronizedMap2. 使用 Concurren

使用Python实现一个优雅的异步定时器

《使用Python实现一个优雅的异步定时器》在Python中实现定时器功能是一个常见需求,尤其是在需要周期性执行任务的场景下,本文给大家介绍了基于asyncio和threading模块,可扩展的异步定... 目录需求背景代码1. 单例事件循环的实现2. 事件循环的运行与关闭3. 定时器核心逻辑4. 启动与停

Spring Boot3虚拟线程的使用步骤详解

《SpringBoot3虚拟线程的使用步骤详解》虚拟线程是Java19中引入的一个新特性,旨在通过简化线程管理来提升应用程序的并发性能,:本文主要介绍SpringBoot3虚拟线程的使用步骤,... 目录问题根源分析解决方案验证验证实验实验1:未启用keep-alive实验2:启用keep-alive扩展建

Springboot如何配置Scheduler定时器

《Springboot如何配置Scheduler定时器》:本文主要介绍Springboot如何配置Scheduler定时器问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录Springboot配置Scheduler定时器1.在启动类上添加 @EnableSchedulin

Java终止正在运行的线程的三种方法

《Java终止正在运行的线程的三种方法》停止一个线程意味着在任务处理完任务之前停掉正在做的操作,也就是放弃当前的操作,停止一个线程可以用Thread.stop()方法,但最好不要用它,本文给大家介绍了... 目录前言1. 停止不了的线程2. 判断线程是否停止状态3. 能停止的线程–异常法4. 在沉睡中停止5