(工作真是太奇妙了,总觉着我在疯狂的给自己立flag)
顺便记录自己第一次说话都打颤的技术交流
1,关于event
event的定义属实难找,教材也没有,spec也没有。
不过它还是很好理解的,作为一个曾经写过如何自学qt博客的人(可惜最后工作完全用不上),我看见它的第一反应就是
这不connect吗
设置一个信号一个槽,当信号被触发时运行槽里的内容
至于怎么用,我举事件类型的例子就可以了
这是常用的三种类型(还有其他类型,类型与类型还可以组合,这不赘述了)
顺便一提,WAIT和SIGHAL是事件的两个互斥状态
下面说gEventQueue[Event->Notify]
2,事件优先级(Notify TPL)
异步事件一定需要事件优先级。如果没有谁重要、谁更重要的概念,那系统干脆按进入的时间顺序运行多好呢。
UEFI提供0~31,共32个优先级。其中,UEFI预定义了四个优先级。包括位于4的TPL_APPLICATION、位于8的TPL_CALLBACK、位于16的TPL_NOTIFY、位于31的TPL_HIGH_LEVEL(在此优先级关闭中断)
哪个函数在哪个优先级自己去spec文档看吧——这里比较有趣的是时间中断和waitforsighal都在TPL_APPLICATION,原因会在第二节提。
然后就是两个数据结构
gEventQueue 保存已经被唤醒、需要执行回调函数的事件,根据优先级分为多个(其实就是32个)链表
gEventPending 使用位图来保存gEventQueue是否含有某个优先级的链表(就是个索引,哪个优先级有就置1,不然置0)
有一些文章认为事件进入gEventQueue就是被执行了——其实不是,虽然时间上差不多。
3,提供接口
CreateEvent 生成一个事件对象
CreateEventEX 生成一个事件对象并加入到一个组内
CloseEvent 关闭事件对象
SignalEvent 触发事件对象
WaitForEvent 等待事件数组中的任意事件触发
CheckEvent 检查事件状态
SetTimer 设置定时器属性
具体参数和用法去spec里看吧,说得很清楚了
需要注意的是这两货
RaiseTPL 提升任务优先级
RestoreTPL 恢复(其实就是降低)任务优先级
剧透一下,事件是在RestoreTPL中执行的
4,加锁
异步需要锁,这是常识。
UEFI怎么加锁呢?
很简单,直接把优先级拉到31,在这个优先级下没有中断,也不会被其他低等级的事件打断
当然,也不都是31
LIST_ENTRY mEfiTimerList = INITIALIZE_LIST_HEAD_VARIABLE (mEfiTimerList);
EFI_LOCK mEfiTimerLock = EFI_INITIALIZE_LOCK_VARIABLE (TPL_HIGH_LEVEL - 1);
EFI_EVENT mEfiCheckTimerEvent = NULL;
EFI_LOCK mEfiSystemTimeLock = EFI_INITIALIZE_LOCK_VARIABLE(TPL_HIGH_LEVEL);
这就有个30,因为有两个锁同时调用的情况,都是31会死锁
5,一个例子
下面是有过期时间的等待触发函数,在edk2里面搜的
下一章讲edk2到底是怎么调度的
下下章讲时钟中断