UEFI开发——编写一个简单的PPI

2024-08-28 22:04
文章标签 简单 开发 编写 uefi ppi

本文主要是介绍UEFI开发——编写一个简单的PPI,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

PEI阶段是由一个个的PEIM实现的,而PPI是PEIM之间互相调用的接口。简单来说,PPI只是一个接口,接口里面有成员函数,想调用某个函数就必须通过该接口。

PPI通过PPI描述符来描述其特性,PPI描述符是一个数据结构,定义如下:

typedef struct {//UINTN是一个无符号的整数,Flags是一系列诸如EFI_PEI_PPI_DESCRIPTOR_***的组合UINTN       Flags;//接口的名字,即它的GUIDEFI_GUID    *Guid;//指向PPI的指针,包含安装服务的信息VOID        *Ppi;
} EFI_PEI_PPI_DESCRIPTOR;

在利用PPI的过程中,有个几个重要的PPI服务:

InstallPpi():安装PPI到PEI foundation

LocatePpi():根据PPI名字GUID从PEI foundation中找到Interface

NotifyPpi():PPI里的function不会再派发时就执行,会有一个判定条件,通知系统这个PPI会在某个PPI被安装时才执行。

1、编写一个InstallPPI的源文件:

该模块通过PeiServicesInstallPpi注册接口,以提供服务

/**This service enables a given PEIM to register an interface into the PEI Foundation.@param  PpiList               A pointer to the list of interfaces that the caller shall install.@retval EFI_SUCCESS           The interface was successfully installed.@retval EFI_INVALID_PARAMETER The PpiList pointer is NULL.@retval EFI_INVALID_PARAMETER Any of the PEI PPI descriptors in the list do not have theEFI_PEI_PPI_DESCRIPTOR_PPI bit set in the Flags field.@retval EFI_OUT_OF_RESOURCES  There is no additional space in the PPI database.**///使特定的PEIM向PEI foundation注册接口
EFI_STATUS
EFIAPI
PeiServicesInstallPpi (IN CONST EFI_PEI_PPI_DESCRIPTOR  *PpiList //指向调用者应安装的接口列表的指针);

 所提供的服务由EFI_PEI_PPI_DESCRIPTOR这个数据结构通过PEIM向PEI foundation进行描述,这个数据结构的原型为

// The data structure through which a PEIM describes available services to the PEI Foundation.
typedef struct {// This field is a set of flags describing the characteristics of this imported table entry.// All flags are defined as EFI_PEI_PPI_DESCRIPTOR_***, which can also be combined into one.UINTN       Flags;// The address of the EFI_GUID that names the interface.EFI_GUID    *Guid;// A pointer to the PPI. It contains the information necessary to install a service.VOID        *Ppi;
} EFI_PEI_PPI_DESCRIPTOR;

完整的 MyHelloWorldInstallPPI模块的源文件为:

#include <uefi.h>
#include <Library/UefiLib.h>
#include <Library/BaseLib.h>
#include <Library/DebugLib.h>
#include <Library/BaseMemoryLib.h>
#include <Library/UefiDriverEntryPoint.h>
#include <Library/PeimEntryPoint.h>
#include <Library/PeiServicesLib.h>
#include <Library/PeiServicesTablePointerLib.h>
#include <Pi/PiHob.h>
#include <Pi/PiPeiCis.h>EFI_GUID gMyHelloWorldPEIGUID = { 0xbdb38129, 0x4d65, 0x39f4, { 0x72, 0x12, 0x68, 0xcf, 0x5a, 0x19, 0xa, 0xf8 }};//定义了一个DEBUG_ERROR级别的信息打印函数,
EFI_STATUS
EFIAPI
PrintHelloMsg (IN CHAR16 * Msg //定义了一个输入参数,类型为CHAR16的指针)
{DEBUG ((EFI_D_ERROR, "[MyHelloWorldPPI] PrintHelloMsg : %s \n",Msg));return EFI_SUCCESS;
}/*
typedef的作用是给已有的数据类型或结构体定义一个新的名称
函数指针:指向函数的指针变量
int maxValue (int a, int b) {return a > b ? a : b;
}    int (*p)(int, int) = NULL;  //定义一个与maxValue兼容的指针
p = maxValue;
p(20, 45);  //通过指针调用
*/
typedef EFI_STATUS (EFIAPI * PRINTMSG)(CHAR16 *Msg);//给已有的数据类型定义了一个新的名称PRINTMSG,原来的数据类型为函数指针类型EFI_STATUS (EFIAPI *)(CHAR16* Msg)
typedef struct _EFI_HELLOWORLD_PROTOCOL  {PRINTMSG PrintMsg;
}EFI_HELLOWORLD_PROTOCOL;//定义了一个结构体,结构体成员为函数指针PrintMsg,并使用typedef将这个结构体的别名为EFI_HELLOWORLD_PROTOCOL//ppi实例
EFI_HELLOWORLD_PROTOCOL mHelloWorldPpi = {PrintHelloMsg
};//定义了一个EFI_HELLOWORLD_PROTOCOL类型的全局变量,并赋初值为PrintHelloMsg(即指向上面的打印函数)EFI_PEI_PPI_DESCRIPTOR     mPpiListCodePpi = { //EFI_PEI_PPI_DESCRIPTOR是一个数据结构,可以向PEIfoundation描述能够提供的服务(EFI_PEI_PPI_DESCRIPTOR_PPI | EFI_PEI_PPI_DESCRIPTOR_TERMINATE_LIST), //EFI_PEI_PPI_DESCRIPTOR_***这样flag的组合,| 表示按位或&gMyHelloWorldPEIGUID, //GUID, PPI的名字&mHelloWorldPpi //ppi实例
};//在这里定义了一个结构体变量mPpiListCodePpi,并进行初始化//安装PPI,指的是把一个或多个PPI注册到PEI foundation中
EFI_STATUS
EFIAPI
MyHelloWorldInstallPPIEntry(IN       EFI_PEI_FILE_HANDLE  FileHandle,IN CONST EFI_PEI_SERVICES     **PeiServices //指向EFI_PEI_SERVICES table的间接指针
)
{EFI_STATUS                              Status = EFI_SUCCESS;DEBUG ((EFI_D_ERROR, "[MyHelloWorldPPI] MyHelloWorldInstallPPIEntry Start..\n"));PeiServicesInstallPpi  (&mPpiListCodePpi);DEBUG ((EFI_D_ERROR, "[MyHelloWorldPPI] MyHelloWorldInstallPPIEntry End..\n"));return Status;
}

2、编写InstallPPI的inf文件

3、编写LocatePPI的源文件:

 PeiServicesLocatePpi服务的作用让PEIM发现接口的给定实例,其原型为:

EFI_STATUS
EFIAPI
PeiServicesLocatePpi (IN CONST EFI_GUID              *Guid, //指向需要被查找的GUIDIN UINTN                       Instance, //所需接口的第几个实例IN OUT EFI_PEI_PPI_DESCRIPTOR  **PpiDescriptor  OPTIONAL, //指向EFI_PEI_PPI_DESCRIPTOR实例的指针IN OUT VOID                    **Ppi //指向接口实例的指针);
#include <uefi.h>
#include <Library/UefiLib.h>
#include <Library/BaseLib.h>
#include <Library/DebugLib.h>
#include <Library/BaseMemoryLib.h>
#include <Library/UefiDriverEntryPoint.h>
#include <Library/PeimEntryPoint.h>
#include <Library/PeiServicesLib.h>
#include <Library/PeiServicesTablePointerLib.h>
#include <Pi/PiHob.h>EFI_GUID gMyHelloWorldPEIGUID = { 0xbdb38129, 0x4d65, 0x39f4, { 0x72, 0x12, 0x68, 0xcf, 0x5a, 0x19, 0xa, 0xf8 }};typedef EFI_STATUS (EFIAPI * PRINTMSG)(CHAR16 *Msg);
typedef struct _EFI_HELLOWORLD_PROTOCOL  {PRINTMSG PrintMsg;
}EFI_HELLOWORLD_PROTOCOL;//ShellCEntryLib call user interface ShellAppMain
EFI_STATUS
EFIAPI
MyHelloWorldLocatePPIEntry(IN       EFI_PEI_FILE_HANDLE  FileHandle,IN CONST EFI_PEI_SERVICES     **PeiServices
)
{EFI_STATUS  Status = EFI_SUCCESS;EFI_HELLOWORLD_PROTOCOL *mHelloWorldPpi = NULL;DEBUG ((EFI_D_ERROR, "[MyHelloWorldPPI] MyHelloWorldLocatePPIEntry Locate PPI Start..\n"));
//定位PPI接口Status = PeiServicesLocatePpi (&gMyHelloWorldPEIGUID, //我们要查找的PPI的GUID0,NULL,(VOID **)&mHelloWorldPpi //指向PPI实例指针的指针);if (EFI_ERROR(Status)){DEBUG ((EFI_D_ERROR, "[MyHelloWorldPPI] MyHelloWorldLocatePPIEntry Locate PPI Fail..%r\n",Status));return Status;}mHelloWorldPpi->PrintMsg(L"2019 CSDN Locate PPI Hello World ...\n");DEBUG ((EFI_D_ERROR, "[MyHelloWorldPPI] MyHelloWorldLocatePPIEntry Locate PPI End..\n"));return Status;
}

4、编写LocatePPI的inf文件

编译整个包,并将生成的OVMF.fd文件加载到qemu中,其

注意这里的模块是PEIM,无法在qemu里运行其二进制文件

制作运行qemu的脚本,新建一个qemurun.bat文件,将以下内容保存在其中:

@echo off
chcp 65001
"E:\Program Files\qemu\qemu-system-x86_64.exe" -bios "E:\UEFIWorkSpace\Build\OvmfX64\DEBUG_VS2017\FV\OVMF.fd" -M pc -m 256 -cpu "qemu64" -boot order=dc -serial stdio -net none -usbdevice disk:HDD_BOOT.img

(注:根据自己的qemu的安装位置和fd文件的位置进行更改) 

然后运行命令:

qemurun.bat | findstr MyHelloWorldPPI

 其运行结果如下

E:\UEFIWorkSpace\edk2>qemurun.bat | findstr MyHelloWorldPPI
WARNING: Image format was not specified for 'HDD_BOOT.img' and probing guessed raw.Automatically detecting the format is dangerous for raw images, write operations on block 0 will be restricted.Specify the 'raw' format explicitly to remove the restrictions.
[MyHelloWorldPPI] MyHelloWorldInstallPPIEntry Start..
[MyHelloWorldPPI] MyHelloWorldInstallPPIEntry End..
[MyHelloWorldPPI] MyHelloWorldLocatePPIEntry Locate PPI Start..
[MyHelloWorldPPI] PrintHelloMsg : 2024 CSDN Locate PPI Hello World ...
[MyHelloWorldPPI] MyHelloWorldLocatePPIEntry Locate PPI End..

 结果总结:

MyHelloWorldInstallPPI模块通过PeiServicesInstallPpi提供了mPpiListCodePpi服务(通过 EFI_PEI_PPI_DESCRIPTOR来指定具体的实例),而MyHelloWorldLocatePPI模块通过PeiServicesLocatePpi获取了mPpiListCodePpi对象,通过访问对象的成员函数即可完成对PEIM驱动MyHelloWorldInstallPPI资源的访问。

(注意,要想在运行的时候输出以上DEBUG信息,需要在编译包的时候加上DEBUG_ON_SERIAL_PORT) 

参考博客:https://blog.csdn.net/xiaopangzi313/article/details/89602882

这篇关于UEFI开发——编写一个简单的PPI的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Spring Boot + MyBatis Plus 高效开发实战从入门到进阶优化(推荐)

《SpringBoot+MyBatisPlus高效开发实战从入门到进阶优化(推荐)》本文将详细介绍SpringBoot+MyBatisPlus的完整开发流程,并深入剖析分页查询、批量操作、动... 目录Spring Boot + MyBATis Plus 高效开发实战:从入门到进阶优化1. MyBatis

Python基于wxPython和FFmpeg开发一个视频标签工具

《Python基于wxPython和FFmpeg开发一个视频标签工具》在当今数字媒体时代,视频内容的管理和标记变得越来越重要,无论是研究人员需要对实验视频进行时间点标记,还是个人用户希望对家庭视频进行... 目录引言1. 应用概述2. 技术栈分析2.1 核心库和模块2.2 wxpython作为GUI选择的优

Mysql表的简单操作(基本技能)

《Mysql表的简单操作(基本技能)》在数据库中,表的操作主要包括表的创建、查看、修改、删除等,了解如何操作这些表是数据库管理和开发的基本技能,本文给大家介绍Mysql表的简单操作,感兴趣的朋友一起看... 目录3.1 创建表 3.2 查看表结构3.3 修改表3.4 实践案例:修改表在数据库中,表的操作主要

分辨率三兄弟LPI、DPI 和 PPI有什么区别? 搞清分辨率的那些事儿

《分辨率三兄弟LPI、DPI和PPI有什么区别?搞清分辨率的那些事儿》分辨率这个东西,真的是让人又爱又恨,为了搞清楚它,我可是翻阅了不少资料,最后发现“小7的背包”的解释最让我茅塞顿开,于是,我... 在谈到分辨率时,我们经常会遇到三个相似的缩写:PPI、DPI 和 LPI。虽然它们看起来差不多,但实际应用

springboot简单集成Security配置的教程

《springboot简单集成Security配置的教程》:本文主要介绍springboot简单集成Security配置的教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录集成Security安全框架引入依赖编写配置类WebSecurityConfig(自定义资源权限规则

利用Python开发Markdown表格结构转换为Excel工具

《利用Python开发Markdown表格结构转换为Excel工具》在数据管理和文档编写过程中,我们经常使用Markdown来记录表格数据,但它没有Excel使用方便,所以本文将使用Python编写一... 目录1.完整代码2. 项目概述3. 代码解析3.1 依赖库3.2 GUI 设计3.3 解析 Mark

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

如何使用Python实现一个简单的window任务管理器

《如何使用Python实现一个简单的window任务管理器》这篇文章主要为大家详细介绍了如何使用Python实现一个简单的window任务管理器,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起... 任务管理器效果图完整代码import tkinter as tkfrom tkinter i

C++中函数模板与类模板的简单使用及区别介绍

《C++中函数模板与类模板的简单使用及区别介绍》这篇文章介绍了C++中的模板机制,包括函数模板和类模板的概念、语法和实际应用,函数模板通过类型参数实现泛型操作,而类模板允许创建可处理多种数据类型的类,... 目录一、函数模板定义语法真实示例二、类模板三、关键区别四、注意事项 ‌在C++中,模板是实现泛型编程

基于Python开发批量提取Excel图片的小工具

《基于Python开发批量提取Excel图片的小工具》这篇文章主要为大家详细介绍了如何使用Python中的openpyxl库开发一个小工具,可以实现批量提取Excel图片,有需要的小伙伴可以参考一下... 目前有一个需求,就是批量读取当前目录下所有文件夹里的Excel文件,去获取出Excel文件中的图片,并