UEFI BIOS之SMBIOS

2024-09-05 08:12
文章标签 uefi bios smbios

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

SMBIO spec:

SMBIOS Specification (dmtf.org)

https://www.dmtf.org/sites/default/files/standards/documents/DSP0134_3.5.0.pdf

SMBIOS

SMBIOS(System Management BIOS)是一种标准化的系统信息数据结构,旨在提供有关计算机硬件配置、系统功能和管理信息的详细描述。SMBIOS是由DMI(Desktop Management Interface)规范化组织制定的,旨在帮助系统管理员和应用

程序开发人员获取有关计算机系统的信息。

简单列一下Spec 中常用的SMBIOS Type,不详细介绍 ,本文主要着重于SMBIOS的生成和解析

不同的Type存储不同的信息,这些信息是BIOS提供OS的一个表,OS的部分配置会参考Smbios。

SMBIOS信息如何生成存取和访问呢?

SystemTable中会存储一些Table ,其中就包括Smbios的EntryTable,顺便提一下还包括ACPI的Entry Table

如何生成;那就是BIOS中去Install相关的Table ,会申请一段内存,然后更新信息到Smbios的存储空间中

EFI_SYSTEM_TABLE 中包含 EFI_CONFIGURATION_TABLE            *ConfigurationTable;

通过以下方式就可以获取到Smbios Table的地址
EFI_STATUS
EFIAPI
EfiGetSystemConfigurationTable (IN  EFI_GUID  *TableGuid,OUT VOID      **Table)
{EFI_SYSTEM_TABLE  *SystemTable;UINTN             Index;ASSERT (TableGuid != NULL);ASSERT (Table != NULL);SystemTable = gST;*Table      = NULL;for (Index = 0; Index < SystemTable->NumberOfTableEntries; Index++) {if (CompareGuid (TableGuid, &(SystemTable->ConfigurationTable[Index].VendorGuid))) {*Table = SystemTable->ConfigurationTable[Index].VendorTable;return EFI_SUCCESS;}}return EFI_NOT_FOUND;
}

SMBIOS 3.0Table的Entry point 结构体:

typedef struct {

  UINT8     AnchorString[5];

  UINT8     EntryPointStructureChecksum;

  UINT8     EntryPointLength;

  UINT8     MajorVersion;

  UINT8     MinorVersion;

  UINT8     DocRev;

  UINT8     EntryPointRevision;

  UINT8     Reserved;

  UINT32    TableMaximumSize;

  UINT64    TableAddress;

} SMBIOS_TABLE_3_0_ENTRY_POINT;

通过标准的Protocol 就可以获取到对应的 TableAddress

从起始地址打印一段地址就是SMBIOS 存储空间的内容了,可以看到SMBIOS 是一段连续的内存空间,

每个Type 和每个Type 的中间是通过两个Ascall 0 0 隔开的,每个Type中若有String 字段,String字段通过一个0 隔开

EFI_STATUS
EFIAPI
FindSMBIOS(IN EFI_HANDLE        ImageHandle,IN EFI_SYSTEM_TABLE  *SystemTable){EFI_STATUS                      Status;SMBIOS_TABLE_ENTRY_POINT        *SmbiosTable;SMBIOS_TABLE_3_0_ENTRY_POINT    *Smbios3Table;EFI_ACPI_ISCSI_BOOT_FIRMWARE_TABLE_HEADER     *Table;EFI_ACPI_3_0_ROOT_SYSTEM_DESCRIPTION_POINTER  *Rsdp;UINT64 address;UINT8 value;UINT8 count;Status = EfiGetSystemConfigurationTable (&gEfiSmbiosTableGuid,(VOID **)&SmbiosTable);Print (L"SmbiosTable status: %r\r\n", Status);          Status = EfiGetSystemConfigurationTable (&gEfiSmbios3TableGuid,(VOID **)&Smbios3Table);
Print (L"Smbios3Table status: %r\r\n", Status);
Print (L"MajorVersion :%x MinorVersion:%x\r\n", Smbios3Table->MajorVersion,Smbios3Table->MinorVersion);
Print (L"Smbios3Table TableAddress: %x\r\n", Smbios3Table->TableAddress);
address=Smbios3Table->TableAddress;count=0;for(UINT64 index=0;index<0x1ff;index++){value=  Mmioread8(address+index);DEBUG ((DEBUG_ERROR, "%x  ", value));count++;if(count==16){count=0;DEBUG ((DEBUG_ERROR, "\n ", value));}} Status = EfiGetSystemConfigurationTable (&gEfiAcpiTableGuid, (VOID **)&Rsdp);Print (L"gEfiAcpiTableGuid status: %r\r\n", Status);}

 打印SMBIOS 空间如下对比Shell下的SMBIOSVIEW信息相同,SMBIOSVIEW 也是相同方式去获取和解析,这是一种方式。

还有一种是传统的SMBIOS 存储在内存地址F0000-FFFFF中

匹配 _smi_ 字符串  然后匹配 _DMI_ 找到EPS Table的地址 EPS的address + length 就是SMBIOS Table Entry 的地址

VOID MySmBiosInit(EFI_HII_HANDLE HiiHandle, UINT16 Class)
{CHAR16  Type0Data[50]={0};CHAR16  Type1Data[50]={0};UINT32 realaddress;UINT32 smaddr;UINT32 SmbiosTableaddr;UINT32 SmbiosLength;UINT32 Dataindex;realaddress=0xf0000;   //find 0xf0000-0xfffff _SM_while(realaddress<0xfffff){if(MyReadaddr32(realaddress)==0x5f4d535f)//find_SM_{               DEBUG((DEBUG_ERROR, "Hyl:SmBiosaddr find _SM_ success "));   if((MyReadaddr32(realaddress+16)==0x494d445f)&&(MyReadaddr8(realaddress+20)==0x5f))//find_DMI_{    smaddr=realaddress;DEBUG((DEBUG_ERROR, "Hyl:SmBiosaddr find _DMI_success "));                   break;} }realaddress = realaddress + 4;} SmbiosTableaddr=MyReadaddr32(smaddr+0x18); //EPSTABLE addeSmbiosLength=  ((MyReadaddr32(SmbiosTableaddr) )&0x0000ff00 )>>8;realaddress=SmbiosTableaddr+SmbiosLength;Dataindex=0;while(1){           if(  MyReadaddr8(realaddress)==0&&MyReadaddr8(realaddress+1)==0) {break;}if( MyReadaddr8(realaddress)>0) {Type0Data[Dataindex]=MyReadaddr8(realaddress);                  Dataindex++;}if( MyReadaddr8(realaddress)==0) {Type0Data[Dataindex]=0x2e ;Dataindex++;}realaddress++;}InitString( HiiHandle, STRING_TOKEN(STR_SMBIOS_VALUE0),L"SMBIOSTYPE0:%s" ,Type0Data);
}

举例分析Type1

需要注意的是STRING 类型的信息 ,这个offset位置代表的是第几个String,而不是描述信息,所有String类型的描述信息都在smbiostype* 的后面开始通过0 隔开

Type1的offset1 length 为 1B ,也就是说1B 开始才是String的开始

Offset 4  Manufacturer    STRING    01 代表第1个string

Offset 5   Product Name  STRING     02代表第2个string

Offset 6  Version  STRING     03代表第2个string

Offset 7  Serial Number  STRING  04代表第3个string

Offset 1B-6A 为String信息

 

  68  74  74  70  3A  2F  2F  77  77  77  2E  74  69

 61  6E  6F  63  6F  72  65  2E  6F  72  67  2F  65  64  6B  32

 2F  0  45  6D  75  6C  61  74  6F  72  50  6B  67  0  31  2E

 30  0  53  79  73  74  65  6D  20  53  65  72  69  61  6C  23

 0  53  79  73  74  65  6D  20  53  4B  55  23  0  65  64  6B

 32  0  0

第一个string   , 0为两个String的分隔符

   68  74  74  70  3A  2F  2F  77  77  77  2E  74  69

 61  6E  6F  63  6F  72  65  2E  6F  72  67  2F  65  64  6B  32

 2F  

  AscalL 对应的字符串就是 http://www.tianocore.org/edk2/。

 后面String相同方式解析

二进制

十进制

十六进制

字符/缩写

解释

00000000

0

00

NUL (NULL)

空字符

00000001

1

01

SOH (Start Of Headling)

标题开始

00000010

2

02

STX (Start Of Text)

正文开始

00000011

3

03

ETX (End Of Text)

正文结束

00000100

4

04

EOT (End Of Transmission)

传输结束

00000101

5

05

ENQ (Enquiry)

请求

00000110

6

06

ACK (Acknowledge)

回应/响应/收到通知

00000111

7

07

BEL (Bell)

响铃

00001000

8

08

BS (Backspace)

退格

00001001

9

09

HT (Horizontal Tab)

水平制表符

00001010

10

0A

LF/NL(Line Feed/New Line)

换行键

00001011

11

0B

VT (Vertical Tab)

垂直制表符

00001100

12

0C

FF/NP (Form Feed/New Page)

换页键

00001101

13

0D

CR (Carriage Return)

回车键

00001110

14

0E

SO (Shift Out)

不用切换

00001111

15

0F

SI (Shift In)

启用切换

00010000

16

10

DLE (Data Link Escape)

数据链路转义

00010001

17

11

DC1/XON

(Device Control 1/Transmission On)

设备控制1/传输开始

00010010

18

12

DC2 (Device Control 2)

设备控制2

00010011

19

13

DC3/XOFF

(Device Control 3/Transmission Off)

设备控制3/传输中断

00010100

20

14

DC4 (Device Control 4)

设备控制4

00010101

21

15

NAK (Negative Acknowledge)

无响应/非正常响应/拒绝接收

00010110

22

16

SYN (Synchronous Idle)

同步空闲

00010111

23

17

ETB (End of Transmission Block)

传输块结束/块传输终止

00011000

24

18

CAN (Cancel)

取消

00011001

25

19

EM (End of Medium)

已到介质末端/介质存储已满/介质中断

00011010

26

1A

SUB (Substitute)

替补/替换

00011011

27

1B

ESC (Escape)

逃离/取消

00011100

28

1C

FS (File Separator)

文件分割符

00011101

29

1D

GS (Group Separator)

组分隔符/分组符

00011110

30

1E

RS (Record Separator)

记录分离符

00011111

31

1F

US (Unit Separator)

单元分隔符

00100000

32

20

(Space)

空格

00100001

33

21

!

 

00100010

34

22

"

 

00100011

35

23

#

 

00100100

36

24

$

 

00100101

37

25

%

 

00100110

38

26

&

 

00100111

39

27

'

 

00101000

40

28

(

 

00101001

41

29

)

 

00101010

42

2A

*

 

00101011

43

2B

+

 

00101100

44

2C

,

 

00101101

45

2D

-

 

00101110

46

2E

.

 

00101111

47

2F

/

 

00110000

48

30

0

 

00110001

49

31

1

 

00110010

50

32

2

 

00110011

51

33

3

 

00110100

52

34

4

 

00110101

53

35

5

 

00110110

54

36

6

 

00110111

55

37

7

 

00111000

56

38

8

 

00111001

57

39

9

 

00111010

58

3A

:

 

00111011

59

3B

;

 

00111100

60

3C

<

 

00111101

61

3D

=

 

00111110

62

3E

>

 

00111111

63

3F

?

 

01000000

64

40

@

 

01000001

65

41

A

 

01000010

66

42

B

 

01000011

67

43

C

 

01000100

68

44

D

 

01000101

69

45

E

 

01000110

70

46

F

 

01000111

71

47

G

 

01001000

72

48

H

 

01001001

73

49

I

 

01001010

74

4A

J

 

01001011

75

4B

K

 

01001100

76

4C

L

 

01001101

77

4D

M

 

01001110

78

4E

N

 

01001111

79

4F

O

 

01010000

80

50

P

 

01010001

81

51

Q

 

01010010

82

52

R

 

01010011

83

53

S

 

01010100

84

54

T

 

01010101

85

55

U

 

01010110

86

56

V

 

01010111

87

57

W

 

01011000

88

58

X

 

01011001

89

59

Y

 

01011010

90

5A

Z

 

01011011

91

5B

[

 

01011100

92

5C

\

 

01011101

93

5D

]

 

01011110

94

5E

^

 

01011111

95

5F

_

 

01100000

96

60

`

 

01100001

97

61

a

 

01100010

98

62

b

 

01100011

99

63

c

 

01100100

100

64

d

 

01100101

101

65

e

 

01100110

102

66

f

 

01100111

103

67

g

 

01101000

104

68

h

 

01101001

105

69

i

 

01101010

106

6A

j

 

01101011

107

6B

k

 

01101100

108

6C

l

 

01101101

109

6D

m

 

01101110

110

6E

n

 

01101111

111

6F

o

 

01110000

112

70

p

 

01110001

113

71

q

 

01110010

114

72

r

 

01110011

115

73

s

 

01110100

116

74

t

 

01110101

117

75

u

 

01110110

118

76

v

 

01110111

119

77

w

 

01111000

120

78

x

 

01111001

121

79

y

 

01111010

122

7A

z

 

01111011

123

7B

{

 

01111100

124

7C

|

 

01111101

125

7D

}

 

01111110

126

7E

~

 

01111111

127

7F

DEL (Delete)

删除

标准 ASCII 编码一览表

0 0 为两个Smbios Type 的分隔符,每个Smbios Type 有相同的Header部

///

/// The Smbios structure header.

///

typedef struct {

  SMBIOS_TYPE      Type;

  UINT8            Length;

  SMBIOS_HANDLE    Handle;

} SMBIOS_STRUCTURE;

根据Length 信息就可以确定String的起点 ,0 0 分隔符就是一个Type的结尾标志

这篇关于UEFI BIOS之SMBIOS的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

win10 gpt分区+uefi引导 卸载双系统ubuntu

1、首先暴力卸载ubuntu 在win10里面磁盘管理中找到对应的linux磁盘分区 删除卷OK 2、重启 出现下面(根据机型不同界面可能不一样 ) 3、exit 退出grub引导 进入uefi引导  选择win10引导项 (当然你要是一直按着进入bios boot的那个按键的话 也不用看第二步了 直接选择windows启动项进去 dell的话是F12) 4、进入

UEFI——Shell下读取SMBIOS信息

一、SMBIOS简介 SMBIOS的全称为System Management BIOS,它不是一个BIOS,只是与BIOS相关。它是一个规范,定义了BIOS传递给操作系统的系统管理信息。它也表示了一系列的数据结构,包含了各类信息,由BIOS启动过程中创建并放在特定的内存,之后操作系统可以拿来用。 整个 SMBIOS_STRUCTURE 结构体定义了 SMBIOS 表中的一个基本单元。每个 SM

【UEFI基础】SMBIOS基础和使用

SMBIOS的定义 SMBIOS的全称是System Management BIOS,关于它的理解包括: 它不是一个BIOS,之所以出现了BIOS字样,是因为它跟BIOS有关,仅此而已。它是一个规范,定义了BIOS传递给操作系统的系统管理信息,具体有哪些信息,可以参考SMBIOS规范。它也可以表示一系列的数据结构,包含了各类信息,由BIOS启动过程中创建并放到特定的内存,之后操作系统可以拿来用

4-7 使用bios 中断 读取磁盘

1 首先是逻辑。 首先来看一下 他的编译代码的逻辑。          可以看到我 生成的 实际上是 Boot.bin ,  这个文件可不止一个扇区, 然后将这个文件写入到了, disk1.img 这里加载了  disk1.img , + disk2.img 我不太理解。 但是可以跑通, 暂时先不管了。 这节的逻辑是: 首先测试的是,  将第0 个扇区后面

4-6 使用bios 中断 显示字符

1 显示的逻辑 bios 首先通过中断,访问到 最前面的中断向量表,然后 通过中断向量表然后 访问到具体的 bios 的函数,这些函数是bios 自带的,具体的位置 , 我也不知道。只知道有这个函数。 3 显示的原理 ; 主要用到的就是 AX  以及BX。 2 显示的代码 // 显示一个字符L 。mov $0xe, %ahmov $'L', %al

华硕笔记本bios设置u盘启动

华硕笔记本bios设置u盘启动 一:快捷启动     ESC 选择U盘---回车 二:进bios改u盘启动     F2进去bios   进入bios: 1:移动到Security选项---找到Security Boot Control     将 Enabled 改为 disabled(关闭) --ok 目的:找到Secure Boot并关闭       这样在bios里面就会读到U盘

联想笔记本bios设置u盘启动

联想笔记本bios设置u盘启动 一:快捷启动     大众化:F12        新款联想:FN + F12 二:进bios改u盘启动     大众化:DEL或F12   新款联想:FN + DEL 进入bios: 1:移动到Security选项---找到Security Boot     改Secure Boot 为 disabled(关闭) --ok 目的:找到Secure Boot并关闭

计算机到底是如何启动的?(传统BIOS)

引言 前先日子在装系统的时候,发觉自己对于计算机到底是如何启动的这一过程认识不清晰,然后下去查了查资料,看了阮一峰的一篇博客计算机是如何启动的,作者写的简单,易懂,我自己 整理了一下,希望分享给大家。 boot 启动用英文来讲是boot,但是boot的意思是靴子,靴子又怎么和计算机的启动联系起来呐,这里的boot是bootstrap(鞋带)的缩写,

计算机到底如何启动?(UEFI boot)

引言 为了把BIOS和UEFI的区别弄清楚,我查了一些关于计算机启动的资料,其中看到一篇翻译自国外技术论坛的文章,(译)UEFI启动:实际工作原理纠正了我之前许多错误的认识,我重新整理、添加了一些东西,把这篇文章分享给大家,能够有一个更清晰的认识。 概念 如果想真正了解有关 UEFI 的权威知识,可以查看 UEFI 规范(可以访问官方UEFI网站)。

BIOS工程师标准作业书 之 PCIE 问题如何处理

bios 工程师作为最基层的岗位,做事情一定要按部就班,肯定不能随意发挥。PCI 作为服务器中最主要的总线,也是最容易出bug 的地方。所以我整理了这篇文章。 当前x86系统PCIe架构中面临的挑战以及问题 系统设计中使用了不同供应商的硬件模块,不同的硬件模块都有各自的错误处理和报错机制。 在这么复杂的系统中,如何快速定位故障的模块,并使系统重新健康的上线工作 如何在复杂的系统中,找到问题