UEFI学习笔记(十):系统表与ACPI表的遍历

一、概述

在 UEFI 系统表中,有几个关键的表用于提供系统信息、服务和硬件抽象。这些表可以通过 EFI_SYSTEM_TABLE 访问,常见的 UEFI 系统表如下:

1、EFI_SYSTEM_TABLE (系统表)

EFI_SYSTEM_TABLE 是一个指针,包含多个服务和系统信息。

包含以下几个重要表:

2、EFI_BOOT_SERVICES (引导服务表):

提供在引导期间可用的功能(如内存分配、事件管理、协议处理等)。

3、EFI_RUNTIME_SERVICES (运行时服务表):

提供操作系统加载后仍可使用的服务(如变量管理、时间功能和固件更新等)。

4、EFI_CONFIGURATION_TABLE (配置表):

包含各种硬件信息和平台信息,包含 ACPI、SMBIOS、HOB(Hand-off Block)等表。

二、遍历ACPI表

c 复制代码
VOID ListAllAcpiTables(VOID)
{
  UINTN     i, j, EntryCount;
  CHAR8     strBuff[20];
  UINT64    *EntryPtr;
  EFI_GUID  AcpiTableGuid  = ACPI_TABLE_GUID;
  EFI_GUID  Acpi2TableGuid = EFI_ACPI_TABLE_GUID;
  EFI_CONFIGURATION_TABLE   *configTab = NULL;  
  EFI_ACPI_DESCRIPTION_HEADER           *XSDT, *Entry;
  EFI_ACPI_5_0_ROOT_SYSTEM_DESCRIPTION_POINTER  *Root;

  Print(L"List All ACPI Tables:\n");
  configTab = gST->ConfigurationTable;
  
  for (i = 0; i < gST->NumberOfTableEntries; i++) {   
    // Step1: 查找 ACPI 表
    if ((CompareGuid(&configTab->VendorGuid, &AcpiTableGuid) == 0) ||
        (CompareGuid(&configTab->VendorGuid, &Acpi2TableGuid) == 0)) { 
      Print(L"Found ACPI table: %g\n", &configTab->VendorGuid); 
      Root = configTab->VendorTable;
      Print(L"ROOT SYSTEM DESCRIPTION @[0x%p]\n", Root);

      // Step2: 检查修订版本(只接受修订版本 >= 2)
      if (Root->Revision >= EFI_ACPI_5_0_ROOT_SYSTEM_DESCRIPTION_POINTER_REVISION) {
        // Step3: 获取 XSDT 表地址
        XSDT = (EFI_ACPI_DESCRIPTION_HEADER *)(UINTN) Root->XsdtAddress;
        EntryCount = (XSDT->Length - sizeof(EFI_ACPI_DESCRIPTION_HEADER)) / sizeof(UINT64);

        Print(L"XSDT address = [0x%p]\n", XSDT);
        Print(L"XSDT-Length = 0x%x\n", XSDT->Length);
        Print(L"Number of ACPI Tables = %d\n", EntryCount);

        // Step4: 遍历每个 ACPI 表并输出信息
        EntryPtr = (UINT64 *)(XSDT + 1);
        for (j = 0; j < EntryCount; j++, EntryPtr++) {
          Entry = (EFI_ACPI_DESCRIPTION_HEADER *)((UINTN)(*EntryPtr));

          // 输出每个表的签名、长度和校验和等基本信息
          ZeroMem(strBuff, sizeof(strBuff));
          CopyMem(strBuff, &(Entry->Signature), sizeof(UINT32));
          Print(L"Table Signature: %a\n", strBuff);
          Print(L"Table Address: 0x%p\n", Entry);
          Print(L"Table Length: 0x%x\n", Entry->Length);
          Print(L"Table Checksum: 0x%x\n", Entry->Checksum);

        }
      }
    }
    configTab++;
  }
}

整个函数 ListAllAcpiTables 的逻辑是:

1)遍历系统配置表。

  1. 查找 ACPI 表,并验证 RSDP 的版本。

  2. 从 RSDP 获取 XSDT 表的地址,并计算其中的 ACPI 表数量。

  3. 遍历所有 ACPI 表,打印它们的基本信息

相关推荐
非凡ghost21 分钟前
AMS PhotoMaster:全方位提升你的照片编辑体验
windows·学习·信息可视化·软件需求
云间月13142 小时前
飞算JavaAI智慧教育场景实践:从个性化学习到教学管理的全链路技术革新
学习·飞算javaai挑战赛
weixin_456904274 小时前
一文讲清楚Pytorch 张量、链式求导、正向传播、反向求导、计算图等基础知识
人工智能·pytorch·学习
Blossom.1184 小时前
把 AI 推理塞进「 8 位 MCU 」——0.5 KB RAM 跑通关键词唤醒的魔幻之旅
人工智能·笔记·单片机·嵌入式硬件·深度学习·机器学习·搜索引擎
草莓熊Lotso5 小时前
《吃透 C++ 类和对象(中):const 成员函数与取地址运算符重载解析》
c语言·开发语言·c++·笔记·其他
Python私教5 小时前
从“Hello World”到“高并发中间件”:Go 语言 2025 系统学习路线图
学习·中间件·golang
玖別ԅ(¯﹃¯ԅ)6 小时前
PID学习笔记6-倒立摆的实现
笔记·stm32·单片机
想学全栈的菜鸟阿董7 小时前
Django5个人笔记
笔记
Brookty7 小时前
【Java学习】锁、线程死锁、线程安全2
java·开发语言·学习·java-ee