MiniGUI 定时器分析

2023-12-11 06:32
文章标签 分析 定时器 minigui

本文主要是介绍MiniGUI 定时器分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

MiniGUI 定时器分析 (注:该MiniGUI库版本为1.6.10 非LITE版本)

MiniGUI几个定时器相关的函数如下:

BOOL GUIAPI ResetTimerEx(HWND hWnd, int id, unsigned int speed, TIMERPROC timer_proc);

BOOL GUIAPI SetTimerEx(HWND hWnd, int id, unsigned int speed, TIMERPROC timer_proc);

#define SetTimer(hwnd, id, speed) SetTimerEx(hwnd, id, speed, NULL)

#define ResetTimer(hwnd, id, speed) ResetTimerEx(hwnd, id, speed, (TIMERPROC)0xFFFFFFFF)

hWnd为创建定时器时传入的窗口句柄。

id 为定时器id号

speed 为定时器时间间隔10ms 为单位。

timer_proc 为定时器回调函数

MiniGUI定时器内部运行原理

src/kernel/init.c 文件下InitGUI函数是在MiniGUI程序初始化的时候被调用的。

int GUIAPI InitGUI (int args, const char *agr[])

{

。。。

  SystemThreads()

。。。

}

InitGUI函数中调用SystemThreads函数

SystemThreads函数中创建了DesktopMain线程。

pthread_create (&__mg_desktop, NULL, DesktopMain, &wait);函数

void* DesktopMain (void* data)

{

    MSG Msg;

。。。

    while (GetMessage(&Msg, HWND_DESKTOP)) {

        int iRet = 0;

        iRet = DesktopWinProc (HWND_DESKTOP, Msg.message, Msg.wParam, Msg.lParam);

。。。      

    }

    return NULL;

}

可以看到在这里MiniGUI创建了一个桌面线程用于处理桌面线程消息。

int DesktopWinProc (HWND hWnd, int message, WPARAM wParam, LPARAM lParam)

{

case MSG_TIMER:      // per 0.01s

        {

            static UINT uCounter = 0;

            DispatchTimerMessage (1);

            if (__mg_timer_counter % 10 != 0)

                break;

            uCounter += 100;

         }

        break;

}

在这里可以看到桌面窗口回调函数对MSG_TIMER消息的处理。调用了DispatchTimerMessage (1);函数。

timerstr该结构体数组里面最大可以放DEF_NR_TIMERS这么多个定时器,轮询每个定时器检查时间是否超时,如超时则设置定时器时间到标志,让PeekMessageEx函数做处理。

void DispatchTimerMessage (unsigned int inter)

{

    int i;

    TIMER_LOCK ();

    for (i=0; i<DEF_NR_TIMERS; i++) {

        if (timerstr[i] && timerstr[i]->msg_queue) {

            timerstr[i]->count += inter;

            if (timerstr[i]->count >= timerstr[i]->speed) {

                if (timerstr[i]->tick_count == 0)

                    timerstr[i]->tick_count = __mg_timer_counter;

                SetMsgQueueTimerFlag (timerstr[i]->msg_queue, i);

                

                timerstr[i]->count -= timerstr[i]->speed;

            }

        }

    }

    TIMER_UNLOCK ();

}

SetMsgQueueTimerFlag 函数中调用了POST_MSGQ (pMsgQueue);

static inline void

SetMsgQueueTimerFlag (PMSGQUEUE pMsgQueue, int slot)

{

    pMsgQueue->TimerMask |= (0x01 << slot);

    POST_MSGQ (pMsgQueue);  //该宏的作用是将窗口消息循环由阻塞状态唤醒。

}

此时创建该定时器的窗口过程中的消息循环PeekMessageEx会被唤醒。

BOOL PeekMessageEx (PMSG pMsg, HWND hWnd, int iMsgFilterMin, int iMsgFilterMax, 

                          BOOL bWait, UINT uRemoveMsg)

{

。。。

      if ((timer = __mg_get_timer (slot))) {

            unsigned int tick_count = timer->tick_count;

            timer->tick_count = 0;

            pMsgQueue->TimerMask &= ~(0x01 << slot);

            if (timer->proc) { //如果该定时器定义回调函数

                BOOL ret_timer_proc;

                UNLOCK_MSGQ (pMsgQueue);

                ret_timer_proc = timer->proc (timer->hWnd,  timer->id, tick_count);

                LOCK_MSGQ (pMsgQueue);

                if (!ret_timer_proc) {

                    __mg_remove_timer (timer, slot);

                }

            }

            else { 

//如果该函数回调函数指针为空则转成消息放入消息队列等着DispatchMessage函数处理

                pMsg->message = MSG_TIMER;

                pMsg->hwnd = timer->hWnd;

                pMsg->wParam = timer->id;

                pMsg->lParam = tick_count;

                SET_PADD (NULL);

                UNLOCK_MSGQ (pMsgQueue);

                return TRUE;

            }

        }

。。。

}

SystemThreads函数中调用__mg_timer_init函数,该函数又启动了 TimerEntry 线程。

int __mg_timer_init (void)

{

    sem_t wait;

    sem_init (&wait, 0, 0);

    pthread_create (&__mg_timer, NULL, TimerEntry, &wait);

    sem_wait (&wait);

    sem_destroy (&wait);

    return 0;

}

TimerEntry 线程调用了_os_timer_loop 时间循环函数

static void* TimerEntry (void* data)

{

    if (!InitTimer ()) {

        fprintf (stderr, "TIMER: Init Timer failure, exit!\n");

        return NULL;

    }

    sem_post ((sem_t*)data);

    _os_timer_loop ();

    return NULL;

}

时间循环函数如下

static inline void _os_timer_loop (void)

{

    while (1) {

        __mg_os_time_delay (10);  //延时10ms

        __mg_timer_action (NULL);      //每10ms调用一次

    }

}

每10ms AlertDesktopTimerEvent 函数被调用

static void __mg_timer_action (void *data)

{

    __mg_timer_counter ++;

    AlertDesktopTimerEvent ();

}

每10ms 由AlertDesktopTimerEvent 给桌面消息循环设置时间到标志,如该循环阻塞,则唤醒该循环。

AlertDesktopTimerEvent (void)

{

    __mg_dsk_msg_queue->TimerMask = 1;

    POST_MSGQ(__mg_dsk_msg_queue);

}


这篇关于MiniGUI 定时器分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

慢sql提前分析预警和动态sql替换-Mybatis-SQL

《慢sql提前分析预警和动态sql替换-Mybatis-SQL》为防止慢SQL问题而开发的MyBatis组件,该组件能够在开发、测试阶段自动分析SQL语句,并在出现慢SQL问题时通过Ducc配置实现动... 目录背景解决思路开源方案调研设计方案详细设计使用方法1、引入依赖jar包2、配置组件XML3、核心配

Java NoClassDefFoundError运行时错误分析解决

《JavaNoClassDefFoundError运行时错误分析解决》在Java开发中,NoClassDefFoundError是一种常见的运行时错误,它通常表明Java虚拟机在尝试加载一个类时未能... 目录前言一、问题分析二、报错原因三、解决思路检查类路径配置检查依赖库检查类文件调试类加载器问题四、常见

Python中的Walrus运算符分析示例详解

《Python中的Walrus运算符分析示例详解》Python中的Walrus运算符(:=)是Python3.8引入的一个新特性,允许在表达式中同时赋值和返回值,它的核心作用是减少重复计算,提升代码简... 目录1. 在循环中避免重复计算2. 在条件判断中同时赋值变量3. 在列表推导式或字典推导式中简化逻辑

Java程序进程起来了但是不打印日志的原因分析

《Java程序进程起来了但是不打印日志的原因分析》:本文主要介绍Java程序进程起来了但是不打印日志的原因分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java程序进程起来了但是不打印日志的原因1、日志配置问题2、日志文件权限问题3、日志文件路径问题4、程序

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

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

Java字符串操作技巧之语法、示例与应用场景分析

《Java字符串操作技巧之语法、示例与应用场景分析》在Java算法题和日常开发中,字符串处理是必备的核心技能,本文全面梳理Java中字符串的常用操作语法,结合代码示例、应用场景和避坑指南,可快速掌握字... 目录引言1. 基础操作1.1 创建字符串1.2 获取长度1.3 访问字符2. 字符串处理2.1 子字

Python 迭代器和生成器概念及场景分析

《Python迭代器和生成器概念及场景分析》yield是Python中实现惰性计算和协程的核心工具,结合send()、throw()、close()等方法,能够构建高效、灵活的数据流和控制流模型,这... 目录迭代器的介绍自定义迭代器省略的迭代器生产器的介绍yield的普通用法yield的高级用法yidle

C++ Sort函数使用场景分析

《C++Sort函数使用场景分析》sort函数是algorithm库下的一个函数,sort函数是不稳定的,即大小相同的元素在排序后相对顺序可能发生改变,如果某些场景需要保持相同元素间的相对顺序,可使... 目录C++ Sort函数详解一、sort函数调用的两种方式二、sort函数使用场景三、sort函数排序

kotlin中const 和val的区别及使用场景分析

《kotlin中const和val的区别及使用场景分析》在Kotlin中,const和val都是用来声明常量的,但它们的使用场景和功能有所不同,下面给大家介绍kotlin中const和val的区别,... 目录kotlin中const 和val的区别1. val:2. const:二 代码示例1 Java

Go标准库常见错误分析和解决办法

《Go标准库常见错误分析和解决办法》Go语言的标准库为开发者提供了丰富且高效的工具,涵盖了从网络编程到文件操作等各个方面,然而,标准库虽好,使用不当却可能适得其反,正所谓工欲善其事,必先利其器,本文将... 目录1. 使用了错误的time.Duration2. time.After导致的内存泄漏3. jsO