基于Win32的GetSystemFirmwareTable获取到的SMBIOS数据简析

本文介绍了如何使用Win32 API GetSystemFirmwareTable获取SMBIOS的原始数据,并解析其中的CPU信息。通过分析任务管理器的实现,解释了如何忽略RawSMBIOSData头部,直接访问SMBIOSTableData,以及如何识别和处理数据段中的硬件信息。
我不是什么硬件专业的人员,可最近有个东西实在不能通过其他方式获取硬件信息,setupapi不行,注册表不行,NtQuerySystemInformation也不行,但是因为匹配需要必需要拿到CPU信息,比如CPU名称,实在无奈了。
纠结了很久,只能通过GetSystemFirmwareTable这个函数拿到SMBIOS的Raw数据,其中有CPU的名称,缓存等信息,可因为GetSystemFirmwareTable返回的数据是原始的SMBIOS数据,对我们不懂做硬件的人来说实在看得头大了,网上的资料不是针对的SMBIOS版本老,就是牛头不对马嘴,而且我也实在没心情去看SMBIOS的资料,就找了个最直接的方法,我发现win8的任务管理器有使用GetSystemFirmwareTable来获取系统信息,就操着OD上马了,小小的逆了点东西出来,还是能简单用的。

GetSystemFirmwareTable调用方法我就不说了,无非就是获取长度,然后malloc,再调用GetSystemFirmwareTable获取数据。
GetSystemFirmwareTable获取到的数据是一个RawSMBIOSData结构体:
struct RawSMBIOSData
{
    BYTE Used20CallingMethod;
    BYTE SMBIOSMajorVersion;
    BYTE SMBIOSMinorVersion;
    BYTE DmiRevision;
    DWORD Length;
    BYTE SMBIOSTableData[];
};

第一个BYTE不懂啥意思,后面3个BYTE都是版本标示,然后Length就是SMBIOSTableData的总长度,一般我们需要直接拿到SMBIOS的Raw,就直接忽略过RawSMBIOSData的前8个字节,就是SMBIOSTableData的开始了。
taskmgr是这样写的:
LEA EDI,DWORD PTR DS:[ECX+8]
//ECX = RawSMBIOSData*
一般获取主板UID都是cmd命令 执行"wmic path Win32_ComputerSystemProduct get uuid" 这里说的是使用API获取,找了半天没找到,最后发现只能通过 GetSystemFirmwareTable 获取 SMBIOS信息 ,在这个结构体里存储的相当复杂,看的有点头晕 首先调用 GetSystemFirmwareTable  先让他返回结构体大小,然后我们申请一块内存,再调用一次,就会把  SMBIOS信息 存储到 我们申请的内存里了. SMBIOS信息 前8个字节就不看了,可以百度慢慢看,从第9个字节开始是TYPE结构,第一个字节是type类型,第二个字节就是这个type类型的大小,从type起始地址计算,后面的就是格式化区域,不同type类型格式化区域代表的信息也不同,格式化区域完后,就会跟一段u字符串,我们知道ascii字符串是0结尾,u字符串是00 结尾,字符串结尾后就是下一个type结构的开始了. 这里就只看下type1类型结构  从资料里看,在第8个字节后的16位就是uid了.   其他类型大家就自己查一下了. 里面的模块命令 就用了下 精益的十六进制转换命令,这个大家应该都有. 代码有点丑,见笑了. 系统信息 (Type 1) : SMBIOS 实现只关联一个单一的系统实例,并且包含且只包含一个系统信息结构。 位置 名称 长度 描述 00h TYPE 号 1BYTE 结构的TYPE 号,此处是1 01h 长度 1BYTE 格式区域总长度,2.0 版为08h ,2.1-2.3.4 版为19h,从2.4 版开始为1Bh 02h 句柄 2BYTE 指向本结构的句柄 04h 电脑制造商 1BYTE 一般为01h ,表示在字符串区域中的编号 05h 产品名称 1BYTE [url=]在字符串区域中的编号[/url] 06h 版本号 1BYTE 在字符串区域中的编号 07h 序列号 1BYTE 在字符串区域中的编号 09h UUID 16BYTE 通用唯一标识符 18h 唤醒类型 BYTE 用来标识导致系统开电启动的事件 19h SKU 号 BYTE 在字符串区域中的编号,SKU 号通常为产品ID 或采购订单号 1Ah 产品家族 1BYTE 在字符串区域中的编号
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值