184ab085aSmws /* 284ab085aSmws * CDDL HEADER START 384ab085aSmws * 484ab085aSmws * The contents of this file are subject to the terms of the 5*074bb90dSTom Pothier * Common Development and Distribution License (the "License"). 6*074bb90dSTom Pothier * You may not use this file except in compliance with the License. 784ab085aSmws * 884ab085aSmws * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 984ab085aSmws * or http://www.opensolaris.org/os/licensing. 1084ab085aSmws * See the License for the specific language governing permissions 1184ab085aSmws * and limitations under the License. 1284ab085aSmws * 1384ab085aSmws * When distributing Covered Code, include this CDDL HEADER in each 1484ab085aSmws * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 1584ab085aSmws * If applicable, add the following below this CDDL HEADER, with the 1684ab085aSmws * fields enclosed by brackets "[]" replaced with your own identifying 1784ab085aSmws * information: Portions Copyright [yyyy] [name of copyright owner] 1884ab085aSmws * 1984ab085aSmws * CDDL HEADER END 2084ab085aSmws */ 2184ab085aSmws 2284ab085aSmws /* 23*074bb90dSTom Pothier * Copyright 2009 Sun Microsystems, Inc. All rights reserved. 2484ab085aSmws * Use is subject to license terms. 2584ab085aSmws */ 2684ab085aSmws 2784ab085aSmws #include <sys/sysmacros.h> 2884ab085aSmws #include <sys/param.h> 2984ab085aSmws 3084ab085aSmws #include <smbios.h> 3184ab085aSmws #include <alloca.h> 3284ab085aSmws #include <limits.h> 3384ab085aSmws #include <unistd.h> 3484ab085aSmws #include <strings.h> 3584ab085aSmws #include <stdlib.h> 3684ab085aSmws #include <stdarg.h> 3784ab085aSmws #include <stdio.h> 3884ab085aSmws #include <fcntl.h> 3984ab085aSmws #include <errno.h> 4084ab085aSmws #include <ctype.h> 4184ab085aSmws 4284ab085aSmws #define SMBIOS_SUCCESS 0 4384ab085aSmws #define SMBIOS_ERROR 1 4484ab085aSmws #define SMBIOS_USAGE 2 4584ab085aSmws 4684ab085aSmws static const char *g_pname; 4784ab085aSmws static int g_hdr; 4884ab085aSmws 4984ab085aSmws static int opt_e; 5084ab085aSmws static int opt_i = -1; 5184ab085aSmws static int opt_O; 5284ab085aSmws static int opt_s; 5384ab085aSmws static int opt_t = -1; 5484ab085aSmws static int opt_x; 5584ab085aSmws 5684ab085aSmws /*PRINTFLIKE2*/ 5784ab085aSmws static void 5884ab085aSmws oprintf(FILE *fp, const char *format, ...) 5984ab085aSmws { 6084ab085aSmws va_list ap; 6184ab085aSmws 6284ab085aSmws va_start(ap, format); 6384ab085aSmws (void) vfprintf(fp, format, ap); 6484ab085aSmws va_end(ap); 6584ab085aSmws } 6684ab085aSmws 6784ab085aSmws /*PRINTFLIKE3*/ 6884ab085aSmws static void 6984ab085aSmws desc_printf(const char *d, FILE *fp, const char *format, ...) 7084ab085aSmws { 7184ab085aSmws va_list ap; 7284ab085aSmws 7384ab085aSmws va_start(ap, format); 7484ab085aSmws (void) vfprintf(fp, format, ap); 7584ab085aSmws va_end(ap); 7684ab085aSmws 7784ab085aSmws if (d != NULL) 7884ab085aSmws (void) fprintf(fp, " (%s)\n", d); 7984ab085aSmws else 8084ab085aSmws (void) fprintf(fp, "\n"); 8184ab085aSmws } 8284ab085aSmws 8384ab085aSmws static void 8484ab085aSmws flag_printf(FILE *fp, const char *s, uint_t flags, size_t bits, 8584ab085aSmws const char *(*flag_name)(uint_t), const char *(*flag_desc)(uint_t)) 8684ab085aSmws { 8784ab085aSmws size_t i; 8884ab085aSmws 8984ab085aSmws oprintf(fp, " %s: 0x%x\n", s, flags); 9084ab085aSmws 9184ab085aSmws for (i = 0; i < bits; i++) { 9284ab085aSmws uint_t f = 1 << i; 9384ab085aSmws const char *n; 9484ab085aSmws 9584ab085aSmws if (!(flags & f)) 9684ab085aSmws continue; 9784ab085aSmws 9884ab085aSmws if ((n = flag_name(f)) != NULL) 9984ab085aSmws desc_printf(flag_desc(f), fp, "\t%s", n); 10084ab085aSmws else 10184ab085aSmws desc_printf(flag_desc(f), fp, "\t0x%x", f); 10284ab085aSmws } 10384ab085aSmws } 10484ab085aSmws 10584ab085aSmws static void 10684ab085aSmws flag64_printf(FILE *fp, const char *s, uint64_t flags, size_t bits, 10784ab085aSmws const char *(*flag_name)(uint64_t), const char *(*flag_desc)(uint64_t)) 10884ab085aSmws { 10984ab085aSmws size_t i; 11084ab085aSmws 11184ab085aSmws oprintf(fp, " %s: 0x%llx\n", s, (u_longlong_t)flags); 11284ab085aSmws 11384ab085aSmws for (i = 0; i < bits; i++) { 11484ab085aSmws u_longlong_t f = 1ULL << i; 11584ab085aSmws const char *n; 11684ab085aSmws 11784ab085aSmws if (!(flags & f)) 11884ab085aSmws continue; 11984ab085aSmws 12084ab085aSmws if ((n = flag_name(f)) != NULL) 12184ab085aSmws desc_printf(flag_desc(f), fp, "\t%s", n); 12284ab085aSmws else 12384ab085aSmws desc_printf(flag_desc(f), fp, "\t0x%llx", f); 12484ab085aSmws } 12584ab085aSmws } 12684ab085aSmws 12784ab085aSmws static void 12884ab085aSmws id_printf(FILE *fp, const char *s, id_t id) 12984ab085aSmws { 13084ab085aSmws switch (id) { 13184ab085aSmws case SMB_ID_NONE: 13284ab085aSmws oprintf(fp, "%sNone\n", s); 13384ab085aSmws break; 13484ab085aSmws case SMB_ID_NOTSUP: 13584ab085aSmws oprintf(fp, "%sNot Supported\n", s); 13684ab085aSmws break; 13784ab085aSmws default: 13884ab085aSmws oprintf(fp, "%s%u\n", s, (uint_t)id); 13984ab085aSmws } 14084ab085aSmws } 14184ab085aSmws 14284ab085aSmws static void 14384ab085aSmws print_smbios(smbios_hdl_t *shp, FILE *fp) 14484ab085aSmws { 14584ab085aSmws smbios_entry_t ep; 14684ab085aSmws int i; 14784ab085aSmws 14884ab085aSmws smbios_info_smbios(shp, &ep); 14984ab085aSmws 15084ab085aSmws oprintf(fp, "Entry Point Anchor Tag: %*.*s\n", 15184ab085aSmws (int)sizeof (ep.smbe_eanchor), (int)sizeof (ep.smbe_eanchor), 15284ab085aSmws ep.smbe_eanchor); 15384ab085aSmws 15484ab085aSmws oprintf(fp, "Entry Point Checksum: 0x%x\n", ep.smbe_ecksum); 15584ab085aSmws oprintf(fp, "Entry Point Length: %u\n", ep.smbe_elen); 15684ab085aSmws oprintf(fp, "Entry Point Version: %u.%u\n", 15784ab085aSmws ep.smbe_major, ep.smbe_minor); 15884ab085aSmws oprintf(fp, "Max Structure Size: %u\n", ep.smbe_maxssize); 15984ab085aSmws oprintf(fp, "Entry Point Revision: 0x%x\n", ep.smbe_revision); 16084ab085aSmws 16184ab085aSmws oprintf(fp, "Entry Point Revision Data:"); 16284ab085aSmws for (i = 0; i < sizeof (ep.smbe_format); i++) 16384ab085aSmws oprintf(fp, " 0x%02x", ep.smbe_format[i]); 16484ab085aSmws oprintf(fp, "\n"); 16584ab085aSmws 16684ab085aSmws oprintf(fp, "Intermediate Anchor Tag: %*.*s\n", 16784ab085aSmws (int)sizeof (ep.smbe_ianchor), (int)sizeof (ep.smbe_ianchor), 16884ab085aSmws ep.smbe_ianchor); 16984ab085aSmws 17084ab085aSmws oprintf(fp, "Intermediate Checksum: 0x%x\n", ep.smbe_icksum); 17184ab085aSmws oprintf(fp, "Structure Table Length: %u\n", ep.smbe_stlen); 17284ab085aSmws oprintf(fp, "Structure Table Address: 0x%x\n", ep.smbe_staddr); 17384ab085aSmws oprintf(fp, "Structure Table Entries: %u\n", ep.smbe_stnum); 17484ab085aSmws oprintf(fp, "DMI BCD Revision: 0x%x\n", ep.smbe_bcdrev); 17584ab085aSmws } 17684ab085aSmws 17784ab085aSmws static void 17884ab085aSmws print_common(const smbios_info_t *ip, FILE *fp) 17984ab085aSmws { 18084ab085aSmws if (ip->smbi_manufacturer[0] != '\0') 18184ab085aSmws oprintf(fp, " Manufacturer: %s\n", ip->smbi_manufacturer); 18284ab085aSmws if (ip->smbi_product[0] != '\0') 18384ab085aSmws oprintf(fp, " Product: %s\n", ip->smbi_product); 18484ab085aSmws if (ip->smbi_version[0] != '\0') 18584ab085aSmws oprintf(fp, " Version: %s\n", ip->smbi_version); 18684ab085aSmws if (ip->smbi_serial[0] != '\0') 18784ab085aSmws oprintf(fp, " Serial Number: %s\n", ip->smbi_serial); 18884ab085aSmws if (ip->smbi_asset[0] != '\0') 18984ab085aSmws oprintf(fp, " Asset Tag: %s\n", ip->smbi_asset); 19084ab085aSmws if (ip->smbi_location[0] != '\0') 19184ab085aSmws oprintf(fp, " Location Tag: %s\n", ip->smbi_location); 19284ab085aSmws if (ip->smbi_part[0] != '\0') 19384ab085aSmws oprintf(fp, " Part Number: %s\n", ip->smbi_part); 19484ab085aSmws } 19584ab085aSmws 19684ab085aSmws static void 19784ab085aSmws print_bios(smbios_hdl_t *shp, FILE *fp) 19884ab085aSmws { 19984ab085aSmws smbios_bios_t b; 20084ab085aSmws 20184ab085aSmws (void) smbios_info_bios(shp, &b); 20284ab085aSmws 20384ab085aSmws oprintf(fp, " Vendor: %s\n", b.smbb_vendor); 20484ab085aSmws oprintf(fp, " Version String: %s\n", b.smbb_version); 20584ab085aSmws oprintf(fp, " Release Date: %s\n", b.smbb_reldate); 20684ab085aSmws oprintf(fp, " Address Segment: 0x%x\n", b.smbb_segment); 20784ab085aSmws oprintf(fp, " ROM Size: %u bytes\n", b.smbb_romsize); 20884ab085aSmws oprintf(fp, " Image Size: %u bytes\n", b.smbb_runsize); 20984ab085aSmws 21084ab085aSmws flag64_printf(fp, "Characteristics", 21184ab085aSmws b.smbb_cflags, sizeof (b.smbb_cflags) * NBBY, 21284ab085aSmws smbios_bios_flag_name, smbios_bios_flag_desc); 21384ab085aSmws 21484ab085aSmws if (b.smbb_nxcflags > SMB_BIOSXB_1) { 21584ab085aSmws flag_printf(fp, "Characteristics Extension Byte 1", 21684ab085aSmws b.smbb_xcflags[SMB_BIOSXB_1], 21784ab085aSmws sizeof (b.smbb_xcflags[SMB_BIOSXB_1]) * NBBY, 21884ab085aSmws smbios_bios_xb1_name, smbios_bios_xb1_desc); 21984ab085aSmws } 22084ab085aSmws 22184ab085aSmws if (b.smbb_nxcflags > SMB_BIOSXB_2) { 22284ab085aSmws flag_printf(fp, "Characteristics Extension Byte 2", 22384ab085aSmws b.smbb_xcflags[SMB_BIOSXB_2], 22484ab085aSmws sizeof (b.smbb_xcflags[SMB_BIOSXB_2]) * NBBY, 22584ab085aSmws smbios_bios_xb2_name, smbios_bios_xb2_desc); 22684ab085aSmws } 22784ab085aSmws 22884ab085aSmws if (b.smbb_nxcflags > SMB_BIOSXB_BIOS_MIN) { 22984ab085aSmws oprintf(fp, " Version Number: %u.%u\n", 23084ab085aSmws b.smbb_biosv.smbv_major, b.smbb_biosv.smbv_minor); 23184ab085aSmws } 23284ab085aSmws 23384ab085aSmws if (b.smbb_nxcflags > SMB_BIOSXB_ECFW_MIN) { 23484ab085aSmws oprintf(fp, " Embedded Ctlr Firmware Version Number: %u.%u\n", 23584ab085aSmws b.smbb_ecfwv.smbv_major, b.smbb_ecfwv.smbv_minor); 23684ab085aSmws } 23784ab085aSmws } 23884ab085aSmws 23984ab085aSmws static void 24084ab085aSmws print_system(smbios_hdl_t *shp, FILE *fp) 24184ab085aSmws { 24284ab085aSmws smbios_system_t s; 24384ab085aSmws uint_t i; 24484ab085aSmws 24584ab085aSmws (void) smbios_info_system(shp, &s); 24684ab085aSmws 24784ab085aSmws oprintf(fp, " UUID: "); 24884ab085aSmws for (i = 0; i < s.smbs_uuidlen; i++) { 24984ab085aSmws oprintf(fp, "%02x", s.smbs_uuid[i]); 25084ab085aSmws if (i == 3 || i == 5 || i == 7 || i == 9) 25184ab085aSmws oprintf(fp, "-"); 25284ab085aSmws } 25384ab085aSmws oprintf(fp, "\n"); 25484ab085aSmws 25584ab085aSmws desc_printf(smbios_system_wakeup_desc(s.smbs_wakeup), 25684ab085aSmws fp, " Wake-Up Event: 0x%x", s.smbs_wakeup); 25784ab085aSmws 25884ab085aSmws oprintf(fp, " SKU Number: %s\n", s.smbs_sku); 25984ab085aSmws oprintf(fp, " Family: %s\n", s.smbs_family); 26084ab085aSmws } 26184ab085aSmws 26284ab085aSmws static void 26384ab085aSmws print_bboard(smbios_hdl_t *shp, id_t id, FILE *fp) 26484ab085aSmws { 26584ab085aSmws smbios_bboard_t b; 266*074bb90dSTom Pothier int chdl_cnt; 26784ab085aSmws 26884ab085aSmws (void) smbios_info_bboard(shp, id, &b); 26984ab085aSmws 27084ab085aSmws oprintf(fp, " Chassis: %u\n", (uint_t)b.smbb_chassis); 27184ab085aSmws 27284ab085aSmws flag_printf(fp, "Flags", b.smbb_flags, sizeof (b.smbb_flags) * NBBY, 27384ab085aSmws smbios_bboard_flag_name, smbios_bboard_flag_desc); 27484ab085aSmws 27584ab085aSmws desc_printf(smbios_bboard_type_desc(b.smbb_type), 27684ab085aSmws fp, " Board Type: 0x%x", b.smbb_type); 277*074bb90dSTom Pothier 278*074bb90dSTom Pothier chdl_cnt = b.smbb_contn; 279*074bb90dSTom Pothier if (chdl_cnt != 0) { 280*074bb90dSTom Pothier id_t *chdl; 281*074bb90dSTom Pothier uint16_t hdl; 282*074bb90dSTom Pothier int i, n, cnt; 283*074bb90dSTom Pothier 284*074bb90dSTom Pothier chdl = alloca(chdl_cnt * sizeof (id_t)); 285*074bb90dSTom Pothier cnt = smbios_info_contains(shp, id, chdl_cnt, chdl); 286*074bb90dSTom Pothier if (cnt > SMB_CONT_MAX) 287*074bb90dSTom Pothier return; 288*074bb90dSTom Pothier n = MIN(chdl_cnt, cnt); 289*074bb90dSTom Pothier 290*074bb90dSTom Pothier oprintf(fp, "\n"); 291*074bb90dSTom Pothier for (i = 0; i < n; i++) { 292*074bb90dSTom Pothier hdl = (uint16_t)chdl[i]; 293*074bb90dSTom Pothier oprintf(fp, " Contained Handle: %u\n", hdl); 294*074bb90dSTom Pothier } 295*074bb90dSTom Pothier } 29684ab085aSmws } 29784ab085aSmws 29884ab085aSmws static void 29984ab085aSmws print_chassis(smbios_hdl_t *shp, id_t id, FILE *fp) 30084ab085aSmws { 30184ab085aSmws smbios_chassis_t c; 302*074bb90dSTom Pothier int elem_cnt; 30384ab085aSmws 30484ab085aSmws (void) smbios_info_chassis(shp, id, &c); 30584ab085aSmws 30684ab085aSmws oprintf(fp, " OEM Data: 0x%x\n", c.smbc_oemdata); 30784ab085aSmws oprintf(fp, " Lock Present: %s\n", c.smbc_lock ? "Y" : "N"); 30884ab085aSmws 30984ab085aSmws desc_printf(smbios_chassis_type_desc(c.smbc_type), 31084ab085aSmws fp, " Chassis Type: 0x%x", c.smbc_type); 31184ab085aSmws 31284ab085aSmws desc_printf(smbios_chassis_state_desc(c.smbc_bustate), 31384ab085aSmws fp, " Boot-Up State: 0x%x", c.smbc_bustate); 31484ab085aSmws 31584ab085aSmws desc_printf(smbios_chassis_state_desc(c.smbc_psstate), 31684ab085aSmws fp, " Power Supply State: 0x%x", c.smbc_psstate); 31784ab085aSmws 31884ab085aSmws desc_printf(smbios_chassis_state_desc(c.smbc_thstate), 31984ab085aSmws fp, " Thermal State: 0x%x", c.smbc_thstate); 32084ab085aSmws 32184ab085aSmws oprintf(fp, " Chassis Height: %uu\n", c.smbc_uheight); 32284ab085aSmws oprintf(fp, " Power Cords: %u\n", c.smbc_cords); 323*074bb90dSTom Pothier 324*074bb90dSTom Pothier elem_cnt = c.smbc_elems; 325*074bb90dSTom Pothier oprintf(fp, " Element Records: %u\n", elem_cnt); 326*074bb90dSTom Pothier 327*074bb90dSTom Pothier if (elem_cnt > 0) { 328*074bb90dSTom Pothier id_t *elems; 329*074bb90dSTom Pothier uint8_t type; 330*074bb90dSTom Pothier int i, n, cnt; 331*074bb90dSTom Pothier 332*074bb90dSTom Pothier elems = alloca(c.smbc_elems * sizeof (id_t)); 333*074bb90dSTom Pothier cnt = smbios_info_contains(shp, id, elem_cnt, elems); 334*074bb90dSTom Pothier if (cnt > SMB_CONT_MAX) 335*074bb90dSTom Pothier return; 336*074bb90dSTom Pothier n = MIN(elem_cnt, cnt); 337*074bb90dSTom Pothier 338*074bb90dSTom Pothier oprintf(fp, "\n"); 339*074bb90dSTom Pothier for (i = 0; i < n; i++) { 340*074bb90dSTom Pothier type = (uint8_t)elems[i]; 341*074bb90dSTom Pothier if (type & 0x80) { 342*074bb90dSTom Pothier /* SMBIOS structrure Type */ 343*074bb90dSTom Pothier desc_printf(smbios_type_name(type & 0x7f), fp, 344*074bb90dSTom Pothier " Contained SMBIOS structure Type: %u", 345*074bb90dSTom Pothier type & 0x80); 346*074bb90dSTom Pothier } else { 347*074bb90dSTom Pothier /* SMBIOS Base Board Type */ 348*074bb90dSTom Pothier desc_printf(smbios_bboard_type_desc(type), fp, 349*074bb90dSTom Pothier " Contained SMBIOS Base Board Type: 0x%x", 350*074bb90dSTom Pothier type); 351*074bb90dSTom Pothier } 352*074bb90dSTom Pothier } 353*074bb90dSTom Pothier } 35484ab085aSmws } 35584ab085aSmws 35684ab085aSmws static void 35784ab085aSmws print_processor(smbios_hdl_t *shp, id_t id, FILE *fp) 35884ab085aSmws { 35984ab085aSmws smbios_processor_t p; 36084ab085aSmws uint_t status; 36184ab085aSmws 36284ab085aSmws (void) smbios_info_processor(shp, id, &p); 36384ab085aSmws status = SMB_PRSTATUS_STATUS(p.smbp_status); 36484ab085aSmws 36584ab085aSmws desc_printf(smbios_processor_family_desc(p.smbp_family), 36684ab085aSmws fp, " Family: %u", p.smbp_family); 36784ab085aSmws 36884ab085aSmws oprintf(fp, " CPUID: 0x%llx\n", (u_longlong_t)p.smbp_cpuid); 36984ab085aSmws 37084ab085aSmws desc_printf(smbios_processor_type_desc(p.smbp_type), 37184ab085aSmws fp, " Type: %u", p.smbp_type); 37284ab085aSmws 37384ab085aSmws desc_printf(smbios_processor_upgrade_desc(p.smbp_upgrade), 37484ab085aSmws fp, " Socket Upgrade: %u", p.smbp_upgrade); 37584ab085aSmws 37684ab085aSmws oprintf(fp, " Socket Status: %s\n", 37784ab085aSmws SMB_PRSTATUS_PRESENT(p.smbp_status) ? 37884ab085aSmws "Populated" : "Not Populated"); 37984ab085aSmws 38084ab085aSmws desc_printf(smbios_processor_status_desc(status), 38184ab085aSmws fp, " Processor Status: %u", status); 38284ab085aSmws 38384ab085aSmws if (SMB_PRV_LEGACY(p.smbp_voltage)) { 38484ab085aSmws oprintf(fp, " Supported Voltages:"); 38584ab085aSmws switch (p.smbp_voltage) { 38684ab085aSmws case SMB_PRV_5V: 38784ab085aSmws oprintf(fp, " 5.0V"); 38884ab085aSmws break; 38984ab085aSmws case SMB_PRV_33V: 39084ab085aSmws oprintf(fp, " 3.3V"); 39184ab085aSmws break; 39284ab085aSmws case SMB_PRV_29V: 39384ab085aSmws oprintf(fp, " 2.9V"); 39484ab085aSmws break; 39584ab085aSmws } 39684ab085aSmws oprintf(fp, "\n"); 39784ab085aSmws } else { 39884ab085aSmws oprintf(fp, " Supported Voltages: %.1fV\n", 39984ab085aSmws (float)SMB_PRV_VOLTAGE(p.smbp_voltage) / 10); 40084ab085aSmws } 40184ab085aSmws 40284ab085aSmws if (p.smbp_clkspeed != 0) 40384ab085aSmws oprintf(fp, " External Clock Speed: %uMHz\n", p.smbp_clkspeed); 40484ab085aSmws else 40584ab085aSmws oprintf(fp, " External Clock Speed: Unknown\n"); 40684ab085aSmws 40784ab085aSmws if (p.smbp_maxspeed != 0) 40884ab085aSmws oprintf(fp, " Maximum Speed: %uMHz\n", p.smbp_maxspeed); 40984ab085aSmws else 41084ab085aSmws oprintf(fp, " Maximum Speed: Unknown\n"); 41184ab085aSmws 41284ab085aSmws if (p.smbp_curspeed != 0) 41384ab085aSmws oprintf(fp, " Current Speed: %uMHz\n", p.smbp_curspeed); 41484ab085aSmws else 41584ab085aSmws oprintf(fp, " Current Speed: Unknown\n"); 41684ab085aSmws 41784ab085aSmws id_printf(fp, " L1 Cache: ", p.smbp_l1cache); 41884ab085aSmws id_printf(fp, " L2 Cache: ", p.smbp_l2cache); 41984ab085aSmws id_printf(fp, " L3 Cache: ", p.smbp_l3cache); 42084ab085aSmws } 42184ab085aSmws 42284ab085aSmws static void 42384ab085aSmws print_cache(smbios_hdl_t *shp, id_t id, FILE *fp) 42484ab085aSmws { 42584ab085aSmws smbios_cache_t c; 42684ab085aSmws 42784ab085aSmws (void) smbios_info_cache(shp, id, &c); 42884ab085aSmws 42984ab085aSmws oprintf(fp, " Level: %u\n", c.smba_level); 43084ab085aSmws oprintf(fp, " Maximum Installed Size: %u bytes\n", c.smba_maxsize); 43184ab085aSmws 43284ab085aSmws if (c.smba_size != 0) 43384ab085aSmws oprintf(fp, " Installed Size: %u bytes\n", c.smba_size); 43484ab085aSmws else 43584ab085aSmws oprintf(fp, " Installed Size: Not Installed\n"); 43684ab085aSmws 43784ab085aSmws if (c.smba_speed != 0) 43884ab085aSmws oprintf(fp, " Speed: %uns\n", c.smba_speed); 43984ab085aSmws else 44084ab085aSmws oprintf(fp, " Speed: Unknown\n"); 44184ab085aSmws 44284ab085aSmws flag_printf(fp, "Supported SRAM Types", 44384ab085aSmws c.smba_stype, sizeof (c.smba_stype) * NBBY, 44484ab085aSmws smbios_cache_ctype_name, smbios_cache_ctype_desc); 44584ab085aSmws 44684ab085aSmws desc_printf(smbios_cache_ctype_desc(c.smba_ctype), 44784ab085aSmws fp, " Current SRAM Type: 0x%x", c.smba_ctype); 44884ab085aSmws 44984ab085aSmws desc_printf(smbios_cache_ecc_desc(c.smba_etype), 45084ab085aSmws fp, " Error Correction Type: %u", c.smba_etype); 45184ab085aSmws 45284ab085aSmws desc_printf(smbios_cache_logical_desc(c.smba_ltype), 45384ab085aSmws fp, " Logical Cache Type: %u", c.smba_ltype); 45484ab085aSmws 45584ab085aSmws desc_printf(smbios_cache_assoc_desc(c.smba_assoc), 45684ab085aSmws fp, " Associativity: %u", c.smba_assoc); 45784ab085aSmws 45884ab085aSmws desc_printf(smbios_cache_mode_desc(c.smba_mode), 45984ab085aSmws fp, " Mode: %u", c.smba_mode); 46084ab085aSmws 46184ab085aSmws desc_printf(smbios_cache_loc_desc(c.smba_location), 46284ab085aSmws fp, " Location: %u", c.smba_location); 46384ab085aSmws 46484ab085aSmws flag_printf(fp, "Flags", c.smba_flags, sizeof (c.smba_flags) * NBBY, 46584ab085aSmws smbios_cache_flag_name, smbios_cache_flag_desc); 46684ab085aSmws } 46784ab085aSmws 46884ab085aSmws static void 46984ab085aSmws print_port(smbios_hdl_t *shp, id_t id, FILE *fp) 47084ab085aSmws { 47184ab085aSmws smbios_port_t p; 47284ab085aSmws 47384ab085aSmws (void) smbios_info_port(shp, id, &p); 47484ab085aSmws 47584ab085aSmws oprintf(fp, " Internal Reference Designator: %s\n", p.smbo_iref); 47684ab085aSmws oprintf(fp, " External Reference Designator: %s\n", p.smbo_eref); 47784ab085aSmws 47884ab085aSmws desc_printf(smbios_port_conn_desc(p.smbo_itype), 47984ab085aSmws fp, " Internal Connector Type: %u", p.smbo_itype); 48084ab085aSmws 48184ab085aSmws desc_printf(smbios_port_conn_desc(p.smbo_etype), 48284ab085aSmws fp, " External Connector Type: %u", p.smbo_etype); 48384ab085aSmws 48484ab085aSmws desc_printf(smbios_port_type_desc(p.smbo_ptype), 48584ab085aSmws fp, " Port Type: %u", p.smbo_ptype); 48684ab085aSmws } 48784ab085aSmws 48884ab085aSmws static void 48984ab085aSmws print_slot(smbios_hdl_t *shp, id_t id, FILE *fp) 49084ab085aSmws { 49184ab085aSmws smbios_slot_t s; 49284ab085aSmws 49384ab085aSmws (void) smbios_info_slot(shp, id, &s); 49484ab085aSmws 49584ab085aSmws oprintf(fp, " Reference Designator: %s\n", s.smbl_name); 49684ab085aSmws oprintf(fp, " Slot ID: 0x%x\n", s.smbl_id); 49784ab085aSmws 49884ab085aSmws desc_printf(smbios_slot_type_desc(s.smbl_type), 49984ab085aSmws fp, " Type: 0x%x", s.smbl_type); 50084ab085aSmws 50184ab085aSmws desc_printf(smbios_slot_width_desc(s.smbl_width), 50284ab085aSmws fp, " Width: 0x%x", s.smbl_width); 50384ab085aSmws 50484ab085aSmws desc_printf(smbios_slot_usage_desc(s.smbl_usage), 50584ab085aSmws fp, " Usage: 0x%x", s.smbl_usage); 50684ab085aSmws 50784ab085aSmws desc_printf(smbios_slot_length_desc(s.smbl_length), 50884ab085aSmws fp, " Length: 0x%x", s.smbl_length); 50984ab085aSmws 51084ab085aSmws flag_printf(fp, "Slot Characteristics 1", 51184ab085aSmws s.smbl_ch1, sizeof (s.smbl_ch1) * NBBY, 51284ab085aSmws smbios_slot_ch1_name, smbios_slot_ch1_desc); 51384ab085aSmws 51484ab085aSmws flag_printf(fp, "Slot Characteristics 2", 51584ab085aSmws s.smbl_ch2, sizeof (s.smbl_ch2) * NBBY, 51684ab085aSmws smbios_slot_ch2_name, smbios_slot_ch2_desc); 51784ab085aSmws } 51884ab085aSmws 51984ab085aSmws static void 52084ab085aSmws print_obdevs(smbios_hdl_t *shp, id_t id, FILE *fp) 52184ab085aSmws { 52284ab085aSmws smbios_obdev_t *argv; 52384ab085aSmws int i, argc; 52484ab085aSmws 52584ab085aSmws if ((argc = smbios_info_obdevs(shp, id, 0, NULL)) > 0) { 52684ab085aSmws argv = alloca(sizeof (smbios_obdev_t) * argc); 52784ab085aSmws (void) smbios_info_obdevs(shp, id, argc, argv); 52884ab085aSmws for (i = 0; i < argc; i++) 52984ab085aSmws oprintf(fp, " %s\n", argv[i].smbd_name); 53084ab085aSmws } 53184ab085aSmws } 53284ab085aSmws 53384ab085aSmws static void 53484ab085aSmws print_strtab(smbios_hdl_t *shp, id_t id, FILE *fp) 53584ab085aSmws { 53684ab085aSmws const char **argv; 53784ab085aSmws int i, argc; 53884ab085aSmws 53984ab085aSmws if ((argc = smbios_info_strtab(shp, id, 0, NULL)) > 0) { 54084ab085aSmws argv = alloca(sizeof (char *) * argc); 54184ab085aSmws (void) smbios_info_strtab(shp, id, argc, argv); 54284ab085aSmws for (i = 0; i < argc; i++) 54384ab085aSmws oprintf(fp, " %s\n", argv[i]); 54484ab085aSmws } 54584ab085aSmws } 54684ab085aSmws 54784ab085aSmws static void 54884ab085aSmws print_lang(smbios_hdl_t *shp, id_t id, FILE *fp) 54984ab085aSmws { 55084ab085aSmws smbios_lang_t l; 55184ab085aSmws 55284ab085aSmws (void) smbios_info_lang(shp, &l); 55384ab085aSmws 55484ab085aSmws oprintf(fp, " Current Language: %s\n", l.smbla_cur); 55584ab085aSmws oprintf(fp, " Language String Format: %u\n", l.smbla_fmt); 55684ab085aSmws oprintf(fp, " Number of Installed Languages: %u\n", l.smbla_num); 55784ab085aSmws oprintf(fp, " Installed Languages:\n"); 55884ab085aSmws 55984ab085aSmws print_strtab(shp, id, fp); 56084ab085aSmws } 56184ab085aSmws 56284ab085aSmws /*ARGSUSED*/ 56384ab085aSmws static void 56484ab085aSmws print_evlog(smbios_hdl_t *shp, id_t id, FILE *fp) 56584ab085aSmws { 56684ab085aSmws smbios_evlog_t ev; 56784ab085aSmws uint32_t i; 56884ab085aSmws 56984ab085aSmws (void) smbios_info_eventlog(shp, &ev); 57084ab085aSmws 57184ab085aSmws oprintf(fp, " Log Area Size: %lu bytes\n", (ulong_t)ev.smbev_size); 57284ab085aSmws oprintf(fp, " Header Offset: %lu\n", (ulong_t)ev.smbev_hdr); 57384ab085aSmws oprintf(fp, " Data Offset: %lu\n", (ulong_t)ev.smbev_data); 57484ab085aSmws 57584ab085aSmws desc_printf(smbios_evlog_method_desc(ev.smbev_method), 57684ab085aSmws fp, " Data Access Method: %u", ev.smbev_method); 57784ab085aSmws 57884ab085aSmws flag_printf(fp, "Log Flags", 57984ab085aSmws ev.smbev_flags, sizeof (ev.smbev_flags) * NBBY, 58084ab085aSmws smbios_evlog_flag_name, smbios_evlog_flag_desc); 58184ab085aSmws 58284ab085aSmws desc_printf(smbios_evlog_format_desc(ev.smbev_format), 58384ab085aSmws fp, " Log Header Format: %u", ev.smbev_format); 58484ab085aSmws 58584ab085aSmws oprintf(fp, " Update Token: 0x%x\n", ev.smbev_token); 58684ab085aSmws oprintf(fp, " Data Access Address: "); 58784ab085aSmws 58884ab085aSmws switch (ev.smbev_method) { 58984ab085aSmws case SMB_EVM_1x1i_1x1d: 59084ab085aSmws case SMB_EVM_2x1i_1x1d: 59184ab085aSmws case SMB_EVM_1x2i_1x1d: 59284ab085aSmws oprintf(fp, "Index Address 0x%x, Data Address 0x%x\n", 59384ab085aSmws ev.smbev_addr.eva_io.evi_iaddr, 59484ab085aSmws ev.smbev_addr.eva_io.evi_daddr); 59584ab085aSmws break; 59684ab085aSmws case SMB_EVM_GPNV: 59784ab085aSmws oprintf(fp, "0x%x\n", ev.smbev_addr.eva_gpnv); 59884ab085aSmws break; 59984ab085aSmws default: 60084ab085aSmws oprintf(fp, "0x%x\n", ev.smbev_addr.eva_addr); 60184ab085aSmws } 60284ab085aSmws 60384ab085aSmws oprintf(fp, " Type Descriptors:\n"); 60484ab085aSmws 60584ab085aSmws for (i = 0; i < ev.smbev_typec; i++) { 60684ab085aSmws oprintf(fp, " %u: Log Type 0x%x, Data Type 0x%x\n", i, 60784ab085aSmws ev.smbev_typev[i].smbevt_ltype, 60884ab085aSmws ev.smbev_typev[i].smbevt_dtype); 60984ab085aSmws } 61084ab085aSmws } 61184ab085aSmws 61284ab085aSmws static void 61384ab085aSmws print_bytes(const uint8_t *data, size_t size, FILE *fp) 61484ab085aSmws { 61584ab085aSmws size_t row, rows = P2ROUNDUP(size, 16) / 16; 61684ab085aSmws size_t col, cols; 61784ab085aSmws 61884ab085aSmws char buf[17]; 61984ab085aSmws uint8_t x; 62084ab085aSmws 62184ab085aSmws oprintf(fp, "\n offset: 0 1 2 3 4 5 6 7 8 9 a b c d e f " 62284ab085aSmws "0123456789abcdef\n"); 62384ab085aSmws 62484ab085aSmws for (row = 0; row < rows; row++) { 62584ab085aSmws oprintf(fp, " %#4lx: ", (ulong_t)row * 16); 62684ab085aSmws cols = MIN(size - row * 16, 16); 62784ab085aSmws 62884ab085aSmws for (col = 0; col < cols; col++) { 62984ab085aSmws if (col % 4 == 0) 63084ab085aSmws oprintf(fp, " "); 63184ab085aSmws x = *data++; 63284ab085aSmws oprintf(fp, "%02x", x); 63384ab085aSmws buf[col] = x <= ' ' || x > '~' ? '.' : x; 63484ab085aSmws } 63584ab085aSmws 63684ab085aSmws for (; col < 16; col++) { 63784ab085aSmws if (col % 4 == 0) 63884ab085aSmws oprintf(fp, " "); 63984ab085aSmws oprintf(fp, " "); 64084ab085aSmws buf[col] = ' '; 64184ab085aSmws } 64284ab085aSmws 64384ab085aSmws buf[col] = '\0'; 64484ab085aSmws oprintf(fp, " %s\n", buf); 64584ab085aSmws } 64684ab085aSmws 64784ab085aSmws oprintf(fp, "\n"); 64884ab085aSmws } 64984ab085aSmws 65084ab085aSmws static void 65184ab085aSmws print_memarray(smbios_hdl_t *shp, id_t id, FILE *fp) 65284ab085aSmws { 65384ab085aSmws smbios_memarray_t ma; 65484ab085aSmws 65584ab085aSmws (void) smbios_info_memarray(shp, id, &ma); 65684ab085aSmws 65784ab085aSmws desc_printf(smbios_memarray_loc_desc(ma.smbma_location), 65884ab085aSmws fp, " Location: %u", ma.smbma_location); 65984ab085aSmws 66084ab085aSmws desc_printf(smbios_memarray_use_desc(ma.smbma_use), 66184ab085aSmws fp, " Use: %u", ma.smbma_use); 66284ab085aSmws 66384ab085aSmws desc_printf(smbios_memarray_ecc_desc(ma.smbma_ecc), 66484ab085aSmws fp, " ECC: %u", ma.smbma_ecc); 66584ab085aSmws 66684ab085aSmws oprintf(fp, " Number of Slots/Sockets: %u\n", ma.smbma_ndevs); 66784ab085aSmws id_printf(fp, " Memory Error Data: ", ma.smbma_err); 66884ab085aSmws oprintf(fp, " Max Capacity: %llu bytes\n", 66984ab085aSmws (u_longlong_t)ma.smbma_size); 67084ab085aSmws } 67184ab085aSmws 67284ab085aSmws static void 67384ab085aSmws print_memdevice(smbios_hdl_t *shp, id_t id, FILE *fp) 67484ab085aSmws { 67584ab085aSmws smbios_memdevice_t md; 67684ab085aSmws 67784ab085aSmws (void) smbios_info_memdevice(shp, id, &md); 67884ab085aSmws 67984ab085aSmws id_printf(fp, " Physical Memory Array: ", md.smbmd_array); 68084ab085aSmws id_printf(fp, " Memory Error Data: ", md.smbmd_error); 68184ab085aSmws 68284ab085aSmws if (md.smbmd_twidth != -1u) 68384ab085aSmws oprintf(fp, " Total Width: %u bits\n", md.smbmd_twidth); 68484ab085aSmws else 68584ab085aSmws oprintf(fp, " Total Width: Unknown\n"); 68684ab085aSmws 68784ab085aSmws if (md.smbmd_dwidth != -1u) 68884ab085aSmws oprintf(fp, " Data Width: %u bits\n", md.smbmd_dwidth); 68984ab085aSmws else 69084ab085aSmws oprintf(fp, " Data Width: Unknown\n"); 69184ab085aSmws 69284ab085aSmws switch (md.smbmd_size) { 69384ab085aSmws case -1ull: 69484ab085aSmws oprintf(fp, " Size: Unknown\n"); 69584ab085aSmws break; 69684ab085aSmws case 0: 69784ab085aSmws oprintf(fp, " Size: Not Populated\n"); 69884ab085aSmws break; 69984ab085aSmws default: 70084ab085aSmws oprintf(fp, " Size: %llu bytes\n", 70184ab085aSmws (u_longlong_t)md.smbmd_size); 70284ab085aSmws } 70384ab085aSmws 70484ab085aSmws desc_printf(smbios_memdevice_form_desc(md.smbmd_form), 70584ab085aSmws fp, " Form Factor: %u", md.smbmd_form); 70684ab085aSmws 70784ab085aSmws if (md.smbmd_set == 0) 70884ab085aSmws oprintf(fp, " Set: None\n"); 70984ab085aSmws else if (md.smbmd_set == (uint8_t)-1u) 71084ab085aSmws oprintf(fp, " Set: Unknown\n"); 71184ab085aSmws else 71284ab085aSmws oprintf(fp, " Set: %u\n", md.smbmd_set); 71384ab085aSmws 71484ab085aSmws desc_printf(smbios_memdevice_type_desc(md.smbmd_type), 71584ab085aSmws fp, " Memory Type: %u", md.smbmd_type); 71684ab085aSmws 71784ab085aSmws flag_printf(fp, "Flags", md.smbmd_flags, sizeof (md.smbmd_flags) * NBBY, 71884ab085aSmws smbios_memdevice_flag_name, smbios_memdevice_flag_desc); 71984ab085aSmws 72084ab085aSmws if (md.smbmd_speed != 0) 72184ab085aSmws oprintf(fp, " Speed: %uns\n", md.smbmd_speed); 72284ab085aSmws else 72384ab085aSmws oprintf(fp, " Speed: Unknown\n"); 72484ab085aSmws 72584ab085aSmws oprintf(fp, " Device Locator: %s\n", md.smbmd_dloc); 72684ab085aSmws oprintf(fp, " Bank Locator: %s\n", md.smbmd_bloc); 72784ab085aSmws } 72884ab085aSmws 72984ab085aSmws static void 73084ab085aSmws print_memarrmap(smbios_hdl_t *shp, id_t id, FILE *fp) 73184ab085aSmws { 73284ab085aSmws smbios_memarrmap_t ma; 73384ab085aSmws 73484ab085aSmws (void) smbios_info_memarrmap(shp, id, &ma); 73584ab085aSmws 73684ab085aSmws id_printf(fp, " Physical Memory Array: ", ma.smbmam_array); 73784ab085aSmws oprintf(fp, " Devices per Row: %u\n", ma.smbmam_width); 73884ab085aSmws 73984ab085aSmws oprintf(fp, " Physical Address: 0x%llx\n Size: %llu bytes\n", 74084ab085aSmws (u_longlong_t)ma.smbmam_addr, (u_longlong_t)ma.smbmam_size); 74184ab085aSmws } 74284ab085aSmws 74384ab085aSmws static void 74484ab085aSmws print_memdevmap(smbios_hdl_t *shp, id_t id, FILE *fp) 74584ab085aSmws { 74684ab085aSmws smbios_memdevmap_t md; 74784ab085aSmws 74884ab085aSmws (void) smbios_info_memdevmap(shp, id, &md); 74984ab085aSmws 75084ab085aSmws id_printf(fp, " Memory Device: ", md.smbmdm_device); 75184ab085aSmws id_printf(fp, " Memory Array Mapped Address: ", md.smbmdm_arrmap); 75284ab085aSmws 75384ab085aSmws oprintf(fp, " Physical Address: 0x%llx\n Size: %llu bytes\n", 75484ab085aSmws (u_longlong_t)md.smbmdm_addr, (u_longlong_t)md.smbmdm_size); 75584ab085aSmws 75684ab085aSmws oprintf(fp, " Partition Row Position: %u\n", md.smbmdm_rpos); 75784ab085aSmws oprintf(fp, " Interleave Position: %u\n", md.smbmdm_ipos); 75884ab085aSmws oprintf(fp, " Interleave Data Depth: %u\n", md.smbmdm_idepth); 75984ab085aSmws } 76084ab085aSmws 76184ab085aSmws static void 76284ab085aSmws print_hwsec(smbios_hdl_t *shp, FILE *fp) 76384ab085aSmws { 76484ab085aSmws smbios_hwsec_t h; 76584ab085aSmws 76684ab085aSmws (void) smbios_info_hwsec(shp, &h); 76784ab085aSmws 76884ab085aSmws desc_printf(smbios_hwsec_desc(h.smbh_pwr_ps), 76984ab085aSmws fp, " Power-On Password Status: %u", h.smbh_pwr_ps); 77084ab085aSmws desc_printf(smbios_hwsec_desc(h.smbh_kbd_ps), 77184ab085aSmws fp, " Keyboard Password Status: %u", h.smbh_kbd_ps); 77284ab085aSmws desc_printf(smbios_hwsec_desc(h.smbh_adm_ps), 77384ab085aSmws fp, " Administrator Password Status: %u", h.smbh_adm_ps); 77484ab085aSmws desc_printf(smbios_hwsec_desc(h.smbh_pan_ps), 77584ab085aSmws fp, " Front Panel Reset Status: %u", h.smbh_pan_ps); 77684ab085aSmws } 77784ab085aSmws 77884ab085aSmws static void 77984ab085aSmws print_boot(smbios_hdl_t *shp, FILE *fp) 78084ab085aSmws { 78184ab085aSmws smbios_boot_t b; 78284ab085aSmws 78384ab085aSmws (void) smbios_info_boot(shp, &b); 78484ab085aSmws 78584ab085aSmws desc_printf(smbios_boot_desc(b.smbt_status), 78684ab085aSmws fp, " Boot Status Code: 0x%x", b.smbt_status); 78784ab085aSmws 78884ab085aSmws if (b.smbt_size != 0) { 78984ab085aSmws oprintf(fp, " Boot Data (%lu bytes):\n", (ulong_t)b.smbt_size); 79084ab085aSmws print_bytes(b.smbt_data, b.smbt_size, fp); 79184ab085aSmws } 79284ab085aSmws } 79384ab085aSmws 79484ab085aSmws static void 79584ab085aSmws print_ipmi(smbios_hdl_t *shp, FILE *fp) 79684ab085aSmws { 79784ab085aSmws smbios_ipmi_t i; 79884ab085aSmws 79984ab085aSmws (void) smbios_info_ipmi(shp, &i); 80084ab085aSmws 80184ab085aSmws desc_printf(smbios_ipmi_type_desc(i.smbip_type), 80284ab085aSmws fp, " Type: %u", i.smbip_type); 80384ab085aSmws 80484ab085aSmws oprintf(fp, " BMC IPMI Version: %u.%u\n", 80584ab085aSmws i.smbip_vers.smbv_major, i.smbip_vers.smbv_minor); 80684ab085aSmws 80784ab085aSmws oprintf(fp, " i2c Bus Slave Address: 0x%x\n", i.smbip_i2c); 80884ab085aSmws oprintf(fp, " NV Storage Device Bus ID: 0x%x\n", i.smbip_bus); 80984ab085aSmws oprintf(fp, " BMC Base Address: 0x%llx\n", (u_longlong_t)i.smbip_addr); 81084ab085aSmws oprintf(fp, " Interrupt Number: %u\n", i.smbip_intr); 81184ab085aSmws oprintf(fp, " Register Spacing: %u\n", i.smbip_regspacing); 81284ab085aSmws 81384ab085aSmws flag_printf(fp, "Flags", i.smbip_flags, sizeof (i.smbip_flags) * NBBY, 81484ab085aSmws smbios_ipmi_flag_name, smbios_ipmi_flag_desc); 81584ab085aSmws } 81684ab085aSmws 81784ab085aSmws static int 818*074bb90dSTom Pothier check_oem(smbios_hdl_t *shp) 819*074bb90dSTom Pothier { 820*074bb90dSTom Pothier int i; 821*074bb90dSTom Pothier int cnt; 822*074bb90dSTom Pothier int rv; 823*074bb90dSTom Pothier id_t oem_id; 824*074bb90dSTom Pothier smbios_struct_t s; 825*074bb90dSTom Pothier const char **oem_str; 826*074bb90dSTom Pothier 827*074bb90dSTom Pothier rv = smbios_lookup_type(shp, SMB_TYPE_OEMSTR, &s); 828*074bb90dSTom Pothier if (rv != 0) { 829*074bb90dSTom Pothier return (-1); 830*074bb90dSTom Pothier } 831*074bb90dSTom Pothier 832*074bb90dSTom Pothier oem_id = s.smbstr_id; 833*074bb90dSTom Pothier 834*074bb90dSTom Pothier cnt = smbios_info_strtab(shp, oem_id, 0, NULL); 835*074bb90dSTom Pothier if (cnt > 0) { 836*074bb90dSTom Pothier oem_str = alloca(sizeof (char *) * cnt); 837*074bb90dSTom Pothier (void) smbios_info_strtab(shp, oem_id, cnt, oem_str); 838*074bb90dSTom Pothier 839*074bb90dSTom Pothier for (i = 0; i < cnt; i++) { 840*074bb90dSTom Pothier if (strncmp(oem_str[i], SMB_PRMS1, 841*074bb90dSTom Pothier strlen(SMB_PRMS1) + 1) == 0) { 842*074bb90dSTom Pothier return (0); 843*074bb90dSTom Pothier } 844*074bb90dSTom Pothier } 845*074bb90dSTom Pothier } 846*074bb90dSTom Pothier 847*074bb90dSTom Pothier return (-1); 848*074bb90dSTom Pothier } 849*074bb90dSTom Pothier 850*074bb90dSTom Pothier static void 851*074bb90dSTom Pothier print_extprocessor(smbios_hdl_t *shp, id_t id, FILE *fp) 852*074bb90dSTom Pothier { 853*074bb90dSTom Pothier int i; 854*074bb90dSTom Pothier smbios_processor_ext_t ep; 855*074bb90dSTom Pothier 856*074bb90dSTom Pothier if (check_oem(shp) != 0) 857*074bb90dSTom Pothier return; 858*074bb90dSTom Pothier 859*074bb90dSTom Pothier (void) smbios_info_extprocessor(shp, id, &ep); 860*074bb90dSTom Pothier 861*074bb90dSTom Pothier oprintf(fp, " Processor: %u\n", ep.smbpe_processor); 862*074bb90dSTom Pothier oprintf(fp, " FRU: %u\n", ep.smbpe_fru); 863*074bb90dSTom Pothier oprintf(fp, " Initial APIC ID count: %u\n\n", ep.smbpe_n); 864*074bb90dSTom Pothier 865*074bb90dSTom Pothier for (i = 0; i < ep.smbpe_n; i++) { 866*074bb90dSTom Pothier oprintf(fp, " Logical Strand %u: Initial APIC ID: %u\n", i, 867*074bb90dSTom Pothier ep.smbpe_apicid[i]); 868*074bb90dSTom Pothier } 869*074bb90dSTom Pothier } 870*074bb90dSTom Pothier 871*074bb90dSTom Pothier static void 872*074bb90dSTom Pothier print_pciexrc(smbios_hdl_t *shp, id_t id, FILE *fp) 873*074bb90dSTom Pothier { 874*074bb90dSTom Pothier smbios_pciexrc_t pcie; 875*074bb90dSTom Pothier 876*074bb90dSTom Pothier if (check_oem(shp) != 0) 877*074bb90dSTom Pothier return; 878*074bb90dSTom Pothier 879*074bb90dSTom Pothier (void) smbios_info_pciexrc(shp, id, &pcie); 880*074bb90dSTom Pothier 881*074bb90dSTom Pothier oprintf(fp, " Component ID: %u\n", pcie.smbpcie_bb); 882*074bb90dSTom Pothier oprintf(fp, " BDF: 0x%x\n", pcie.smbpcie_bdf); 883*074bb90dSTom Pothier } 884*074bb90dSTom Pothier 885*074bb90dSTom Pothier static void 886*074bb90dSTom Pothier print_extmemarray(smbios_hdl_t *shp, id_t id, FILE *fp) 887*074bb90dSTom Pothier { 888*074bb90dSTom Pothier smbios_memarray_ext_t em; 889*074bb90dSTom Pothier 890*074bb90dSTom Pothier if (check_oem(shp) != 0) 891*074bb90dSTom Pothier return; 892*074bb90dSTom Pothier 893*074bb90dSTom Pothier (void) smbios_info_extmemarray(shp, id, &em); 894*074bb90dSTom Pothier 895*074bb90dSTom Pothier oprintf(fp, " Physical Memory Array Handle: %u\n", em.smbmae_ma); 896*074bb90dSTom Pothier oprintf(fp, " Component Parent Handle: %u\n", em.smbmae_comp); 897*074bb90dSTom Pothier oprintf(fp, " BDF: 0x%x\n", em.smbmae_bdf); 898*074bb90dSTom Pothier } 899*074bb90dSTom Pothier 900*074bb90dSTom Pothier static void 901*074bb90dSTom Pothier print_extmemdevice(smbios_hdl_t *shp, id_t id, FILE *fp) 902*074bb90dSTom Pothier { 903*074bb90dSTom Pothier int i; 904*074bb90dSTom Pothier smbios_memdevice_ext_t emd; 905*074bb90dSTom Pothier 906*074bb90dSTom Pothier if (check_oem(shp) != 0) 907*074bb90dSTom Pothier return; 908*074bb90dSTom Pothier 909*074bb90dSTom Pothier (void) smbios_info_extmemdevice(shp, id, &emd); 910*074bb90dSTom Pothier 911*074bb90dSTom Pothier oprintf(fp, " Memory Device Handle: %u\n", emd.smbmdeve_md); 912*074bb90dSTom Pothier oprintf(fp, " DRAM Channel: %u\n", emd.smbmdeve_drch); 913*074bb90dSTom Pothier oprintf(fp, " Number of Chip Selects: %u\n", emd.smbmdeve_ncs); 914*074bb90dSTom Pothier 915*074bb90dSTom Pothier for (i = 0; i < emd.smbmdeve_ncs; i++) { 916*074bb90dSTom Pothier oprintf(fp, " Chip Select: %u\n", emd.smbmdeve_cs[i]); 917*074bb90dSTom Pothier } 918*074bb90dSTom Pothier } 919*074bb90dSTom Pothier 920*074bb90dSTom Pothier static int 92184ab085aSmws print_struct(smbios_hdl_t *shp, const smbios_struct_t *sp, void *fp) 92284ab085aSmws { 92384ab085aSmws smbios_info_t info; 92484ab085aSmws int hex = opt_x; 92584ab085aSmws const char *s; 92684ab085aSmws 92784ab085aSmws if (opt_t != -1 && opt_t != sp->smbstr_type) 92884ab085aSmws return (0); /* skip struct if type doesn't match -t */ 92984ab085aSmws 93084ab085aSmws if (!opt_O && (sp->smbstr_type == SMB_TYPE_MEMCTL || 93184ab085aSmws sp->smbstr_type == SMB_TYPE_MEMMOD)) 93284ab085aSmws return (0); /* skip struct if type is obsolete */ 93384ab085aSmws 93484ab085aSmws if (g_hdr++ == 0 || !opt_s) 93584ab085aSmws oprintf(fp, "%-5s %-4s %s\n", "ID", "SIZE", "TYPE"); 93684ab085aSmws 93784ab085aSmws oprintf(fp, "%-5u %-4lu", 93884ab085aSmws (uint_t)sp->smbstr_id, (ulong_t)sp->smbstr_size); 93984ab085aSmws 94084ab085aSmws if ((s = smbios_type_name(sp->smbstr_type)) != NULL) 94184ab085aSmws oprintf(fp, " %s", s); 94284ab085aSmws else if (sp->smbstr_type > SMB_TYPE_OEM_LO && 94384ab085aSmws sp->smbstr_type < SMB_TYPE_OEM_HI) 94484ab085aSmws oprintf(fp, " %s+%u", "SMB_TYPE_OEM_LO", 94584ab085aSmws sp->smbstr_type - SMB_TYPE_OEM_LO); 94684ab085aSmws else 94784ab085aSmws oprintf(fp, " %u", sp->smbstr_type); 94884ab085aSmws 94984ab085aSmws if ((s = smbios_type_desc(sp->smbstr_type)) != NULL) 95084ab085aSmws oprintf(fp, " (%s)\n", s); 95184ab085aSmws else 95284ab085aSmws oprintf(fp, "\n"); 95384ab085aSmws 95484ab085aSmws if (opt_s) 95584ab085aSmws return (0); /* only print header line if -s specified */ 95684ab085aSmws 95784ab085aSmws if (smbios_info_common(shp, sp->smbstr_id, &info) == 0) { 95884ab085aSmws oprintf(fp, "\n"); 95984ab085aSmws print_common(&info, fp); 96084ab085aSmws } 96184ab085aSmws 96284ab085aSmws switch (sp->smbstr_type) { 96384ab085aSmws case SMB_TYPE_BIOS: 96484ab085aSmws oprintf(fp, "\n"); 96584ab085aSmws print_bios(shp, fp); 96684ab085aSmws break; 96784ab085aSmws case SMB_TYPE_SYSTEM: 96884ab085aSmws oprintf(fp, "\n"); 96984ab085aSmws print_system(shp, fp); 97084ab085aSmws break; 97184ab085aSmws case SMB_TYPE_BASEBOARD: 97284ab085aSmws oprintf(fp, "\n"); 97384ab085aSmws print_bboard(shp, sp->smbstr_id, fp); 97484ab085aSmws break; 97584ab085aSmws case SMB_TYPE_CHASSIS: 97684ab085aSmws oprintf(fp, "\n"); 97784ab085aSmws print_chassis(shp, sp->smbstr_id, fp); 97884ab085aSmws break; 97984ab085aSmws case SMB_TYPE_PROCESSOR: 98084ab085aSmws oprintf(fp, "\n"); 98184ab085aSmws print_processor(shp, sp->smbstr_id, fp); 98284ab085aSmws break; 98384ab085aSmws case SMB_TYPE_CACHE: 98484ab085aSmws oprintf(fp, "\n"); 98584ab085aSmws print_cache(shp, sp->smbstr_id, fp); 98684ab085aSmws break; 98784ab085aSmws case SMB_TYPE_PORT: 98884ab085aSmws oprintf(fp, "\n"); 98984ab085aSmws print_port(shp, sp->smbstr_id, fp); 99084ab085aSmws break; 99184ab085aSmws case SMB_TYPE_SLOT: 99284ab085aSmws oprintf(fp, "\n"); 99384ab085aSmws print_slot(shp, sp->smbstr_id, fp); 99484ab085aSmws break; 99584ab085aSmws case SMB_TYPE_OBDEVS: 99684ab085aSmws oprintf(fp, "\n"); 99784ab085aSmws print_obdevs(shp, sp->smbstr_id, fp); 99884ab085aSmws break; 99984ab085aSmws case SMB_TYPE_OEMSTR: 100084ab085aSmws case SMB_TYPE_SYSCONFSTR: 100184ab085aSmws oprintf(fp, "\n"); 100284ab085aSmws print_strtab(shp, sp->smbstr_id, fp); 100384ab085aSmws break; 100484ab085aSmws case SMB_TYPE_LANG: 100584ab085aSmws oprintf(fp, "\n"); 100684ab085aSmws print_lang(shp, sp->smbstr_id, fp); 100784ab085aSmws break; 100884ab085aSmws case SMB_TYPE_EVENTLOG: 100984ab085aSmws oprintf(fp, "\n"); 101084ab085aSmws print_evlog(shp, sp->smbstr_id, fp); 101184ab085aSmws break; 101284ab085aSmws case SMB_TYPE_MEMARRAY: 101384ab085aSmws oprintf(fp, "\n"); 101484ab085aSmws print_memarray(shp, sp->smbstr_id, fp); 101584ab085aSmws break; 101684ab085aSmws case SMB_TYPE_MEMDEVICE: 101784ab085aSmws oprintf(fp, "\n"); 101884ab085aSmws print_memdevice(shp, sp->smbstr_id, fp); 101984ab085aSmws break; 102084ab085aSmws case SMB_TYPE_MEMARRAYMAP: 102184ab085aSmws oprintf(fp, "\n"); 102284ab085aSmws print_memarrmap(shp, sp->smbstr_id, fp); 102384ab085aSmws break; 102484ab085aSmws case SMB_TYPE_MEMDEVICEMAP: 102584ab085aSmws oprintf(fp, "\n"); 102684ab085aSmws print_memdevmap(shp, sp->smbstr_id, fp); 102784ab085aSmws break; 102884ab085aSmws case SMB_TYPE_SECURITY: 102984ab085aSmws oprintf(fp, "\n"); 103084ab085aSmws print_hwsec(shp, fp); 103184ab085aSmws break; 103284ab085aSmws case SMB_TYPE_BOOT: 103384ab085aSmws oprintf(fp, "\n"); 103484ab085aSmws print_boot(shp, fp); 103584ab085aSmws break; 103684ab085aSmws case SMB_TYPE_IPMIDEV: 103784ab085aSmws oprintf(fp, "\n"); 103884ab085aSmws print_ipmi(shp, fp); 103984ab085aSmws break; 1040*074bb90dSTom Pothier case SUN_OEM_EXT_PROCESSOR: 1041*074bb90dSTom Pothier oprintf(fp, "\n"); 1042*074bb90dSTom Pothier print_extprocessor(shp, sp->smbstr_id, fp); 1043*074bb90dSTom Pothier break; 1044*074bb90dSTom Pothier case SUN_OEM_PCIEXRC: 1045*074bb90dSTom Pothier oprintf(fp, "\n"); 1046*074bb90dSTom Pothier print_pciexrc(shp, sp->smbstr_id, fp); 1047*074bb90dSTom Pothier break; 1048*074bb90dSTom Pothier case SUN_OEM_EXT_MEMARRAY: 1049*074bb90dSTom Pothier oprintf(fp, "\n"); 1050*074bb90dSTom Pothier print_extmemarray(shp, sp->smbstr_id, fp); 1051*074bb90dSTom Pothier break; 1052*074bb90dSTom Pothier case SUN_OEM_EXT_MEMDEVICE: 1053*074bb90dSTom Pothier oprintf(fp, "\n"); 1054*074bb90dSTom Pothier print_extmemdevice(shp, sp->smbstr_id, fp); 1055*074bb90dSTom Pothier break; 105684ab085aSmws default: 105784ab085aSmws hex++; 105884ab085aSmws } 105984ab085aSmws 106084ab085aSmws if (hex) 106184ab085aSmws print_bytes(sp->smbstr_data, sp->smbstr_size, fp); 106284ab085aSmws else 106384ab085aSmws oprintf(fp, "\n"); 106484ab085aSmws 106584ab085aSmws return (0); 106684ab085aSmws } 106784ab085aSmws 106884ab085aSmws static uint16_t 106984ab085aSmws getu16(const char *name, const char *s) 107084ab085aSmws { 107184ab085aSmws u_longlong_t val; 107284ab085aSmws char *p; 107384ab085aSmws 107484ab085aSmws errno = 0; 107584ab085aSmws val = strtoull(s, &p, 0); 107684ab085aSmws 107784ab085aSmws if (errno != 0 || p == s || *p != '\0' || val > UINT16_MAX) { 107884ab085aSmws (void) fprintf(stderr, "%s: invalid %s argument -- %s\n", 107984ab085aSmws g_pname, name, s); 108084ab085aSmws exit(SMBIOS_USAGE); 108184ab085aSmws } 108284ab085aSmws 108384ab085aSmws return ((uint16_t)val); 108484ab085aSmws } 108584ab085aSmws 108684ab085aSmws static uint16_t 108784ab085aSmws getstype(const char *name, const char *s) 108884ab085aSmws { 108984ab085aSmws const char *ts; 109084ab085aSmws uint16_t t; 109184ab085aSmws 109284ab085aSmws for (t = 0; t < SMB_TYPE_OEM_LO; t++) { 109384ab085aSmws if ((ts = smbios_type_name(t)) != NULL && strcmp(s, ts) == 0) 109484ab085aSmws return (t); 109584ab085aSmws } 109684ab085aSmws 109784ab085aSmws (void) fprintf(stderr, "%s: invalid %s argument -- %s\n", 109884ab085aSmws g_pname, name, s); 109984ab085aSmws 110084ab085aSmws exit(SMBIOS_USAGE); 110184ab085aSmws /*NOTREACHED*/ 110284ab085aSmws } 110384ab085aSmws 110484ab085aSmws static int 110584ab085aSmws usage(FILE *fp) 110684ab085aSmws { 110784ab085aSmws (void) fprintf(fp, "Usage: %s " 110884ab085aSmws "[-BeOsx] [-i id] [-t type] [-w file] [file]\n\n", g_pname); 110984ab085aSmws 111084ab085aSmws (void) fprintf(fp, 111184ab085aSmws "\t-B disable header validation for broken BIOSes\n" 111284ab085aSmws "\t-e display SMBIOS entry point information\n" 111384ab085aSmws "\t-i display only the specified structure\n" 111484ab085aSmws "\t-O display obsolete structure types\n" 111584ab085aSmws "\t-s display only a summary of structure identifiers and types\n" 111684ab085aSmws "\t-t display only the specified structure type\n" 111784ab085aSmws "\t-w write the raw data to the specified file\n" 111884ab085aSmws "\t-x display raw data for structures\n"); 111984ab085aSmws 112084ab085aSmws return (SMBIOS_USAGE); 112184ab085aSmws } 112284ab085aSmws 112384ab085aSmws int 112484ab085aSmws main(int argc, char *argv[]) 112584ab085aSmws { 112684ab085aSmws const char *ifile = NULL; 112784ab085aSmws const char *ofile = NULL; 112884ab085aSmws int oflags = 0; 112984ab085aSmws 113084ab085aSmws smbios_hdl_t *shp; 113184ab085aSmws smbios_struct_t s; 113284ab085aSmws int err, fd, c; 113384ab085aSmws char *p; 113484ab085aSmws 113584ab085aSmws if ((p = strrchr(argv[0], '/')) == NULL) 113684ab085aSmws g_pname = argv[0]; 113784ab085aSmws else 113884ab085aSmws g_pname = p + 1; 113984ab085aSmws 114084ab085aSmws while (optind < argc) { 114184ab085aSmws while ((c = getopt(argc, argv, "Bei:Ost:w:xZ")) != EOF) { 114284ab085aSmws switch (c) { 114384ab085aSmws case 'B': 114484ab085aSmws oflags |= SMB_O_NOCKSUM | SMB_O_NOVERS; 114584ab085aSmws break; 114684ab085aSmws case 'e': 114784ab085aSmws opt_e++; 114884ab085aSmws break; 114984ab085aSmws case 'i': 115084ab085aSmws opt_i = getu16("struct ID", optarg); 115184ab085aSmws break; 115284ab085aSmws case 'O': 115384ab085aSmws opt_O++; 115484ab085aSmws break; 115584ab085aSmws case 's': 115684ab085aSmws opt_s++; 115784ab085aSmws break; 115884ab085aSmws case 't': 115984ab085aSmws if (isdigit(optarg[0])) 116084ab085aSmws opt_t = getu16("struct type", optarg); 116184ab085aSmws else 116284ab085aSmws opt_t = getstype("struct type", optarg); 116384ab085aSmws break; 116484ab085aSmws case 'w': 116584ab085aSmws ofile = optarg; 116684ab085aSmws break; 116784ab085aSmws case 'x': 116884ab085aSmws opt_x++; 116984ab085aSmws break; 117084ab085aSmws case 'Z': 117184ab085aSmws oflags |= SMB_O_ZIDS; /* undocumented */ 117284ab085aSmws break; 117384ab085aSmws default: 117484ab085aSmws return (usage(stderr)); 117584ab085aSmws } 117684ab085aSmws } 117784ab085aSmws 117884ab085aSmws if (optind < argc) { 117984ab085aSmws if (ifile != NULL) { 118084ab085aSmws (void) fprintf(stderr, "%s: illegal " 118184ab085aSmws "argument -- %s\n", g_pname, argv[optind]); 118284ab085aSmws return (SMBIOS_USAGE); 118384ab085aSmws } 118484ab085aSmws ifile = argv[optind++]; 118584ab085aSmws } 118684ab085aSmws } 118784ab085aSmws 118884ab085aSmws if ((shp = smbios_open(ifile, SMB_VERSION, oflags, &err)) == NULL) { 118984ab085aSmws (void) fprintf(stderr, "%s: failed to load SMBIOS: %s\n", 119084ab085aSmws g_pname, smbios_errmsg(err)); 119184ab085aSmws return (SMBIOS_ERROR); 119284ab085aSmws } 119384ab085aSmws 119484ab085aSmws if (ofile != NULL) { 119584ab085aSmws if ((fd = open(ofile, O_WRONLY|O_CREAT|O_TRUNC, 0666)) == -1) { 119684ab085aSmws (void) fprintf(stderr, "%s: failed to open %s: %s\n", 119784ab085aSmws g_pname, ofile, strerror(errno)); 119884ab085aSmws err = SMBIOS_ERROR; 119984ab085aSmws } else if (smbios_write(shp, fd) != 0) { 120084ab085aSmws (void) fprintf(stderr, "%s: failed to write %s: %s\n", 120184ab085aSmws g_pname, ofile, smbios_errmsg(smbios_errno(shp))); 120284ab085aSmws err = SMBIOS_ERROR; 120384ab085aSmws } 120484ab085aSmws smbios_close(shp); 120584ab085aSmws return (err); 120684ab085aSmws } 120784ab085aSmws 120884ab085aSmws if (opt_e) { 120984ab085aSmws print_smbios(shp, stdout); 121084ab085aSmws smbios_close(shp); 121184ab085aSmws return (SMBIOS_SUCCESS); 121284ab085aSmws } 121384ab085aSmws 121484ab085aSmws if (opt_O && (opt_i != -1 || opt_t != -1)) 121584ab085aSmws opt_O++; /* -i or -t imply displaying obsolete records */ 121684ab085aSmws 121784ab085aSmws if (opt_i != -1) 121884ab085aSmws err = smbios_lookup_id(shp, opt_i, &s); 121984ab085aSmws else 122084ab085aSmws err = smbios_iter(shp, print_struct, stdout); 122184ab085aSmws 122284ab085aSmws if (err != 0) { 122384ab085aSmws (void) fprintf(stderr, "%s: failed to access SMBIOS: %s\n", 122484ab085aSmws g_pname, smbios_errmsg(smbios_errno(shp))); 122584ab085aSmws smbios_close(shp); 122684ab085aSmws return (SMBIOS_ERROR); 122784ab085aSmws } 122884ab085aSmws 122984ab085aSmws if (opt_i != -1) 123084ab085aSmws (void) print_struct(shp, &s, stdout); 123184ab085aSmws 123284ab085aSmws smbios_close(shp); 123384ab085aSmws return (SMBIOS_SUCCESS); 123484ab085aSmws } 1235