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

相关文章

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

Java捕获ThreadPoolExecutor内部线程异常的四种方法

《Java捕获ThreadPoolExecutor内部线程异常的四种方法》这篇文章主要为大家详细介绍了Java捕获ThreadPoolExecutor内部线程异常的四种方法,文中的示例代码讲解详细,感... 目录方案 1方案 2方案 3方案 4结论方案 1使用 execute + try-catch 记录

Spring Boot 中正确地在异步线程中使用 HttpServletRequest的方法

《SpringBoot中正确地在异步线程中使用HttpServletRequest的方法》文章讨论了在SpringBoot中如何在异步线程中正确使用HttpServletRequest的问题,... 目录前言一、问题的来源:为什么异步线程中无法访问 HttpServletRequest?1. 请求上下文与线

在 Spring Boot 中使用异步线程时的 HttpServletRequest 复用问题记录

《在SpringBoot中使用异步线程时的HttpServletRequest复用问题记录》文章讨论了在SpringBoot中使用异步线程时,由于HttpServletRequest复用导致... 目录一、问题描述:异步线程操作导致请求复用时 Cookie 解析失败1. 场景背景2. 问题根源二、问题详细分

Java多线程父线程向子线程传值问题及解决

《Java多线程父线程向子线程传值问题及解决》文章总结了5种解决父子之间数据传递困扰的解决方案,包括ThreadLocal+TaskDecorator、UserUtils、CustomTaskDeco... 目录1 背景2 ThreadLocal+TaskDecorator3 RequestContextH

java父子线程之间实现共享传递数据

《java父子线程之间实现共享传递数据》本文介绍了Java中父子线程间共享传递数据的几种方法,包括ThreadLocal变量、并发集合和内存队列或消息队列,并提醒注意并发安全问题... 目录通过 ThreadLocal 变量共享数据通过并发集合共享数据通过内存队列或消息队列共享数据注意并发安全问题总结在 J

异步线程traceId如何实现传递

《异步线程traceId如何实现传递》文章介绍了如何在异步请求中传递traceId,通过重写ThreadPoolTaskExecutor的方法和实现TaskDecorator接口来增强线程池,确保异步... 目录前言重写ThreadPoolTaskExecutor中方法线程池增强总结前言在日常问题排查中,

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

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