BIOS实战之读取文件---获取文件路径+最终读取

2024-06-07 05:18

本文主要是介绍BIOS实战之读取文件---获取文件路径+最终读取,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

接上一节,获取文件路径实际上也是获取当前这个APP的路径,怎么获取这个APP的路径,这里需要用到一个EFI_LOADED_IMAGE_PROTOCOL,原理如下:

根据当前 Application 的 ImageHandle 取得对应的 Handle。
使用 HandleProtcol 直接打开加载在Application 上面的 EFI_LOADED_IMAGE_PROTOCOL。
在这个 Protocol上有 FilePath,就是当前 Image 的 Device Path Protocol

通过引用看下面的代码一目了然:

EFI_LOADED_IMAGE_PROTOCOL *LoadedImage;Status = gBS->HandleProtocol(gImageHandle, &gEfiLoadedImageProtocolGuid,(void **)&LoadedImage);if(EFI_ERROR(Status)) {Print(L"Application path error!\n");return Status;}EFI_DEVICE_PATH_PROTOCOL  *ptFilePath;ptFilePath  = LoadedImage->FilePath;if(!ptFilePath) {Print(L"Unkown media!\n");return Status;}

ok,又到了Device Path Protocol的环节,在设备路径,找到我们放置APP的设备:

while(!IsDevicePathEnd(ptFilePath)){if(DevicePathType(ptFilePath)==MEDIA_DEVICE_PATH && DevicePathSubType(ptFilePath)==MEDIA_FILEPATH_DP){break;}ptFilePath = NextDevicePathNode(ptFilePath);    }TmpPath = ((FILEPATH_DEVICE_PATH*)ptFilePath)->PathName;

那么这个TmpPath就是app的路径了,如\xxx\xxx\xxx\xx.efi,但是我们只需要路径,不需要app的名字,那么咱们就可以通过判断‘\’来将这个app名字去掉,那么最终的函数如下:

EFI_STATUS
GetAPPFolderPathStr(CHAR16 **Path)
{EFI_STATUS                Status;EFI_DEVICE_PATH_PROTOCOL  *ptFilePath;CHAR16                    *PathName, *TmpPath;UINTN                     StrSize;Status = EFI_SUCCESS;  PathName    = NULL;EFI_LOADED_IMAGE_PROTOCOL *LoadedImage;Status = gBS->HandleProtocol(gImageHandle, &gEfiLoadedImageProtocolGuid,(void **)&LoadedImage);if(EFI_ERROR(Status)) {Print(L"Application path error!\n");return Status;}ptFilePath  = LoadedImage->FilePath;if(!ptFilePath) {Print(L"Unkown media!\n");return Status;}while(!IsDevicePathEnd(ptFilePath)){if(DevicePathType(ptFilePath)==MEDIA_DEVICE_PATH && DevicePathSubType(ptFilePath)==MEDIA_FILEPATH_DP){break;}ptFilePath = NextDevicePathNode(ptFilePath);    }TmpPath = ((FILEPATH_DEVICE_PATH*)ptFilePath)->PathName;while ((TmpPath = StrStr (TmpPath+1, L"\\")) != NULL) {PathName = TmpPath;}if(PathName != NULL){StrSize = (PathName - ((FILEPATH_DEVICE_PATH*)ptFilePath)->PathName)*sizeof(CHAR16);}else{StrSize = 2;}PathName = (CHAR16 *)AllocateZeroPool(StrSize + 2);if(PathName==NULL) {Status = EFI_OUT_OF_RESOURCES;goto ProcExit; }TmpPath = ((FILEPATH_DEVICE_PATH*)ptFilePath)->PathName;CopyMem (PathName, TmpPath, StrSize);if(PathName[0] != L'\\'){Status = EFI_UNSUPPORTED;goto ProcExit;}*Path = PathName;ProcExit:if(EFI_ERROR(Status) && PathName != NULL){FreePool(PathName);PathName = NULL; }return Status;
}

获取这个路径之后咱们还需要将文件的名字加上,这个才是最终需要的:

EFI_STATUS
GetFullPathBaseOnAPPFolder (IN       CHAR16 *FileName,IN OUT   CHAR16 **FullPath)
{EFI_STATUS  Status;CHAR16      *AppFolder;UINTN       FullPathSize;CHAR16      *AppFullPath;CHAR16      Slash = L'\\';Status      = EFI_SUCCESS;AppFolder   = NULL;AppFullPath = NULL;if(FileName == NULL || FullPath == NULL){Status = EFI_INVALID_PARAMETER;goto ProcExit;}GetAPPFolderPathStr(&AppFolder);//上个函数的返回路径if (AppFolder == NULL) {Status = EFI_OUT_OF_RESOURCES;goto ProcExit;}FullPathSize = strsize(AppFolder) + strsize(FileName);AppFullPath  = AllocateZeroPool(FullPathSize);if(AppFullPath == NULL){Status = EFI_OUT_OF_RESOURCES;goto ProcExit;}if (strsize(AppFolder) > 4) { // '\' and '\0'CopyMem (AppFullPath, AppFolder, strsize(AppFolder) - 2); // skip '\0'CopyMem (AppFullPath + strsize(AppFolder)/2 - 1, &Slash, 2);CopyMem (AppFullPath + strsize(AppFolder)/2, FileName, strsize(FileName));} else {CopyMem (AppFullPath, FileName, strsize(FileName));}*FullPath = AppFullPath;

这个AppFullPath的内容就是xxx\xx\xxx.bmp

最终的读取函数,我们就可以把上述的函数串联起来:

EFI_STATUS
ReadFile (IN     CHAR16  *FileName,IN OUT UINT8   **FileBuffer,OUT    UINTN   *FileSize
)
{EFI_STATUS  Status;CHAR16      *FullPath;// get File full pathStatus = GetFullPathBaseOnAPPFolder(FileName, &FullPath);if(EFI_ERROR(Status)) {return Status;}Status = ReadFileInFS(FullPath, FileBuffer, FileSize);FreePool (FullPath);return Status;
}

这篇关于BIOS实战之读取文件---获取文件路径+最终读取的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Golang操作DuckDB实战案例分享

《Golang操作DuckDB实战案例分享》DuckDB是一个嵌入式SQL数据库引擎,它与众所周知的SQLite非常相似,但它是为olap风格的工作负载设计的,DuckDB支持各种数据类型和SQL特性... 目录DuckDB的主要优点环境准备初始化表和数据查询单行或多行错误处理和事务完整代码最后总结Duck

Python MySQL如何通过Binlog获取变更记录恢复数据

《PythonMySQL如何通过Binlog获取变更记录恢复数据》本文介绍了如何使用Python和pymysqlreplication库通过MySQL的二进制日志(Binlog)获取数据库的变更记录... 目录python mysql通过Binlog获取变更记录恢复数据1.安装pymysqlreplicat

C#实现获取电脑中的端口号和硬件信息

《C#实现获取电脑中的端口号和硬件信息》这篇文章主要为大家详细介绍了C#实现获取电脑中的端口号和硬件信息的相关方法,文中的示例代码讲解详细,有需要的小伙伴可以参考一下... 我们经常在使用一个串口软件的时候,发现软件中的端口号并不是普通的COM1,而是带有硬件信息的。那么如果我们使用C#编写软件时候,如

C#实现WinForm控件焦点的获取与失去

《C#实现WinForm控件焦点的获取与失去》在一个数据输入表单中,当用户从一个文本框切换到另一个文本框时,需要准确地判断焦点的转移,以便进行数据验证、提示信息显示等操作,本文将探讨Winform控件... 目录前言获取焦点改变TabIndex属性值调用Focus方法失去焦点总结最后前言在一个数据输入表单

通过C#获取PDF中指定文本或所有文本的字体信息

《通过C#获取PDF中指定文本或所有文本的字体信息》在设计和出版行业中,字体的选择和使用对最终作品的质量有着重要影响,然而,有时我们可能会遇到包含未知字体的PDF文件,这使得我们无法准确地复制或修改文... 目录引言C# 获取PDF中指定文本的字体信息C# 获取PDF文档中用到的所有字体信息引言在设计和出

Python中的随机森林算法与实战

《Python中的随机森林算法与实战》本文详细介绍了随机森林算法,包括其原理、实现步骤、分类和回归案例,并讨论了其优点和缺点,通过面向对象编程实现了一个简单的随机森林模型,并应用于鸢尾花分类和波士顿房... 目录1、随机森林算法概述2、随机森林的原理3、实现步骤4、分类案例:使用随机森林预测鸢尾花品种4.1

python中os.stat().st_size、os.path.getsize()获取文件大小

《python中os.stat().st_size、os.path.getsize()获取文件大小》本文介绍了使用os.stat()和os.path.getsize()函数获取文件大小,文中通过示例代... 目录一、os.stat().st_size二、os.path.getsize()三、函数封装一、os

四种简单方法 轻松进入电脑主板 BIOS 或 UEFI 固件设置

《四种简单方法轻松进入电脑主板BIOS或UEFI固件设置》设置BIOS/UEFI是计算机维护和管理中的一项重要任务,它允许用户配置计算机的启动选项、硬件设置和其他关键参数,该怎么进入呢?下面... 随着计算机技术的发展,大多数主流 PC 和笔记本已经从传统 BIOS 转向了 UEFI 固件。很多时候,我们也

Java读取InfluxDB数据库的方法详解

《Java读取InfluxDB数据库的方法详解》本文介绍基于Java语言,读取InfluxDB数据库的方法,包括读取InfluxDB的所有数据库,以及指定数据库中的measurement、field、... 首先,创建一个Java项目,用于撰写代码。接下来,配置所需要的依赖;这里我们就选择可用于与Infl

C#读取本地网络配置信息全攻略分享

《C#读取本地网络配置信息全攻略分享》在当今数字化时代,网络已深度融入我们生活与工作的方方面面,对于软件开发而言,掌握本地计算机的网络配置信息显得尤为关键,而在C#编程的世界里,我们又该如何巧妙地读取... 目录一、引言二、C# 读取本地网络配置信息的基础准备2.1 引入关键命名空间2.2 理解核心类与方法