xref: /titanic_53/usr/src/cmd/smbios/smbios.c (revision 45807aa8d5f1e5e3815664c4c0e9b8f314f79d16)
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