/*
* \brief Utilities for accessing information of the System Management BIOS
* \author Martin Stein
* \date 2019-07-02
*/
/*
* Copyright (C) 2019 Genode Labs GmbH
*
* This file is part of the Genode OS framework, which is distributed
* under the terms of the GNU Affero General Public License version 3.
*/
#ifndef _OS__SMBIOS_H_
#define _OS__SMBIOS_H_
/* Genode includes */
#include
#include
#include
namespace Genode {
namespace Smbios_table { };
inline bool smbios_checksum_correct(uint8_t const *base, uint8_t size)
{
uint8_t sum { 0 };
for (uint8_t idx = 0; idx < size; idx++) {
sum = (uint8_t)(sum + base[idx]);
}
return sum == 0;
}
struct Smbios_3_entry_point;
struct Smbios_entry_point;
struct Dmi_entry_point;
struct Smbios_structure;
}
struct Genode::Smbios_structure
{
enum Type {
BIOS = 0,
SYSTEM = 1,
BASE_BOARD = 2,
};
uint8_t type;
uint8_t length;
uint16_t handle;
} __attribute__((packed));
struct Genode::Dmi_entry_point
{
enum { LENGTH = 15 };
uint8_t anchor_string[5];
uint8_t checksum;
uint16_t struct_table_length;
uint32_t struct_table_addr;
uint16_t nr_of_structs;
uint8_t bcd_revision;
bool checksum_correct() const
{
return smbios_checksum_correct(anchor_string, LENGTH);
}
} __attribute__((packed));
struct Genode::Smbios_3_entry_point
{
enum { MAX_LENGTH = 32 };
uint8_t anchor_string[5];
uint8_t checksum;
uint8_t length;
uint8_t version_major;
uint8_t version_minor;
uint8_t docrev;
uint8_t revision;
uint8_t reserved_0;
uint32_t struct_table_max_size;
uint64_t struct_table_addr;
bool length_valid() const
{
return length <= MAX_LENGTH;
}
bool checksum_correct() const
{
return smbios_checksum_correct(anchor_string, length);
}
} __attribute__((packed));
struct Genode::Smbios_entry_point
{
enum { MAX_LENGTH = 32 };
enum { INTERM_LENGTH = 15 };
uint8_t anchor_string[4];
uint8_t checksum;
uint8_t length;
uint8_t version_major;
uint8_t version_minor;
uint16_t max_struct_size;
uint8_t revision;
uint8_t formatted_area[5];
uint8_t interm_anchor_string[5];
uint8_t interm_checksum;
uint16_t struct_table_length;
uint32_t struct_table_addr;
uint16_t nr_of_structs;
uint8_t bcd_revision;
bool length_valid() const
{
return length <= MAX_LENGTH;
}
bool checksum_correct() const
{
return smbios_checksum_correct(anchor_string, length);
}
bool interm_checksum_correct() const
{
return smbios_checksum_correct(interm_anchor_string, INTERM_LENGTH);
}
Dmi_entry_point const &dmi_ep() const
{
return *(Dmi_entry_point *)&interm_anchor_string;
}
} __attribute__((packed));
namespace Genode::Smbios_table
{
bool smbios_3(addr_t const anchor,
addr_t const ep_phy,
auto const &phy_mem_fn,
auto const &handle_ep_fn)
{
if (memcmp((char *)anchor, "_SM3_", 5)) {
return false;
}
Smbios_3_entry_point const &ep { *(Smbios_3_entry_point *)
phy_mem_fn(ep_phy, sizeof(Smbios_3_entry_point)) };
if (!ep.length_valid()) {
warning("SMBIOS 3 entry point has bad length");
return false;
}
if (!ep.checksum_correct()) {
warning("SMBIOS 3 entry point has bad checksum");
return false;
}
if (ep.struct_table_addr > (uint64_t)(~(addr_t)0)) {
warning("SMBIOS 3 entry point has bad structure-table address");
return false;
}
log("SMBIOS 3 table (entry point: ", Hex(anchor), " structures: ", Hex(ep.struct_table_addr), ")");
handle_ep_fn(ep);
return true;
}
bool smbios(addr_t const anchor,
addr_t const ep_phy,
auto const &phy_mem_fn,
auto const &handle_ep_fn)
{
if (memcmp((char *)anchor, "_SM_", 4)) {
return false;
}
Smbios_entry_point const &ep { *(Smbios_entry_point *)
phy_mem_fn(ep_phy, sizeof(Smbios_entry_point)) };
if (!ep.length_valid()) {
warning("SMBIOS entry point has bad length");
return false;
}
if (!ep.checksum_correct()) {
warning("SMBIOS entry point has bad checksum");
return false;
}
if (String<6>((char const *)&ep.interm_anchor_string) != "_DMI_") {
warning("SMBIOS entry point has bad intermediate anchor string");
return false;
}
if (!ep.interm_checksum_correct()) {
warning("SMBIOS entry point has bad intermediate checksum");
return false;
}
log("SMBIOS table (entry point: ", Hex(anchor), " structures: ", Hex(ep.struct_table_addr), ")");
handle_ep_fn(ep);
return true;
}
bool dmi(addr_t const anchor,
addr_t const ep_phy,
auto const &phy_mem_fn,
auto const &handle_ep_fn)
{
if (memcmp((char *)anchor, "_DMI_", 5)) {
return false;
}
Dmi_entry_point const &ep { *(Dmi_entry_point *)
phy_mem_fn(ep_phy, sizeof(Dmi_entry_point)) };
if (!ep.checksum_correct()) {
warning("DMI entry point has bad checksum");
return false;
}
log("DMI table (entry point: ", Hex(anchor), " structures: ", Hex(ep.struct_table_addr), ")");
handle_ep_fn(ep);
return true;
}
void from_scan(auto const &phy_mem_fn,
auto const &handle_smbios_3_ep_fn,
auto const &handle_smbios_ep_fn,
auto const &handle_dmi_ep_fn)
{
enum { SCAN_BASE_PHY = 0xf0000 };
enum { SCAN_SIZE = 0x10000 };
enum { SCAN_SIZE_SMBIOS = 0xfff0 };
enum { SCAN_STEP = 0x10 };
addr_t const scan_base { (addr_t)phy_mem_fn(SCAN_BASE_PHY, SCAN_SIZE) };
try {
addr_t const scan_end { scan_base + SCAN_SIZE };
size_t const scan_end_smbios { scan_base + SCAN_SIZE_SMBIOS };
for (addr_t curr { scan_base }; curr < scan_end_smbios; curr += SCAN_STEP ) {
if (smbios_3(curr, SCAN_BASE_PHY + (curr - scan_base), phy_mem_fn, handle_smbios_3_ep_fn)) {
return;
}
}
for (addr_t curr { scan_base }; curr < scan_end_smbios; curr += SCAN_STEP ) {
if (smbios(curr, SCAN_BASE_PHY + (curr - scan_base), phy_mem_fn, handle_smbios_ep_fn)) {
return;
}
}
for (addr_t curr { scan_base }; curr < scan_end; curr += SCAN_STEP ) {
if (dmi(curr, SCAN_BASE_PHY + (curr - scan_base), phy_mem_fn, handle_dmi_ep_fn)) {
return;
}
}
} catch (...) { }
}
void from_pointer(addr_t const table_phy,
auto const &phy_mem_fn,
auto const &handle_smbios_3_ep_fn,
auto const &handle_smbios_ep_fn,
auto const &handle_dmi_ep_fn)
{
addr_t const anchor { (addr_t)phy_mem_fn(table_phy, 5) };
if (smbios_3(anchor, table_phy, phy_mem_fn, handle_smbios_3_ep_fn)) {
return;
}
if (smbios(anchor, table_phy, phy_mem_fn, handle_smbios_ep_fn)) {
return;
}
dmi(anchor, table_phy, phy_mem_fn, handle_dmi_ep_fn);
}
};
#endif /* _OS__SMBIOS_H_ */