184ab085aSmws /* 284ab085aSmws * CDDL HEADER START 384ab085aSmws * 484ab085aSmws * The contents of this file are subject to the terms of the 5074bb90dSTom Pothier * Common Development and Distribution License (the "License"). 6074bb90dSTom 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 /* 234e901881SDale Ghent * Copyright 2015 OmniTI Computer Consulting, Inc. All rights reserved. 241f6171acSRobert Mustacchi * Copyright (c) 2017, Joyent, Inc. 2503f9f63dSTom Pothier * Copyright 2010 Sun Microsystems, Inc. All rights reserved. 2684ab085aSmws * Use is subject to license terms. 2784ab085aSmws */ 2884ab085aSmws 2984ab085aSmws #include <sys/sysmacros.h> 3084ab085aSmws #include <sys/param.h> 3184ab085aSmws 3284ab085aSmws #include <smbios.h> 3384ab085aSmws #include <alloca.h> 3484ab085aSmws #include <limits.h> 3584ab085aSmws #include <unistd.h> 3684ab085aSmws #include <strings.h> 3784ab085aSmws #include <stdlib.h> 3884ab085aSmws #include <stdarg.h> 3984ab085aSmws #include <stdio.h> 4084ab085aSmws #include <fcntl.h> 4184ab085aSmws #include <errno.h> 4284ab085aSmws #include <ctype.h> 4384ab085aSmws 4484ab085aSmws #define SMBIOS_SUCCESS 0 4584ab085aSmws #define SMBIOS_ERROR 1 4684ab085aSmws #define SMBIOS_USAGE 2 4784ab085aSmws 4884ab085aSmws static const char *g_pname; 4984ab085aSmws static int g_hdr; 5084ab085aSmws 5184ab085aSmws static int opt_e; 5284ab085aSmws static int opt_i = -1; 5384ab085aSmws static int opt_O; 5484ab085aSmws static int opt_s; 5584ab085aSmws static int opt_t = -1; 5684ab085aSmws static int opt_x; 5784ab085aSmws 5884ab085aSmws /*PRINTFLIKE2*/ 5984ab085aSmws static void 602b9d2074SRobert Mustacchi smbios_warn(smbios_hdl_t *shp, const char *format, ...) 612b9d2074SRobert Mustacchi { 622b9d2074SRobert Mustacchi va_list ap; 632b9d2074SRobert Mustacchi 642b9d2074SRobert Mustacchi va_start(ap, format); 652b9d2074SRobert Mustacchi (void) vfprintf(stderr, format, ap); 662b9d2074SRobert Mustacchi va_end(ap); 672b9d2074SRobert Mustacchi 682b9d2074SRobert Mustacchi if (shp != NULL) { 692b9d2074SRobert Mustacchi (void) fprintf(stderr, ": %s", 702b9d2074SRobert Mustacchi smbios_errmsg(smbios_errno(shp))); 712b9d2074SRobert Mustacchi } 722b9d2074SRobert Mustacchi 732b9d2074SRobert Mustacchi (void) fprintf(stderr, "\n"); 742b9d2074SRobert Mustacchi } 752b9d2074SRobert Mustacchi 762b9d2074SRobert Mustacchi /*PRINTFLIKE2*/ 772b9d2074SRobert Mustacchi static void 7884ab085aSmws oprintf(FILE *fp, const char *format, ...) 7984ab085aSmws { 8084ab085aSmws va_list ap; 8184ab085aSmws 8284ab085aSmws va_start(ap, format); 8384ab085aSmws (void) vfprintf(fp, format, ap); 8484ab085aSmws va_end(ap); 8584ab085aSmws } 8684ab085aSmws 8784ab085aSmws /*PRINTFLIKE3*/ 8884ab085aSmws static void 8984ab085aSmws desc_printf(const char *d, FILE *fp, const char *format, ...) 9084ab085aSmws { 9184ab085aSmws va_list ap; 9284ab085aSmws 9384ab085aSmws va_start(ap, format); 9484ab085aSmws (void) vfprintf(fp, format, ap); 9584ab085aSmws va_end(ap); 9684ab085aSmws 9784ab085aSmws if (d != NULL) 9884ab085aSmws (void) fprintf(fp, " (%s)\n", d); 9984ab085aSmws else 10084ab085aSmws (void) fprintf(fp, "\n"); 10184ab085aSmws } 10284ab085aSmws 10384ab085aSmws static void 10484ab085aSmws flag_printf(FILE *fp, const char *s, uint_t flags, size_t bits, 10584ab085aSmws const char *(*flag_name)(uint_t), const char *(*flag_desc)(uint_t)) 10684ab085aSmws { 10784ab085aSmws size_t i; 10884ab085aSmws 10984ab085aSmws oprintf(fp, " %s: 0x%x\n", s, flags); 11084ab085aSmws 11184ab085aSmws for (i = 0; i < bits; i++) { 11284ab085aSmws uint_t f = 1 << i; 11384ab085aSmws const char *n; 11484ab085aSmws 11584ab085aSmws if (!(flags & f)) 11684ab085aSmws continue; 11784ab085aSmws 11884ab085aSmws if ((n = flag_name(f)) != NULL) 11984ab085aSmws desc_printf(flag_desc(f), fp, "\t%s", n); 12084ab085aSmws else 12184ab085aSmws desc_printf(flag_desc(f), fp, "\t0x%x", f); 12284ab085aSmws } 12384ab085aSmws } 12484ab085aSmws 12584ab085aSmws static void 12684ab085aSmws flag64_printf(FILE *fp, const char *s, uint64_t flags, size_t bits, 12784ab085aSmws const char *(*flag_name)(uint64_t), const char *(*flag_desc)(uint64_t)) 12884ab085aSmws { 12984ab085aSmws size_t i; 13084ab085aSmws 13184ab085aSmws oprintf(fp, " %s: 0x%llx\n", s, (u_longlong_t)flags); 13284ab085aSmws 13384ab085aSmws for (i = 0; i < bits; i++) { 13484ab085aSmws u_longlong_t f = 1ULL << i; 13584ab085aSmws const char *n; 13684ab085aSmws 13784ab085aSmws if (!(flags & f)) 13884ab085aSmws continue; 13984ab085aSmws 14084ab085aSmws if ((n = flag_name(f)) != NULL) 14184ab085aSmws desc_printf(flag_desc(f), fp, "\t%s", n); 14284ab085aSmws else 14384ab085aSmws desc_printf(flag_desc(f), fp, "\t0x%llx", f); 14484ab085aSmws } 14584ab085aSmws } 14684ab085aSmws 14784ab085aSmws static void 14884ab085aSmws id_printf(FILE *fp, const char *s, id_t id) 14984ab085aSmws { 15084ab085aSmws switch (id) { 15184ab085aSmws case SMB_ID_NONE: 15284ab085aSmws oprintf(fp, "%sNone\n", s); 15384ab085aSmws break; 15484ab085aSmws case SMB_ID_NOTSUP: 15584ab085aSmws oprintf(fp, "%sNot Supported\n", s); 15684ab085aSmws break; 15784ab085aSmws default: 15884ab085aSmws oprintf(fp, "%s%u\n", s, (uint_t)id); 15984ab085aSmws } 16084ab085aSmws } 16184ab085aSmws 16203f9f63dSTom Pothier static int 16303f9f63dSTom Pothier check_oem(smbios_hdl_t *shp) 16403f9f63dSTom Pothier { 16503f9f63dSTom Pothier int i; 16603f9f63dSTom Pothier int cnt; 16703f9f63dSTom Pothier int rv; 16803f9f63dSTom Pothier id_t oem_id; 16903f9f63dSTom Pothier smbios_struct_t s; 17003f9f63dSTom Pothier const char **oem_str; 17103f9f63dSTom Pothier 17203f9f63dSTom Pothier rv = smbios_lookup_type(shp, SMB_TYPE_OEMSTR, &s); 17303f9f63dSTom Pothier if (rv != 0) { 17403f9f63dSTom Pothier return (-1); 17503f9f63dSTom Pothier } 17603f9f63dSTom Pothier 17703f9f63dSTom Pothier oem_id = s.smbstr_id; 17803f9f63dSTom Pothier 17903f9f63dSTom Pothier cnt = smbios_info_strtab(shp, oem_id, 0, NULL); 18003f9f63dSTom Pothier if (cnt > 0) { 18103f9f63dSTom Pothier oem_str = alloca(sizeof (char *) * cnt); 18203f9f63dSTom Pothier (void) smbios_info_strtab(shp, oem_id, cnt, oem_str); 18303f9f63dSTom Pothier 18403f9f63dSTom Pothier for (i = 0; i < cnt; i++) { 18503f9f63dSTom Pothier if (strncmp(oem_str[i], SMB_PRMS1, 18603f9f63dSTom Pothier strlen(SMB_PRMS1) + 1) == 0) { 18703f9f63dSTom Pothier return (0); 18803f9f63dSTom Pothier } 18903f9f63dSTom Pothier } 19003f9f63dSTom Pothier } 19103f9f63dSTom Pothier 19203f9f63dSTom Pothier return (-1); 19303f9f63dSTom Pothier } 19403f9f63dSTom Pothier 19584ab085aSmws static void 196c325726fSToomas Soome print_smbios_21(smbios_21_entry_t *ep, FILE *fp) 19784ab085aSmws { 19884ab085aSmws int i; 19984ab085aSmws 20084ab085aSmws oprintf(fp, "Entry Point Anchor Tag: %*.*s\n", 201c325726fSToomas Soome (int)sizeof (ep->smbe_eanchor), (int)sizeof (ep->smbe_eanchor), 202c325726fSToomas Soome ep->smbe_eanchor); 20384ab085aSmws 204c325726fSToomas Soome oprintf(fp, "Entry Point Checksum: 0x%x\n", ep->smbe_ecksum); 205c325726fSToomas Soome oprintf(fp, "Entry Point Length: %u\n", ep->smbe_elen); 20684ab085aSmws oprintf(fp, "Entry Point Version: %u.%u\n", 207c325726fSToomas Soome ep->smbe_major, ep->smbe_minor); 208c325726fSToomas Soome oprintf(fp, "Max Structure Size: %u\n", ep->smbe_maxssize); 209c325726fSToomas Soome oprintf(fp, "Entry Point Revision: 0x%x\n", ep->smbe_revision); 21084ab085aSmws 21184ab085aSmws oprintf(fp, "Entry Point Revision Data:"); 212c325726fSToomas Soome for (i = 0; i < sizeof (ep->smbe_format); i++) 213c325726fSToomas Soome oprintf(fp, " 0x%02x", ep->smbe_format[i]); 21484ab085aSmws oprintf(fp, "\n"); 21584ab085aSmws 21684ab085aSmws oprintf(fp, "Intermediate Anchor Tag: %*.*s\n", 217c325726fSToomas Soome (int)sizeof (ep->smbe_ianchor), (int)sizeof (ep->smbe_ianchor), 218c325726fSToomas Soome ep->smbe_ianchor); 21984ab085aSmws 220c325726fSToomas Soome oprintf(fp, "Intermediate Checksum: 0x%x\n", ep->smbe_icksum); 221c325726fSToomas Soome oprintf(fp, "Structure Table Length: %u\n", ep->smbe_stlen); 222c325726fSToomas Soome oprintf(fp, "Structure Table Address: 0x%x\n", ep->smbe_staddr); 223c325726fSToomas Soome oprintf(fp, "Structure Table Entries: %u\n", ep->smbe_stnum); 224c325726fSToomas Soome oprintf(fp, "DMI BCD Revision: 0x%x\n", ep->smbe_bcdrev); 225c325726fSToomas Soome } 226c325726fSToomas Soome 227c325726fSToomas Soome static void 228c325726fSToomas Soome print_smbios_30(smbios_30_entry_t *ep, FILE *fp) 229c325726fSToomas Soome { 230c325726fSToomas Soome oprintf(fp, "Entry Point Anchor Tag: %*.*s\n", 231c325726fSToomas Soome (int)sizeof (ep->smbe_eanchor), (int)sizeof (ep->smbe_eanchor), 232c325726fSToomas Soome ep->smbe_eanchor); 233c325726fSToomas Soome 234c325726fSToomas Soome oprintf(fp, "Entry Point Checksum: 0x%x\n", ep->smbe_ecksum); 235c325726fSToomas Soome oprintf(fp, "Entry Point Length: %u\n", ep->smbe_elen); 236c325726fSToomas Soome oprintf(fp, "SMBIOS Version: %u.%u\n", 237c325726fSToomas Soome ep->smbe_major, ep->smbe_minor); 238c325726fSToomas Soome oprintf(fp, "SMBIOS DocRev: 0x%x\n", ep->smbe_docrev); 239c325726fSToomas Soome oprintf(fp, "Entry Point Revision: 0x%x\n", ep->smbe_revision); 240c325726fSToomas Soome 241c325726fSToomas Soome oprintf(fp, "Structure Table Length: %u\n", ep->smbe_stlen); 242c325726fSToomas Soome oprintf(fp, "Structure Table Address: 0x%" PRIx64 "\n", 243c325726fSToomas Soome ep->smbe_staddr); 244c325726fSToomas Soome } 245c325726fSToomas Soome 246c325726fSToomas Soome static void 247c325726fSToomas Soome print_smbios(smbios_hdl_t *shp, FILE *fp) 248c325726fSToomas Soome { 249c325726fSToomas Soome smbios_entry_t ep; 250c325726fSToomas Soome 251c325726fSToomas Soome switch (smbios_info_smbios(shp, &ep)) { 252c325726fSToomas Soome case SMBIOS_ENTRY_POINT_21: 253c325726fSToomas Soome print_smbios_21(&ep.ep21, fp); 254c325726fSToomas Soome break; 255c325726fSToomas Soome case SMBIOS_ENTRY_POINT_30: 256c325726fSToomas Soome print_smbios_30(&ep.ep30, fp); 257c325726fSToomas Soome break; 258c325726fSToomas Soome } 25984ab085aSmws } 26084ab085aSmws 26184ab085aSmws static void 26284ab085aSmws print_common(const smbios_info_t *ip, FILE *fp) 26384ab085aSmws { 26484ab085aSmws if (ip->smbi_manufacturer[0] != '\0') 26584ab085aSmws oprintf(fp, " Manufacturer: %s\n", ip->smbi_manufacturer); 26684ab085aSmws if (ip->smbi_product[0] != '\0') 26784ab085aSmws oprintf(fp, " Product: %s\n", ip->smbi_product); 26884ab085aSmws if (ip->smbi_version[0] != '\0') 26984ab085aSmws oprintf(fp, " Version: %s\n", ip->smbi_version); 27084ab085aSmws if (ip->smbi_serial[0] != '\0') 27184ab085aSmws oprintf(fp, " Serial Number: %s\n", ip->smbi_serial); 27284ab085aSmws if (ip->smbi_asset[0] != '\0') 27384ab085aSmws oprintf(fp, " Asset Tag: %s\n", ip->smbi_asset); 27484ab085aSmws if (ip->smbi_location[0] != '\0') 27584ab085aSmws oprintf(fp, " Location Tag: %s\n", ip->smbi_location); 27684ab085aSmws if (ip->smbi_part[0] != '\0') 27784ab085aSmws oprintf(fp, " Part Number: %s\n", ip->smbi_part); 27884ab085aSmws } 27984ab085aSmws 28084ab085aSmws static void 28184ab085aSmws print_bios(smbios_hdl_t *shp, FILE *fp) 28284ab085aSmws { 28384ab085aSmws smbios_bios_t b; 28484ab085aSmws 28584ab085aSmws (void) smbios_info_bios(shp, &b); 28684ab085aSmws 28784ab085aSmws oprintf(fp, " Vendor: %s\n", b.smbb_vendor); 28884ab085aSmws oprintf(fp, " Version String: %s\n", b.smbb_version); 28984ab085aSmws oprintf(fp, " Release Date: %s\n", b.smbb_reldate); 29084ab085aSmws oprintf(fp, " Address Segment: 0x%x\n", b.smbb_segment); 29138d76b18SRobert Mustacchi oprintf(fp, " ROM Size: %" PRIu64 " bytes\n", b.smbb_extromsize); 29284ab085aSmws oprintf(fp, " Image Size: %u bytes\n", b.smbb_runsize); 29384ab085aSmws 29484ab085aSmws flag64_printf(fp, "Characteristics", 29584ab085aSmws b.smbb_cflags, sizeof (b.smbb_cflags) * NBBY, 29684ab085aSmws smbios_bios_flag_name, smbios_bios_flag_desc); 29784ab085aSmws 29884ab085aSmws if (b.smbb_nxcflags > SMB_BIOSXB_1) { 29984ab085aSmws flag_printf(fp, "Characteristics Extension Byte 1", 30084ab085aSmws b.smbb_xcflags[SMB_BIOSXB_1], 30184ab085aSmws sizeof (b.smbb_xcflags[SMB_BIOSXB_1]) * NBBY, 30284ab085aSmws smbios_bios_xb1_name, smbios_bios_xb1_desc); 30384ab085aSmws } 30484ab085aSmws 30584ab085aSmws if (b.smbb_nxcflags > SMB_BIOSXB_2) { 30684ab085aSmws flag_printf(fp, "Characteristics Extension Byte 2", 30784ab085aSmws b.smbb_xcflags[SMB_BIOSXB_2], 30884ab085aSmws sizeof (b.smbb_xcflags[SMB_BIOSXB_2]) * NBBY, 30984ab085aSmws smbios_bios_xb2_name, smbios_bios_xb2_desc); 31084ab085aSmws } 31184ab085aSmws 31284ab085aSmws if (b.smbb_nxcflags > SMB_BIOSXB_BIOS_MIN) { 31384ab085aSmws oprintf(fp, " Version Number: %u.%u\n", 31484ab085aSmws b.smbb_biosv.smbv_major, b.smbb_biosv.smbv_minor); 31584ab085aSmws } 31684ab085aSmws 317*45807aa8SRobert Mustacchi /* 318*45807aa8SRobert Mustacchi * If the major and minor versions are 0xff then that indicates that the 319*45807aa8SRobert Mustacchi * embedded controller does not exist. 320*45807aa8SRobert Mustacchi */ 321*45807aa8SRobert Mustacchi if (b.smbb_nxcflags > SMB_BIOSXB_ECFW_MIN && 322*45807aa8SRobert Mustacchi b.smbb_ecfwv.smbv_major != 0xff && 323*45807aa8SRobert Mustacchi b.smbb_ecfwv.smbv_minor != 0xff) { 32484ab085aSmws oprintf(fp, " Embedded Ctlr Firmware Version Number: %u.%u\n", 32584ab085aSmws b.smbb_ecfwv.smbv_major, b.smbb_ecfwv.smbv_minor); 32684ab085aSmws } 32784ab085aSmws } 32884ab085aSmws 32984ab085aSmws static void 33084ab085aSmws print_system(smbios_hdl_t *shp, FILE *fp) 33184ab085aSmws { 33284ab085aSmws smbios_system_t s; 33384ab085aSmws uint_t i; 33484ab085aSmws 33584ab085aSmws (void) smbios_info_system(shp, &s); 33684ab085aSmws 33784ab085aSmws oprintf(fp, " UUID: "); 33884ab085aSmws for (i = 0; i < s.smbs_uuidlen; i++) { 33984ab085aSmws oprintf(fp, "%02x", s.smbs_uuid[i]); 34084ab085aSmws if (i == 3 || i == 5 || i == 7 || i == 9) 34184ab085aSmws oprintf(fp, "-"); 34284ab085aSmws } 34384ab085aSmws oprintf(fp, "\n"); 34484ab085aSmws 34584ab085aSmws desc_printf(smbios_system_wakeup_desc(s.smbs_wakeup), 34684ab085aSmws fp, " Wake-Up Event: 0x%x", s.smbs_wakeup); 34784ab085aSmws 34884ab085aSmws oprintf(fp, " SKU Number: %s\n", s.smbs_sku); 34984ab085aSmws oprintf(fp, " Family: %s\n", s.smbs_family); 35084ab085aSmws } 35184ab085aSmws 35284ab085aSmws static void 35384ab085aSmws print_bboard(smbios_hdl_t *shp, id_t id, FILE *fp) 35484ab085aSmws { 35584ab085aSmws smbios_bboard_t b; 356074bb90dSTom Pothier int chdl_cnt; 35784ab085aSmws 35884ab085aSmws (void) smbios_info_bboard(shp, id, &b); 35984ab085aSmws 36084ab085aSmws oprintf(fp, " Chassis: %u\n", (uint_t)b.smbb_chassis); 36184ab085aSmws 36284ab085aSmws flag_printf(fp, "Flags", b.smbb_flags, sizeof (b.smbb_flags) * NBBY, 36384ab085aSmws smbios_bboard_flag_name, smbios_bboard_flag_desc); 36484ab085aSmws 36584ab085aSmws desc_printf(smbios_bboard_type_desc(b.smbb_type), 36684ab085aSmws fp, " Board Type: 0x%x", b.smbb_type); 367074bb90dSTom Pothier 368074bb90dSTom Pothier chdl_cnt = b.smbb_contn; 369074bb90dSTom Pothier if (chdl_cnt != 0) { 370074bb90dSTom Pothier id_t *chdl; 371074bb90dSTom Pothier uint16_t hdl; 372074bb90dSTom Pothier int i, n, cnt; 373074bb90dSTom Pothier 374074bb90dSTom Pothier chdl = alloca(chdl_cnt * sizeof (id_t)); 375074bb90dSTom Pothier cnt = smbios_info_contains(shp, id, chdl_cnt, chdl); 376074bb90dSTom Pothier if (cnt > SMB_CONT_MAX) 377074bb90dSTom Pothier return; 378074bb90dSTom Pothier n = MIN(chdl_cnt, cnt); 379074bb90dSTom Pothier 380074bb90dSTom Pothier oprintf(fp, "\n"); 381074bb90dSTom Pothier for (i = 0; i < n; i++) { 382074bb90dSTom Pothier hdl = (uint16_t)chdl[i]; 383074bb90dSTom Pothier oprintf(fp, " Contained Handle: %u\n", hdl); 384074bb90dSTom Pothier } 385074bb90dSTom Pothier } 38684ab085aSmws } 38784ab085aSmws 38884ab085aSmws static void 38984ab085aSmws print_chassis(smbios_hdl_t *shp, id_t id, FILE *fp) 39084ab085aSmws { 39184ab085aSmws smbios_chassis_t c; 392074bb90dSTom Pothier int elem_cnt; 39384ab085aSmws 39484ab085aSmws (void) smbios_info_chassis(shp, id, &c); 39584ab085aSmws 39684ab085aSmws oprintf(fp, " OEM Data: 0x%x\n", c.smbc_oemdata); 3974e901881SDale Ghent oprintf(fp, " SKU number: %s\n", 39873b433d1SYuri Pankov c.smbc_sku[0] == '\0' ? "<unknown>" : c.smbc_sku); 39984ab085aSmws oprintf(fp, " Lock Present: %s\n", c.smbc_lock ? "Y" : "N"); 40084ab085aSmws 40184ab085aSmws desc_printf(smbios_chassis_type_desc(c.smbc_type), 40284ab085aSmws fp, " Chassis Type: 0x%x", c.smbc_type); 40384ab085aSmws 40484ab085aSmws desc_printf(smbios_chassis_state_desc(c.smbc_bustate), 40584ab085aSmws fp, " Boot-Up State: 0x%x", c.smbc_bustate); 40684ab085aSmws 40784ab085aSmws desc_printf(smbios_chassis_state_desc(c.smbc_psstate), 40884ab085aSmws fp, " Power Supply State: 0x%x", c.smbc_psstate); 40984ab085aSmws 41084ab085aSmws desc_printf(smbios_chassis_state_desc(c.smbc_thstate), 41184ab085aSmws fp, " Thermal State: 0x%x", c.smbc_thstate); 41284ab085aSmws 41384ab085aSmws oprintf(fp, " Chassis Height: %uu\n", c.smbc_uheight); 41484ab085aSmws oprintf(fp, " Power Cords: %u\n", c.smbc_cords); 415074bb90dSTom Pothier 416074bb90dSTom Pothier elem_cnt = c.smbc_elems; 417074bb90dSTom Pothier oprintf(fp, " Element Records: %u\n", elem_cnt); 418074bb90dSTom Pothier 419074bb90dSTom Pothier if (elem_cnt > 0) { 420074bb90dSTom Pothier id_t *elems; 421074bb90dSTom Pothier uint8_t type; 422074bb90dSTom Pothier int i, n, cnt; 423074bb90dSTom Pothier 424074bb90dSTom Pothier elems = alloca(c.smbc_elems * sizeof (id_t)); 425074bb90dSTom Pothier cnt = smbios_info_contains(shp, id, elem_cnt, elems); 426074bb90dSTom Pothier if (cnt > SMB_CONT_MAX) 427074bb90dSTom Pothier return; 428074bb90dSTom Pothier n = MIN(elem_cnt, cnt); 429074bb90dSTom Pothier 430074bb90dSTom Pothier oprintf(fp, "\n"); 431074bb90dSTom Pothier for (i = 0; i < n; i++) { 432074bb90dSTom Pothier type = (uint8_t)elems[i]; 433074bb90dSTom Pothier if (type & 0x80) { 434074bb90dSTom Pothier /* SMBIOS structrure Type */ 435074bb90dSTom Pothier desc_printf(smbios_type_name(type & 0x7f), fp, 436074bb90dSTom Pothier " Contained SMBIOS structure Type: %u", 437074bb90dSTom Pothier type & 0x80); 438074bb90dSTom Pothier } else { 439074bb90dSTom Pothier /* SMBIOS Base Board Type */ 440074bb90dSTom Pothier desc_printf(smbios_bboard_type_desc(type), fp, 441074bb90dSTom Pothier " Contained SMBIOS Base Board Type: 0x%x", 442074bb90dSTom Pothier type); 443074bb90dSTom Pothier } 444074bb90dSTom Pothier } 445074bb90dSTom Pothier } 44684ab085aSmws } 44784ab085aSmws 44884ab085aSmws static void 44984ab085aSmws print_processor(smbios_hdl_t *shp, id_t id, FILE *fp) 45084ab085aSmws { 45184ab085aSmws smbios_processor_t p; 45284ab085aSmws uint_t status; 45384ab085aSmws 45484ab085aSmws (void) smbios_info_processor(shp, id, &p); 45584ab085aSmws status = SMB_PRSTATUS_STATUS(p.smbp_status); 45684ab085aSmws 45784ab085aSmws desc_printf(smbios_processor_family_desc(p.smbp_family), 45884ab085aSmws fp, " Family: %u", p.smbp_family); 45984ab085aSmws 4604e901881SDale Ghent if (p.smbp_family2 != 0) 4614e901881SDale Ghent desc_printf(smbios_processor_family_desc(p.smbp_family2), 4624e901881SDale Ghent fp, " Family Ext: %u", p.smbp_family2); 4634e901881SDale Ghent 46484ab085aSmws oprintf(fp, " CPUID: 0x%llx\n", (u_longlong_t)p.smbp_cpuid); 46584ab085aSmws 46684ab085aSmws desc_printf(smbios_processor_type_desc(p.smbp_type), 46784ab085aSmws fp, " Type: %u", p.smbp_type); 46884ab085aSmws 46984ab085aSmws desc_printf(smbios_processor_upgrade_desc(p.smbp_upgrade), 47084ab085aSmws fp, " Socket Upgrade: %u", p.smbp_upgrade); 47184ab085aSmws 47284ab085aSmws oprintf(fp, " Socket Status: %s\n", 47384ab085aSmws SMB_PRSTATUS_PRESENT(p.smbp_status) ? 47484ab085aSmws "Populated" : "Not Populated"); 47584ab085aSmws 47684ab085aSmws desc_printf(smbios_processor_status_desc(status), 47784ab085aSmws fp, " Processor Status: %u", status); 47884ab085aSmws 47984ab085aSmws if (SMB_PRV_LEGACY(p.smbp_voltage)) { 48084ab085aSmws oprintf(fp, " Supported Voltages:"); 48184ab085aSmws switch (p.smbp_voltage) { 48284ab085aSmws case SMB_PRV_5V: 48384ab085aSmws oprintf(fp, " 5.0V"); 48484ab085aSmws break; 48584ab085aSmws case SMB_PRV_33V: 48684ab085aSmws oprintf(fp, " 3.3V"); 48784ab085aSmws break; 48884ab085aSmws case SMB_PRV_29V: 48984ab085aSmws oprintf(fp, " 2.9V"); 49084ab085aSmws break; 49184ab085aSmws } 49284ab085aSmws oprintf(fp, "\n"); 49384ab085aSmws } else { 49484ab085aSmws oprintf(fp, " Supported Voltages: %.1fV\n", 49584ab085aSmws (float)SMB_PRV_VOLTAGE(p.smbp_voltage) / 10); 49684ab085aSmws } 49784ab085aSmws 4986734c4b0SRobert Mustacchi if (p.smbp_corecount != 0) { 4996734c4b0SRobert Mustacchi if (p.smbp_corecount != 0xff || p.smbp_corecount2 == 0) 5004e901881SDale Ghent oprintf(fp, " Core Count: %u\n", p.smbp_corecount); 5014e901881SDale Ghent else 5026734c4b0SRobert Mustacchi oprintf(fp, " Core Count: %u\n", p.smbp_corecount2); 5036734c4b0SRobert Mustacchi } else { 5044e901881SDale Ghent oprintf(fp, " Core Count: Unknown\n"); 5056734c4b0SRobert Mustacchi } 5064e901881SDale Ghent 5076734c4b0SRobert Mustacchi if (p.smbp_coresenabled != 0) { 5086734c4b0SRobert Mustacchi if (p.smbp_coresenabled != 0xff || p.smbp_coresenabled2 == 0) { 5096734c4b0SRobert Mustacchi oprintf(fp, " Cores Enabled: %u\n", 5106734c4b0SRobert Mustacchi p.smbp_coresenabled); 5116734c4b0SRobert Mustacchi } else { 5126734c4b0SRobert Mustacchi oprintf(fp, " Cores Enabled: %u\n", 5136734c4b0SRobert Mustacchi p.smbp_coresenabled2); 5146734c4b0SRobert Mustacchi } 5156734c4b0SRobert Mustacchi } else { 5164e901881SDale Ghent oprintf(fp, " Cores Enabled: Unknown\n"); 5176734c4b0SRobert Mustacchi } 5184e901881SDale Ghent 5196734c4b0SRobert Mustacchi if (p.smbp_threadcount != 0) { 5206734c4b0SRobert Mustacchi if (p.smbp_threadcount != 0xff || p.smbp_threadcount2 == 0) { 5216734c4b0SRobert Mustacchi oprintf(fp, " Thread Count: %u\n", 5226734c4b0SRobert Mustacchi p.smbp_threadcount); 5236734c4b0SRobert Mustacchi } else { 5246734c4b0SRobert Mustacchi oprintf(fp, " Thread Count: %u\n", 5256734c4b0SRobert Mustacchi p.smbp_threadcount2); 5266734c4b0SRobert Mustacchi } 5276734c4b0SRobert Mustacchi } else { 5284e901881SDale Ghent oprintf(fp, " Thread Count: Unknown\n"); 5296734c4b0SRobert Mustacchi } 5304e901881SDale Ghent 5314e901881SDale Ghent if (p.smbp_cflags) { 5324e901881SDale Ghent flag_printf(fp, "Processor Characteristics", 5334e901881SDale Ghent p.smbp_cflags, sizeof (p.smbp_cflags) * NBBY, 5344e901881SDale Ghent smbios_processor_core_flag_name, 5354e901881SDale Ghent smbios_processor_core_flag_desc); 5364e901881SDale Ghent } 5374e901881SDale Ghent 53884ab085aSmws if (p.smbp_clkspeed != 0) 53984ab085aSmws oprintf(fp, " External Clock Speed: %uMHz\n", p.smbp_clkspeed); 54084ab085aSmws else 54184ab085aSmws oprintf(fp, " External Clock Speed: Unknown\n"); 54284ab085aSmws 54384ab085aSmws if (p.smbp_maxspeed != 0) 54484ab085aSmws oprintf(fp, " Maximum Speed: %uMHz\n", p.smbp_maxspeed); 54584ab085aSmws else 54684ab085aSmws oprintf(fp, " Maximum Speed: Unknown\n"); 54784ab085aSmws 54884ab085aSmws if (p.smbp_curspeed != 0) 54984ab085aSmws oprintf(fp, " Current Speed: %uMHz\n", p.smbp_curspeed); 55084ab085aSmws else 55184ab085aSmws oprintf(fp, " Current Speed: Unknown\n"); 55284ab085aSmws 5532b9d2074SRobert Mustacchi id_printf(fp, " L1 Cache Handle: ", p.smbp_l1cache); 5542b9d2074SRobert Mustacchi id_printf(fp, " L2 Cache Handle: ", p.smbp_l2cache); 5552b9d2074SRobert Mustacchi id_printf(fp, " L3 Cache Handle: ", p.smbp_l3cache); 55684ab085aSmws } 55784ab085aSmws 55884ab085aSmws static void 55984ab085aSmws print_cache(smbios_hdl_t *shp, id_t id, FILE *fp) 56084ab085aSmws { 56184ab085aSmws smbios_cache_t c; 56284ab085aSmws 56384ab085aSmws (void) smbios_info_cache(shp, id, &c); 56484ab085aSmws 56584ab085aSmws oprintf(fp, " Level: %u\n", c.smba_level); 56638d76b18SRobert Mustacchi oprintf(fp, " Maximum Installed Size: %" PRIu64 " bytes\n", 56738d76b18SRobert Mustacchi c.smba_maxsize2); 56884ab085aSmws 56938d76b18SRobert Mustacchi if (c.smba_size2 != 0) { 57038d76b18SRobert Mustacchi oprintf(fp, " Installed Size: %" PRIu64 " bytes\n", 57138d76b18SRobert Mustacchi c.smba_size2); 57238d76b18SRobert Mustacchi } else { 57384ab085aSmws oprintf(fp, " Installed Size: Not Installed\n"); 57438d76b18SRobert Mustacchi } 57584ab085aSmws 57684ab085aSmws if (c.smba_speed != 0) 57784ab085aSmws oprintf(fp, " Speed: %uns\n", c.smba_speed); 57884ab085aSmws else 57984ab085aSmws oprintf(fp, " Speed: Unknown\n"); 58084ab085aSmws 58184ab085aSmws flag_printf(fp, "Supported SRAM Types", 58284ab085aSmws c.smba_stype, sizeof (c.smba_stype) * NBBY, 58384ab085aSmws smbios_cache_ctype_name, smbios_cache_ctype_desc); 58484ab085aSmws 58584ab085aSmws desc_printf(smbios_cache_ctype_desc(c.smba_ctype), 58684ab085aSmws fp, " Current SRAM Type: 0x%x", c.smba_ctype); 58784ab085aSmws 58884ab085aSmws desc_printf(smbios_cache_ecc_desc(c.smba_etype), 58984ab085aSmws fp, " Error Correction Type: %u", c.smba_etype); 59084ab085aSmws 59184ab085aSmws desc_printf(smbios_cache_logical_desc(c.smba_ltype), 59284ab085aSmws fp, " Logical Cache Type: %u", c.smba_ltype); 59384ab085aSmws 59484ab085aSmws desc_printf(smbios_cache_assoc_desc(c.smba_assoc), 59584ab085aSmws fp, " Associativity: %u", c.smba_assoc); 59684ab085aSmws 59784ab085aSmws desc_printf(smbios_cache_mode_desc(c.smba_mode), 59884ab085aSmws fp, " Mode: %u", c.smba_mode); 59984ab085aSmws 60084ab085aSmws desc_printf(smbios_cache_loc_desc(c.smba_location), 60184ab085aSmws fp, " Location: %u", c.smba_location); 60284ab085aSmws 60384ab085aSmws flag_printf(fp, "Flags", c.smba_flags, sizeof (c.smba_flags) * NBBY, 60484ab085aSmws smbios_cache_flag_name, smbios_cache_flag_desc); 60584ab085aSmws } 60684ab085aSmws 60784ab085aSmws static void 60884ab085aSmws print_port(smbios_hdl_t *shp, id_t id, FILE *fp) 60984ab085aSmws { 61084ab085aSmws smbios_port_t p; 61184ab085aSmws 61284ab085aSmws (void) smbios_info_port(shp, id, &p); 61384ab085aSmws 61484ab085aSmws oprintf(fp, " Internal Reference Designator: %s\n", p.smbo_iref); 61584ab085aSmws oprintf(fp, " External Reference Designator: %s\n", p.smbo_eref); 61684ab085aSmws 61784ab085aSmws desc_printf(smbios_port_conn_desc(p.smbo_itype), 61884ab085aSmws fp, " Internal Connector Type: %u", p.smbo_itype); 61984ab085aSmws 62084ab085aSmws desc_printf(smbios_port_conn_desc(p.smbo_etype), 62184ab085aSmws fp, " External Connector Type: %u", p.smbo_etype); 62284ab085aSmws 62384ab085aSmws desc_printf(smbios_port_type_desc(p.smbo_ptype), 62484ab085aSmws fp, " Port Type: %u", p.smbo_ptype); 62584ab085aSmws } 62684ab085aSmws 62784ab085aSmws static void 62884ab085aSmws print_slot(smbios_hdl_t *shp, id_t id, FILE *fp) 62984ab085aSmws { 63084ab085aSmws smbios_slot_t s; 631c325726fSToomas Soome smbios_version_t v; 63284ab085aSmws 63384ab085aSmws (void) smbios_info_slot(shp, id, &s); 634c325726fSToomas Soome smbios_info_smbios_version(shp, &v); 63584ab085aSmws 63684ab085aSmws oprintf(fp, " Reference Designator: %s\n", s.smbl_name); 63784ab085aSmws oprintf(fp, " Slot ID: 0x%x\n", s.smbl_id); 63884ab085aSmws 63984ab085aSmws desc_printf(smbios_slot_type_desc(s.smbl_type), 64084ab085aSmws fp, " Type: 0x%x", s.smbl_type); 64184ab085aSmws 64284ab085aSmws desc_printf(smbios_slot_width_desc(s.smbl_width), 64384ab085aSmws fp, " Width: 0x%x", s.smbl_width); 64484ab085aSmws 64584ab085aSmws desc_printf(smbios_slot_usage_desc(s.smbl_usage), 64684ab085aSmws fp, " Usage: 0x%x", s.smbl_usage); 64784ab085aSmws 64884ab085aSmws desc_printf(smbios_slot_length_desc(s.smbl_length), 64984ab085aSmws fp, " Length: 0x%x", s.smbl_length); 65084ab085aSmws 65184ab085aSmws flag_printf(fp, "Slot Characteristics 1", 65284ab085aSmws s.smbl_ch1, sizeof (s.smbl_ch1) * NBBY, 65384ab085aSmws smbios_slot_ch1_name, smbios_slot_ch1_desc); 65484ab085aSmws 65584ab085aSmws flag_printf(fp, "Slot Characteristics 2", 65684ab085aSmws s.smbl_ch2, sizeof (s.smbl_ch2) * NBBY, 65784ab085aSmws smbios_slot_ch2_name, smbios_slot_ch2_desc); 65803f9f63dSTom Pothier 659c325726fSToomas Soome if (check_oem(shp) != 0 && (v.smbv_major < 2 || v.smbv_minor < 6)) 66003f9f63dSTom Pothier return; 66103f9f63dSTom Pothier 66203f9f63dSTom Pothier oprintf(fp, " Segment Group: %u\n", s.smbl_sg); 66303f9f63dSTom Pothier oprintf(fp, " Bus Number: %u\n", s.smbl_bus); 66403f9f63dSTom Pothier oprintf(fp, " Device/Function Number: %u\n", s.smbl_df); 66503f9f63dSTom Pothier } 66603f9f63dSTom Pothier 66703f9f63dSTom Pothier static void 66803f9f63dSTom Pothier print_obdevs_ext(smbios_hdl_t *shp, id_t id, FILE *fp) 66903f9f63dSTom Pothier { 6706734c4b0SRobert Mustacchi boolean_t enabled; 67103f9f63dSTom Pothier smbios_obdev_ext_t oe; 6726734c4b0SRobert Mustacchi const char *type; 67303f9f63dSTom Pothier 67403f9f63dSTom Pothier (void) smbios_info_obdevs_ext(shp, id, &oe); 67503f9f63dSTom Pothier 6766734c4b0SRobert Mustacchi /* 6776734c4b0SRobert Mustacchi * Bit 7 is always whether or not the device is enabled while bits 0:6 6786734c4b0SRobert Mustacchi * are the actual device type. 6796734c4b0SRobert Mustacchi */ 6806734c4b0SRobert Mustacchi enabled = oe.smboe_dtype >> 7; 6816734c4b0SRobert Mustacchi type = smbios_onboard_type_desc(oe.smboe_dtype & 0x7f); 6826734c4b0SRobert Mustacchi 68303f9f63dSTom Pothier oprintf(fp, " Reference Designator: %s\n", oe.smboe_name); 6846734c4b0SRobert Mustacchi oprintf(fp, " Device Enabled: %s\n", enabled == B_TRUE ? "true" : 6856734c4b0SRobert Mustacchi "false"); 6866734c4b0SRobert Mustacchi oprintf(fp, " Device Type: %s\n", type); 68703f9f63dSTom Pothier oprintf(fp, " Device Type Instance: %u\n", oe.smboe_dti); 68803f9f63dSTom Pothier oprintf(fp, " Segment Group Number: %u\n", oe.smboe_sg); 68903f9f63dSTom Pothier oprintf(fp, " Bus Number: %u\n", oe.smboe_bus); 69003f9f63dSTom Pothier oprintf(fp, " Device/Function Number: %u\n", oe.smboe_df); 69184ab085aSmws } 69284ab085aSmws 69384ab085aSmws static void 69484ab085aSmws print_obdevs(smbios_hdl_t *shp, id_t id, FILE *fp) 69584ab085aSmws { 69684ab085aSmws smbios_obdev_t *argv; 69784ab085aSmws int i, argc; 69884ab085aSmws 69984ab085aSmws if ((argc = smbios_info_obdevs(shp, id, 0, NULL)) > 0) { 70084ab085aSmws argv = alloca(sizeof (smbios_obdev_t) * argc); 70184ab085aSmws (void) smbios_info_obdevs(shp, id, argc, argv); 70284ab085aSmws for (i = 0; i < argc; i++) 70384ab085aSmws oprintf(fp, " %s\n", argv[i].smbd_name); 70484ab085aSmws } 70584ab085aSmws } 70684ab085aSmws 70784ab085aSmws static void 70884ab085aSmws print_strtab(smbios_hdl_t *shp, id_t id, FILE *fp) 70984ab085aSmws { 71084ab085aSmws const char **argv; 71184ab085aSmws int i, argc; 71284ab085aSmws 71384ab085aSmws if ((argc = smbios_info_strtab(shp, id, 0, NULL)) > 0) { 71484ab085aSmws argv = alloca(sizeof (char *) * argc); 71584ab085aSmws (void) smbios_info_strtab(shp, id, argc, argv); 71684ab085aSmws for (i = 0; i < argc; i++) 71784ab085aSmws oprintf(fp, " %s\n", argv[i]); 71884ab085aSmws } 71984ab085aSmws } 72084ab085aSmws 72184ab085aSmws static void 72284ab085aSmws print_lang(smbios_hdl_t *shp, id_t id, FILE *fp) 72384ab085aSmws { 72484ab085aSmws smbios_lang_t l; 72584ab085aSmws 72684ab085aSmws (void) smbios_info_lang(shp, &l); 72784ab085aSmws 72884ab085aSmws oprintf(fp, " Current Language: %s\n", l.smbla_cur); 72984ab085aSmws oprintf(fp, " Language String Format: %u\n", l.smbla_fmt); 73084ab085aSmws oprintf(fp, " Number of Installed Languages: %u\n", l.smbla_num); 73184ab085aSmws oprintf(fp, " Installed Languages:\n"); 73284ab085aSmws 73384ab085aSmws print_strtab(shp, id, fp); 73484ab085aSmws } 73584ab085aSmws 73684ab085aSmws /*ARGSUSED*/ 73784ab085aSmws static void 73884ab085aSmws print_evlog(smbios_hdl_t *shp, id_t id, FILE *fp) 73984ab085aSmws { 74084ab085aSmws smbios_evlog_t ev; 74184ab085aSmws uint32_t i; 74284ab085aSmws 74384ab085aSmws (void) smbios_info_eventlog(shp, &ev); 74484ab085aSmws 74584ab085aSmws oprintf(fp, " Log Area Size: %lu bytes\n", (ulong_t)ev.smbev_size); 74684ab085aSmws oprintf(fp, " Header Offset: %lu\n", (ulong_t)ev.smbev_hdr); 74784ab085aSmws oprintf(fp, " Data Offset: %lu\n", (ulong_t)ev.smbev_data); 74884ab085aSmws 74984ab085aSmws desc_printf(smbios_evlog_method_desc(ev.smbev_method), 75084ab085aSmws fp, " Data Access Method: %u", ev.smbev_method); 75184ab085aSmws 75284ab085aSmws flag_printf(fp, "Log Flags", 75384ab085aSmws ev.smbev_flags, sizeof (ev.smbev_flags) * NBBY, 75484ab085aSmws smbios_evlog_flag_name, smbios_evlog_flag_desc); 75584ab085aSmws 75684ab085aSmws desc_printf(smbios_evlog_format_desc(ev.smbev_format), 75784ab085aSmws fp, " Log Header Format: %u", ev.smbev_format); 75884ab085aSmws 75984ab085aSmws oprintf(fp, " Update Token: 0x%x\n", ev.smbev_token); 76084ab085aSmws oprintf(fp, " Data Access Address: "); 76184ab085aSmws 76284ab085aSmws switch (ev.smbev_method) { 76384ab085aSmws case SMB_EVM_1x1i_1x1d: 76484ab085aSmws case SMB_EVM_2x1i_1x1d: 76584ab085aSmws case SMB_EVM_1x2i_1x1d: 76684ab085aSmws oprintf(fp, "Index Address 0x%x, Data Address 0x%x\n", 76784ab085aSmws ev.smbev_addr.eva_io.evi_iaddr, 76884ab085aSmws ev.smbev_addr.eva_io.evi_daddr); 76984ab085aSmws break; 77084ab085aSmws case SMB_EVM_GPNV: 77184ab085aSmws oprintf(fp, "0x%x\n", ev.smbev_addr.eva_gpnv); 77284ab085aSmws break; 77384ab085aSmws default: 77484ab085aSmws oprintf(fp, "0x%x\n", ev.smbev_addr.eva_addr); 77584ab085aSmws } 77684ab085aSmws 77784ab085aSmws oprintf(fp, " Type Descriptors:\n"); 77884ab085aSmws 77984ab085aSmws for (i = 0; i < ev.smbev_typec; i++) { 78084ab085aSmws oprintf(fp, " %u: Log Type 0x%x, Data Type 0x%x\n", i, 78184ab085aSmws ev.smbev_typev[i].smbevt_ltype, 78284ab085aSmws ev.smbev_typev[i].smbevt_dtype); 78384ab085aSmws } 78484ab085aSmws } 78584ab085aSmws 78684ab085aSmws static void 78784ab085aSmws print_bytes(const uint8_t *data, size_t size, FILE *fp) 78884ab085aSmws { 78984ab085aSmws size_t row, rows = P2ROUNDUP(size, 16) / 16; 79084ab085aSmws size_t col, cols; 79184ab085aSmws 79284ab085aSmws char buf[17]; 79384ab085aSmws uint8_t x; 79484ab085aSmws 79584ab085aSmws oprintf(fp, "\n offset: 0 1 2 3 4 5 6 7 8 9 a b c d e f " 79684ab085aSmws "0123456789abcdef\n"); 79784ab085aSmws 79884ab085aSmws for (row = 0; row < rows; row++) { 7991f6171acSRobert Mustacchi oprintf(fp, " %#6lx: ", (ulong_t)row * 16); 80084ab085aSmws cols = MIN(size - row * 16, 16); 80184ab085aSmws 80284ab085aSmws for (col = 0; col < cols; col++) { 80384ab085aSmws if (col % 4 == 0) 80484ab085aSmws oprintf(fp, " "); 80584ab085aSmws x = *data++; 80684ab085aSmws oprintf(fp, "%02x", x); 80784ab085aSmws buf[col] = x <= ' ' || x > '~' ? '.' : x; 80884ab085aSmws } 80984ab085aSmws 81084ab085aSmws for (; col < 16; col++) { 81184ab085aSmws if (col % 4 == 0) 81284ab085aSmws oprintf(fp, " "); 81384ab085aSmws oprintf(fp, " "); 81484ab085aSmws buf[col] = ' '; 81584ab085aSmws } 81684ab085aSmws 81784ab085aSmws buf[col] = '\0'; 81884ab085aSmws oprintf(fp, " %s\n", buf); 81984ab085aSmws } 82084ab085aSmws 82184ab085aSmws oprintf(fp, "\n"); 82284ab085aSmws } 82384ab085aSmws 82484ab085aSmws static void 82584ab085aSmws print_memarray(smbios_hdl_t *shp, id_t id, FILE *fp) 82684ab085aSmws { 82784ab085aSmws smbios_memarray_t ma; 82884ab085aSmws 82984ab085aSmws (void) smbios_info_memarray(shp, id, &ma); 83084ab085aSmws 83184ab085aSmws desc_printf(smbios_memarray_loc_desc(ma.smbma_location), 83284ab085aSmws fp, " Location: %u", ma.smbma_location); 83384ab085aSmws 83484ab085aSmws desc_printf(smbios_memarray_use_desc(ma.smbma_use), 83584ab085aSmws fp, " Use: %u", ma.smbma_use); 83684ab085aSmws 83784ab085aSmws desc_printf(smbios_memarray_ecc_desc(ma.smbma_ecc), 83884ab085aSmws fp, " ECC: %u", ma.smbma_ecc); 83984ab085aSmws 84084ab085aSmws oprintf(fp, " Number of Slots/Sockets: %u\n", ma.smbma_ndevs); 84184ab085aSmws id_printf(fp, " Memory Error Data: ", ma.smbma_err); 84284ab085aSmws oprintf(fp, " Max Capacity: %llu bytes\n", 84384ab085aSmws (u_longlong_t)ma.smbma_size); 84484ab085aSmws } 84584ab085aSmws 84684ab085aSmws static void 84784ab085aSmws print_memdevice(smbios_hdl_t *shp, id_t id, FILE *fp) 84884ab085aSmws { 84984ab085aSmws smbios_memdevice_t md; 85084ab085aSmws 85184ab085aSmws (void) smbios_info_memdevice(shp, id, &md); 85284ab085aSmws 85384ab085aSmws id_printf(fp, " Physical Memory Array: ", md.smbmd_array); 85484ab085aSmws id_printf(fp, " Memory Error Data: ", md.smbmd_error); 85584ab085aSmws 85684ab085aSmws if (md.smbmd_twidth != -1u) 85784ab085aSmws oprintf(fp, " Total Width: %u bits\n", md.smbmd_twidth); 85884ab085aSmws else 85984ab085aSmws oprintf(fp, " Total Width: Unknown\n"); 86084ab085aSmws 86184ab085aSmws if (md.smbmd_dwidth != -1u) 86284ab085aSmws oprintf(fp, " Data Width: %u bits\n", md.smbmd_dwidth); 86384ab085aSmws else 86484ab085aSmws oprintf(fp, " Data Width: Unknown\n"); 86584ab085aSmws 86684ab085aSmws switch (md.smbmd_size) { 86784ab085aSmws case -1ull: 86884ab085aSmws oprintf(fp, " Size: Unknown\n"); 86984ab085aSmws break; 87084ab085aSmws case 0: 87184ab085aSmws oprintf(fp, " Size: Not Populated\n"); 87284ab085aSmws break; 87384ab085aSmws default: 87484ab085aSmws oprintf(fp, " Size: %llu bytes\n", 87584ab085aSmws (u_longlong_t)md.smbmd_size); 87684ab085aSmws } 87784ab085aSmws 87884ab085aSmws desc_printf(smbios_memdevice_form_desc(md.smbmd_form), 87984ab085aSmws fp, " Form Factor: %u", md.smbmd_form); 88084ab085aSmws 88184ab085aSmws if (md.smbmd_set == 0) 88284ab085aSmws oprintf(fp, " Set: None\n"); 88384ab085aSmws else if (md.smbmd_set == (uint8_t)-1u) 88484ab085aSmws oprintf(fp, " Set: Unknown\n"); 88584ab085aSmws else 88684ab085aSmws oprintf(fp, " Set: %u\n", md.smbmd_set); 88784ab085aSmws 8884e901881SDale Ghent if (md.smbmd_rank != 0) { 8894e901881SDale Ghent desc_printf(smbios_memdevice_rank_desc(md.smbmd_rank), 8904e901881SDale Ghent fp, " Rank: %u", md.smbmd_rank); 8914e901881SDale Ghent } else { 8924e901881SDale Ghent oprintf(fp, " Rank: Unknown\n"); 8934e901881SDale Ghent } 8944e901881SDale Ghent 89584ab085aSmws desc_printf(smbios_memdevice_type_desc(md.smbmd_type), 89684ab085aSmws fp, " Memory Type: %u", md.smbmd_type); 89784ab085aSmws 89884ab085aSmws flag_printf(fp, "Flags", md.smbmd_flags, sizeof (md.smbmd_flags) * NBBY, 89984ab085aSmws smbios_memdevice_flag_name, smbios_memdevice_flag_desc); 90084ab085aSmws 90184ab085aSmws if (md.smbmd_speed != 0) 90238d76b18SRobert Mustacchi oprintf(fp, " Speed: %u MT/s\n", md.smbmd_speed); 90384ab085aSmws else 90484ab085aSmws oprintf(fp, " Speed: Unknown\n"); 90584ab085aSmws 9064e901881SDale Ghent if (md.smbmd_clkspeed != 0) 90738d76b18SRobert Mustacchi oprintf(fp, " Configured Speed: %u MT/s\n", md.smbmd_clkspeed); 9084e901881SDale Ghent else 9094e901881SDale Ghent oprintf(fp, " Configured Speed: Unknown\n"); 9104e901881SDale Ghent 91184ab085aSmws oprintf(fp, " Device Locator: %s\n", md.smbmd_dloc); 91284ab085aSmws oprintf(fp, " Bank Locator: %s\n", md.smbmd_bloc); 9134e901881SDale Ghent 9144e901881SDale Ghent if (md.smbmd_minvolt != 0) { 9154e901881SDale Ghent oprintf(fp, " Minimum Voltage: %.2fV\n", 9164e901881SDale Ghent md.smbmd_minvolt / 1000.0); 9174e901881SDale Ghent } else { 9184e901881SDale Ghent oprintf(fp, " Minimum Voltage: Unknown\n"); 9194e901881SDale Ghent } 9204e901881SDale Ghent 9214e901881SDale Ghent if (md.smbmd_maxvolt != 0) { 9224e901881SDale Ghent oprintf(fp, " Maximum Voltage: %.2fV\n", 9234e901881SDale Ghent md.smbmd_maxvolt / 1000.0); 9244e901881SDale Ghent } else { 9254e901881SDale Ghent oprintf(fp, " Maximum Voltage: Unknown\n"); 9264e901881SDale Ghent } 9274e901881SDale Ghent 9284e901881SDale Ghent if (md.smbmd_confvolt != 0) { 9294e901881SDale Ghent oprintf(fp, " Configured Voltage: %.2fV\n", 9304e901881SDale Ghent md.smbmd_confvolt / 1000.0); 9314e901881SDale Ghent } else { 9324e901881SDale Ghent oprintf(fp, " Configured Voltage: Unknown\n"); 9334e901881SDale Ghent } 93484ab085aSmws } 93584ab085aSmws 93684ab085aSmws static void 93784ab085aSmws print_memarrmap(smbios_hdl_t *shp, id_t id, FILE *fp) 93884ab085aSmws { 93984ab085aSmws smbios_memarrmap_t ma; 94084ab085aSmws 94184ab085aSmws (void) smbios_info_memarrmap(shp, id, &ma); 94284ab085aSmws 94384ab085aSmws id_printf(fp, " Physical Memory Array: ", ma.smbmam_array); 94484ab085aSmws oprintf(fp, " Devices per Row: %u\n", ma.smbmam_width); 94584ab085aSmws 94684ab085aSmws oprintf(fp, " Physical Address: 0x%llx\n Size: %llu bytes\n", 94784ab085aSmws (u_longlong_t)ma.smbmam_addr, (u_longlong_t)ma.smbmam_size); 94884ab085aSmws } 94984ab085aSmws 95084ab085aSmws static void 95184ab085aSmws print_memdevmap(smbios_hdl_t *shp, id_t id, FILE *fp) 95284ab085aSmws { 95384ab085aSmws smbios_memdevmap_t md; 95484ab085aSmws 95584ab085aSmws (void) smbios_info_memdevmap(shp, id, &md); 95684ab085aSmws 95784ab085aSmws id_printf(fp, " Memory Device: ", md.smbmdm_device); 95884ab085aSmws id_printf(fp, " Memory Array Mapped Address: ", md.smbmdm_arrmap); 95984ab085aSmws 96084ab085aSmws oprintf(fp, " Physical Address: 0x%llx\n Size: %llu bytes\n", 96184ab085aSmws (u_longlong_t)md.smbmdm_addr, (u_longlong_t)md.smbmdm_size); 96284ab085aSmws 96384ab085aSmws oprintf(fp, " Partition Row Position: %u\n", md.smbmdm_rpos); 96484ab085aSmws oprintf(fp, " Interleave Position: %u\n", md.smbmdm_ipos); 96584ab085aSmws oprintf(fp, " Interleave Data Depth: %u\n", md.smbmdm_idepth); 96684ab085aSmws } 96784ab085aSmws 96884ab085aSmws static void 96984ab085aSmws print_hwsec(smbios_hdl_t *shp, FILE *fp) 97084ab085aSmws { 97184ab085aSmws smbios_hwsec_t h; 97284ab085aSmws 97384ab085aSmws (void) smbios_info_hwsec(shp, &h); 97484ab085aSmws 97584ab085aSmws desc_printf(smbios_hwsec_desc(h.smbh_pwr_ps), 97684ab085aSmws fp, " Power-On Password Status: %u", h.smbh_pwr_ps); 97784ab085aSmws desc_printf(smbios_hwsec_desc(h.smbh_kbd_ps), 97884ab085aSmws fp, " Keyboard Password Status: %u", h.smbh_kbd_ps); 97984ab085aSmws desc_printf(smbios_hwsec_desc(h.smbh_adm_ps), 98084ab085aSmws fp, " Administrator Password Status: %u", h.smbh_adm_ps); 98184ab085aSmws desc_printf(smbios_hwsec_desc(h.smbh_pan_ps), 98284ab085aSmws fp, " Front Panel Reset Status: %u", h.smbh_pan_ps); 98384ab085aSmws } 98484ab085aSmws 98584ab085aSmws static void 986*45807aa8SRobert Mustacchi print_vprobe(smbios_hdl_t *shp, id_t id, FILE *fp) 987*45807aa8SRobert Mustacchi { 988*45807aa8SRobert Mustacchi smbios_vprobe_t vp; 989*45807aa8SRobert Mustacchi 990*45807aa8SRobert Mustacchi if (smbios_info_vprobe(shp, id, &vp) != 0) { 991*45807aa8SRobert Mustacchi smbios_warn(shp, "failed to read voltage probe information"); 992*45807aa8SRobert Mustacchi return; 993*45807aa8SRobert Mustacchi } 994*45807aa8SRobert Mustacchi 995*45807aa8SRobert Mustacchi oprintf(fp, " Description: %s\n", vp.smbvp_description != NULL ? 996*45807aa8SRobert Mustacchi vp.smbvp_description : "unknown"); 997*45807aa8SRobert Mustacchi desc_printf(smbios_vprobe_loc_desc(vp.smbvp_location), 998*45807aa8SRobert Mustacchi fp, " Location: %u", vp.smbvp_location); 999*45807aa8SRobert Mustacchi desc_printf(smbios_vprobe_status_desc(vp.smbvp_status), 1000*45807aa8SRobert Mustacchi fp, " Status: %u", vp.smbvp_status); 1001*45807aa8SRobert Mustacchi 1002*45807aa8SRobert Mustacchi if (vp.smbvp_maxval != SMB_PROBE_UNKNOWN_VALUE) { 1003*45807aa8SRobert Mustacchi oprintf(fp, " Maximum Possible Voltage: %u mV\n", 1004*45807aa8SRobert Mustacchi vp.smbvp_maxval); 1005*45807aa8SRobert Mustacchi } else { 1006*45807aa8SRobert Mustacchi oprintf(fp, " Maximum Possible Voltage: unknown\n"); 1007*45807aa8SRobert Mustacchi } 1008*45807aa8SRobert Mustacchi 1009*45807aa8SRobert Mustacchi if (vp.smbvp_minval != SMB_PROBE_UNKNOWN_VALUE) { 1010*45807aa8SRobert Mustacchi oprintf(fp, " Minimum Possible Voltage: %u mV\n", 1011*45807aa8SRobert Mustacchi vp.smbvp_minval); 1012*45807aa8SRobert Mustacchi } else { 1013*45807aa8SRobert Mustacchi oprintf(fp, " Minimum Possible Voltage: unknown\n"); 1014*45807aa8SRobert Mustacchi } 1015*45807aa8SRobert Mustacchi 1016*45807aa8SRobert Mustacchi if (vp.smbvp_resolution != SMB_PROBE_UNKNOWN_VALUE) { 1017*45807aa8SRobert Mustacchi oprintf(fp, " Probe Resolution: %u.%u mV\n", 1018*45807aa8SRobert Mustacchi vp.smbvp_resolution / 10, 1019*45807aa8SRobert Mustacchi vp.smbvp_resolution % 10); 1020*45807aa8SRobert Mustacchi } else { 1021*45807aa8SRobert Mustacchi oprintf(fp, " Probe Resolution: unknown\n"); 1022*45807aa8SRobert Mustacchi } 1023*45807aa8SRobert Mustacchi 1024*45807aa8SRobert Mustacchi if (vp.smbvp_tolerance != SMB_PROBE_UNKNOWN_VALUE) { 1025*45807aa8SRobert Mustacchi oprintf(fp, " Probe Tolerance: +/-%u mV\n", 1026*45807aa8SRobert Mustacchi vp.smbvp_tolerance); 1027*45807aa8SRobert Mustacchi } else { 1028*45807aa8SRobert Mustacchi oprintf(fp, " Probe Tolerance: unknown\n"); 1029*45807aa8SRobert Mustacchi } 1030*45807aa8SRobert Mustacchi 1031*45807aa8SRobert Mustacchi if (vp.smbvp_accuracy != SMB_PROBE_UNKNOWN_VALUE) { 1032*45807aa8SRobert Mustacchi oprintf(fp, " Probe Accuracy: +/-%u.%02u%%\n", 1033*45807aa8SRobert Mustacchi vp.smbvp_accuracy / 100, 1034*45807aa8SRobert Mustacchi vp.smbvp_accuracy % 100); 1035*45807aa8SRobert Mustacchi } else { 1036*45807aa8SRobert Mustacchi oprintf(fp, " Probe Accuracy: unknown\n"); 1037*45807aa8SRobert Mustacchi } 1038*45807aa8SRobert Mustacchi 1039*45807aa8SRobert Mustacchi oprintf(fp, " OEM- or BIOS- defined value: 0x%x\n", vp.smbvp_oem); 1040*45807aa8SRobert Mustacchi 1041*45807aa8SRobert Mustacchi if (vp.smbvp_nominal != SMB_PROBE_UNKNOWN_VALUE) { 1042*45807aa8SRobert Mustacchi oprintf(fp, " Probe Nominal Value: %u mV\n", vp.smbvp_nominal); 1043*45807aa8SRobert Mustacchi } else { 1044*45807aa8SRobert Mustacchi oprintf(fp, " Probe Nominal Value: unknown\n"); 1045*45807aa8SRobert Mustacchi } 1046*45807aa8SRobert Mustacchi } 1047*45807aa8SRobert Mustacchi 1048*45807aa8SRobert Mustacchi static void 1049*45807aa8SRobert Mustacchi print_cooldev(smbios_hdl_t *shp, id_t id, FILE *fp) 1050*45807aa8SRobert Mustacchi { 1051*45807aa8SRobert Mustacchi smbios_cooldev_t cd; 1052*45807aa8SRobert Mustacchi 1053*45807aa8SRobert Mustacchi if (smbios_info_cooldev(shp, id, &cd) != 0) { 1054*45807aa8SRobert Mustacchi smbios_warn(shp, "failed to read cooling device " 1055*45807aa8SRobert Mustacchi "information"); 1056*45807aa8SRobert Mustacchi return; 1057*45807aa8SRobert Mustacchi } 1058*45807aa8SRobert Mustacchi 1059*45807aa8SRobert Mustacchi id_printf(fp, " Temperature Probe Handle: ", cd.smbcd_tprobe); 1060*45807aa8SRobert Mustacchi desc_printf(smbios_cooldev_type_desc(cd.smbcd_type), 1061*45807aa8SRobert Mustacchi fp, " Device Type: %u", cd.smbcd_type); 1062*45807aa8SRobert Mustacchi desc_printf(smbios_cooldev_status_desc(cd.smbcd_status), 1063*45807aa8SRobert Mustacchi fp, " Status: %u", cd.smbcd_status); 1064*45807aa8SRobert Mustacchi oprintf(fp, " Cooling Unit Group: %u\n", cd.smbcd_group); 1065*45807aa8SRobert Mustacchi oprintf(fp, " OEM- or BIOS- defined data: 0x%x\n", cd.smbcd_oem); 1066*45807aa8SRobert Mustacchi if (cd.smbcd_nominal != SMB_PROBE_UNKNOWN_VALUE) { 1067*45807aa8SRobert Mustacchi oprintf(fp, " Nominal Speed: %u RPM\n", cd.smbcd_nominal); 1068*45807aa8SRobert Mustacchi } else { 1069*45807aa8SRobert Mustacchi oprintf(fp, " Nominal Speed: unknown\n"); 1070*45807aa8SRobert Mustacchi } 1071*45807aa8SRobert Mustacchi 1072*45807aa8SRobert Mustacchi if (cd.smbcd_descr != NULL && cd.smbcd_descr[0] != '\0') { 1073*45807aa8SRobert Mustacchi oprintf(fp, " Description: %s\n", cd.smbcd_descr); 1074*45807aa8SRobert Mustacchi } 1075*45807aa8SRobert Mustacchi } 1076*45807aa8SRobert Mustacchi 1077*45807aa8SRobert Mustacchi static void 1078*45807aa8SRobert Mustacchi print_tprobe(smbios_hdl_t *shp, id_t id, FILE *fp) 1079*45807aa8SRobert Mustacchi { 1080*45807aa8SRobert Mustacchi smbios_tprobe_t tp; 1081*45807aa8SRobert Mustacchi 1082*45807aa8SRobert Mustacchi if (smbios_info_tprobe(shp, id, &tp) != 0) { 1083*45807aa8SRobert Mustacchi smbios_warn(shp, "failed to read temperature probe " 1084*45807aa8SRobert Mustacchi "information"); 1085*45807aa8SRobert Mustacchi return; 1086*45807aa8SRobert Mustacchi } 1087*45807aa8SRobert Mustacchi 1088*45807aa8SRobert Mustacchi oprintf(fp, " Description: %s\n", tp.smbtp_description != NULL ? 1089*45807aa8SRobert Mustacchi tp.smbtp_description : "unknown"); 1090*45807aa8SRobert Mustacchi desc_printf(smbios_tprobe_loc_desc(tp.smbtp_location), 1091*45807aa8SRobert Mustacchi fp, " Location: %u", tp.smbtp_location); 1092*45807aa8SRobert Mustacchi desc_printf(smbios_tprobe_status_desc(tp.smbtp_status), 1093*45807aa8SRobert Mustacchi fp, " Status: %u", tp.smbtp_status); 1094*45807aa8SRobert Mustacchi 1095*45807aa8SRobert Mustacchi if (tp.smbtp_maxval != SMB_PROBE_UNKNOWN_VALUE) { 1096*45807aa8SRobert Mustacchi oprintf(fp, " Maximum Possible Temperature: %u.%u C\n", 1097*45807aa8SRobert Mustacchi tp.smbtp_maxval / 10, tp.smbtp_maxval % 10); 1098*45807aa8SRobert Mustacchi } else { 1099*45807aa8SRobert Mustacchi oprintf(fp, " Maximum Possible Temperature: unknown\n"); 1100*45807aa8SRobert Mustacchi } 1101*45807aa8SRobert Mustacchi 1102*45807aa8SRobert Mustacchi if (tp.smbtp_minval != SMB_PROBE_UNKNOWN_VALUE) { 1103*45807aa8SRobert Mustacchi oprintf(fp, " Minimum Possible Temperature: %u.%u C\n", 1104*45807aa8SRobert Mustacchi tp.smbtp_minval / 10, tp.smbtp_minval % 10); 1105*45807aa8SRobert Mustacchi } else { 1106*45807aa8SRobert Mustacchi oprintf(fp, " Minimum Possible Temperature: unknown\n"); 1107*45807aa8SRobert Mustacchi } 1108*45807aa8SRobert Mustacchi 1109*45807aa8SRobert Mustacchi if (tp.smbtp_resolution != SMB_PROBE_UNKNOWN_VALUE) { 1110*45807aa8SRobert Mustacchi oprintf(fp, " Probe Resolution: %u.%03u C\n", 1111*45807aa8SRobert Mustacchi tp.smbtp_resolution / 1000, 1112*45807aa8SRobert Mustacchi tp.smbtp_resolution % 1000); 1113*45807aa8SRobert Mustacchi } else { 1114*45807aa8SRobert Mustacchi oprintf(fp, " Probe Resolution: unknown\n"); 1115*45807aa8SRobert Mustacchi } 1116*45807aa8SRobert Mustacchi 1117*45807aa8SRobert Mustacchi if (tp.smbtp_tolerance != SMB_PROBE_UNKNOWN_VALUE) { 1118*45807aa8SRobert Mustacchi oprintf(fp, " Probe Tolerance: +/-%u.%u C\n", 1119*45807aa8SRobert Mustacchi tp.smbtp_tolerance / 10, tp.smbtp_tolerance % 10); 1120*45807aa8SRobert Mustacchi } else { 1121*45807aa8SRobert Mustacchi oprintf(fp, " Probe Tolerance: unknown\n"); 1122*45807aa8SRobert Mustacchi } 1123*45807aa8SRobert Mustacchi 1124*45807aa8SRobert Mustacchi if (tp.smbtp_accuracy != SMB_PROBE_UNKNOWN_VALUE) { 1125*45807aa8SRobert Mustacchi oprintf(fp, " Probe Accuracy: +/-%u.%02u%%\n", 1126*45807aa8SRobert Mustacchi tp.smbtp_accuracy / 100, 1127*45807aa8SRobert Mustacchi tp.smbtp_accuracy % 100); 1128*45807aa8SRobert Mustacchi } else { 1129*45807aa8SRobert Mustacchi oprintf(fp, " Probe Accuracy: unknown\n"); 1130*45807aa8SRobert Mustacchi } 1131*45807aa8SRobert Mustacchi 1132*45807aa8SRobert Mustacchi oprintf(fp, " OEM- or BIOS- defined value: 0x%x\n", tp.smbtp_oem); 1133*45807aa8SRobert Mustacchi 1134*45807aa8SRobert Mustacchi if (tp.smbtp_nominal != SMB_PROBE_UNKNOWN_VALUE) { 1135*45807aa8SRobert Mustacchi oprintf(fp, " Probe Nominal Value: %u.%u C\n", 1136*45807aa8SRobert Mustacchi tp.smbtp_nominal / 10, tp.smbtp_nominal % 10); 1137*45807aa8SRobert Mustacchi } else { 1138*45807aa8SRobert Mustacchi oprintf(fp, " Probe Nominal Value: unknown\n"); 1139*45807aa8SRobert Mustacchi } 1140*45807aa8SRobert Mustacchi } 1141*45807aa8SRobert Mustacchi 1142*45807aa8SRobert Mustacchi static void 1143*45807aa8SRobert Mustacchi print_iprobe(smbios_hdl_t *shp, id_t id, FILE *fp) 1144*45807aa8SRobert Mustacchi { 1145*45807aa8SRobert Mustacchi smbios_iprobe_t ip; 1146*45807aa8SRobert Mustacchi 1147*45807aa8SRobert Mustacchi if (smbios_info_iprobe(shp, id, &ip) != 0) { 1148*45807aa8SRobert Mustacchi smbios_warn(shp, "failed to read current probe information"); 1149*45807aa8SRobert Mustacchi return; 1150*45807aa8SRobert Mustacchi } 1151*45807aa8SRobert Mustacchi 1152*45807aa8SRobert Mustacchi oprintf(fp, " Description: %s\n", ip.smbip_description != NULL ? 1153*45807aa8SRobert Mustacchi ip.smbip_description : "unknown"); 1154*45807aa8SRobert Mustacchi desc_printf(smbios_iprobe_loc_desc(ip.smbip_location), 1155*45807aa8SRobert Mustacchi fp, " Location: %u", ip.smbip_location); 1156*45807aa8SRobert Mustacchi desc_printf(smbios_iprobe_status_desc(ip.smbip_status), 1157*45807aa8SRobert Mustacchi fp, " Status: %u", ip.smbip_status); 1158*45807aa8SRobert Mustacchi 1159*45807aa8SRobert Mustacchi if (ip.smbip_maxval != SMB_PROBE_UNKNOWN_VALUE) { 1160*45807aa8SRobert Mustacchi oprintf(fp, " Maximum Possible Current: %u mA\n", 1161*45807aa8SRobert Mustacchi ip.smbip_maxval); 1162*45807aa8SRobert Mustacchi } else { 1163*45807aa8SRobert Mustacchi oprintf(fp, " Maximum Possible Current: unknown\n"); 1164*45807aa8SRobert Mustacchi } 1165*45807aa8SRobert Mustacchi 1166*45807aa8SRobert Mustacchi if (ip.smbip_minval != SMB_PROBE_UNKNOWN_VALUE) { 1167*45807aa8SRobert Mustacchi oprintf(fp, " Minimum Possible Current: %u mA\n", 1168*45807aa8SRobert Mustacchi ip.smbip_minval); 1169*45807aa8SRobert Mustacchi } else { 1170*45807aa8SRobert Mustacchi oprintf(fp, " Minimum Possible Current: unknown\n"); 1171*45807aa8SRobert Mustacchi } 1172*45807aa8SRobert Mustacchi 1173*45807aa8SRobert Mustacchi if (ip.smbip_resolution != SMB_PROBE_UNKNOWN_VALUE) { 1174*45807aa8SRobert Mustacchi oprintf(fp, " Probe Resolution: %u.%u mA\n", 1175*45807aa8SRobert Mustacchi ip.smbip_resolution / 10, 1176*45807aa8SRobert Mustacchi ip.smbip_resolution % 10); 1177*45807aa8SRobert Mustacchi } else { 1178*45807aa8SRobert Mustacchi oprintf(fp, " Probe Resolution: unknown\n"); 1179*45807aa8SRobert Mustacchi } 1180*45807aa8SRobert Mustacchi 1181*45807aa8SRobert Mustacchi if (ip.smbip_tolerance != SMB_PROBE_UNKNOWN_VALUE) { 1182*45807aa8SRobert Mustacchi oprintf(fp, " Probe Tolerance: +/-%u mA\n", 1183*45807aa8SRobert Mustacchi ip.smbip_tolerance); 1184*45807aa8SRobert Mustacchi } else { 1185*45807aa8SRobert Mustacchi oprintf(fp, " Probe Tolerance: unknown\n"); 1186*45807aa8SRobert Mustacchi } 1187*45807aa8SRobert Mustacchi 1188*45807aa8SRobert Mustacchi if (ip.smbip_accuracy != SMB_PROBE_UNKNOWN_VALUE) { 1189*45807aa8SRobert Mustacchi oprintf(fp, " Probe Accuracy: +/-%u.%02u%%\n", 1190*45807aa8SRobert Mustacchi ip.smbip_accuracy / 100, 1191*45807aa8SRobert Mustacchi ip.smbip_accuracy % 100); 1192*45807aa8SRobert Mustacchi } else { 1193*45807aa8SRobert Mustacchi oprintf(fp, " Probe Accuracy: unknown\n"); 1194*45807aa8SRobert Mustacchi } 1195*45807aa8SRobert Mustacchi 1196*45807aa8SRobert Mustacchi oprintf(fp, " OEM- or BIOS- defined value: 0x%x\n", ip.smbip_oem); 1197*45807aa8SRobert Mustacchi 1198*45807aa8SRobert Mustacchi if (ip.smbip_nominal != SMB_PROBE_UNKNOWN_VALUE) { 1199*45807aa8SRobert Mustacchi oprintf(fp, " Probe Nominal Value: %u mA\n", ip.smbip_nominal); 1200*45807aa8SRobert Mustacchi } else { 1201*45807aa8SRobert Mustacchi oprintf(fp, " Probe Nominal Value: unknown\n"); 1202*45807aa8SRobert Mustacchi } 1203*45807aa8SRobert Mustacchi } 1204*45807aa8SRobert Mustacchi 1205*45807aa8SRobert Mustacchi 1206*45807aa8SRobert Mustacchi static void 120784ab085aSmws print_boot(smbios_hdl_t *shp, FILE *fp) 120884ab085aSmws { 120984ab085aSmws smbios_boot_t b; 121084ab085aSmws 121184ab085aSmws (void) smbios_info_boot(shp, &b); 121284ab085aSmws 121384ab085aSmws desc_printf(smbios_boot_desc(b.smbt_status), 121484ab085aSmws fp, " Boot Status Code: 0x%x", b.smbt_status); 121584ab085aSmws 121684ab085aSmws if (b.smbt_size != 0) { 121784ab085aSmws oprintf(fp, " Boot Data (%lu bytes):\n", (ulong_t)b.smbt_size); 121884ab085aSmws print_bytes(b.smbt_data, b.smbt_size, fp); 121984ab085aSmws } 122084ab085aSmws } 122184ab085aSmws 122284ab085aSmws static void 122384ab085aSmws print_ipmi(smbios_hdl_t *shp, FILE *fp) 122484ab085aSmws { 122584ab085aSmws smbios_ipmi_t i; 122684ab085aSmws 122784ab085aSmws (void) smbios_info_ipmi(shp, &i); 122884ab085aSmws 122984ab085aSmws desc_printf(smbios_ipmi_type_desc(i.smbip_type), 123084ab085aSmws fp, " Type: %u", i.smbip_type); 123184ab085aSmws 123284ab085aSmws oprintf(fp, " BMC IPMI Version: %u.%u\n", 123384ab085aSmws i.smbip_vers.smbv_major, i.smbip_vers.smbv_minor); 123484ab085aSmws 123584ab085aSmws oprintf(fp, " i2c Bus Slave Address: 0x%x\n", i.smbip_i2c); 123684ab085aSmws oprintf(fp, " NV Storage Device Bus ID: 0x%x\n", i.smbip_bus); 123784ab085aSmws oprintf(fp, " BMC Base Address: 0x%llx\n", (u_longlong_t)i.smbip_addr); 123884ab085aSmws oprintf(fp, " Interrupt Number: %u\n", i.smbip_intr); 123984ab085aSmws oprintf(fp, " Register Spacing: %u\n", i.smbip_regspacing); 124084ab085aSmws 124184ab085aSmws flag_printf(fp, "Flags", i.smbip_flags, sizeof (i.smbip_flags) * NBBY, 124284ab085aSmws smbios_ipmi_flag_name, smbios_ipmi_flag_desc); 124384ab085aSmws } 124484ab085aSmws 1245074bb90dSTom Pothier static void 12462b9d2074SRobert Mustacchi print_powersup(smbios_hdl_t *shp, id_t id, FILE *fp) 12472b9d2074SRobert Mustacchi { 12482b9d2074SRobert Mustacchi smbios_powersup_t p; 12492b9d2074SRobert Mustacchi 12502b9d2074SRobert Mustacchi if (smbios_info_powersup(shp, id, &p) != 0) { 12512b9d2074SRobert Mustacchi smbios_warn(shp, "failed to read power supply information"); 12522b9d2074SRobert Mustacchi return; 12532b9d2074SRobert Mustacchi } 12542b9d2074SRobert Mustacchi 12552b9d2074SRobert Mustacchi oprintf(fp, " Power Supply Group: %u\n", p.smbps_group); 12562b9d2074SRobert Mustacchi if (p.smbps_maxout != 0x8000) { 12572b9d2074SRobert Mustacchi oprintf(fp, " Maximum Output: %llu mW\n", p.smbps_maxout); 12582b9d2074SRobert Mustacchi } else { 12592b9d2074SRobert Mustacchi oprintf(fp, " Maximum Output: unknown\n"); 12602b9d2074SRobert Mustacchi } 12612b9d2074SRobert Mustacchi 12622b9d2074SRobert Mustacchi flag_printf(fp, "Characteristics", p.smbps_flags, 12632b9d2074SRobert Mustacchi sizeof (p.smbps_flags) * NBBY, smbios_powersup_flag_name, 12642b9d2074SRobert Mustacchi smbios_powersup_flag_desc); 12652b9d2074SRobert Mustacchi 12662b9d2074SRobert Mustacchi desc_printf(smbios_powersup_input_desc(p.smbps_ivrs), 12672b9d2074SRobert Mustacchi fp, " Input Voltage Range Switching: %u", p.smbps_ivrs); 12682b9d2074SRobert Mustacchi desc_printf(smbios_powersup_status_desc(p.smbps_status), 12692b9d2074SRobert Mustacchi fp, " Status: %u", p.smbps_status); 12702b9d2074SRobert Mustacchi desc_printf(smbios_powersup_type_desc(p.smbps_pstype), 12712b9d2074SRobert Mustacchi fp, " Type: %u", p.smbps_pstype); 12722b9d2074SRobert Mustacchi 12732b9d2074SRobert Mustacchi if (p.smbps_vprobe != 0xffff) { 12742b9d2074SRobert Mustacchi oprintf(fp, " Voltage Probe Handle: %lu\n", p.smbps_vprobe); 12752b9d2074SRobert Mustacchi } 12762b9d2074SRobert Mustacchi 12772b9d2074SRobert Mustacchi if (p.smbps_cooldev != 0xffff) { 12782b9d2074SRobert Mustacchi oprintf(fp, " Cooling Device Handle: %lu\n", p.smbps_cooldev); 12792b9d2074SRobert Mustacchi } 12802b9d2074SRobert Mustacchi 12812b9d2074SRobert Mustacchi if (p.smbps_iprobe != 0xffff) { 12822b9d2074SRobert Mustacchi oprintf(fp, " Current Probe Handle: %lu\n", p.smbps_iprobe); 12832b9d2074SRobert Mustacchi } 12842b9d2074SRobert Mustacchi } 12852b9d2074SRobert Mustacchi 12862b9d2074SRobert Mustacchi static void 1287074bb90dSTom Pothier print_extprocessor(smbios_hdl_t *shp, id_t id, FILE *fp) 1288074bb90dSTom Pothier { 1289074bb90dSTom Pothier int i; 1290074bb90dSTom Pothier smbios_processor_ext_t ep; 1291074bb90dSTom Pothier 1292074bb90dSTom Pothier if (check_oem(shp) != 0) 1293074bb90dSTom Pothier return; 1294074bb90dSTom Pothier 1295074bb90dSTom Pothier (void) smbios_info_extprocessor(shp, id, &ep); 1296074bb90dSTom Pothier 1297074bb90dSTom Pothier oprintf(fp, " Processor: %u\n", ep.smbpe_processor); 1298074bb90dSTom Pothier oprintf(fp, " FRU: %u\n", ep.smbpe_fru); 1299074bb90dSTom Pothier oprintf(fp, " Initial APIC ID count: %u\n\n", ep.smbpe_n); 1300074bb90dSTom Pothier 1301074bb90dSTom Pothier for (i = 0; i < ep.smbpe_n; i++) { 1302074bb90dSTom Pothier oprintf(fp, " Logical Strand %u: Initial APIC ID: %u\n", i, 1303074bb90dSTom Pothier ep.smbpe_apicid[i]); 1304074bb90dSTom Pothier } 1305074bb90dSTom Pothier } 1306074bb90dSTom Pothier 1307074bb90dSTom Pothier static void 130803f9f63dSTom Pothier print_extport(smbios_hdl_t *shp, id_t id, FILE *fp) 130903f9f63dSTom Pothier { 131003f9f63dSTom Pothier smbios_port_ext_t epo; 131103f9f63dSTom Pothier 131203f9f63dSTom Pothier if (check_oem(shp) != 0) 131303f9f63dSTom Pothier return; 131403f9f63dSTom Pothier 131503f9f63dSTom Pothier (void) smbios_info_extport(shp, id, &epo); 131603f9f63dSTom Pothier 131703f9f63dSTom Pothier oprintf(fp, " Chassis Handle: %u\n", epo.smbporte_chassis); 131803f9f63dSTom Pothier oprintf(fp, " Port Connector Handle: %u\n", epo.smbporte_port); 131903f9f63dSTom Pothier oprintf(fp, " Device Type: %u\n", epo.smbporte_dtype); 132003f9f63dSTom Pothier oprintf(fp, " Device Handle: %u\n", epo.smbporte_devhdl); 132103f9f63dSTom Pothier oprintf(fp, " PHY: %u\n", epo.smbporte_phy); 132203f9f63dSTom Pothier } 132303f9f63dSTom Pothier 132403f9f63dSTom Pothier static void 1325074bb90dSTom Pothier print_pciexrc(smbios_hdl_t *shp, id_t id, FILE *fp) 1326074bb90dSTom Pothier { 1327074bb90dSTom Pothier smbios_pciexrc_t pcie; 1328074bb90dSTom Pothier 1329074bb90dSTom Pothier if (check_oem(shp) != 0) 1330074bb90dSTom Pothier return; 1331074bb90dSTom Pothier 1332074bb90dSTom Pothier (void) smbios_info_pciexrc(shp, id, &pcie); 1333074bb90dSTom Pothier 1334074bb90dSTom Pothier oprintf(fp, " Component ID: %u\n", pcie.smbpcie_bb); 1335074bb90dSTom Pothier oprintf(fp, " BDF: 0x%x\n", pcie.smbpcie_bdf); 1336074bb90dSTom Pothier } 1337074bb90dSTom Pothier 1338074bb90dSTom Pothier static void 1339074bb90dSTom Pothier print_extmemarray(smbios_hdl_t *shp, id_t id, FILE *fp) 1340074bb90dSTom Pothier { 1341074bb90dSTom Pothier smbios_memarray_ext_t em; 1342074bb90dSTom Pothier 1343074bb90dSTom Pothier if (check_oem(shp) != 0) 1344074bb90dSTom Pothier return; 1345074bb90dSTom Pothier 1346074bb90dSTom Pothier (void) smbios_info_extmemarray(shp, id, &em); 1347074bb90dSTom Pothier 1348074bb90dSTom Pothier oprintf(fp, " Physical Memory Array Handle: %u\n", em.smbmae_ma); 1349074bb90dSTom Pothier oprintf(fp, " Component Parent Handle: %u\n", em.smbmae_comp); 1350074bb90dSTom Pothier oprintf(fp, " BDF: 0x%x\n", em.smbmae_bdf); 1351074bb90dSTom Pothier } 1352074bb90dSTom Pothier 1353074bb90dSTom Pothier static void 1354074bb90dSTom Pothier print_extmemdevice(smbios_hdl_t *shp, id_t id, FILE *fp) 1355074bb90dSTom Pothier { 1356074bb90dSTom Pothier int i; 1357074bb90dSTom Pothier smbios_memdevice_ext_t emd; 1358074bb90dSTom Pothier 1359074bb90dSTom Pothier if (check_oem(shp) != 0) 1360074bb90dSTom Pothier return; 1361074bb90dSTom Pothier 1362074bb90dSTom Pothier (void) smbios_info_extmemdevice(shp, id, &emd); 1363074bb90dSTom Pothier 1364074bb90dSTom Pothier oprintf(fp, " Memory Device Handle: %u\n", emd.smbmdeve_md); 1365074bb90dSTom Pothier oprintf(fp, " DRAM Channel: %u\n", emd.smbmdeve_drch); 1366074bb90dSTom Pothier oprintf(fp, " Number of Chip Selects: %u\n", emd.smbmdeve_ncs); 1367074bb90dSTom Pothier 1368074bb90dSTom Pothier for (i = 0; i < emd.smbmdeve_ncs; i++) { 1369074bb90dSTom Pothier oprintf(fp, " Chip Select: %u\n", emd.smbmdeve_cs[i]); 1370074bb90dSTom Pothier } 1371074bb90dSTom Pothier } 1372074bb90dSTom Pothier 1373074bb90dSTom Pothier static int 137484ab085aSmws print_struct(smbios_hdl_t *shp, const smbios_struct_t *sp, void *fp) 137584ab085aSmws { 137684ab085aSmws smbios_info_t info; 137784ab085aSmws int hex = opt_x; 137884ab085aSmws const char *s; 137984ab085aSmws 138084ab085aSmws if (opt_t != -1 && opt_t != sp->smbstr_type) 138184ab085aSmws return (0); /* skip struct if type doesn't match -t */ 138284ab085aSmws 138384ab085aSmws if (!opt_O && (sp->smbstr_type == SMB_TYPE_MEMCTL || 138484ab085aSmws sp->smbstr_type == SMB_TYPE_MEMMOD)) 138584ab085aSmws return (0); /* skip struct if type is obsolete */ 138684ab085aSmws 138784ab085aSmws if (g_hdr++ == 0 || !opt_s) 138884ab085aSmws oprintf(fp, "%-5s %-4s %s\n", "ID", "SIZE", "TYPE"); 138984ab085aSmws 139084ab085aSmws oprintf(fp, "%-5u %-4lu", 139184ab085aSmws (uint_t)sp->smbstr_id, (ulong_t)sp->smbstr_size); 139284ab085aSmws 139384ab085aSmws if ((s = smbios_type_name(sp->smbstr_type)) != NULL) 139495ae7286SDan McDonald oprintf(fp, " %s (type %u)", s, sp->smbstr_type); 139584ab085aSmws else if (sp->smbstr_type > SMB_TYPE_OEM_LO && 139684ab085aSmws sp->smbstr_type < SMB_TYPE_OEM_HI) 139795ae7286SDan McDonald oprintf(fp, " %s+%u (type %u)", "SMB_TYPE_OEM_LO", 139895ae7286SDan McDonald sp->smbstr_type - SMB_TYPE_OEM_LO, sp->smbstr_type); 139984ab085aSmws else 140084ab085aSmws oprintf(fp, " %u", sp->smbstr_type); 140184ab085aSmws 140284ab085aSmws if ((s = smbios_type_desc(sp->smbstr_type)) != NULL) 140384ab085aSmws oprintf(fp, " (%s)\n", s); 140484ab085aSmws else 140584ab085aSmws oprintf(fp, "\n"); 140684ab085aSmws 140784ab085aSmws if (opt_s) 140884ab085aSmws return (0); /* only print header line if -s specified */ 140984ab085aSmws 141084ab085aSmws if (smbios_info_common(shp, sp->smbstr_id, &info) == 0) { 141184ab085aSmws oprintf(fp, "\n"); 141284ab085aSmws print_common(&info, fp); 141384ab085aSmws } 141484ab085aSmws 141584ab085aSmws switch (sp->smbstr_type) { 141684ab085aSmws case SMB_TYPE_BIOS: 141784ab085aSmws oprintf(fp, "\n"); 141884ab085aSmws print_bios(shp, fp); 141984ab085aSmws break; 142084ab085aSmws case SMB_TYPE_SYSTEM: 142184ab085aSmws oprintf(fp, "\n"); 142284ab085aSmws print_system(shp, fp); 142384ab085aSmws break; 142484ab085aSmws case SMB_TYPE_BASEBOARD: 142584ab085aSmws oprintf(fp, "\n"); 142684ab085aSmws print_bboard(shp, sp->smbstr_id, fp); 142784ab085aSmws break; 142884ab085aSmws case SMB_TYPE_CHASSIS: 142984ab085aSmws oprintf(fp, "\n"); 143084ab085aSmws print_chassis(shp, sp->smbstr_id, fp); 143184ab085aSmws break; 143284ab085aSmws case SMB_TYPE_PROCESSOR: 143384ab085aSmws oprintf(fp, "\n"); 143484ab085aSmws print_processor(shp, sp->smbstr_id, fp); 143584ab085aSmws break; 143684ab085aSmws case SMB_TYPE_CACHE: 143784ab085aSmws oprintf(fp, "\n"); 143884ab085aSmws print_cache(shp, sp->smbstr_id, fp); 143984ab085aSmws break; 144084ab085aSmws case SMB_TYPE_PORT: 144184ab085aSmws oprintf(fp, "\n"); 144284ab085aSmws print_port(shp, sp->smbstr_id, fp); 144384ab085aSmws break; 144484ab085aSmws case SMB_TYPE_SLOT: 144584ab085aSmws oprintf(fp, "\n"); 144684ab085aSmws print_slot(shp, sp->smbstr_id, fp); 144784ab085aSmws break; 144884ab085aSmws case SMB_TYPE_OBDEVS: 144984ab085aSmws oprintf(fp, "\n"); 145084ab085aSmws print_obdevs(shp, sp->smbstr_id, fp); 145184ab085aSmws break; 145284ab085aSmws case SMB_TYPE_OEMSTR: 145384ab085aSmws case SMB_TYPE_SYSCONFSTR: 145484ab085aSmws oprintf(fp, "\n"); 145584ab085aSmws print_strtab(shp, sp->smbstr_id, fp); 145684ab085aSmws break; 145784ab085aSmws case SMB_TYPE_LANG: 145884ab085aSmws oprintf(fp, "\n"); 145984ab085aSmws print_lang(shp, sp->smbstr_id, fp); 146084ab085aSmws break; 146184ab085aSmws case SMB_TYPE_EVENTLOG: 146284ab085aSmws oprintf(fp, "\n"); 146384ab085aSmws print_evlog(shp, sp->smbstr_id, fp); 146484ab085aSmws break; 146584ab085aSmws case SMB_TYPE_MEMARRAY: 146684ab085aSmws oprintf(fp, "\n"); 146784ab085aSmws print_memarray(shp, sp->smbstr_id, fp); 146884ab085aSmws break; 146984ab085aSmws case SMB_TYPE_MEMDEVICE: 147084ab085aSmws oprintf(fp, "\n"); 147184ab085aSmws print_memdevice(shp, sp->smbstr_id, fp); 147284ab085aSmws break; 147384ab085aSmws case SMB_TYPE_MEMARRAYMAP: 147484ab085aSmws oprintf(fp, "\n"); 147584ab085aSmws print_memarrmap(shp, sp->smbstr_id, fp); 147684ab085aSmws break; 147784ab085aSmws case SMB_TYPE_MEMDEVICEMAP: 147884ab085aSmws oprintf(fp, "\n"); 147984ab085aSmws print_memdevmap(shp, sp->smbstr_id, fp); 148084ab085aSmws break; 148184ab085aSmws case SMB_TYPE_SECURITY: 148284ab085aSmws oprintf(fp, "\n"); 148384ab085aSmws print_hwsec(shp, fp); 148484ab085aSmws break; 1485*45807aa8SRobert Mustacchi case SMB_TYPE_VPROBE: 1486*45807aa8SRobert Mustacchi oprintf(fp, "\n"); 1487*45807aa8SRobert Mustacchi print_vprobe(shp, sp->smbstr_id, fp); 1488*45807aa8SRobert Mustacchi break; 1489*45807aa8SRobert Mustacchi case SMB_TYPE_COOLDEV: 1490*45807aa8SRobert Mustacchi oprintf(fp, "\n"); 1491*45807aa8SRobert Mustacchi print_cooldev(shp, sp->smbstr_id, fp); 1492*45807aa8SRobert Mustacchi break; 1493*45807aa8SRobert Mustacchi case SMB_TYPE_TPROBE: 1494*45807aa8SRobert Mustacchi oprintf(fp, "\n"); 1495*45807aa8SRobert Mustacchi print_tprobe(shp, sp->smbstr_id, fp); 1496*45807aa8SRobert Mustacchi break; 1497*45807aa8SRobert Mustacchi case SMB_TYPE_IPROBE: 1498*45807aa8SRobert Mustacchi oprintf(fp, "\n"); 1499*45807aa8SRobert Mustacchi print_iprobe(shp, sp->smbstr_id, fp); 1500*45807aa8SRobert Mustacchi break; 150184ab085aSmws case SMB_TYPE_BOOT: 150284ab085aSmws oprintf(fp, "\n"); 150384ab085aSmws print_boot(shp, fp); 150484ab085aSmws break; 150584ab085aSmws case SMB_TYPE_IPMIDEV: 150684ab085aSmws oprintf(fp, "\n"); 150784ab085aSmws print_ipmi(shp, fp); 150884ab085aSmws break; 15092b9d2074SRobert Mustacchi case SMB_TYPE_POWERSUP: 15102b9d2074SRobert Mustacchi oprintf(fp, "\n"); 15112b9d2074SRobert Mustacchi print_powersup(shp, sp->smbstr_id, fp); 15122b9d2074SRobert Mustacchi break; 151303f9f63dSTom Pothier case SMB_TYPE_OBDEVEXT: 151403f9f63dSTom Pothier oprintf(fp, "\n"); 151503f9f63dSTom Pothier print_obdevs_ext(shp, sp->smbstr_id, fp); 151603f9f63dSTom Pothier break; 1517074bb90dSTom Pothier case SUN_OEM_EXT_PROCESSOR: 1518074bb90dSTom Pothier oprintf(fp, "\n"); 1519074bb90dSTom Pothier print_extprocessor(shp, sp->smbstr_id, fp); 1520074bb90dSTom Pothier break; 152103f9f63dSTom Pothier case SUN_OEM_EXT_PORT: 152203f9f63dSTom Pothier oprintf(fp, "\n"); 152303f9f63dSTom Pothier print_extport(shp, sp->smbstr_id, fp); 152403f9f63dSTom Pothier break; 1525074bb90dSTom Pothier case SUN_OEM_PCIEXRC: 1526074bb90dSTom Pothier oprintf(fp, "\n"); 1527074bb90dSTom Pothier print_pciexrc(shp, sp->smbstr_id, fp); 1528074bb90dSTom Pothier break; 1529074bb90dSTom Pothier case SUN_OEM_EXT_MEMARRAY: 1530074bb90dSTom Pothier oprintf(fp, "\n"); 1531074bb90dSTom Pothier print_extmemarray(shp, sp->smbstr_id, fp); 1532074bb90dSTom Pothier break; 1533074bb90dSTom Pothier case SUN_OEM_EXT_MEMDEVICE: 1534074bb90dSTom Pothier oprintf(fp, "\n"); 1535074bb90dSTom Pothier print_extmemdevice(shp, sp->smbstr_id, fp); 1536074bb90dSTom Pothier break; 153784ab085aSmws default: 153884ab085aSmws hex++; 153984ab085aSmws } 154084ab085aSmws 154184ab085aSmws if (hex) 154284ab085aSmws print_bytes(sp->smbstr_data, sp->smbstr_size, fp); 154384ab085aSmws else 154484ab085aSmws oprintf(fp, "\n"); 154584ab085aSmws 154684ab085aSmws return (0); 154784ab085aSmws } 154884ab085aSmws 154984ab085aSmws static uint16_t 155084ab085aSmws getu16(const char *name, const char *s) 155184ab085aSmws { 155284ab085aSmws u_longlong_t val; 155384ab085aSmws char *p; 155484ab085aSmws 155584ab085aSmws errno = 0; 155684ab085aSmws val = strtoull(s, &p, 0); 155784ab085aSmws 155884ab085aSmws if (errno != 0 || p == s || *p != '\0' || val > UINT16_MAX) { 155984ab085aSmws (void) fprintf(stderr, "%s: invalid %s argument -- %s\n", 156084ab085aSmws g_pname, name, s); 156184ab085aSmws exit(SMBIOS_USAGE); 156284ab085aSmws } 156384ab085aSmws 156484ab085aSmws return ((uint16_t)val); 156584ab085aSmws } 156684ab085aSmws 156784ab085aSmws static uint16_t 156884ab085aSmws getstype(const char *name, const char *s) 156984ab085aSmws { 157084ab085aSmws const char *ts; 157184ab085aSmws uint16_t t; 157284ab085aSmws 157384ab085aSmws for (t = 0; t < SMB_TYPE_OEM_LO; t++) { 157484ab085aSmws if ((ts = smbios_type_name(t)) != NULL && strcmp(s, ts) == 0) 157584ab085aSmws return (t); 157684ab085aSmws } 157784ab085aSmws 157884ab085aSmws (void) fprintf(stderr, "%s: invalid %s argument -- %s\n", 157984ab085aSmws g_pname, name, s); 158084ab085aSmws 158184ab085aSmws exit(SMBIOS_USAGE); 158284ab085aSmws /*NOTREACHED*/ 158384ab085aSmws } 158484ab085aSmws 158584ab085aSmws static int 158684ab085aSmws usage(FILE *fp) 158784ab085aSmws { 158884ab085aSmws (void) fprintf(fp, "Usage: %s " 158984ab085aSmws "[-BeOsx] [-i id] [-t type] [-w file] [file]\n\n", g_pname); 159084ab085aSmws 159184ab085aSmws (void) fprintf(fp, 159284ab085aSmws "\t-B disable header validation for broken BIOSes\n" 159384ab085aSmws "\t-e display SMBIOS entry point information\n" 159484ab085aSmws "\t-i display only the specified structure\n" 159584ab085aSmws "\t-O display obsolete structure types\n" 159684ab085aSmws "\t-s display only a summary of structure identifiers and types\n" 159784ab085aSmws "\t-t display only the specified structure type\n" 159884ab085aSmws "\t-w write the raw data to the specified file\n" 159984ab085aSmws "\t-x display raw data for structures\n"); 160084ab085aSmws 160184ab085aSmws return (SMBIOS_USAGE); 160284ab085aSmws } 160384ab085aSmws 160484ab085aSmws int 160584ab085aSmws main(int argc, char *argv[]) 160684ab085aSmws { 160784ab085aSmws const char *ifile = NULL; 160884ab085aSmws const char *ofile = NULL; 160984ab085aSmws int oflags = 0; 161084ab085aSmws 161184ab085aSmws smbios_hdl_t *shp; 161284ab085aSmws smbios_struct_t s; 161384ab085aSmws int err, fd, c; 161484ab085aSmws char *p; 161584ab085aSmws 161684ab085aSmws if ((p = strrchr(argv[0], '/')) == NULL) 161784ab085aSmws g_pname = argv[0]; 161884ab085aSmws else 161984ab085aSmws g_pname = p + 1; 162084ab085aSmws 162184ab085aSmws while (optind < argc) { 162284ab085aSmws while ((c = getopt(argc, argv, "Bei:Ost:w:xZ")) != EOF) { 162384ab085aSmws switch (c) { 162484ab085aSmws case 'B': 162584ab085aSmws oflags |= SMB_O_NOCKSUM | SMB_O_NOVERS; 162684ab085aSmws break; 162784ab085aSmws case 'e': 162884ab085aSmws opt_e++; 162984ab085aSmws break; 163084ab085aSmws case 'i': 163184ab085aSmws opt_i = getu16("struct ID", optarg); 163284ab085aSmws break; 163384ab085aSmws case 'O': 163484ab085aSmws opt_O++; 163584ab085aSmws break; 163684ab085aSmws case 's': 163784ab085aSmws opt_s++; 163884ab085aSmws break; 163984ab085aSmws case 't': 164084ab085aSmws if (isdigit(optarg[0])) 164184ab085aSmws opt_t = getu16("struct type", optarg); 164284ab085aSmws else 164384ab085aSmws opt_t = getstype("struct type", optarg); 164484ab085aSmws break; 164584ab085aSmws case 'w': 164684ab085aSmws ofile = optarg; 164784ab085aSmws break; 164884ab085aSmws case 'x': 164984ab085aSmws opt_x++; 165084ab085aSmws break; 165184ab085aSmws case 'Z': 165284ab085aSmws oflags |= SMB_O_ZIDS; /* undocumented */ 165384ab085aSmws break; 165484ab085aSmws default: 165584ab085aSmws return (usage(stderr)); 165684ab085aSmws } 165784ab085aSmws } 165884ab085aSmws 165984ab085aSmws if (optind < argc) { 166084ab085aSmws if (ifile != NULL) { 166184ab085aSmws (void) fprintf(stderr, "%s: illegal " 166284ab085aSmws "argument -- %s\n", g_pname, argv[optind]); 166384ab085aSmws return (SMBIOS_USAGE); 166484ab085aSmws } 166584ab085aSmws ifile = argv[optind++]; 166684ab085aSmws } 166784ab085aSmws } 166884ab085aSmws 166984ab085aSmws if ((shp = smbios_open(ifile, SMB_VERSION, oflags, &err)) == NULL) { 167084ab085aSmws (void) fprintf(stderr, "%s: failed to load SMBIOS: %s\n", 167184ab085aSmws g_pname, smbios_errmsg(err)); 167284ab085aSmws return (SMBIOS_ERROR); 167384ab085aSmws } 167484ab085aSmws 1675b60ae21dSJonathan Matthew if (opt_i == -1 && opt_t == -1 && opt_e == 0 && 1676b60ae21dSJonathan Matthew smbios_truncated(shp)) 1677b60ae21dSJonathan Matthew (void) fprintf(stderr, "%s: SMBIOS table is truncated\n", 1678b60ae21dSJonathan Matthew g_pname); 1679b60ae21dSJonathan Matthew 168084ab085aSmws if (ofile != NULL) { 168184ab085aSmws if ((fd = open(ofile, O_WRONLY|O_CREAT|O_TRUNC, 0666)) == -1) { 168284ab085aSmws (void) fprintf(stderr, "%s: failed to open %s: %s\n", 168384ab085aSmws g_pname, ofile, strerror(errno)); 168484ab085aSmws err = SMBIOS_ERROR; 168584ab085aSmws } else if (smbios_write(shp, fd) != 0) { 168684ab085aSmws (void) fprintf(stderr, "%s: failed to write %s: %s\n", 168784ab085aSmws g_pname, ofile, smbios_errmsg(smbios_errno(shp))); 168884ab085aSmws err = SMBIOS_ERROR; 168984ab085aSmws } 169084ab085aSmws smbios_close(shp); 169184ab085aSmws return (err); 169284ab085aSmws } 169384ab085aSmws 169484ab085aSmws if (opt_e) { 169584ab085aSmws print_smbios(shp, stdout); 169684ab085aSmws smbios_close(shp); 169784ab085aSmws return (SMBIOS_SUCCESS); 169884ab085aSmws } 169984ab085aSmws 170084ab085aSmws if (opt_O && (opt_i != -1 || opt_t != -1)) 170184ab085aSmws opt_O++; /* -i or -t imply displaying obsolete records */ 170284ab085aSmws 170384ab085aSmws if (opt_i != -1) 170484ab085aSmws err = smbios_lookup_id(shp, opt_i, &s); 170584ab085aSmws else 170684ab085aSmws err = smbios_iter(shp, print_struct, stdout); 170784ab085aSmws 170884ab085aSmws if (err != 0) { 170984ab085aSmws (void) fprintf(stderr, "%s: failed to access SMBIOS: %s\n", 171084ab085aSmws g_pname, smbios_errmsg(smbios_errno(shp))); 171184ab085aSmws smbios_close(shp); 171284ab085aSmws return (SMBIOS_ERROR); 171384ab085aSmws } 171484ab085aSmws 171584ab085aSmws if (opt_i != -1) 171684ab085aSmws (void) print_struct(shp, &s, stdout); 171784ab085aSmws 171884ab085aSmws smbios_close(shp); 171984ab085aSmws return (SMBIOS_SUCCESS); 172084ab085aSmws } 1721