UEFI 事件
UEFI 不再支持中断(准确地说,UEFI 不再为开发者提供中断支持,但在UEFI内部还是使用了时钟中断),所有的异步操作都要通过事件(Event)来完成。
启动服务为开发者提供了用于操作事件、定时器及TPL(任务优先级)的函数。这些函数可以分为三类:事件相关函数、定时器相关函数及 TPL相关函数。

事件函数
启动服务中事件相关函数有 6 个,函数名大部分以 Event 结尾。提供给事件生产者的函数有 CreateEvent/CreateEventEx、SignalEvent及CloseEvent。提供给事件使用者的有WaitForEvent和CheckEvent。
等待事件的服务 WaitForEvent
WaitForEvent 用于等待事件的发生,类似于 Windows 提供的 WaitForMultipleObjects(...)。
 
 WaitForEvent 是阻塞操作,直到 Event 数组内任一事件被触发,或任一事件导致错误出现,WaitForEvent 才返回。WaitForEvent 从前到后依次检查 Event 数组内的事件,发现有被触发的事件或遇到错误则返回,如果所有事件都没有被触发,则从头开始重新检查。
当检查到某个事件处于触发态时,*Index赋值为该事件在Event数组中的下标,返回前该事件将重置为非触发态。
当检查到某个事件是EVT_NOTIFY_SIGNAL类型时,*Index赋值为该事件在Event数组中的下标,并返回EFI_INVALID_PARAMETER。
 WaitForEvent必须运行在 TPL_APPLICAION 级别,否则将返回 EFI_UNSUPPORTED。
WaitForEvent 没有超时属性,如果想让 WaitForEvent 只等待一定的时间,则需要在事件等待数组加入定时器事件。
生成事件的服务 CreateEvent
CreateEvent 用于生成一个事件。
 
1.事件的类型
 事件的类型可以是以下一种或几种基本类型的组合:
 
- EVT_TIMER:定时器事件。普通Timer事件,没有Notification函数。生成事件后需调用SetTimer服务设置时钟属性;
- EVT_NOTIFY_WAIT:普通事件。这个事件有一个Notification函数,当这个事件通过CheckEvent()检查状态或通过WaitForEvent()等待时,这个Notification函数会被放到待执行队列gEventQueue[Event->NotifyTpl]中;
- EVT_NOTIFY_SIGNAL:普通事件。这个事件有一个Notification函数,当这个事件通过SignalEvent()被触发时,这个Notifcation函数会被放到待执行队列gEventQueue[Event->NotifyTpl]中等待执行;
- 0x00000000:普通事件。此类事件没有Notification 函数。
还有两种特殊的事件,它们用在操作系统系统加载器从启动期向运行时期转换的过程中:
- EVT_SIGNAL_EXIT_BOOT_SERVICES:此类事件是一种特殊的 EVT_NOTIFY_SIGNAL,实际上它是EVT_NOTIFY_SIGNAL和0x00000001的组合。当ExitBootServices()执行时,事件被触发。EVT_SIGNAL_EXIT_BOOT_SERVICES不能和其他类型混合使用。它的Notification函数和子函数不能使用启动服务中的内存分配服务;在Notification函数执行前所有的定时器服务都已失效,因而在Notificaiton函数中也不能使用定时器服务。
- EVT_SIGNAL_VIRTUAL_ADDRESS_CHANGE:它是EVT_RUNTIME_CONTEXT、EVT_RUNTIME、EVT_NOTIFY_SIGNAL和0x00000002的组合。它不能和这 4 种类型之外的类型组合使用。当SetVirtualAddressMap()被调用时触发此类事件。
2.优先级
 CreateEvent 的第二个参数为NotifyTPL(即任务优先级),它可以是 0 ~ 31 的一个整数。
 UEFI 预定义了以下 4 个优先级:
 
- TPL_APPLICATION:预定义的 4 个级别中最低的一个优先级。应用程序运行(包括 Boot Manager 和 OS Loader)在这个级别。当程序运行在这个级别时,任务队列中没有任何处于就绪状态的事件 Notification函数;
- TPL_CALLBACK:比较耗时的操作通常在这个优先级执行,如文件系统、磁盘操作等;
- TPL_NOTIFY:运行在这个级别的程序不允许阻塞必须尽快执行完毕并且返回。如果需要更多操作,则需要使用Event由内核重新调度。通常,底层的 IO 操作允许在这个级别。大部分Event的Notification函数允许在这个级别;
- TPL_HIGH_LEVEL:优先级最高级别。在此级别,中断被禁止。UEFI 内核全局变量的修改需要允许在这个级别。
3.Notification 函数 NotifyFunction
 CreateEvent的第三个参数NotifyFunction是EFI_EVENT_NOTIFY类型的函数指针。
 
 如果事件的类型是 EVT_NOTIFY_WAIT,则 EFI_EVENT_NOTIFY 函数会在等待此事件的过程中调用;如果事件的类型是EVT_NOTIFY_SIGNAL,则 EFI_EVENT_NOTIFY 函数会在事件触发时调用。既没有 EVT_NOTIFY_WAIT 属性也没有EVT_NOTIFY_SIGNAL属性的事件,Notification 参数将被忽略。
CreateEvent 的第4个参数是NotifyContext,将在Notification 函数被调用时作为第 2 个参数传递给该函数,用于指向这个Notification 函数的上下文。
CreateEventEx 服务
CreateEventEx 服务用于生成事件并将事件加入事件组。
 
 由 CreateEventEx 生成的事件会加入到EventGroup中。当EventGroup中的任一事件被触发后,组中的所有其他事件都会被触发,进而同组内所有的Notification函数都将被加入到待执行队列。同组内 NotifyTpl(优先级)高的Notification函数会先被执行。
如果输入参数EventGroup为NULL,则CreateEventEx退化为 CreateEvent。
Type 不能是 EVT_SIGNAL_EXIT_BOOT_SERVICES 或 EVT_SIGNAL_VIRTUAL_ADDRESS_CHANGE,因为这两种类型有各自对应的Group。
UEFI 预定义的 4 个 Event 组:
 
事件相关的其他函数
1.检查事件状态的服务CheckEvent()
 CheckEvent 用于检测事件的状态。与WaitForEvent不同的是,CheckEvent调用后立刻返回。
 
2.触发事件的服务 SignalEvent
 SignalEvent 用于将事件的状态设置为触发态。如果事件类型为EVT_NOTIFY_SIGNAL,则将其 Notification函数添加到就绪队列准备执行。如果该事件属于一个组,则将该组内所有事件都设置为触发态,并将组内所有EVT_NOTIFY_SIGNAL事件的Notification函数添加到就绪队列准备执行。
 
3.关闭事件的服务CloseEvent
 事件使用完毕后,必须调用CloseEvent关闭这个事件:
 
通常的原则是由事件的所有者(即调用CreateEvent产生该事件的调用者)调用CloseEvent函数。调用该函数后,指定的事件将从内核中删除。
定时器函数
定时器是一类特殊的事件,生成定时器事件后,可以通过SetTimer服务设置定时器属性。
SetTimer 服务的函数原型:
 
 Type 是定时器类别:
 
任务优先级
UEFI标准虽然不支持多线程,但是UEFI中有任务的概念:一个程序是一个任务,事件的Notification函数也是一个任务。UEFI没有给开发者提供中断接口,但 UEFI 内核的运行需要时钟中断的支持,时钟中断处理函数也是一个任务。
 例如,时钟中断任务的重要性要大于定时器的Notification函数,而定时器的 Notification函数的重要性大于普通应用程序。UEFI为任务定义了任务级别,以便有限的计算机资源可以相对合理地分配给众多的任务。
提升和恢复任务优先级
RaiseTPL(NewTpl)用于提升当前任务的任务优先级至NewTpl,该函数的返回值为原来的任务优先级。RestoreTPL用于恢复(通常是降低)任务优先级至原来的优先级。

 RaiseTPL 和 RestoreTPL 必须成对出现,执行了RaiseTPL后,必须尽快调用 RaiseTPL 将任务优先级恢复到原来的值。
当任务优先级提升至TPL_HIGH_LEVEL时,将关闭中断。当任务优先级从TPL_HIGH_LEVEL恢复到原来的(比TPL_HIGH_LEVEL低的)值时,中断被重新打开。
在任务优先级恢复到原优先级之前,所有高于原优先级的触发态事件的Notification函数都要执行完毕。 
UEFI是单CPU单线程系统,产生数据竞争的唯一可能来自中断处理函数,因而可以利用这一特性实现锁,这正是 UEFI 锁的实现机制。
UEFI 中的时钟中断
UEFI 用事件机制取代了 BIOS 中的中断机制,虽然 UEFI 不再提供中断接口,但其实现却离不开中断尤其是时钟中断。时钟中断是事件机制的基础。
1.时钟处理函数 CoreTimerTick
 在时钟中断中调用,是时钟中断处理函数的主体。该函数执行期间必须关中断并且不能被其他任何任务干扰,因而进入函数时需要加锁,离开函数时需要解锁。它的主要功能是维持系统时间,检查定时器事件列表中是否有到期的事件。
 
2.设置时钟处理函数及安装时钟中断
 
 
mArchProtocols 数组是 UEFI 系统 DXE 阶段的全局变量,存放了体系结构相关的 Protocol,系统初始化时会为 mArchProtocols 中的每个元素生成一个事件,当这个元素对应的 Protocol 安装时,该事件会触发,在事件的回调函数中会对该Protocol做相应的初始化。
Protocol 在内核中的组织:
 
 所有的 Protocol均放在 mProtocolDatabase指向的PROTOCOL_ENTRY链表中。PROTOCOL_ENTRY 包含三个链表。
 AllEntries 是PROTOCOL_ENTRY 链。
 Protocols 指向此Protocol的所有实例。
 Notify指向 PROTOCOL_NOTIFY链表,当 PROTOCOL_ENTRY.ProtocolD 对应的 Protocol 安装时,Notify 链表中所有Event都会触发。
例如,mArchProtocols[3]为{&gEfiTimerArchProtocolGuid, (VOID**)&gTimer, NULL, NULL, FALSE},是EFI_TIMER_ARCH_PROTOCOL 对应的 EFI_CORE_PROTOCOL_NOTIFY_ENTRY 项;
 CoreNotifyOnProtocolInstallation 执行后,mArchProtocols[3]为{&gEfiTimerArchProtocolGuid, ( VOlD** )&gTimer, timerEvent, Registration, FALSE} ,TimerEvent 的 Notification 函数被 CoreRegisterProtocolNotify 函数注册到系统。
 
 
 CoreRegisterProtocolNotify(…)函数的主要功能是在 mProtocolDatabase 数据库中注册PROTOCOL_NOTIFY。当Protocol安装时,会检查该Protocol对应的PROTOCOL_ENTRY.Notify。如果PROTOCOL_ENTRY.Notify存在,则触发它指向的事件。
 
当 EFI_TIMER_ARCH_PROTOCOL安装时,mArchProtocols[3].Event事件会触发,然后这个事件的响应函数 GenericProtocolNotify会执行,在 GenericProtocoINotify中通过 EFI_TIMER_ARCH_PROTOCOL的RegisterHandler 时钟中断处理函数。
向 gimer 注册 CoreTimerTick 函数
 gTimer 是EFI_TIMER_ARCH_PROTOCOL*类型的全局变量。
 
 gTimer->RegisterHandler 这个函数指针指向了函数 TimerDriverRegisterHandler。
 
mTimerNotifyFunction这个函数指针将在时钟中断处理函数TimerInterruptHandler中被调用。
 
向 CPU 注册中断处理函数 TimerInterruptHandler
 
 
 
 在 TimerDriverlnitialize 中,最终通过 EFI_CPU_ARCH_PROTOCOL 的 RegisterInterruptHandler注册了 TimerInterruptHandler。另外,在中断处理函数中将会调用 TimerInterruptHandler,而TimerlnterruptHandler 又会调用 mTimerNotifyFunction, 即 CoreTimerTick。
 
 而 mCpu->RegisterInterruptHandler 将会调用CpuRegisterInterruptHandler 函数:
 
 此函数注册并启用由 InterruptHandler 为处理器中断或由InterruptType指定的异常类型指定的处理程序。如果InterruptHandler为NULL,则取消安装由InterruptType指定的处理器中断或异常类型的处理程序。安装的处理程序对于每个处理器中断或异常调用一次。
在 CPU 时钟中断向量中调用时钟中断处理函数
默认的中断向量主要功能是调用CommonInterruptEntry。该函数主要完成以下任务:
 1)保存寄存器;
 2)调用ExternalVectorTable[InterruptType];
 3)恢复寄存器,从中断处理返回。
 在时钟中断向量中,ExternalVectorTable[InterruptType]指向函数 TimerInterruptHandler。
时钟中断的执行过程:
 
 注册时钟中断函数和注册时钟处理函数的过程:
 
UEFI 事件 Notification 函数的派发
Event的一个重要作用是实现异步操作,事件 Notification函数的派发是在gBS->RestoreTpl服务中完成的。gBS->RestoreTpl实际指向CoreRestoreTpl函数:
 
内容来源于 《UEFI 原理与编程》。。。
相关文章:
 
UEFI 事件
UEFI 不再支持中断(准确地说,UEFI 不再为开发者提供中断支持,但在UEFI内部还是使用了时钟中断),所有的异步操作都要通过事件(Event)来完成。 启动服务为开发者提供了用于操作事件、定时器及TPL…...
大师开讲-图形学领域顶级专家王锐开讲Vulkan、VSG开源引擎
王锐,毕业于清华大学,图形学领域顶级专家,开源技术社区的贡献者与推广者。三维引擎OpenSceneGraph的核心基石开发者与维护者,倾斜摄影数据格式osgb的发明人。著有《OpenSceneGraph 3 Cookbook》,《OpenSceneGraph 3 Beginers Guid…...
 
小F的矩阵值调整
问题描述 小F得到了一个矩阵。如果矩阵中某一个格子的值是偶数,则该值变为它的三倍;如果是奇数,则保持不变。小F想知道调整后的矩阵是什么样子的。 测试样例 样例1: 输入:a [[1, 2, 3], [4, 5, 6]] 输出:…...
 
ORB-SLAM2 ----- LocalMapping::SearchInNeighbors()
文章目录 一、函数意义二、函数讲解三、函数代码四、本函数使用的匹配方法ORBmatcher::Fuse()1. 函数讲解2. 函数代码 四、总结 一、函数意义 本函数是用于地图点融合的函数,前面的函数生成了新的地图点,但这些地图点可能在前面的关键帧中已经生成过了&a…...
 
给UE5优化一丢丢编辑器性能
背后的原理 先看FActorIterator的定义 /*** Actor iterator* Note that when Playing In Editor, this will find actors only in CurrentWorld*/ class FActorIterator : public TActorIteratorBase<FActorIterator> {//..... }找到基类TActorIteratorBase /*** Temp…...
 
【Docker】常用命令汇总
Docker 是1个开源的应用容器引擎,基于Go 语言并遵从 Apache2.0 协议开源。 可以让开发者打包他们的应用以及依赖包到一个轻量级、可移植的容器中,然后发布到任何流行的 Linux 机器上,也可以实现虚拟化。 容器是完全使用沙箱机制,相…...
 
Mybatis:CRUD数据操作之多条件查询及动态SQL
Mybatis基础环境准备请看:Mybatis基础环境准备 本篇讲解Mybati数据CRUD数据操作之多条件查询 1,编写接口方法 在 com.itheima.mapper 包写创建名为 BrandMapper 的接口。在 BrandMapper 接口中定义多条件查询的方法。 而该功能有三个参数,…...
【笔记】轻型民用无人驾驶航空器安全操控
《轻型民用无人驾驶航空器安全操控》 理论考试培训材料 法规部分 【民用无人驾驶航空器的分类】 1、如何定义微型、轻型无人驾驶航空器? 微型无人驾驶航空器,是指空机重量小于0.25千克,最大平飞速度不超过40千米/小时,无线电发…...
TouchGFX设计模式代码实例说明
一)Model - View - Presenter (MVP) 模式在 TouchGFX 中的应用 1)Model(模型): 模型代表应用程序的数据和业务逻辑。例如,在一个简单的计数器应用中,模型可以是一个包含计数器当前值的类。 class CounterModel { pri…...
 
flink学习(7)——window
概述 窗口的长度(大小): 决定了要计算最近多长时间的数据 窗口的间隔: 决定了每隔多久计算一次 举例:每隔10min,计算最近24h的热搜词,24小时是长度,每隔10分钟是间隔。 窗口的分类 1、根据window前是否调用keyBy分为键控窗口和非键控窗口…...
restTemplate get请求
报错解释: 这个报错信息表明在使用RestTemplate进行GET请求时,需要提供一个请求类型(reqType),但是传入的值为空。这通常意味着在构建请求或者调用方法时,没有正确设置请求的Content-Type头部,…...
ffmpeg 预设的值 加速
centos 安装ffmpeg 编译安装 官网获取最新的linux ffmpeg 代码 https://ffmpeg.org//releases/ mkdir -p /data/app/ffmpeg cd /data/app/ffmpeg wget http://www.ffmpeg.org/releases/ffmpeg-7.1.tar.gz tar -zxvf ffmpeg-7.1.tar.gz#安装所需的编译环境 yum install -y \…...
maven <scope>compile</scope>作用
在 Maven 项目中, 元素用于定义依赖项的作用范围。 元素可以有多个值,每个值表示不同的作用范围。其中,scope compile scope 是默认的作用范围,表示该依赖项在编译、测试和运行时都需要。 scope compile scope 的含义 1、编译时…...
 
Ubuntu Server 22.04.5 从零到一:详尽安装部署指南
文章目录 Ubuntu Server 22.04.5 从零到一:详尽安装部署指南一、部署环境二、安装系统2.1 安装2.1.1 选择安装方式2.1.2 选择语言2.1.3 选择不更新2.1.4 选择键盘标准2.1.5 选择安装版本2.1.6 设置网卡2.1.7 配置代理2.1.8 设置镜像源2.1.9 选择装系统的硬盘2.1.10 …...
反射机制了解
反射概念 了解反射背景 存在某些变量或形参的声明类型是Object类型,但是程序却需要调用该对象运行时类型的方法,该方法不是Object中的方法,如何解决。转到如何获取该对象运行时类型的方法。 只能运行时才能获取,这就用到反射。 …...
机器学习策略Ⅰ
机器学习策略Ⅰ 在构建一个好的监督学习系统时,通常需要确保以下四个方面: 系统需要在训练集上能够很好地拟合数据,达到某种可接受的性能水平(如接近人类水平)。如果训练集表现不好,可以使用更大的模型&…...
redis中的bigkey及读取优化
一、bigKey介绍 1、简介 在 Redis 中,Big Key(大键)指的是占用大量内存的单个键。通常,Redis 是一个高性能的内存数据库,但是当某些键变得非常大时,会带来性能上的影响。例如,大量的内存消耗、长时间的操作延迟,甚至可能导致 Redis 停止响应或崩溃。 通俗的来说,指…...
 
【西瓜书】支持向量机(SVM)
支持向量机(Support Vector Machine,简称SVM)。 超平面 分类学习最基本的想法就是基于训练集合D在样本空间中找到一个划分超平面,将不同类别的样本分开。 但能将训练样本分开的划分超平面可能有很多,应该努力去找到哪…...
 
三维渲染中顺序无关的半透明混合(OIT)(二——Stencil Route)
1、A-Buffer算法。 在谈到Stencil Route之前,需要先讨论A-Buffer算法。A-Buffer是一种图形学(渲染方向)上的用于可见面分析(Visble Surface Detection)的技术,是Z-Buffer的衍生方法。 Z-Buffer是用于剔除 不透明 物体的算法。假…...
(SAST检测规则-3)固定的 SessionID 缺陷详解
漏洞类型: 会话固定攻击(Session Fixation Attack) 漏洞描述: 会话固定攻击是利用服务器的会话管理机制存在漏洞,攻击者通过提前控制或预测用户的会话标识符(Session ID),当用户登录…...
 
前端开发面试题总结-JavaScript篇(一)
文章目录 JavaScript高频问答一、作用域与闭包1.什么是闭包(Closure)?闭包有什么应用场景和潜在问题?2.解释 JavaScript 的作用域链(Scope Chain) 二、原型与继承3.原型链是什么?如何实现继承&a…...
什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南
文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果 
以光量子为例,详解量子获取方式
光量子技术获取量子比特可在室温下进行。该方式有望通过与名为硅光子学(silicon photonics)的光波导(optical waveguide)芯片制造技术和光纤等光通信技术相结合来实现量子计算机。量子力学中,光既是波又是粒子。光子本…...
Python ROS2【机器人中间件框架】 简介
销量过万TEEIS德国护膝夏天用薄款 优惠券冠生园 百花蜂蜜428g 挤压瓶纯蜂蜜巨奇严选 鞋子除臭剂360ml 多芬身体磨砂膏280g健70%-75%酒精消毒棉片湿巾1418cm 80片/袋3袋大包清洁食品用消毒 优惠券AIMORNY52朵红玫瑰永生香皂花同城配送非鲜花七夕情人节生日礼物送女友 热卖妙洁棉…...
 
技术栈RabbitMq的介绍和使用
目录 1. 什么是消息队列?2. 消息队列的优点3. RabbitMQ 消息队列概述4. RabbitMQ 安装5. Exchange 四种类型5.1 direct 精准匹配5.2 fanout 广播5.3 topic 正则匹配 6. RabbitMQ 队列模式6.1 简单队列模式6.2 工作队列模式6.3 发布/订阅模式6.4 路由模式6.5 主题模式…...
 
20个超级好用的 CSS 动画库
分享 20 个最佳 CSS 动画库。 它们中的大多数将生成纯 CSS 代码,而不需要任何外部库。 1.Animate.css 一个开箱即用型的跨浏览器动画库,可供你在项目中使用。 2.Magic Animations CSS3 一组简单的动画,可以包含在你的网页或应用项目中。 3.An…...
Redis:现代应用开发的高效内存数据存储利器
一、Redis的起源与发展 Redis最初由意大利程序员Salvatore Sanfilippo在2009年开发,其初衷是为了满足他自己的一个项目需求,即需要一个高性能的键值存储系统来解决传统数据库在高并发场景下的性能瓶颈。随着项目的开源,Redis凭借其简单易用、…...
 
永磁同步电机无速度算法--基于卡尔曼滤波器的滑模观测器
一、原理介绍 传统滑模观测器采用如下结构: 传统SMO中LPF会带来相位延迟和幅值衰减,并且需要额外的相位补偿。 采用扩展卡尔曼滤波器代替常用低通滤波器(LPF),可以去除高次谐波,并且不用相位补偿就可以获得一个误差较小的转子位…...
redis和redission的区别
Redis 和 Redisson 是两个密切相关但又本质不同的技术,它们扮演着完全不同的角色: Redis: 内存数据库/数据结构存储 本质: 它是一个开源的、高性能的、基于内存的 键值存储数据库。它也可以将数据持久化到磁盘。 核心功能: 提供丰…...
