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

相关文章

MySQL 多列 IN 查询之语法、性能与实战技巧(最新整理)

《MySQL多列IN查询之语法、性能与实战技巧(最新整理)》本文详解MySQL多列IN查询,对比传统OR写法,强调其简洁高效,适合批量匹配复合键,通过联合索引、分批次优化提升性能,兼容多种数据库... 目录一、基础语法:多列 IN 的两种写法1. 直接值列表2. 子查询二、对比传统 OR 的写法三、性能分析

Java中读取YAML文件配置信息常见问题及解决方法

《Java中读取YAML文件配置信息常见问题及解决方法》:本文主要介绍Java中读取YAML文件配置信息常见问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录1 使用Spring Boot的@ConfigurationProperties2. 使用@Valu

Python办公自动化实战之打造智能邮件发送工具

《Python办公自动化实战之打造智能邮件发送工具》在数字化办公场景中,邮件自动化是提升工作效率的关键技能,本文将演示如何使用Python的smtplib和email库构建一个支持图文混排,多附件,多... 目录前言一、基础配置:搭建邮件发送框架1.1 邮箱服务准备1.2 核心库导入1.3 基础发送函数二、

PowerShell中15个提升运维效率关键命令实战指南

《PowerShell中15个提升运维效率关键命令实战指南》作为网络安全专业人员的必备技能,PowerShell在系统管理、日志分析、威胁检测和自动化响应方面展现出强大能力,下面我们就来看看15个提升... 目录一、PowerShell在网络安全中的战略价值二、网络安全关键场景命令实战1. 系统安全基线核查

Python使用OpenCV实现获取视频时长的小工具

《Python使用OpenCV实现获取视频时长的小工具》在处理视频数据时,获取视频的时长是一项常见且基础的需求,本文将详细介绍如何使用Python和OpenCV获取视频时长,并对每一行代码进行深入解析... 目录一、代码实现二、代码解析1. 导入 OpenCV 库2. 定义获取视频时长的函数3. 打开视频文

从原理到实战深入理解Java 断言assert

《从原理到实战深入理解Java断言assert》本文深入解析Java断言机制,涵盖语法、工作原理、启用方式及与异常的区别,推荐用于开发阶段的条件检查与状态验证,并强调生产环境应使用参数验证工具类替代... 目录深入理解 Java 断言(assert):从原理到实战引言:为什么需要断言?一、断言基础1.1 语

Java MQTT实战应用

《JavaMQTT实战应用》本文详解MQTT协议,涵盖其发布/订阅机制、低功耗高效特性、三种服务质量等级(QoS0/1/2),以及客户端、代理、主题的核心概念,最后提供Linux部署教程、Sprin... 目录一、MQTT协议二、MQTT优点三、三种服务质量等级四、客户端、代理、主题1. 客户端(Clien

在Spring Boot中集成RabbitMQ的实战记录

《在SpringBoot中集成RabbitMQ的实战记录》本文介绍SpringBoot集成RabbitMQ的步骤,涵盖配置连接、消息发送与接收,并对比两种定义Exchange与队列的方式:手动声明(... 目录前言准备工作1. 安装 RabbitMQ2. 消息发送者(Producer)配置1. 创建 Spr

MySQL 获取字符串长度及注意事项

《MySQL获取字符串长度及注意事项》本文通过实例代码给大家介绍MySQL获取字符串长度及注意事项,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql 获取字符串长度详解 核心长度函数对比⚠️ 六大关键注意事项1. 字符编码决定字节长度2

深度解析Spring Boot拦截器Interceptor与过滤器Filter的区别与实战指南

《深度解析SpringBoot拦截器Interceptor与过滤器Filter的区别与实战指南》本文深度解析SpringBoot中拦截器与过滤器的区别,涵盖执行顺序、依赖关系、异常处理等核心差异,并... 目录Spring Boot拦截器(Interceptor)与过滤器(Filter)深度解析:区别、实现