HOB(hand off block)简介

有些时候,我们需要将一些information从PEI阶段传递到DXE阶段,那么这个时候我们就需要hob当做桥梁来给我们进行传递了。总的来说,PEI生产HOB,DXE消费HOB,在这里需要注意HOB只有在PEI阶段是可修改的,在DXE阶段是只读的,每一个Block都有自己的GUID与Struct

运作模式

当PEI阶段的module需要传递信息给DXE时,会Call PEI Service去建立一个HOB的block,并加载在HOB的list上。

HOB使用的实例

在这里我是将上一章所使用的setup的item选项联系起来,在PEI阶段获取item的值,并建立HOB,将item的值进行保存,然后由DXE阶段获取。

具体需要看源码的请移步我的github的homework项目

HOB在PEI阶段

新建pei文件并获取item的值

在sdl文件中加入

TOKEN
    Name  = "HomeWorkPeim_INF_SUPPORT"
    Value  = "1"
    Help  = "Main switch to enable HomeWorkDxe support in Project"
    TokenType = Boolean
    Token = "HomeWork_SUPPORT" "=" "1"
End

INFComponent
    Name  = "HomeWorkPeim"
    File  = "Pei/HomeWorkPeim.inf"
    Package  = "Homework"
    ModuleTypes  = "PEIM"
    Token = "HomeWorkPeim_INF_SUPPORT" "=" "1"
End

这样就已经将PEI的inf文件定义好了,此时要做的是在module的目录中建立Pei目录并在其中新建HomeWorkPeim.c/HomeWorkPeim.inf
并在inf文件中

[Defines]
  INF_VERSION           = 0x00010015
  BASE_NAME             = HomeWorkPeim
  FILE_GUID             = d133c000-7911-4e7f-abff-71f14f3337a8
  MODULE_TYPE           = PEIM
  VERSION_STRING        = 1.0
  ENTRY_POINT           = HomeWorkPeimEntryPoint

定义这个PEI文件的name 与入口参数,还有一些guid,依赖等,这个在后续专门出博客来讲解inf的一些定义,这里主要讲hob,我就不描述了。

注意:这里的入口函数一定要与C文件对应

同时你需要在module建立一个homework.h文件用于建立hob所需要的结构体与GUID

#define HOMEWORK_HOB_GUID \
{0x6ff0434a, 0x9598, 0x42cc, 0xb7, 0x38, 0xad, 0x17, 0x55, 0x48, 0x42, 0x9c}
typedef struct {
  EFI_HOB_GUID_TYPE    EfiHobGuidType; 
  UINT8    homeworkdata;  
} HOMEWORK_HOB;

如前面所说,hob就是通过guid来获取相关的信息,那么这个GUID就必须可以让PEI阶段的文件抓取的到,有可以让DXE文件抓取的到。

首先要做的是讲setup的值获取到,此时你需要用到的是gEfiPeiReadOnlyVariable2PpiGuid这个PPI。至于什么是PPI请移步我前面发布的一些博客

EFI_GUID                          SetupGuid = SETUP_GUID;
    EFI_PEI_READ_ONLY_VARIABLE2_PPI   *ReadOnlyVariable;
    UINTN                              VariableSize;
    SETUP_DATA                         SetupData;
 Status = PeiServicesLocatePpi(&gEfiPeiReadOnlyVariable2PpiGuid, 0, NULL, &ReadOnlyVariable);
  //  ASSERT_EFI_ERROR(Status);
    VariableSize = sizeof(SETUP_DATA);
    Status = ReadOnlyVariable->GetVariable(ReadOnlyVariable, L"Setup", &SetupGuid, NULL, &VariableSize, &SetupData);
 
    if (EFI_ERROR(Status)) {
        return Status;
    }

通过ReadOnlyVariable->GetVariable方法来获取setup的所有值。

创建一个hob

HOMEWORK_HOB                      *homeworkhob = NULL;
    EFI_GUID                           homeworkHobGuid = HOMEWORK_HOB_GUID;

Status = (*PeiServices)->CreateHob (
                               PeiServices,
                               EFI_HOB_TYPE_GUID_EXTENSION,
                               sizeof(HOMEWORK_HOB),
                               &homeworkhob);
    if (EFI_ERROR(Status)) {
        return Status;
    }

仔细看这些code你会发现用到了h文件中定义的GUID与结构体,当然这边只是做了一个简单的数据类型,其实你可以做得更加多的东西。

讲取到的值放入hob中

homeworkhob->EfiHobGuidType.Name=homeworkHobGuid;
    homeworkhob->homeworkdata=SetupData.HomeWorkoption;

这里的HomeWorkoption就是我们上一文所加的新选项

HOB在Dxe阶段

新建inf文件

同样的道理需要新建inf文件,加入相关的依赖项

[Defines]
  INF_VERSION           = 0x00010015
  BASE_NAME             = HomeWorkDxe
  FILE_GUID             = 43a12ef5-0372-4119-9f0f-000a8c104c62
  MODULE_TYPE           = DXE_DRIVER
  VERSION_STRING        = 1.0
  ENTRY_POINT           = HomeWorkDxeEntryPoint

新建dxe的 c文件

EFI_GUID                GuidHob = HOB_LIST_GUID;
    EFI_STATUS              Status;
    HOMEWORK_HOB *Homeworkhob=NULL;
    EFI_GUID    HomeworkHobGuid=HOMEWORK_HOB_GUID;
    VOID       *pHobList = NULL;
pHobList = GetEfiConfigurationTable(SystemTable, &GuidHob);
    if (!pHobList) return EFI_NOT_READY;
    
    Homeworkhob = (HOMEWORK_HOB*)pHobList;
    while (!EFI_ERROR(Status = FindNextHobByType(EFI_HOB_TYPE_GUID_EXTENSION, &Homeworkhob)))
    {
            if (guidcmp(&(*Homeworkhob).EfiHobGuidType.Name, &HomeworkHobGuid) == 0)
                break;
    }
       if (EFI_ERROR(Status)) 
      {
          return EFI_NOT_FOUND;
       }else{
             OEM_TRACE("data=%d\n",Homeworkhob->homeworkdata);
       }

在dxe阶段,你需要通过GetEfiConfigurationTable拿到hob list的hander,也相当与一个链表的头指针,然后通过FindNextHobByType去遍历这个链表,通过比较GUID进行锁定,并把值写出来。

在这里我提醒一下,所谓的OEM_TRACE是我们自主写的一个module用于串口debug的。如果有可能,我会在纯净的UEFI中再做这个module,让大家可以进行自行编译,而不是现在的只能看博客。