smbios: add detailed smbios information

Add detailed SMBIOS information as following:
1. Missing fields in Type #3 and #4
2. Type #7

Add support to dynamic length of contained object handles and
elements.

As SMBIOS is a fundamental feature which is enabled for all
boards, in order to avoid increasing rom size, all detailed SMBIOS
information is under kconfig GENERATE_SMBIOS_TABLE_VERBOSE.
Board maintainers can determine whether to select this feature.

Signed-off-by: Raymond Mao <raymond.mao@linaro.org>
This commit is contained in:
Raymond Mao
2024-12-06 14:54:23 -08:00
committed by Simon Glass
parent 1d03c4b28d
commit 87f1e537cb

View File

@@ -528,6 +528,8 @@ static int smbios_write_type3(ulong *current, int handle,
int len = sizeof(*t);
u8 *eos_addr;
size_t elem_size = 0;
__maybe_unused u8 *elem_addr;
__maybe_unused u8 *sku_num_addr;
/*
* reserve the space for the dynamic bytes of contained elements.
@@ -540,7 +542,10 @@ static int smbios_write_type3(ulong *current, int handle,
t = map_sysmem(*current, len);
memset(t, 0, len);
fill_smbios_header(t, SMBIOS_SYSTEM_ENCLOSURE, len, handle);
#if IS_ENABLED(CONFIG_GENERATE_SMBIOS_TABLE_VERBOSE)
elem_addr = (u8 *)t + offsetof(struct smbios_type3, sku_number);
sku_num_addr = elem_addr + elem_size;
#endif
/* eos is at the end of the structure */
eos_addr = (u8 *)t + len - sizeof(t->eos);
smbios_set_eos(ctx, eos_addr);
@@ -564,6 +569,32 @@ static int smbios_write_type3(ulong *current, int handle,
SYSID_SM_ENCLOSURE_SECURITY,
SMBIOS_SECURITY_UNKNOWN);
#if IS_ENABLED(CONFIG_GENERATE_SMBIOS_TABLE_VERBOSE)
t->version = smbios_add_prop_si(ctx, "version",
SYSID_SM_ENCLOSURE_VERSION, NULL);
t->serial_number = smbios_add_prop_si(ctx, "serial",
SYSID_SM_ENCLOSURE_SERIAL, NULL);
t->asset_tag_number = smbios_add_prop_si(ctx, "asset-tag",
SYSID_SM_BASEBOARD_ASSET_TAG,
NULL);
t->oem_defined = smbios_get_val_si(ctx, "oem-defined",
SYSID_SM_ENCLOSURE_OEM, 0);
t->height = smbios_get_val_si(ctx, "height",
SYSID_SM_ENCLOSURE_HEIGHT, 0);
t->number_of_power_cords =
smbios_get_val_si(ctx, "number-of-power-cords",
SYSID_SM_ENCLOSURE_POWCORE_NUM, 0);
/*
* TODO: Populate the Contained Element Record if they exist
* t->element_count = <element_num>;
* t->element_record_length = <element_len>;
*/
*sku_num_addr = smbios_add_prop_si(ctx, "sku", SYSID_SM_ENCLOSURE_SKU,
NULL);
#endif
len = t->hdr.length + smbios_string_table_len(ctx);
*current += len;
unmap_sysmem(t);
@@ -577,6 +608,8 @@ static void smbios_write_type4_dm(struct smbios_type4 *t,
u16 processor_family = SMBIOS_PROCESSOR_FAMILY_UNKNOWN;
const char *vendor = NULL;
const char *name = NULL;
__maybe_unused void *id_data = NULL;
__maybe_unused size_t id_size = 0;
#ifdef CONFIG_CPU
char processor_name[49];
@@ -617,6 +650,16 @@ static void smbios_write_type4_dm(struct smbios_type4 *t,
t->processor_version = smbios_add_prop_si(ctx, "version",
SYSID_SM_PROCESSOR_VERSION,
name);
#if IS_ENABLED(CONFIG_GENERATE_SMBIOS_TABLE_VERBOSE)
if (t->processor_id[0] || t->processor_id[1] ||
sysinfo_get_data(ctx->dev, SYSID_SM_PROCESSOR_ID, &id_data,
&id_size))
return;
if (id_data && id_size == sizeof(t->processor_id))
memcpy((u8 *)t->processor_id, id_data, id_size);
#endif
}
static int smbios_write_type4(ulong *current, int handle,
@@ -624,6 +667,8 @@ static int smbios_write_type4(ulong *current, int handle,
{
struct smbios_type4 *t;
int len = sizeof(*t);
__maybe_unused void *hdl;
__maybe_unused size_t hdl_size;
t = map_sysmem(*current, len);
memset(t, 0, len);
@@ -647,6 +692,60 @@ static int smbios_write_type4(ulong *current, int handle,
t->l2_cache_handle = SMBIOS_CACHE_HANDLE_NONE;
t->l3_cache_handle = SMBIOS_CACHE_HANDLE_NONE;
#if IS_ENABLED(CONFIG_GENERATE_SMBIOS_TABLE_VERBOSE)
t->voltage = smbios_get_val_si(ctx, "voltage",
SYSID_SM_PROCESSOR_VOLTAGE, 0);
t->external_clock = smbios_get_val_si(ctx, "external-clock",
SYSID_SM_PROCESSOR_EXT_CLOCK, 0);
t->max_speed = smbios_get_val_si(ctx, "max-speed",
SYSID_SM_PROCESSOR_MAX_SPEED, 0);
t->current_speed = smbios_get_val_si(ctx, "current-speed",
SYSID_SM_PROCESSOR_CUR_SPEED, 0);
/* Read the cache handles */
if (!sysinfo_get_data(ctx->dev, SYSID_SM_CACHE_HANDLE, &hdl,
&hdl_size) &&
(hdl_size == SYSINFO_CACHE_LVL_MAX * sizeof(u16))) {
u16 *handle = (u16 *)hdl;
if (*handle)
t->l1_cache_handle = *handle;
handle++;
if (*handle)
t->l2_cache_handle = *handle;
handle++;
if (*handle)
t->l3_cache_handle = *handle;
}
t->serial_number = smbios_add_prop_si(ctx, "serial",
SYSID_SM_PROCESSOR_SN, NULL);
t->asset_tag = smbios_add_prop_si(ctx, "asset-tag",
SYSID_SM_PROCESSOR_ASSET_TAG, NULL);
t->part_number = smbios_add_prop_si(ctx, "part-number",
SYSID_SM_PROCESSOR_PN, NULL);
t->core_count = smbios_get_val_si(ctx, "core-count",
SYSID_SM_PROCESSOR_CORE_CNT, 0);
t->core_enabled = smbios_get_val_si(ctx, "core-enabled",
SYSID_SM_PROCESSOR_CORE_EN, 0);
t->thread_count = smbios_get_val_si(ctx, "thread-count",
SYSID_SM_PROCESSOR_THREAD_CNT, 0);
t->processor_characteristics =
smbios_get_val_si(ctx, "characteristics",
SYSID_SM_PROCESSOR_CHARA,
SMBIOS_PROCESSOR_UND);
t->core_count2 = smbios_get_val_si(ctx, "core-count2",
SYSID_SM_PROCESSOR_CORE_CNT2, 0);
t->core_enabled2 = smbios_get_val_si(ctx, "core-enabled2",
SYSID_SM_PROCESSOR_CORE_EN2, 0);
t->thread_count2 = smbios_get_val_si(ctx, "thread-count2",
SYSID_SM_PROCESSOR_THREAD_CNT2, 0);
t->thread_enabled = smbios_get_val_si(ctx, "thread-enabled",
SYSID_SM_PROCESSOR_THREAD_EN, 0);
#endif
len = t->hdr.length + smbios_string_table_len(ctx);
*current += len;
unmap_sysmem(t);
@@ -654,6 +753,104 @@ static int smbios_write_type4(ulong *current, int handle,
return len;
}
#if IS_ENABLED(CONFIG_GENERATE_SMBIOS_TABLE_VERBOSE)
static int smbios_write_type7_1level(ulong *current, int handle,
struct smbios_ctx *ctx, int level)
{
struct smbios_type7 *t;
int len = sizeof(*t);
void *hdl;
size_t hdl_size;
t = map_sysmem(*current, len);
memset(t, 0, len);
fill_smbios_header(t, SMBIOS_CACHE_INFORMATION, len, handle);
smbios_set_eos(ctx, t->eos);
t->socket_design = smbios_add_prop_si(ctx, "socket-design",
SYSID_SM_CACHE_SOCKET + level,
NULL);
t->config.data = smbios_get_val_si(ctx, "config",
SYSID_SM_CACHE_CONFIG + level,
(level - 1) | SMBIOS_CACHE_OP_UND);
t->max_size.data = smbios_get_val_si(ctx, "max-size",
SYSID_SM_CACHE_MAX_SIZE + level,
0);
t->inst_size.data = smbios_get_val_si(ctx, "installed-size",
SYSID_SM_CACHE_INST_SIZE + level,
0);
t->supp_sram_type.data =
smbios_get_val_si(ctx, "supported-sram-type",
SYSID_SM_CACHE_SUPSRAM_TYPE + level,
SMBIOS_CACHE_SRAM_TYPE_UNKNOWN);
t->curr_sram_type.data =
smbios_get_val_si(ctx, "current-sram-type",
SYSID_SM_CACHE_CURSRAM_TYPE + level,
SMBIOS_CACHE_SRAM_TYPE_UNKNOWN);
t->speed = smbios_get_val_si(ctx, "speed", SYSID_SM_CACHE_SPEED + level,
0);
t->err_corr_type = smbios_get_val_si(ctx, "error-correction-type",
SYSID_SM_CACHE_ERRCOR_TYPE + level,
SMBIOS_CACHE_ERRCORR_UNKNOWN);
t->sys_cache_type =
smbios_get_val_si(ctx, "system-cache-type",
SYSID_SM_CACHE_SCACHE_TYPE + level,
SMBIOS_CACHE_SYSCACHE_TYPE_UNKNOWN);
t->associativity = smbios_get_val_si(ctx, "associativity",
SYSID_SM_CACHE_ASSOC + level,
SMBIOS_CACHE_ASSOC_UNKNOWN);
t->max_size2.data = smbios_get_val_si(ctx, "max-size2",
SYSID_SM_CACHE_MAX_SIZE2 + level,
0);
t->inst_size2.data =
smbios_get_val_si(ctx, "installed-size2",
SYSID_SM_CACHE_INST_SIZE2 + level, 0);
/* Save the cache handles */
if (!sysinfo_get_data(ctx->dev, SYSID_SM_CACHE_HANDLE, &hdl,
&hdl_size)) {
if (hdl_size == SYSINFO_CACHE_LVL_MAX * sizeof(u16))
*((u16 *)hdl + level) = handle;
}
len = t->hdr.length + smbios_string_table_len(ctx);
*current += len;
unmap_sysmem(t);
return len;
}
static int smbios_write_type7(ulong *current, int handle,
struct smbios_ctx *ctx)
{
int len = 0;
int i, level;
ofnode parent = ctx->node;
struct smbios_ctx ctx_bak;
memcpy(&ctx_bak, ctx, sizeof(ctx_bak));
/* Get the number of level */
level = smbios_get_val_si(ctx, NULL, SYSID_SM_CACHE_LEVEL, 0);
if (level >= SYSINFO_CACHE_LVL_MAX) /* Error, return 0-length */
return 0;
for (i = 0; i <= level; i++) {
char buf[9] = "";
if (!snprintf(buf, sizeof(buf), "l%d-cache", i + 1))
return 0;
ctx->subnode_name = buf;
ctx->node = ofnode_find_subnode(parent, ctx->subnode_name);
len += smbios_write_type7_1level(current, handle++, ctx, i);
memcpy(ctx, &ctx_bak, sizeof(*ctx));
}
return len;
}
#endif /* #if IS_ENABLED(CONFIG_GENERATE_SMBIOS_TABLE_VERBOSE) */
static int smbios_write_type32(ulong *current, int handle,
struct smbios_ctx *ctx)
{
@@ -693,6 +890,10 @@ static struct smbios_write_method smbios_write_funcs[] = {
{ smbios_write_type2, "baseboard", },
/* Type 3 must immediately follow type 2 due to chassis handle. */
{ smbios_write_type3, "chassis", },
#if IS_ENABLED(CONFIG_GENERATE_SMBIOS_TABLE_VERBOSE)
/* Type 7 must ahead of type 4 to get cache handles. */
{ smbios_write_type7, "cache", },
#endif
{ smbios_write_type4, "processor"},
{ smbios_write_type32, },
{ smbios_write_type127 },