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 OS
    ResetGopDriver ();
  }
  
  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);
  }

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值