BIOS实战之事件的使用

2024-06-07 05:18
文章标签 实战 使用 事件 bios

本文主要是介绍BIOS实战之事件的使用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这里主要介绍两个示例,第一个是在之前Phoenix代码中看到的,第二个是国产平台中遇到的,首先看第一个。

问题:开机显示logo后进入shell或者pxe,logo依然存在,进入pxe好解决,咱们在进入之前进行清屏就行,那么进入shell后的问题该怎么去解决呢

这里我们就可以用到一个事件(为什么不在进入shell前进行清屏?这个我也没实践过,不晓得当初写这个解决方案的是否验证过了,因为相比较事件,直接清屏更加简单有效),看过UEFI原理与编程实践中事件的简单使用,这里新增一个知识点RegisterProtocolNotify的使用

STATIC VOID  *mShellEnvProtocolCallbackReg = NULL;
STATIC EFI_GUID  gEfiShellEnvironment2Guid = {0x47c7b221, 0xc42a, 0x11d2, {0x8e, 0x57, 0x00, 0xa0, 0xc9, 0x69, 0x72, 0x3b}};
VOID
EFIAPI
ShellEnvProtocolCallback (IN EFI_EVENT                          Event,IN VOID                               *Context)
{gST->ConOut->ClearScreen(gST->ConOut);
}
EFI_EVENT                         ShellImageEvent;if (mShellEnvProtocolCallbackReg == NULL) {Status = gBS->CreateEvent (EVT_NOTIFY_SIGNAL,TPL_CALLBACK,ShellEnvProtocolCallback,NULL,&ShellImageEvent);if (!EFI_ERROR (Status)) {Status = gBS->RegisterProtocolNotify (&gEfiShellEnvironment2Guid,ShellImageEvent,&mShellEnvProtocolCallbackReg);}}

上述代码的逻辑:新增了一个事件,这个事件的响应函数就是一个清屏的功能,那么这个事件怎么进入呢,需要先明白RegisterProtocolNotify函数的作用,它就是判断某个protocol在安装的那一刻,然后运行某个事件,也就是说在shell安装gEfiShellEnvironment2Guid后这个事件就起作用了,进行了清屏的操作,顺利解决上面提到的问题

问题二:当UEFI设置显示分辨率为1024x768的时候,麒麟系统通过VGA进行显示,会导致分屏

有的时候,分辨率高一些,当然是最好的,比如开机显示logo的时候,看着模糊的logo图片,总让人怀疑机器的性能,分辨率高了,logo的分辨率同样跟着提升,这对用户来说绝对是一个好的体验,但是好的体验总不可能以牺牲正常使用为代价,所以这个问题必须解决,那么怎么解决呢,就是通过事件去解决。

Status = gBS->CreateEventEx (EVT_NOTIFY_SIGNAL,TPL_NOTIFY,ExitBootService,NULL,&gEfiEventExitBootServicesGuid,&Event);
VOID
ExitBootService (EFI_EVENT  Event,VOID       *Context)
{EFI_STATUS                      Status;UINT16                          OptionNumber;UINTN                           VarSize;CHAR16                          OptionName[9] = L"Boot";EFI_BOOT_MANAGER_LOAD_OPTION    BootOption;EFI_SIMPLE_FILE_SYSTEM_PROTOCOL *Volume = NULL;EFI_TPL  Tpl;Tpl = gBS->RaiseTPL (TPL_HIGH_LEVEL);gBS->RestoreTPL (TPL_APPLICATION);VarSize = sizeof (UINT16);Status = gRT->GetVariable (L"BootCurrent",&gEfiGlobalVariableGuid,NULL,&VarSize,&OptionNumber);if(EFI_ERROR(Status)) {return;}UnicodeSPrint (OptionName, sizeof (OptionName), L"%s%04x",OptionName, OptionNumber);Status = EfiBootManagerVariableToLoadOption (OptionName, &BootOption);if(EFI_ERROR(Status)) {return;}Volume = GetFsVolume (gBS, BootOption.FilePath, L"\\grub\\grub_ba.efi");if (Volume != NULL || !IsUefiAhciHddDp (gBS, BootOption.FilePath, NULL)) {// patch for kylin OSResetGopDriver ();}gBS->RaiseTPL (Tpl);
}
void
ResetGopDriver ()
{EFI_HANDLE          *HandleBuffer;UINTN               Index;UINTN               HandleCount;EFI_STATUS          Status;EFI_PCI_IO_PROTOCOL *PciIo;UINT32              Data32;BOOLEAN             IsAmdVga = FALSE;DEBUG ((EFI_D_INFO,"%a()\n",__FUNCTION__));Status = gBS->LocateHandleBuffer (ByProtocol,&gEfiPciIoProtocolGuid,NULL,&HandleCount,&HandleBuffer);if (!EFI_ERROR (Status)) {for (Index = 0; Index < HandleCount; Index++) {Status = gBS->HandleProtocol (HandleBuffer[Index],&gEfiPciIoProtocolGuid,(VOID **)&PciIo);if (!EFI_ERROR (Status)) {Status = PciIo->Pci.Read (PciIo, EfiPciIoWidthUint32, 8, 1, &Data32);Data32 = Data32 & 0xFFFFFF00;if (Data32 == 0x3000000 || Data32 == 0x00010000) {PciIo->Pci.Read (PciIo, EfiPciIoWidthUint32, 0, 1, &Data32);Data32 = Data32 & 0xFFff;if (Data32 == 0x1002) {IsAmdVga = TRUE;break;}}}} }if (IsAmdVga == FALSE) {if (HandleBuffer != NULL) {FreePool (HandleBuffer);}return;}PcdSet32 (PcdVideoHorizontalResolution, 640);PcdSet32 (PcdVideoVerticalResolution, 480);gBS->DisconnectController (HandleBuffer[Index], NULL, NULL);gBS->ConnectController (HandleBuffer[Index], NULL, NULL, TRUE);if (HandleBuffer != NULL) {FreePool (HandleBuffer);}}

这篇关于BIOS实战之事件的使用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Linux使用fdisk进行磁盘的相关操作

《Linux使用fdisk进行磁盘的相关操作》fdisk命令是Linux中用于管理磁盘分区的强大文本实用程序,这篇文章主要为大家详细介绍了如何使用fdisk进行磁盘的相关操作,需要的可以了解下... 目录简介基本语法示例用法列出所有分区查看指定磁盘的区分管理指定的磁盘进入交互式模式创建一个新的分区删除一个存

C#使用HttpClient进行Post请求出现超时问题的解决及优化

《C#使用HttpClient进行Post请求出现超时问题的解决及优化》最近我的控制台程序发现有时候总是出现请求超时等问题,通常好几分钟最多只有3-4个请求,在使用apipost发现并发10个5分钟也... 目录优化结论单例HttpClient连接池耗尽和并发并发异步最终优化后优化结论我直接上优化结论吧,

Golang操作DuckDB实战案例分享

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

SpringBoot使用Apache Tika检测敏感信息

《SpringBoot使用ApacheTika检测敏感信息》ApacheTika是一个功能强大的内容分析工具,它能够从多种文件格式中提取文本、元数据以及其他结构化信息,下面我们来看看如何使用Ap... 目录Tika 主要特性1. 多格式支持2. 自动文件类型检测3. 文本和元数据提取4. 支持 OCR(光学

JAVA系统中Spring Boot应用程序的配置文件application.yml使用详解

《JAVA系统中SpringBoot应用程序的配置文件application.yml使用详解》:本文主要介绍JAVA系统中SpringBoot应用程序的配置文件application.yml的... 目录文件路径文件内容解释1. Server 配置2. Spring 配置3. Logging 配置4. Ma

Linux使用dd命令来复制和转换数据的操作方法

《Linux使用dd命令来复制和转换数据的操作方法》Linux中的dd命令是一个功能强大的数据复制和转换实用程序,它以较低级别运行,通常用于创建可启动的USB驱动器、克隆磁盘和生成随机数据等任务,本文... 目录简介功能和能力语法常用选项示例用法基础用法创建可启动www.chinasem.cn的 USB 驱动

C#使用yield关键字实现提升迭代性能与效率

《C#使用yield关键字实现提升迭代性能与效率》yield关键字在C#中简化了数据迭代的方式,实现了按需生成数据,自动维护迭代状态,本文主要来聊聊如何使用yield关键字实现提升迭代性能与效率,感兴... 目录前言传统迭代和yield迭代方式对比yield延迟加载按需获取数据yield break显式示迭

使用SQL语言查询多个Excel表格的操作方法

《使用SQL语言查询多个Excel表格的操作方法》本文介绍了如何使用SQL语言查询多个Excel表格,通过将所有Excel表格放入一个.xlsx文件中,并使用pandas和pandasql库进行读取和... 目录如何用SQL语言查询多个Excel表格如何使用sql查询excel内容1. 简介2. 实现思路3

java脚本使用不同版本jdk的说明介绍

《java脚本使用不同版本jdk的说明介绍》本文介绍了在Java中执行JavaScript脚本的几种方式,包括使用ScriptEngine、Nashorn和GraalVM,ScriptEngine适用... 目录Java脚本使用不同版本jdk的说明1.使用ScriptEngine执行javascript2.

c# checked和unchecked关键字的使用

《c#checked和unchecked关键字的使用》C#中的checked关键字用于启用整数运算的溢出检查,可以捕获并抛出System.OverflowException异常,而unchecked... 目录在 C# 中,checked 关键字用于启用整数运算的溢出检查。默认情况下,C# 的整数运算不会自