UEFI PEI CORE的二个阶段与PeiCore函数的3次调用流程简介

本文介绍了UEFI PEI CORE的两个阶段(PreMemory和PostMemory)以及PeiCore函数的三次调用流程。在PreMemory阶段,PeiCore首次运行在temp RAM,接着在主内存初始化后第二次运行,进行栈空间分配、PEI_CORE_INSTANCE实例初始化等操作。在PostMemory阶段,PeiCore第三次运行,继续初始化和服务调度,以完成硬件设置和PEI阶段的任务。

PeiCore的3次执行流程简介

         PeiCore()函数作为PEI阶段最重要的组成部分,在整个PEI生命周期中分为二个阶段,分别是主内存初始化之前(PreMemory)和内存初始化完成之后(PostMemory),在这二个阶段中PeiCore()函数会被调用3次,简化的调用流程如下。示例描述了二个阶段内存以及堆栈的分布情况。蓝色的地址空间被映射到BIOS ROM,黄色的被映射栈空间,紫色被映射到堆空间,高地址是PreMemory阶段,低地址是PostMemory阶段。

  1. PeiCore()                                 //第1次在temp ram运行
  2. PeiDispatcher()                       //查找并调度PEIM
  3. PeiCheckAndSwitchStack()    //切换堆栈
  4. PeiCore()                                 //第2次在主内存运行
  5. ShadowedPeiCore()               //加载并重映射到主内存
  6. PeiCore()                                //第3次在主内存运行

一.PeiCore()第1次执行流程

         第1次调用PeiCore (SecCoreDataPtr, PpiList, NULL);此时PeiCore存储于flash,主内存未初始化。

VOID

EFIAPI

PeiCore (

  IN CONST EFI_SEC_PEI_HAND_OFF    *SecCoreDataPtr,

  IN CONST EFI_PEI_PPI_DESCRIPTOR  *PpiList,

  IN VOID                          *Data

  )
  1. 在栈空间分配内存,用来临时保存PEI_CORE_INSTANCE实例(PrivateData)、EFI_SEC_PEI_HAND_OFF(NewSecCoreData)及其他数据。
   PEI_CORE_INSTANCE               PrivateData;

        EFI_SEC_PEI_HAND_OFF            NewSecCoreData;

          OldCoreData = (PEI_CORE_INSTANCE *)Data;//等于NULL

        SecCoreData = (EFI_SEC_PEI_HAND_OFF *)SecCoreDataPtr; //等于NULL
  1. 此时Data==NULL,PeiCore()是是第1次进入。
  2. 初始化PEI_CORE_INSTANCE(PrivateData)实例,复制EFI_PEI_SERVICES表到上述内存地址。由MdePkg\Library\BaseMemoryLibRepStr\Ia32\CopyMem.nasm实现拷贝。
CopyMem (&PrivateData.ServiceTableShadow, &gPs, sizeof (gPs));
  1. 重定位PeiCore()的私有数据地址,PrivateData.Ps 指向上述#3所述地址。
PrivateData.Ps = &PrivateData.ServiceTableShadow;
  1. 使用中断描述符表IDT来保存PEI service表指针Ps。
 SetPeiServicesTablePointer ((CONST EFI_PEI_SERVICES **)&PrivateData.Ps);
  1. 初始化PeiCore()用到的所有库,调用由工具自动生成AutoGen.c文件内的初始化函数。
ProcessLibraryConstructorList (NULL, (CONST EFI_PEI_SERVICES **)&PrivateData.Ps);
  1. 初始化调试及性能追踪服务,可以用来调试、追踪优化代码流程。
  2. 初始化PeiCore服务。
InitializeMemoryServices (&PrivateData, SecCoreData, OldCoreData);

InitializeSecurityServices (&PrivateData.Ps, OldCoreData);

InitializeDispatcherData (&PrivateData, OldCoreData, SecCoreData);

InitializeImageServices (&PrivateData, OldCoreData);
  1. 如果是第1次进入,且PpiList不为NULL时则可能安装SecCore提供的PPI
ProcessPpiListFromSec ((CONST EFI_PEI_SERVICES **)&PrivateData.Ps, PpiList);
  1. 完成了基本初始化之后,开始执行PEI dispatch,PeiDispatcher()之后的代码在第1次PeiCore()时不会执行到函数返回。PEI dispatch调度其他的PEIM完成基本的硬件初始化,比如查找并调用FSP提供的服务来初始化内存,芯片组等信息。
PeiDispatcher (SecCoreData, &PrivateData);

         VOID

         PeiCheckAndSwitchStack (

       IN CONST EFI_SEC_PEI_HAND_OFF         *SecCoreData,

       IN PEI_CORE_INSTANCE                  *Private

        )

         以上完成了第1次进入PeiCore的全部详细流程,#10步完成之后,主内存会被FSP初始化完成,然后再次进入PEI CORE,而不是函数返回,也就是进行第2次执行PeiCore (;;)函数,不同于第1次的是,第2次执行的时候第三个参数Data!=NULL,因此执行流程会稍有不同。

一.PeiCore()第2次执行流程

...

一.PeiCore()第3次执行流程

...

跟多关注公众号.固件C字营

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值