xref: /titanic_44/usr/src/cmd/smbios/smbios.c (revision a60349c89adffc0902b2353230891d8e7f2b24d9)
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.
24697d799cSRobert 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>
43*a60349c8SRobert Mustacchi #include <libjedec.h>
4484ab085aSmws 
4584ab085aSmws #define	SMBIOS_SUCCESS	0
4684ab085aSmws #define	SMBIOS_ERROR	1
4784ab085aSmws #define	SMBIOS_USAGE	2
4884ab085aSmws 
4984ab085aSmws static const char *g_pname;
5084ab085aSmws static int g_hdr;
5184ab085aSmws 
5284ab085aSmws static int opt_e;
5384ab085aSmws static int opt_i = -1;
5484ab085aSmws static int opt_O;
5584ab085aSmws static int opt_s;
5684ab085aSmws static int opt_t = -1;
5784ab085aSmws static int opt_x;
5884ab085aSmws 
59*a60349c8SRobert Mustacchi static boolean_t
smbios_vergteq(smbios_version_t * v,uint_t major,uint_t minor)60*a60349c8SRobert Mustacchi smbios_vergteq(smbios_version_t *v, uint_t major, uint_t minor)
61*a60349c8SRobert Mustacchi {
62*a60349c8SRobert Mustacchi 	if (v->smbv_major > major)
63*a60349c8SRobert Mustacchi 		return (B_TRUE);
64*a60349c8SRobert Mustacchi 	if (v->smbv_major == major &&
65*a60349c8SRobert Mustacchi 	    v->smbv_minor >= minor)
66*a60349c8SRobert Mustacchi 		return (B_TRUE);
67*a60349c8SRobert Mustacchi 	return (B_FALSE);
68*a60349c8SRobert Mustacchi }
69*a60349c8SRobert Mustacchi 
7084ab085aSmws /*PRINTFLIKE2*/
7184ab085aSmws static void
smbios_warn(smbios_hdl_t * shp,const char * format,...)7265fa020fSRobert Mustacchi smbios_warn(smbios_hdl_t *shp, const char *format, ...)
7365fa020fSRobert Mustacchi {
7465fa020fSRobert Mustacchi 	va_list ap;
7565fa020fSRobert Mustacchi 
7665fa020fSRobert Mustacchi 	va_start(ap, format);
7765fa020fSRobert Mustacchi 	(void) vfprintf(stderr, format, ap);
7865fa020fSRobert Mustacchi 	va_end(ap);
7965fa020fSRobert Mustacchi 
8065fa020fSRobert Mustacchi 	if (shp != NULL) {
8165fa020fSRobert Mustacchi 		(void) fprintf(stderr, ": %s",
8265fa020fSRobert Mustacchi 		    smbios_errmsg(smbios_errno(shp)));
8365fa020fSRobert Mustacchi 	}
8465fa020fSRobert Mustacchi 
8565fa020fSRobert Mustacchi 	(void) fprintf(stderr, "\n");
8665fa020fSRobert Mustacchi }
8765fa020fSRobert Mustacchi 
8865fa020fSRobert Mustacchi /*PRINTFLIKE2*/
8965fa020fSRobert Mustacchi static void
oprintf(FILE * fp,const char * format,...)9084ab085aSmws oprintf(FILE *fp, const char *format, ...)
9184ab085aSmws {
9284ab085aSmws 	va_list ap;
9384ab085aSmws 
9484ab085aSmws 	va_start(ap, format);
9584ab085aSmws 	(void) vfprintf(fp, format, ap);
9684ab085aSmws 	va_end(ap);
9784ab085aSmws }
9884ab085aSmws 
9984ab085aSmws /*PRINTFLIKE3*/
10084ab085aSmws static void
desc_printf(const char * d,FILE * fp,const char * format,...)10184ab085aSmws desc_printf(const char *d, FILE *fp, const char *format, ...)
10284ab085aSmws {
10384ab085aSmws 	va_list ap;
10484ab085aSmws 
10584ab085aSmws 	va_start(ap, format);
10684ab085aSmws 	(void) vfprintf(fp, format, ap);
10784ab085aSmws 	va_end(ap);
10884ab085aSmws 
10984ab085aSmws 	if (d != NULL)
11084ab085aSmws 		(void) fprintf(fp, " (%s)\n", d);
11184ab085aSmws 	else
11284ab085aSmws 		(void) fprintf(fp, "\n");
11384ab085aSmws }
11484ab085aSmws 
11584ab085aSmws static void
flag_printf(FILE * fp,const char * s,uint_t flags,size_t bits,const char * (* flag_name)(uint_t),const char * (* flag_desc)(uint_t))11684ab085aSmws flag_printf(FILE *fp, const char *s, uint_t flags, size_t bits,
11784ab085aSmws     const char *(*flag_name)(uint_t), const char *(*flag_desc)(uint_t))
11884ab085aSmws {
11984ab085aSmws 	size_t i;
12084ab085aSmws 
12184ab085aSmws 	oprintf(fp, "  %s: 0x%x\n", s, flags);
12284ab085aSmws 
12384ab085aSmws 	for (i = 0; i < bits; i++) {
12484ab085aSmws 		uint_t f = 1 << i;
12584ab085aSmws 		const char *n;
12684ab085aSmws 
12784ab085aSmws 		if (!(flags & f))
12884ab085aSmws 			continue;
12984ab085aSmws 
13084ab085aSmws 		if ((n = flag_name(f)) != NULL)
13184ab085aSmws 			desc_printf(flag_desc(f), fp, "\t%s", n);
13284ab085aSmws 		else
13384ab085aSmws 			desc_printf(flag_desc(f), fp, "\t0x%x", f);
13484ab085aSmws 	}
13584ab085aSmws }
13684ab085aSmws 
13784ab085aSmws static void
flag64_printf(FILE * fp,const char * s,uint64_t flags,size_t bits,const char * (* flag_name)(uint64_t),const char * (* flag_desc)(uint64_t))13884ab085aSmws flag64_printf(FILE *fp, const char *s, uint64_t flags, size_t bits,
13984ab085aSmws     const char *(*flag_name)(uint64_t), const char *(*flag_desc)(uint64_t))
14084ab085aSmws {
14184ab085aSmws 	size_t i;
14284ab085aSmws 
14384ab085aSmws 	oprintf(fp, "  %s: 0x%llx\n", s, (u_longlong_t)flags);
14484ab085aSmws 
14584ab085aSmws 	for (i = 0; i < bits; i++) {
14684ab085aSmws 		u_longlong_t f = 1ULL << i;
14784ab085aSmws 		const char *n;
14884ab085aSmws 
14984ab085aSmws 		if (!(flags & f))
15084ab085aSmws 			continue;
15184ab085aSmws 
15284ab085aSmws 		if ((n = flag_name(f)) != NULL)
15384ab085aSmws 			desc_printf(flag_desc(f), fp, "\t%s", n);
15484ab085aSmws 		else
15584ab085aSmws 			desc_printf(flag_desc(f), fp, "\t0x%llx", f);
15684ab085aSmws 	}
15784ab085aSmws }
15884ab085aSmws 
15984ab085aSmws static void
id_printf(FILE * fp,const char * s,id_t id)16084ab085aSmws id_printf(FILE *fp, const char *s, id_t id)
16184ab085aSmws {
16284ab085aSmws 	switch (id) {
16384ab085aSmws 	case SMB_ID_NONE:
16484ab085aSmws 		oprintf(fp, "%sNone\n", s);
16584ab085aSmws 		break;
16684ab085aSmws 	case SMB_ID_NOTSUP:
16784ab085aSmws 		oprintf(fp, "%sNot Supported\n", s);
16884ab085aSmws 		break;
16984ab085aSmws 	default:
17084ab085aSmws 		oprintf(fp, "%s%u\n", s, (uint_t)id);
17184ab085aSmws 	}
17284ab085aSmws }
17384ab085aSmws 
174*a60349c8SRobert Mustacchi static void
jedec_print(FILE * fp,const char * desc,uint_t id)175*a60349c8SRobert Mustacchi jedec_print(FILE *fp, const char *desc, uint_t id)
176*a60349c8SRobert Mustacchi {
177*a60349c8SRobert Mustacchi 	const char *name;
178*a60349c8SRobert Mustacchi 	uint_t cont, vendor;
179*a60349c8SRobert Mustacchi 
180*a60349c8SRobert Mustacchi 	vendor = id & 0xff;
181*a60349c8SRobert Mustacchi 	cont = (id >> 8) & 0xff;
182*a60349c8SRobert Mustacchi 	name = libjedec_vendor_string(cont, vendor);
183*a60349c8SRobert Mustacchi 	if (name == NULL) {
184*a60349c8SRobert Mustacchi 		oprintf(fp, "  %s: 0x%x\n", desc, id);
185*a60349c8SRobert Mustacchi 	} else {
186*a60349c8SRobert Mustacchi 		oprintf(fp, "  %s: 0x%x (%s)\n", desc, id, name);
187*a60349c8SRobert Mustacchi 	}
188*a60349c8SRobert Mustacchi }
189*a60349c8SRobert Mustacchi 
19003f9f63dSTom Pothier static int
check_oem(smbios_hdl_t * shp)19103f9f63dSTom Pothier check_oem(smbios_hdl_t *shp)
19203f9f63dSTom Pothier {
19303f9f63dSTom Pothier 	int i;
19403f9f63dSTom Pothier 	int cnt;
19503f9f63dSTom Pothier 	int rv;
19603f9f63dSTom Pothier 	id_t oem_id;
19703f9f63dSTom Pothier 	smbios_struct_t s;
19803f9f63dSTom Pothier 	const char **oem_str;
19903f9f63dSTom Pothier 
20003f9f63dSTom Pothier 	rv = smbios_lookup_type(shp, SMB_TYPE_OEMSTR, &s);
20103f9f63dSTom Pothier 	if (rv != 0) {
20203f9f63dSTom Pothier 		return (-1);
20303f9f63dSTom Pothier 	}
20403f9f63dSTom Pothier 
20503f9f63dSTom Pothier 	oem_id = s.smbstr_id;
20603f9f63dSTom Pothier 
20703f9f63dSTom Pothier 	cnt = smbios_info_strtab(shp, oem_id, 0, NULL);
20803f9f63dSTom Pothier 	if (cnt > 0) {
20903f9f63dSTom Pothier 		oem_str =  alloca(sizeof (char *) * cnt);
21003f9f63dSTom Pothier 		(void) smbios_info_strtab(shp, oem_id, cnt, oem_str);
21103f9f63dSTom Pothier 
21203f9f63dSTom Pothier 		for (i = 0; i < cnt; i++) {
21303f9f63dSTom Pothier 			if (strncmp(oem_str[i], SMB_PRMS1,
21403f9f63dSTom Pothier 			    strlen(SMB_PRMS1) + 1) == 0) {
21503f9f63dSTom Pothier 				return (0);
21603f9f63dSTom Pothier 			}
21703f9f63dSTom Pothier 		}
21803f9f63dSTom Pothier 	}
21903f9f63dSTom Pothier 
22003f9f63dSTom Pothier 	return (-1);
22103f9f63dSTom Pothier }
22203f9f63dSTom Pothier 
22384ab085aSmws static void
print_smbios_21(smbios_21_entry_t * ep,FILE * fp)224af349cd6SToomas Soome print_smbios_21(smbios_21_entry_t *ep, FILE *fp)
22584ab085aSmws {
22684ab085aSmws 	int i;
22784ab085aSmws 
22884ab085aSmws 	oprintf(fp, "Entry Point Anchor Tag: %*.*s\n",
229af349cd6SToomas Soome 	    (int)sizeof (ep->smbe_eanchor), (int)sizeof (ep->smbe_eanchor),
230af349cd6SToomas Soome 	    ep->smbe_eanchor);
23184ab085aSmws 
232af349cd6SToomas Soome 	oprintf(fp, "Entry Point Checksum: 0x%x\n", ep->smbe_ecksum);
233af349cd6SToomas Soome 	oprintf(fp, "Entry Point Length: %u\n", ep->smbe_elen);
23484ab085aSmws 	oprintf(fp, "Entry Point Version: %u.%u\n",
235af349cd6SToomas Soome 	    ep->smbe_major, ep->smbe_minor);
236af349cd6SToomas Soome 	oprintf(fp, "Max Structure Size: %u\n", ep->smbe_maxssize);
237af349cd6SToomas Soome 	oprintf(fp, "Entry Point Revision: 0x%x\n", ep->smbe_revision);
23884ab085aSmws 
23984ab085aSmws 	oprintf(fp, "Entry Point Revision Data:");
240af349cd6SToomas Soome 	for (i = 0; i < sizeof (ep->smbe_format); i++)
241af349cd6SToomas Soome 		oprintf(fp, " 0x%02x", ep->smbe_format[i]);
24284ab085aSmws 	oprintf(fp, "\n");
24384ab085aSmws 
24484ab085aSmws 	oprintf(fp, "Intermediate Anchor Tag: %*.*s\n",
245af349cd6SToomas Soome 	    (int)sizeof (ep->smbe_ianchor), (int)sizeof (ep->smbe_ianchor),
246af349cd6SToomas Soome 	    ep->smbe_ianchor);
24784ab085aSmws 
248af349cd6SToomas Soome 	oprintf(fp, "Intermediate Checksum: 0x%x\n", ep->smbe_icksum);
249af349cd6SToomas Soome 	oprintf(fp, "Structure Table Length: %u\n", ep->smbe_stlen);
250af349cd6SToomas Soome 	oprintf(fp, "Structure Table Address: 0x%x\n", ep->smbe_staddr);
251af349cd6SToomas Soome 	oprintf(fp, "Structure Table Entries: %u\n", ep->smbe_stnum);
252af349cd6SToomas Soome 	oprintf(fp, "DMI BCD Revision: 0x%x\n", ep->smbe_bcdrev);
253af349cd6SToomas Soome }
254af349cd6SToomas Soome 
255af349cd6SToomas Soome static void
print_smbios_30(smbios_30_entry_t * ep,FILE * fp)256af349cd6SToomas Soome print_smbios_30(smbios_30_entry_t *ep, FILE *fp)
257af349cd6SToomas Soome {
258af349cd6SToomas Soome 	oprintf(fp, "Entry Point Anchor Tag: %*.*s\n",
259af349cd6SToomas Soome 	    (int)sizeof (ep->smbe_eanchor), (int)sizeof (ep->smbe_eanchor),
260af349cd6SToomas Soome 	    ep->smbe_eanchor);
261af349cd6SToomas Soome 
262af349cd6SToomas Soome 	oprintf(fp, "Entry Point Checksum: 0x%x\n", ep->smbe_ecksum);
263af349cd6SToomas Soome 	oprintf(fp, "Entry Point Length: %u\n", ep->smbe_elen);
264af349cd6SToomas Soome 	oprintf(fp, "SMBIOS Version: %u.%u\n",
265af349cd6SToomas Soome 	    ep->smbe_major, ep->smbe_minor);
266af349cd6SToomas Soome 	oprintf(fp, "SMBIOS DocRev: 0x%x\n", ep->smbe_docrev);
267af349cd6SToomas Soome 	oprintf(fp, "Entry Point Revision: 0x%x\n", ep->smbe_revision);
268af349cd6SToomas Soome 
269af349cd6SToomas Soome 	oprintf(fp, "Structure Table Length: %u\n", ep->smbe_stlen);
270af349cd6SToomas Soome 	oprintf(fp, "Structure Table Address: 0x%" PRIx64 "\n",
271af349cd6SToomas Soome 	    ep->smbe_staddr);
272af349cd6SToomas Soome }
273af349cd6SToomas Soome 
274af349cd6SToomas Soome static void
print_smbios(smbios_hdl_t * shp,FILE * fp)275af349cd6SToomas Soome print_smbios(smbios_hdl_t *shp, FILE *fp)
276af349cd6SToomas Soome {
277af349cd6SToomas Soome 	smbios_entry_t ep;
278af349cd6SToomas Soome 
279af349cd6SToomas Soome 	switch (smbios_info_smbios(shp, &ep)) {
280af349cd6SToomas Soome 	case SMBIOS_ENTRY_POINT_21:
281af349cd6SToomas Soome 		print_smbios_21(&ep.ep21, fp);
282af349cd6SToomas Soome 		break;
283af349cd6SToomas Soome 	case SMBIOS_ENTRY_POINT_30:
284af349cd6SToomas Soome 		print_smbios_30(&ep.ep30, fp);
285af349cd6SToomas Soome 		break;
286af349cd6SToomas Soome 	}
28784ab085aSmws }
28884ab085aSmws 
28984ab085aSmws static void
print_common(const smbios_info_t * ip,FILE * fp)29084ab085aSmws print_common(const smbios_info_t *ip, FILE *fp)
29184ab085aSmws {
29284ab085aSmws 	if (ip->smbi_manufacturer[0] != '\0')
29384ab085aSmws 		oprintf(fp, "  Manufacturer: %s\n", ip->smbi_manufacturer);
29484ab085aSmws 	if (ip->smbi_product[0] != '\0')
29584ab085aSmws 		oprintf(fp, "  Product: %s\n", ip->smbi_product);
29684ab085aSmws 	if (ip->smbi_version[0] != '\0')
29784ab085aSmws 		oprintf(fp, "  Version: %s\n", ip->smbi_version);
29884ab085aSmws 	if (ip->smbi_serial[0] != '\0')
29984ab085aSmws 		oprintf(fp, "  Serial Number: %s\n", ip->smbi_serial);
30084ab085aSmws 	if (ip->smbi_asset[0] != '\0')
30184ab085aSmws 		oprintf(fp, "  Asset Tag: %s\n", ip->smbi_asset);
30284ab085aSmws 	if (ip->smbi_location[0] != '\0')
30384ab085aSmws 		oprintf(fp, "  Location Tag: %s\n", ip->smbi_location);
30484ab085aSmws 	if (ip->smbi_part[0] != '\0')
30584ab085aSmws 		oprintf(fp, "  Part Number: %s\n", ip->smbi_part);
30684ab085aSmws }
30784ab085aSmws 
30884ab085aSmws static void
print_bios(smbios_hdl_t * shp,FILE * fp)30984ab085aSmws print_bios(smbios_hdl_t *shp, FILE *fp)
31084ab085aSmws {
31184ab085aSmws 	smbios_bios_t b;
31284ab085aSmws 
31384ab085aSmws 	(void) smbios_info_bios(shp, &b);
31484ab085aSmws 
31584ab085aSmws 	oprintf(fp, "  Vendor: %s\n", b.smbb_vendor);
31684ab085aSmws 	oprintf(fp, "  Version String: %s\n", b.smbb_version);
31784ab085aSmws 	oprintf(fp, "  Release Date: %s\n", b.smbb_reldate);
31884ab085aSmws 	oprintf(fp, "  Address Segment: 0x%x\n", b.smbb_segment);
3191eb9e3eaSRobert Mustacchi 	oprintf(fp, "  ROM Size: %" PRIu64 " bytes\n", b.smbb_extromsize);
32084ab085aSmws 	oprintf(fp, "  Image Size: %u bytes\n", b.smbb_runsize);
32184ab085aSmws 
32284ab085aSmws 	flag64_printf(fp, "Characteristics",
32384ab085aSmws 	    b.smbb_cflags, sizeof (b.smbb_cflags) * NBBY,
32484ab085aSmws 	    smbios_bios_flag_name, smbios_bios_flag_desc);
32584ab085aSmws 
32684ab085aSmws 	if (b.smbb_nxcflags > SMB_BIOSXB_1) {
32784ab085aSmws 		flag_printf(fp, "Characteristics Extension Byte 1",
32884ab085aSmws 		    b.smbb_xcflags[SMB_BIOSXB_1],
32984ab085aSmws 		    sizeof (b.smbb_xcflags[SMB_BIOSXB_1]) * NBBY,
33084ab085aSmws 		    smbios_bios_xb1_name, smbios_bios_xb1_desc);
33184ab085aSmws 	}
33284ab085aSmws 
33384ab085aSmws 	if (b.smbb_nxcflags > SMB_BIOSXB_2) {
33484ab085aSmws 		flag_printf(fp, "Characteristics Extension Byte 2",
33584ab085aSmws 		    b.smbb_xcflags[SMB_BIOSXB_2],
33684ab085aSmws 		    sizeof (b.smbb_xcflags[SMB_BIOSXB_2]) * NBBY,
33784ab085aSmws 		    smbios_bios_xb2_name, smbios_bios_xb2_desc);
33884ab085aSmws 	}
33984ab085aSmws 
34084ab085aSmws 	if (b.smbb_nxcflags > SMB_BIOSXB_BIOS_MIN) {
34184ab085aSmws 		oprintf(fp, "  Version Number: %u.%u\n",
34284ab085aSmws 		    b.smbb_biosv.smbv_major, b.smbb_biosv.smbv_minor);
34384ab085aSmws 	}
34484ab085aSmws 
345a448814aSRobert Mustacchi 	/*
346a448814aSRobert Mustacchi 	 * If the major and minor versions are 0xff then that indicates that the
347a448814aSRobert Mustacchi 	 * embedded controller does not exist.
348a448814aSRobert Mustacchi 	 */
349a448814aSRobert Mustacchi 	if (b.smbb_nxcflags > SMB_BIOSXB_ECFW_MIN &&
350a448814aSRobert Mustacchi 	    b.smbb_ecfwv.smbv_major != 0xff &&
351a448814aSRobert Mustacchi 	    b.smbb_ecfwv.smbv_minor != 0xff) {
35284ab085aSmws 		oprintf(fp, "  Embedded Ctlr Firmware Version Number: %u.%u\n",
35384ab085aSmws 		    b.smbb_ecfwv.smbv_major, b.smbb_ecfwv.smbv_minor);
35484ab085aSmws 	}
35584ab085aSmws }
35684ab085aSmws 
35784ab085aSmws static void
print_system(smbios_hdl_t * shp,FILE * fp)35884ab085aSmws print_system(smbios_hdl_t *shp, FILE *fp)
35984ab085aSmws {
36084ab085aSmws 	smbios_system_t s;
36184ab085aSmws 	uint_t i;
36284ab085aSmws 
36384ab085aSmws 	(void) smbios_info_system(shp, &s);
36484ab085aSmws 
36584ab085aSmws 	oprintf(fp, "  UUID: ");
36684ab085aSmws 	for (i = 0; i < s.smbs_uuidlen; i++) {
36784ab085aSmws 		oprintf(fp, "%02x", s.smbs_uuid[i]);
36884ab085aSmws 		if (i == 3 || i == 5 || i == 7 || i == 9)
36984ab085aSmws 			oprintf(fp, "-");
37084ab085aSmws 	}
37184ab085aSmws 	oprintf(fp, "\n");
37284ab085aSmws 
37384ab085aSmws 	desc_printf(smbios_system_wakeup_desc(s.smbs_wakeup),
37484ab085aSmws 	    fp, "  Wake-Up Event: 0x%x", s.smbs_wakeup);
37584ab085aSmws 
37684ab085aSmws 	oprintf(fp, "  SKU Number: %s\n", s.smbs_sku);
37784ab085aSmws 	oprintf(fp, "  Family: %s\n", s.smbs_family);
37884ab085aSmws }
37984ab085aSmws 
38084ab085aSmws static void
print_bboard(smbios_hdl_t * shp,id_t id,FILE * fp)38184ab085aSmws print_bboard(smbios_hdl_t *shp, id_t id, FILE *fp)
38284ab085aSmws {
38384ab085aSmws 	smbios_bboard_t b;
384074bb90dSTom Pothier 	int chdl_cnt;
38584ab085aSmws 
38684ab085aSmws 	(void) smbios_info_bboard(shp, id, &b);
38784ab085aSmws 
38884ab085aSmws 	oprintf(fp, "  Chassis: %u\n", (uint_t)b.smbb_chassis);
38984ab085aSmws 
39084ab085aSmws 	flag_printf(fp, "Flags", b.smbb_flags, sizeof (b.smbb_flags) * NBBY,
39184ab085aSmws 	    smbios_bboard_flag_name, smbios_bboard_flag_desc);
39284ab085aSmws 
39384ab085aSmws 	desc_printf(smbios_bboard_type_desc(b.smbb_type),
39484ab085aSmws 	    fp, "  Board Type: 0x%x", b.smbb_type);
395074bb90dSTom Pothier 
396074bb90dSTom Pothier 	chdl_cnt = b.smbb_contn;
397074bb90dSTom Pothier 	if (chdl_cnt != 0) {
398074bb90dSTom Pothier 		id_t *chdl;
399074bb90dSTom Pothier 		uint16_t hdl;
400074bb90dSTom Pothier 		int i, n, cnt;
401074bb90dSTom Pothier 
402074bb90dSTom Pothier 		chdl = alloca(chdl_cnt * sizeof (id_t));
403074bb90dSTom Pothier 		cnt = smbios_info_contains(shp, id, chdl_cnt, chdl);
404074bb90dSTom Pothier 		if (cnt > SMB_CONT_MAX)
405074bb90dSTom Pothier 			return;
406074bb90dSTom Pothier 		n = MIN(chdl_cnt, cnt);
407074bb90dSTom Pothier 
408074bb90dSTom Pothier 		oprintf(fp, "\n");
409074bb90dSTom Pothier 		for (i = 0; i < n; i++) {
410074bb90dSTom Pothier 			hdl = (uint16_t)chdl[i];
411074bb90dSTom Pothier 			oprintf(fp, "  Contained Handle: %u\n", hdl);
412074bb90dSTom Pothier 		}
413074bb90dSTom Pothier 	}
41484ab085aSmws }
41584ab085aSmws 
41684ab085aSmws static void
print_chassis(smbios_hdl_t * shp,id_t id,FILE * fp)41784ab085aSmws print_chassis(smbios_hdl_t *shp, id_t id, FILE *fp)
41884ab085aSmws {
41984ab085aSmws 	smbios_chassis_t c;
420074bb90dSTom Pothier 	int elem_cnt;
42184ab085aSmws 
42284ab085aSmws 	(void) smbios_info_chassis(shp, id, &c);
42384ab085aSmws 
42484ab085aSmws 	oprintf(fp, "  OEM Data: 0x%x\n", c.smbc_oemdata);
4254e901881SDale Ghent 	oprintf(fp, "  SKU number: %s\n",
426b2be8c60SYuri Pankov 	    c.smbc_sku[0] == '\0' ? "<unknown>" : c.smbc_sku);
42784ab085aSmws 	oprintf(fp, "  Lock Present: %s\n", c.smbc_lock ? "Y" : "N");
42884ab085aSmws 
42984ab085aSmws 	desc_printf(smbios_chassis_type_desc(c.smbc_type),
43084ab085aSmws 	    fp, "  Chassis Type: 0x%x", c.smbc_type);
43184ab085aSmws 
43284ab085aSmws 	desc_printf(smbios_chassis_state_desc(c.smbc_bustate),
43384ab085aSmws 	    fp, "  Boot-Up State: 0x%x", c.smbc_bustate);
43484ab085aSmws 
43584ab085aSmws 	desc_printf(smbios_chassis_state_desc(c.smbc_psstate),
43684ab085aSmws 	    fp, "  Power Supply State: 0x%x", c.smbc_psstate);
43784ab085aSmws 
43884ab085aSmws 	desc_printf(smbios_chassis_state_desc(c.smbc_thstate),
43984ab085aSmws 	    fp, "  Thermal State: 0x%x", c.smbc_thstate);
44084ab085aSmws 
44184ab085aSmws 	oprintf(fp, "  Chassis Height: %uu\n", c.smbc_uheight);
44284ab085aSmws 	oprintf(fp, "  Power Cords: %u\n", c.smbc_cords);
443074bb90dSTom Pothier 
444074bb90dSTom Pothier 	elem_cnt = c.smbc_elems;
445074bb90dSTom Pothier 	oprintf(fp, "  Element Records: %u\n", elem_cnt);
446074bb90dSTom Pothier 
447074bb90dSTom Pothier 	if (elem_cnt > 0) {
448074bb90dSTom Pothier 		id_t *elems;
449074bb90dSTom Pothier 		uint8_t type;
450074bb90dSTom Pothier 		int i, n, cnt;
451074bb90dSTom Pothier 
452074bb90dSTom Pothier 		elems = alloca(c.smbc_elems * sizeof (id_t));
453074bb90dSTom Pothier 		cnt = smbios_info_contains(shp, id, elem_cnt, elems);
454074bb90dSTom Pothier 		if (cnt > SMB_CONT_MAX)
455074bb90dSTom Pothier 			return;
456074bb90dSTom Pothier 		n = MIN(elem_cnt, cnt);
457074bb90dSTom Pothier 
458074bb90dSTom Pothier 		oprintf(fp, "\n");
459074bb90dSTom Pothier 		for (i = 0; i < n; i++) {
460074bb90dSTom Pothier 			type = (uint8_t)elems[i];
461074bb90dSTom Pothier 			if (type & 0x80) {
462074bb90dSTom Pothier 				/* SMBIOS structrure Type */
463074bb90dSTom Pothier 				desc_printf(smbios_type_name(type & 0x7f), fp,
464074bb90dSTom Pothier 				    "  Contained SMBIOS structure Type: %u",
465074bb90dSTom Pothier 				    type & 0x80);
466074bb90dSTom Pothier 			} else {
467074bb90dSTom Pothier 				/* SMBIOS Base Board Type */
468074bb90dSTom Pothier 				desc_printf(smbios_bboard_type_desc(type), fp,
469074bb90dSTom Pothier 				    "  Contained SMBIOS Base Board Type: 0x%x",
470074bb90dSTom Pothier 				    type);
471074bb90dSTom Pothier 			}
472074bb90dSTom Pothier 		}
473074bb90dSTom Pothier 	}
47484ab085aSmws }
47584ab085aSmws 
47684ab085aSmws static void
print_processor(smbios_hdl_t * shp,id_t id,FILE * fp)47784ab085aSmws print_processor(smbios_hdl_t *shp, id_t id, FILE *fp)
47884ab085aSmws {
47984ab085aSmws 	smbios_processor_t p;
48084ab085aSmws 	uint_t status;
48184ab085aSmws 
48284ab085aSmws 	(void) smbios_info_processor(shp, id, &p);
48384ab085aSmws 	status = SMB_PRSTATUS_STATUS(p.smbp_status);
48484ab085aSmws 
48584ab085aSmws 	desc_printf(smbios_processor_family_desc(p.smbp_family),
48684ab085aSmws 	    fp, "  Family: %u", p.smbp_family);
48784ab085aSmws 
4884e901881SDale Ghent 	if (p.smbp_family2 != 0)
4894e901881SDale Ghent 		desc_printf(smbios_processor_family_desc(p.smbp_family2),
4904e901881SDale Ghent 		    fp, "  Family Ext: %u", p.smbp_family2);
4914e901881SDale Ghent 
49284ab085aSmws 	oprintf(fp, "  CPUID: 0x%llx\n", (u_longlong_t)p.smbp_cpuid);
49384ab085aSmws 
49484ab085aSmws 	desc_printf(smbios_processor_type_desc(p.smbp_type),
49584ab085aSmws 	    fp, "  Type: %u", p.smbp_type);
49684ab085aSmws 
49784ab085aSmws 	desc_printf(smbios_processor_upgrade_desc(p.smbp_upgrade),
49884ab085aSmws 	    fp, "  Socket Upgrade: %u", p.smbp_upgrade);
49984ab085aSmws 
50084ab085aSmws 	oprintf(fp, "  Socket Status: %s\n",
50184ab085aSmws 	    SMB_PRSTATUS_PRESENT(p.smbp_status) ?
50284ab085aSmws 	    "Populated" : "Not Populated");
50384ab085aSmws 
50484ab085aSmws 	desc_printf(smbios_processor_status_desc(status),
50584ab085aSmws 	    fp, "  Processor Status: %u", status);
50684ab085aSmws 
50784ab085aSmws 	if (SMB_PRV_LEGACY(p.smbp_voltage)) {
50884ab085aSmws 		oprintf(fp, "  Supported Voltages:");
50984ab085aSmws 		switch (p.smbp_voltage) {
51084ab085aSmws 		case SMB_PRV_5V:
51184ab085aSmws 			oprintf(fp, " 5.0V");
51284ab085aSmws 			break;
51384ab085aSmws 		case SMB_PRV_33V:
51484ab085aSmws 			oprintf(fp, " 3.3V");
51584ab085aSmws 			break;
51684ab085aSmws 		case SMB_PRV_29V:
51784ab085aSmws 			oprintf(fp, " 2.9V");
51884ab085aSmws 			break;
51984ab085aSmws 		}
52084ab085aSmws 		oprintf(fp, "\n");
52184ab085aSmws 	} else {
52284ab085aSmws 		oprintf(fp, "  Supported Voltages: %.1fV\n",
52384ab085aSmws 		    (float)SMB_PRV_VOLTAGE(p.smbp_voltage) / 10);
52484ab085aSmws 	}
52584ab085aSmws 
52604cdb0e3SRobert Mustacchi 	if (p.smbp_corecount != 0) {
52704cdb0e3SRobert Mustacchi 		if (p.smbp_corecount != 0xff || p.smbp_corecount2 == 0)
5284e901881SDale Ghent 			oprintf(fp, "  Core Count: %u\n", p.smbp_corecount);
5294e901881SDale Ghent 		else
53004cdb0e3SRobert Mustacchi 			oprintf(fp, "  Core Count: %u\n", p.smbp_corecount2);
53104cdb0e3SRobert Mustacchi 	} else {
5324e901881SDale Ghent 		oprintf(fp, "  Core Count: Unknown\n");
53304cdb0e3SRobert Mustacchi 	}
5344e901881SDale Ghent 
53504cdb0e3SRobert Mustacchi 	if (p.smbp_coresenabled != 0) {
53604cdb0e3SRobert Mustacchi 		if (p.smbp_coresenabled != 0xff || p.smbp_coresenabled2 == 0) {
53704cdb0e3SRobert Mustacchi 			oprintf(fp, "  Cores Enabled: %u\n",
53804cdb0e3SRobert Mustacchi 			    p.smbp_coresenabled);
53904cdb0e3SRobert Mustacchi 		} else {
54004cdb0e3SRobert Mustacchi 			oprintf(fp, "  Cores Enabled: %u\n",
54104cdb0e3SRobert Mustacchi 			    p.smbp_coresenabled2);
54204cdb0e3SRobert Mustacchi 		}
54304cdb0e3SRobert Mustacchi 	} else {
5444e901881SDale Ghent 		oprintf(fp, "  Cores Enabled: Unknown\n");
54504cdb0e3SRobert Mustacchi 	}
5464e901881SDale Ghent 
54704cdb0e3SRobert Mustacchi 	if (p.smbp_threadcount != 0) {
54804cdb0e3SRobert Mustacchi 		if (p.smbp_threadcount != 0xff || p.smbp_threadcount2 == 0) {
54904cdb0e3SRobert Mustacchi 			oprintf(fp, "  Thread Count: %u\n",
55004cdb0e3SRobert Mustacchi 			    p.smbp_threadcount);
55104cdb0e3SRobert Mustacchi 		} else {
55204cdb0e3SRobert Mustacchi 			oprintf(fp, "  Thread Count: %u\n",
55304cdb0e3SRobert Mustacchi 			    p.smbp_threadcount2);
55404cdb0e3SRobert Mustacchi 		}
55504cdb0e3SRobert Mustacchi 	} else {
5564e901881SDale Ghent 		oprintf(fp, "  Thread Count: Unknown\n");
55704cdb0e3SRobert Mustacchi 	}
5584e901881SDale Ghent 
5594e901881SDale Ghent 	if (p.smbp_cflags) {
5604e901881SDale Ghent 		flag_printf(fp, "Processor Characteristics",
5614e901881SDale Ghent 		    p.smbp_cflags, sizeof (p.smbp_cflags) * NBBY,
5624e901881SDale Ghent 		    smbios_processor_core_flag_name,
5634e901881SDale Ghent 		    smbios_processor_core_flag_desc);
5644e901881SDale Ghent 	}
5654e901881SDale Ghent 
56684ab085aSmws 	if (p.smbp_clkspeed != 0)
56784ab085aSmws 		oprintf(fp, "  External Clock Speed: %uMHz\n", p.smbp_clkspeed);
56884ab085aSmws 	else
56984ab085aSmws 		oprintf(fp, "  External Clock Speed: Unknown\n");
57084ab085aSmws 
57184ab085aSmws 	if (p.smbp_maxspeed != 0)
57284ab085aSmws 		oprintf(fp, "  Maximum Speed: %uMHz\n", p.smbp_maxspeed);
57384ab085aSmws 	else
57484ab085aSmws 		oprintf(fp, "  Maximum Speed: Unknown\n");
57584ab085aSmws 
57684ab085aSmws 	if (p.smbp_curspeed != 0)
57784ab085aSmws 		oprintf(fp, "  Current Speed: %uMHz\n", p.smbp_curspeed);
57884ab085aSmws 	else
57984ab085aSmws 		oprintf(fp, "  Current Speed: Unknown\n");
58084ab085aSmws 
58165fa020fSRobert Mustacchi 	id_printf(fp, "  L1 Cache Handle: ", p.smbp_l1cache);
58265fa020fSRobert Mustacchi 	id_printf(fp, "  L2 Cache Handle: ", p.smbp_l2cache);
58365fa020fSRobert Mustacchi 	id_printf(fp, "  L3 Cache Handle: ", p.smbp_l3cache);
58484ab085aSmws }
58584ab085aSmws 
58684ab085aSmws static void
print_cache(smbios_hdl_t * shp,id_t id,FILE * fp)58784ab085aSmws print_cache(smbios_hdl_t *shp, id_t id, FILE *fp)
58884ab085aSmws {
58984ab085aSmws 	smbios_cache_t c;
59084ab085aSmws 
59184ab085aSmws 	(void) smbios_info_cache(shp, id, &c);
59284ab085aSmws 
59384ab085aSmws 	oprintf(fp, "  Level: %u\n", c.smba_level);
5941eb9e3eaSRobert Mustacchi 	oprintf(fp, "  Maximum Installed Size: %" PRIu64 " bytes\n",
5951eb9e3eaSRobert Mustacchi 	    c.smba_maxsize2);
59684ab085aSmws 
5971eb9e3eaSRobert Mustacchi 	if (c.smba_size2 != 0) {
5981eb9e3eaSRobert Mustacchi 		oprintf(fp, "  Installed Size: %" PRIu64 " bytes\n",
5991eb9e3eaSRobert Mustacchi 		    c.smba_size2);
6001eb9e3eaSRobert Mustacchi 	} else {
60184ab085aSmws 		oprintf(fp, "  Installed Size: Not Installed\n");
6021eb9e3eaSRobert Mustacchi 	}
60384ab085aSmws 
60484ab085aSmws 	if (c.smba_speed != 0)
60584ab085aSmws 		oprintf(fp, "  Speed: %uns\n", c.smba_speed);
60684ab085aSmws 	else
60784ab085aSmws 		oprintf(fp, "  Speed: Unknown\n");
60884ab085aSmws 
60984ab085aSmws 	flag_printf(fp, "Supported SRAM Types",
61084ab085aSmws 	    c.smba_stype, sizeof (c.smba_stype) * NBBY,
61184ab085aSmws 	    smbios_cache_ctype_name, smbios_cache_ctype_desc);
61284ab085aSmws 
61384ab085aSmws 	desc_printf(smbios_cache_ctype_desc(c.smba_ctype),
61484ab085aSmws 	    fp, "  Current SRAM Type: 0x%x", c.smba_ctype);
61584ab085aSmws 
61684ab085aSmws 	desc_printf(smbios_cache_ecc_desc(c.smba_etype),
61784ab085aSmws 	    fp, "  Error Correction Type: %u", c.smba_etype);
61884ab085aSmws 
61984ab085aSmws 	desc_printf(smbios_cache_logical_desc(c.smba_ltype),
62084ab085aSmws 	    fp, "  Logical Cache Type: %u", c.smba_ltype);
62184ab085aSmws 
62284ab085aSmws 	desc_printf(smbios_cache_assoc_desc(c.smba_assoc),
62384ab085aSmws 	    fp, "  Associativity: %u", c.smba_assoc);
62484ab085aSmws 
62584ab085aSmws 	desc_printf(smbios_cache_mode_desc(c.smba_mode),
62684ab085aSmws 	    fp, "  Mode: %u", c.smba_mode);
62784ab085aSmws 
62884ab085aSmws 	desc_printf(smbios_cache_loc_desc(c.smba_location),
62984ab085aSmws 	    fp, "  Location: %u", c.smba_location);
63084ab085aSmws 
63184ab085aSmws 	flag_printf(fp, "Flags", c.smba_flags, sizeof (c.smba_flags) * NBBY,
63284ab085aSmws 	    smbios_cache_flag_name, smbios_cache_flag_desc);
63384ab085aSmws }
63484ab085aSmws 
63584ab085aSmws static void
print_port(smbios_hdl_t * shp,id_t id,FILE * fp)63684ab085aSmws print_port(smbios_hdl_t *shp, id_t id, FILE *fp)
63784ab085aSmws {
63884ab085aSmws 	smbios_port_t p;
63984ab085aSmws 
64084ab085aSmws 	(void) smbios_info_port(shp, id, &p);
64184ab085aSmws 
64284ab085aSmws 	oprintf(fp, "  Internal Reference Designator: %s\n", p.smbo_iref);
64384ab085aSmws 	oprintf(fp, "  External Reference Designator: %s\n", p.smbo_eref);
64484ab085aSmws 
64584ab085aSmws 	desc_printf(smbios_port_conn_desc(p.smbo_itype),
64684ab085aSmws 	    fp, "  Internal Connector Type: %u", p.smbo_itype);
64784ab085aSmws 
64884ab085aSmws 	desc_printf(smbios_port_conn_desc(p.smbo_etype),
64984ab085aSmws 	    fp, "  External Connector Type: %u", p.smbo_etype);
65084ab085aSmws 
65184ab085aSmws 	desc_printf(smbios_port_type_desc(p.smbo_ptype),
65284ab085aSmws 	    fp, "  Port Type: %u", p.smbo_ptype);
65384ab085aSmws }
65484ab085aSmws 
65584ab085aSmws static void
print_slot(smbios_hdl_t * shp,id_t id,FILE * fp)65684ab085aSmws print_slot(smbios_hdl_t *shp, id_t id, FILE *fp)
65784ab085aSmws {
65884ab085aSmws 	smbios_slot_t s;
659af349cd6SToomas Soome 	smbios_version_t v;
66084ab085aSmws 
66184ab085aSmws 	(void) smbios_info_slot(shp, id, &s);
662af349cd6SToomas Soome 	smbios_info_smbios_version(shp, &v);
66384ab085aSmws 
66484ab085aSmws 	oprintf(fp, "  Reference Designator: %s\n", s.smbl_name);
66584ab085aSmws 	oprintf(fp, "  Slot ID: 0x%x\n", s.smbl_id);
66684ab085aSmws 
66784ab085aSmws 	desc_printf(smbios_slot_type_desc(s.smbl_type),
66884ab085aSmws 	    fp, "  Type: 0x%x", s.smbl_type);
66984ab085aSmws 
67084ab085aSmws 	desc_printf(smbios_slot_width_desc(s.smbl_width),
67184ab085aSmws 	    fp, "  Width: 0x%x", s.smbl_width);
67284ab085aSmws 
67384ab085aSmws 	desc_printf(smbios_slot_usage_desc(s.smbl_usage),
67484ab085aSmws 	    fp, "  Usage: 0x%x", s.smbl_usage);
67584ab085aSmws 
67684ab085aSmws 	desc_printf(smbios_slot_length_desc(s.smbl_length),
67784ab085aSmws 	    fp, "  Length: 0x%x", s.smbl_length);
67884ab085aSmws 
67984ab085aSmws 	flag_printf(fp, "Slot Characteristics 1",
68084ab085aSmws 	    s.smbl_ch1, sizeof (s.smbl_ch1) * NBBY,
68184ab085aSmws 	    smbios_slot_ch1_name, smbios_slot_ch1_desc);
68284ab085aSmws 
68384ab085aSmws 	flag_printf(fp, "Slot Characteristics 2",
68484ab085aSmws 	    s.smbl_ch2, sizeof (s.smbl_ch2) * NBBY,
68584ab085aSmws 	    smbios_slot_ch2_name, smbios_slot_ch2_desc);
68603f9f63dSTom Pothier 
687*a60349c8SRobert Mustacchi 	if (check_oem(shp) != 0 && !smbios_vergteq(&v, 2, 6))
68803f9f63dSTom Pothier 		return;
68903f9f63dSTom Pothier 
69003f9f63dSTom Pothier 	oprintf(fp, "  Segment Group: %u\n", s.smbl_sg);
69103f9f63dSTom Pothier 	oprintf(fp, "  Bus Number: %u\n", s.smbl_bus);
692*a60349c8SRobert Mustacchi 	oprintf(fp, "  Device/Function Number: %u/%u\n", s.smbl_df >> 3,
693*a60349c8SRobert Mustacchi 	    s.smbl_df & 0x7);
694*a60349c8SRobert Mustacchi 
695*a60349c8SRobert Mustacchi 	if (s.smbl_dbw != 0) {
696*a60349c8SRobert Mustacchi 		oprintf(fp, "  Data Bus Width: %d\n", s.smbl_dbw);
697*a60349c8SRobert Mustacchi 	}
698*a60349c8SRobert Mustacchi 
699*a60349c8SRobert Mustacchi 	if (s.smbl_npeers > 0) {
700*a60349c8SRobert Mustacchi 		smbios_slot_peer_t *peer;
701*a60349c8SRobert Mustacchi 		uint_t i, npeers;
702*a60349c8SRobert Mustacchi 
703*a60349c8SRobert Mustacchi 		if (smbios_info_slot_peers(shp, id, &npeers, &peer) != 0) {
704*a60349c8SRobert Mustacchi 			smbios_warn(shp, "failed to read slot peer "
705*a60349c8SRobert Mustacchi 			    "information");
706*a60349c8SRobert Mustacchi 			return;
707*a60349c8SRobert Mustacchi 		}
708*a60349c8SRobert Mustacchi 
709*a60349c8SRobert Mustacchi 		for (i = 0; i < npeers; i++) {
710*a60349c8SRobert Mustacchi 			oprintf(fp, "  Slot Peer %u:\n", i);
711*a60349c8SRobert Mustacchi 			oprintf(fp, "    Segment group: %u\n",
712*a60349c8SRobert Mustacchi 			    peer[i].smblp_group);
713*a60349c8SRobert Mustacchi 			oprintf(fp, "    Bus/Device/Function: %u/%u/%u",
714*a60349c8SRobert Mustacchi 			    peer[i].smblp_bus, peer[i].smblp_device,
715*a60349c8SRobert Mustacchi 			    peer[i].smblp_function);
716*a60349c8SRobert Mustacchi 			oprintf(fp, "    Electrical width: %u\n",
717*a60349c8SRobert Mustacchi 			    peer[i].smblp_data_width);
718*a60349c8SRobert Mustacchi 		}
719*a60349c8SRobert Mustacchi 
720*a60349c8SRobert Mustacchi 		smbios_info_slot_peers_free(shp, npeers, peer);
721*a60349c8SRobert Mustacchi 	}
72203f9f63dSTom Pothier }
72303f9f63dSTom Pothier 
72403f9f63dSTom Pothier static void
print_obdevs_ext(smbios_hdl_t * shp,id_t id,FILE * fp)72503f9f63dSTom Pothier print_obdevs_ext(smbios_hdl_t *shp, id_t id, FILE *fp)
72603f9f63dSTom Pothier {
72704cdb0e3SRobert Mustacchi 	boolean_t enabled;
72803f9f63dSTom Pothier 	smbios_obdev_ext_t oe;
72904cdb0e3SRobert Mustacchi 	const char *type;
73003f9f63dSTom Pothier 
73103f9f63dSTom Pothier 	(void) smbios_info_obdevs_ext(shp, id, &oe);
73203f9f63dSTom Pothier 
73304cdb0e3SRobert Mustacchi 	/*
73404cdb0e3SRobert Mustacchi 	 * Bit 7 is always whether or not the device is enabled while bits 0:6
73504cdb0e3SRobert Mustacchi 	 * are the actual device type.
73604cdb0e3SRobert Mustacchi 	 */
73704cdb0e3SRobert Mustacchi 	enabled = oe.smboe_dtype >> 7;
73804cdb0e3SRobert Mustacchi 	type = smbios_onboard_type_desc(oe.smboe_dtype & 0x7f);
73904cdb0e3SRobert Mustacchi 
74003f9f63dSTom Pothier 	oprintf(fp, "  Reference Designator: %s\n", oe.smboe_name);
74104cdb0e3SRobert Mustacchi 	oprintf(fp, "  Device Enabled: %s\n", enabled == B_TRUE ? "true" :
74204cdb0e3SRobert Mustacchi 	    "false");
74304cdb0e3SRobert Mustacchi 	oprintf(fp, "  Device Type: %s\n", type);
74403f9f63dSTom Pothier 	oprintf(fp, "  Device Type Instance: %u\n", oe.smboe_dti);
74503f9f63dSTom Pothier 	oprintf(fp, "  Segment Group Number: %u\n", oe.smboe_sg);
74603f9f63dSTom Pothier 	oprintf(fp, "  Bus Number: %u\n", oe.smboe_bus);
74703f9f63dSTom Pothier 	oprintf(fp, "  Device/Function Number: %u\n", oe.smboe_df);
74884ab085aSmws }
74984ab085aSmws 
75084ab085aSmws static void
print_obdevs(smbios_hdl_t * shp,id_t id,FILE * fp)75184ab085aSmws print_obdevs(smbios_hdl_t *shp, id_t id, FILE *fp)
75284ab085aSmws {
75384ab085aSmws 	smbios_obdev_t *argv;
75484ab085aSmws 	int i, argc;
75584ab085aSmws 
75684ab085aSmws 	if ((argc = smbios_info_obdevs(shp, id, 0, NULL)) > 0) {
75784ab085aSmws 		argv = alloca(sizeof (smbios_obdev_t) * argc);
75884ab085aSmws 		(void) smbios_info_obdevs(shp, id, argc, argv);
75984ab085aSmws 		for (i = 0; i < argc; i++)
76084ab085aSmws 			oprintf(fp, "  %s\n", argv[i].smbd_name);
76184ab085aSmws 	}
76284ab085aSmws }
76384ab085aSmws 
76484ab085aSmws static void
print_strtab(smbios_hdl_t * shp,id_t id,FILE * fp)76584ab085aSmws print_strtab(smbios_hdl_t *shp, id_t id, FILE *fp)
76684ab085aSmws {
76784ab085aSmws 	const char **argv;
76884ab085aSmws 	int i, argc;
76984ab085aSmws 
77084ab085aSmws 	if ((argc = smbios_info_strtab(shp, id, 0, NULL)) > 0) {
77184ab085aSmws 		argv = alloca(sizeof (char *) * argc);
77284ab085aSmws 		(void) smbios_info_strtab(shp, id, argc, argv);
77384ab085aSmws 		for (i = 0; i < argc; i++)
77484ab085aSmws 			oprintf(fp, "  %s\n", argv[i]);
77584ab085aSmws 	}
77684ab085aSmws }
77784ab085aSmws 
77884ab085aSmws static void
print_lang(smbios_hdl_t * shp,id_t id,FILE * fp)77984ab085aSmws print_lang(smbios_hdl_t *shp, id_t id, FILE *fp)
78084ab085aSmws {
78184ab085aSmws 	smbios_lang_t l;
78284ab085aSmws 
78384ab085aSmws 	(void) smbios_info_lang(shp, &l);
78484ab085aSmws 
78584ab085aSmws 	oprintf(fp, "  Current Language: %s\n", l.smbla_cur);
78684ab085aSmws 	oprintf(fp, "  Language String Format: %u\n", l.smbla_fmt);
78784ab085aSmws 	oprintf(fp, "  Number of Installed Languages: %u\n", l.smbla_num);
78884ab085aSmws 	oprintf(fp, "  Installed Languages:\n");
78984ab085aSmws 
79084ab085aSmws 	print_strtab(shp, id, fp);
79184ab085aSmws }
79284ab085aSmws 
79384ab085aSmws /*ARGSUSED*/
79484ab085aSmws static void
print_evlog(smbios_hdl_t * shp,id_t id,FILE * fp)79584ab085aSmws print_evlog(smbios_hdl_t *shp, id_t id, FILE *fp)
79684ab085aSmws {
79784ab085aSmws 	smbios_evlog_t ev;
79884ab085aSmws 	uint32_t i;
79984ab085aSmws 
80084ab085aSmws 	(void) smbios_info_eventlog(shp, &ev);
80184ab085aSmws 
80284ab085aSmws 	oprintf(fp, "  Log Area Size: %lu bytes\n", (ulong_t)ev.smbev_size);
80384ab085aSmws 	oprintf(fp, "  Header Offset: %lu\n", (ulong_t)ev.smbev_hdr);
80484ab085aSmws 	oprintf(fp, "  Data Offset: %lu\n", (ulong_t)ev.smbev_data);
80584ab085aSmws 
80684ab085aSmws 	desc_printf(smbios_evlog_method_desc(ev.smbev_method),
80784ab085aSmws 	    fp, "  Data Access Method: %u", ev.smbev_method);
80884ab085aSmws 
80984ab085aSmws 	flag_printf(fp, "Log Flags",
81084ab085aSmws 	    ev.smbev_flags, sizeof (ev.smbev_flags) * NBBY,
81184ab085aSmws 	    smbios_evlog_flag_name, smbios_evlog_flag_desc);
81284ab085aSmws 
81384ab085aSmws 	desc_printf(smbios_evlog_format_desc(ev.smbev_format),
81484ab085aSmws 	    fp, "  Log Header Format: %u", ev.smbev_format);
81584ab085aSmws 
81684ab085aSmws 	oprintf(fp, "  Update Token: 0x%x\n", ev.smbev_token);
81784ab085aSmws 	oprintf(fp, "  Data Access Address: ");
81884ab085aSmws 
81984ab085aSmws 	switch (ev.smbev_method) {
82084ab085aSmws 	case SMB_EVM_1x1i_1x1d:
82184ab085aSmws 	case SMB_EVM_2x1i_1x1d:
82284ab085aSmws 	case SMB_EVM_1x2i_1x1d:
82384ab085aSmws 		oprintf(fp, "Index Address 0x%x, Data Address 0x%x\n",
82484ab085aSmws 		    ev.smbev_addr.eva_io.evi_iaddr,
82584ab085aSmws 		    ev.smbev_addr.eva_io.evi_daddr);
82684ab085aSmws 		break;
82784ab085aSmws 	case SMB_EVM_GPNV:
82884ab085aSmws 		oprintf(fp, "0x%x\n", ev.smbev_addr.eva_gpnv);
82984ab085aSmws 		break;
83084ab085aSmws 	default:
83184ab085aSmws 		oprintf(fp, "0x%x\n", ev.smbev_addr.eva_addr);
83284ab085aSmws 	}
83384ab085aSmws 
83484ab085aSmws 	oprintf(fp, "  Type Descriptors:\n");
83584ab085aSmws 
83684ab085aSmws 	for (i = 0; i < ev.smbev_typec; i++) {
83784ab085aSmws 		oprintf(fp, "  %u: Log Type 0x%x, Data Type 0x%x\n", i,
83884ab085aSmws 		    ev.smbev_typev[i].smbevt_ltype,
83984ab085aSmws 		    ev.smbev_typev[i].smbevt_dtype);
84084ab085aSmws 	}
84184ab085aSmws }
84284ab085aSmws 
84384ab085aSmws static void
print_bytes(const uint8_t * data,size_t size,FILE * fp)84484ab085aSmws print_bytes(const uint8_t *data, size_t size, FILE *fp)
84584ab085aSmws {
84684ab085aSmws 	size_t row, rows = P2ROUNDUP(size, 16) / 16;
84784ab085aSmws 	size_t col, cols;
84884ab085aSmws 
84984ab085aSmws 	char buf[17];
85084ab085aSmws 	uint8_t x;
85184ab085aSmws 
85284ab085aSmws 	oprintf(fp, "\n  offset:   0 1 2 3  4 5 6 7  8 9 a b  c d e f  "
85384ab085aSmws 	    "0123456789abcdef\n");
85484ab085aSmws 
85584ab085aSmws 	for (row = 0; row < rows; row++) {
856697d799cSRobert Mustacchi 		oprintf(fp, "  %#6lx: ", (ulong_t)row * 16);
85784ab085aSmws 		cols = MIN(size - row * 16, 16);
85884ab085aSmws 
85984ab085aSmws 		for (col = 0; col < cols; col++) {
86084ab085aSmws 			if (col % 4 == 0)
86184ab085aSmws 				oprintf(fp, " ");
86284ab085aSmws 			x = *data++;
86384ab085aSmws 			oprintf(fp, "%02x", x);
86484ab085aSmws 			buf[col] = x <= ' ' || x > '~' ? '.' : x;
86584ab085aSmws 		}
86684ab085aSmws 
86784ab085aSmws 		for (; col < 16; col++) {
86884ab085aSmws 			if (col % 4 == 0)
86984ab085aSmws 				oprintf(fp, " ");
87084ab085aSmws 			oprintf(fp, "  ");
87184ab085aSmws 			buf[col] = ' ';
87284ab085aSmws 		}
87384ab085aSmws 
87484ab085aSmws 		buf[col] = '\0';
87584ab085aSmws 		oprintf(fp, "  %s\n", buf);
87684ab085aSmws 	}
87784ab085aSmws 
87884ab085aSmws 	oprintf(fp, "\n");
87984ab085aSmws }
88084ab085aSmws 
88184ab085aSmws static void
print_memarray(smbios_hdl_t * shp,id_t id,FILE * fp)88284ab085aSmws print_memarray(smbios_hdl_t *shp, id_t id, FILE *fp)
88384ab085aSmws {
88484ab085aSmws 	smbios_memarray_t ma;
88584ab085aSmws 
88684ab085aSmws 	(void) smbios_info_memarray(shp, id, &ma);
88784ab085aSmws 
88884ab085aSmws 	desc_printf(smbios_memarray_loc_desc(ma.smbma_location),
88984ab085aSmws 	    fp, "  Location: %u", ma.smbma_location);
89084ab085aSmws 
89184ab085aSmws 	desc_printf(smbios_memarray_use_desc(ma.smbma_use),
89284ab085aSmws 	    fp, "  Use: %u", ma.smbma_use);
89384ab085aSmws 
89484ab085aSmws 	desc_printf(smbios_memarray_ecc_desc(ma.smbma_ecc),
89584ab085aSmws 	    fp, "  ECC: %u", ma.smbma_ecc);
89684ab085aSmws 
89784ab085aSmws 	oprintf(fp, "  Number of Slots/Sockets: %u\n", ma.smbma_ndevs);
89884ab085aSmws 	id_printf(fp, "  Memory Error Data: ", ma.smbma_err);
89984ab085aSmws 	oprintf(fp, "  Max Capacity: %llu bytes\n",
90084ab085aSmws 	    (u_longlong_t)ma.smbma_size);
90184ab085aSmws }
90284ab085aSmws 
90384ab085aSmws static void
print_memdevice(smbios_hdl_t * shp,id_t id,FILE * fp)90484ab085aSmws print_memdevice(smbios_hdl_t *shp, id_t id, FILE *fp)
90584ab085aSmws {
90684ab085aSmws 	smbios_memdevice_t md;
90784ab085aSmws 
90884ab085aSmws 	(void) smbios_info_memdevice(shp, id, &md);
90984ab085aSmws 
91084ab085aSmws 	id_printf(fp, "  Physical Memory Array: ", md.smbmd_array);
91184ab085aSmws 	id_printf(fp, "  Memory Error Data: ", md.smbmd_error);
91284ab085aSmws 
91384ab085aSmws 	if (md.smbmd_twidth != -1u)
91484ab085aSmws 		oprintf(fp, "  Total Width: %u bits\n", md.smbmd_twidth);
91584ab085aSmws 	else
91684ab085aSmws 		oprintf(fp, "  Total Width: Unknown\n");
91784ab085aSmws 
91884ab085aSmws 	if (md.smbmd_dwidth != -1u)
91984ab085aSmws 		oprintf(fp, "  Data Width: %u bits\n", md.smbmd_dwidth);
92084ab085aSmws 	else
92184ab085aSmws 		oprintf(fp, "  Data Width: Unknown\n");
92284ab085aSmws 
92384ab085aSmws 	switch (md.smbmd_size) {
92484ab085aSmws 	case -1ull:
92584ab085aSmws 		oprintf(fp, "  Size: Unknown\n");
92684ab085aSmws 		break;
92784ab085aSmws 	case 0:
92884ab085aSmws 		oprintf(fp, "  Size: Not Populated\n");
92984ab085aSmws 		break;
93084ab085aSmws 	default:
93184ab085aSmws 		oprintf(fp, "  Size: %llu bytes\n",
93284ab085aSmws 		    (u_longlong_t)md.smbmd_size);
93384ab085aSmws 	}
93484ab085aSmws 
93584ab085aSmws 	desc_printf(smbios_memdevice_form_desc(md.smbmd_form),
93684ab085aSmws 	    fp, "  Form Factor: %u", md.smbmd_form);
93784ab085aSmws 
93884ab085aSmws 	if (md.smbmd_set == 0)
93984ab085aSmws 		oprintf(fp, "  Set: None\n");
94084ab085aSmws 	else if (md.smbmd_set == (uint8_t)-1u)
94184ab085aSmws 		oprintf(fp, "  Set: Unknown\n");
94284ab085aSmws 	else
94384ab085aSmws 		oprintf(fp, "  Set: %u\n", md.smbmd_set);
94484ab085aSmws 
9454e901881SDale Ghent 	if (md.smbmd_rank != 0) {
9464e901881SDale Ghent 		desc_printf(smbios_memdevice_rank_desc(md.smbmd_rank),
9474e901881SDale Ghent 		    fp, "  Rank: %u", md.smbmd_rank);
9484e901881SDale Ghent 	} else {
9494e901881SDale Ghent 		oprintf(fp, "  Rank: Unknown\n");
9504e901881SDale Ghent 	}
9514e901881SDale Ghent 
95284ab085aSmws 	desc_printf(smbios_memdevice_type_desc(md.smbmd_type),
95384ab085aSmws 	    fp, "  Memory Type: %u", md.smbmd_type);
95484ab085aSmws 
95584ab085aSmws 	flag_printf(fp, "Flags", md.smbmd_flags, sizeof (md.smbmd_flags) * NBBY,
95684ab085aSmws 	    smbios_memdevice_flag_name, smbios_memdevice_flag_desc);
95784ab085aSmws 
95884ab085aSmws 	if (md.smbmd_speed != 0)
9591eb9e3eaSRobert Mustacchi 		oprintf(fp, "  Speed: %u MT/s\n", md.smbmd_speed);
96084ab085aSmws 	else
96184ab085aSmws 		oprintf(fp, "  Speed: Unknown\n");
96284ab085aSmws 
9634e901881SDale Ghent 	if (md.smbmd_clkspeed != 0)
9641eb9e3eaSRobert Mustacchi 		oprintf(fp, "  Configured Speed: %u MT/s\n", md.smbmd_clkspeed);
9654e901881SDale Ghent 	else
9664e901881SDale Ghent 		oprintf(fp, "  Configured Speed: Unknown\n");
9674e901881SDale Ghent 
96884ab085aSmws 	oprintf(fp, "  Device Locator: %s\n", md.smbmd_dloc);
96984ab085aSmws 	oprintf(fp, "  Bank Locator: %s\n", md.smbmd_bloc);
9704e901881SDale Ghent 
9714e901881SDale Ghent 	if (md.smbmd_minvolt != 0) {
9724e901881SDale Ghent 		oprintf(fp, "  Minimum Voltage: %.2fV\n",
9734e901881SDale Ghent 		    md.smbmd_minvolt / 1000.0);
9744e901881SDale Ghent 	} else {
9754e901881SDale Ghent 		oprintf(fp, "  Minimum Voltage: Unknown\n");
9764e901881SDale Ghent 	}
9774e901881SDale Ghent 
9784e901881SDale Ghent 	if (md.smbmd_maxvolt != 0) {
9794e901881SDale Ghent 		oprintf(fp, "  Maximum Voltage: %.2fV\n",
9804e901881SDale Ghent 		    md.smbmd_maxvolt / 1000.0);
9814e901881SDale Ghent 	} else {
9824e901881SDale Ghent 		oprintf(fp, "  Maximum Voltage: Unknown\n");
9834e901881SDale Ghent 	}
9844e901881SDale Ghent 
9854e901881SDale Ghent 	if (md.smbmd_confvolt != 0) {
9864e901881SDale Ghent 		oprintf(fp, "  Configured Voltage: %.2fV\n",
9874e901881SDale Ghent 		    md.smbmd_confvolt / 1000.0);
9884e901881SDale Ghent 	} else {
9894e901881SDale Ghent 		oprintf(fp, "  Configured Voltage: Unknown\n");
9904e901881SDale Ghent 	}
991*a60349c8SRobert Mustacchi 
992*a60349c8SRobert Mustacchi 	if (md.smbmd_memtech != 0) {
993*a60349c8SRobert Mustacchi 		desc_printf(smbios_memdevice_memtech_desc(md.smbmd_memtech),
994*a60349c8SRobert Mustacchi 		    fp, "  Memory Technology: %u", md.smbmd_memtech);
995*a60349c8SRobert Mustacchi 	}
996*a60349c8SRobert Mustacchi 
997*a60349c8SRobert Mustacchi 	if (md.smbmd_opcap_flags != 0) {
998*a60349c8SRobert Mustacchi 		flag_printf(fp, "  Operating Mode Capabilities",
999*a60349c8SRobert Mustacchi 		    md.smbmd_opcap_flags, sizeof (md.smbmd_opcap_flags) * NBBY,
1000*a60349c8SRobert Mustacchi 		    smbios_memdevice_op_capab_name,
1001*a60349c8SRobert Mustacchi 		    smbios_memdevice_op_capab_desc);
1002*a60349c8SRobert Mustacchi 	}
1003*a60349c8SRobert Mustacchi 
1004*a60349c8SRobert Mustacchi 	if (md.smbmd_firmware_rev[0] != '\0') {
1005*a60349c8SRobert Mustacchi 		oprintf(fp, "  Firmware Revision: %s\n", md.smbmd_firmware_rev);
1006*a60349c8SRobert Mustacchi 	}
1007*a60349c8SRobert Mustacchi 
1008*a60349c8SRobert Mustacchi 	if (md.smbmd_modmfg_id != 0) {
1009*a60349c8SRobert Mustacchi 		jedec_print(fp, "Module Manufacturer ID", md.smbmd_modmfg_id);
1010*a60349c8SRobert Mustacchi 	}
1011*a60349c8SRobert Mustacchi 
1012*a60349c8SRobert Mustacchi 	if (md.smbmd_modprod_id  != 0) {
1013*a60349c8SRobert Mustacchi 		jedec_print(fp, "Module Product ID", md.smbmd_modprod_id);
1014*a60349c8SRobert Mustacchi 	}
1015*a60349c8SRobert Mustacchi 
1016*a60349c8SRobert Mustacchi 	if (md.smbmd_cntrlmfg_id != 0) {
1017*a60349c8SRobert Mustacchi 		jedec_print(fp, "Memory Subsystem Controller Manufacturer ID",
1018*a60349c8SRobert Mustacchi 		    md.smbmd_cntrlmfg_id);
1019*a60349c8SRobert Mustacchi 	}
1020*a60349c8SRobert Mustacchi 
1021*a60349c8SRobert Mustacchi 	if (md.smbmd_cntrlprod_id != 0) {
1022*a60349c8SRobert Mustacchi 		jedec_print(fp, "Memory Subsystem Controller Product ID",
1023*a60349c8SRobert Mustacchi 		    md.smbmd_cntrlprod_id);
1024*a60349c8SRobert Mustacchi 	}
1025*a60349c8SRobert Mustacchi 
1026*a60349c8SRobert Mustacchi 	if (md.smbmd_nvsize == UINT64_MAX) {
1027*a60349c8SRobert Mustacchi 		oprintf(fp, "  Non-volatile Size: Unknown\n");
1028*a60349c8SRobert Mustacchi 	} else if (md.smbmd_nvsize != 0) {
1029*a60349c8SRobert Mustacchi 		oprintf(fp, "  Non-volatile Size: %llu bytes\n",
1030*a60349c8SRobert Mustacchi 		    (u_longlong_t)md.smbmd_nvsize);
1031*a60349c8SRobert Mustacchi 	}
1032*a60349c8SRobert Mustacchi 
1033*a60349c8SRobert Mustacchi 	if (md.smbmd_volatile_size == UINT64_MAX) {
1034*a60349c8SRobert Mustacchi 		oprintf(fp, "  Volatile Size: Unknown\n");
1035*a60349c8SRobert Mustacchi 	} else if (md.smbmd_volatile_size != 0) {
1036*a60349c8SRobert Mustacchi 		oprintf(fp, "  Volatile Size: %llu bytes\n",
1037*a60349c8SRobert Mustacchi 		    (u_longlong_t)md.smbmd_volatile_size);
1038*a60349c8SRobert Mustacchi 	}
1039*a60349c8SRobert Mustacchi 
1040*a60349c8SRobert Mustacchi 	if (md.smbmd_cache_size == UINT64_MAX) {
1041*a60349c8SRobert Mustacchi 		oprintf(fp, "  Cache Size: Unknown\n");
1042*a60349c8SRobert Mustacchi 	} else if (md.smbmd_cache_size != 0) {
1043*a60349c8SRobert Mustacchi 		oprintf(fp, "  Cache Size: %llu bytes\n",
1044*a60349c8SRobert Mustacchi 		    (u_longlong_t)md.smbmd_cache_size);
1045*a60349c8SRobert Mustacchi 	}
1046*a60349c8SRobert Mustacchi 
1047*a60349c8SRobert Mustacchi 	if (md.smbmd_logical_size == UINT64_MAX) {
1048*a60349c8SRobert Mustacchi 		oprintf(fp, "  Logical Size: Unknown\n");
1049*a60349c8SRobert Mustacchi 	} else if (md.smbmd_logical_size != 0) {
1050*a60349c8SRobert Mustacchi 		oprintf(fp, "  Logical Size: %llu bytes\n",
1051*a60349c8SRobert Mustacchi 		    (u_longlong_t)md.smbmd_logical_size);
1052*a60349c8SRobert Mustacchi 	}
105384ab085aSmws }
105484ab085aSmws 
105584ab085aSmws static void
print_memarrmap(smbios_hdl_t * shp,id_t id,FILE * fp)105684ab085aSmws print_memarrmap(smbios_hdl_t *shp, id_t id, FILE *fp)
105784ab085aSmws {
105884ab085aSmws 	smbios_memarrmap_t ma;
105984ab085aSmws 
106084ab085aSmws 	(void) smbios_info_memarrmap(shp, id, &ma);
106184ab085aSmws 
106284ab085aSmws 	id_printf(fp, "  Physical Memory Array: ", ma.smbmam_array);
106384ab085aSmws 	oprintf(fp, "  Devices per Row: %u\n", ma.smbmam_width);
106484ab085aSmws 
106584ab085aSmws 	oprintf(fp, "  Physical Address: 0x%llx\n  Size: %llu bytes\n",
106684ab085aSmws 	    (u_longlong_t)ma.smbmam_addr, (u_longlong_t)ma.smbmam_size);
106784ab085aSmws }
106884ab085aSmws 
106984ab085aSmws static void
print_memdevmap(smbios_hdl_t * shp,id_t id,FILE * fp)107084ab085aSmws print_memdevmap(smbios_hdl_t *shp, id_t id, FILE *fp)
107184ab085aSmws {
107284ab085aSmws 	smbios_memdevmap_t md;
107384ab085aSmws 
107484ab085aSmws 	(void) smbios_info_memdevmap(shp, id, &md);
107584ab085aSmws 
107684ab085aSmws 	id_printf(fp, "  Memory Device: ", md.smbmdm_device);
107784ab085aSmws 	id_printf(fp, "  Memory Array Mapped Address: ", md.smbmdm_arrmap);
107884ab085aSmws 
107984ab085aSmws 	oprintf(fp, "  Physical Address: 0x%llx\n  Size: %llu bytes\n",
108084ab085aSmws 	    (u_longlong_t)md.smbmdm_addr, (u_longlong_t)md.smbmdm_size);
108184ab085aSmws 
108284ab085aSmws 	oprintf(fp, "  Partition Row Position: %u\n", md.smbmdm_rpos);
108384ab085aSmws 	oprintf(fp, "  Interleave Position: %u\n", md.smbmdm_ipos);
108484ab085aSmws 	oprintf(fp, "  Interleave Data Depth: %u\n", md.smbmdm_idepth);
108584ab085aSmws }
108684ab085aSmws 
108784ab085aSmws static void
print_hwsec(smbios_hdl_t * shp,FILE * fp)108884ab085aSmws print_hwsec(smbios_hdl_t *shp, FILE *fp)
108984ab085aSmws {
109084ab085aSmws 	smbios_hwsec_t h;
109184ab085aSmws 
109284ab085aSmws 	(void) smbios_info_hwsec(shp, &h);
109384ab085aSmws 
109484ab085aSmws 	desc_printf(smbios_hwsec_desc(h.smbh_pwr_ps),
109584ab085aSmws 	    fp, "  Power-On Password Status: %u", h.smbh_pwr_ps);
109684ab085aSmws 	desc_printf(smbios_hwsec_desc(h.smbh_kbd_ps),
109784ab085aSmws 	    fp, "  Keyboard Password Status: %u", h.smbh_kbd_ps);
109884ab085aSmws 	desc_printf(smbios_hwsec_desc(h.smbh_adm_ps),
109984ab085aSmws 	    fp, "  Administrator Password Status: %u", h.smbh_adm_ps);
110084ab085aSmws 	desc_printf(smbios_hwsec_desc(h.smbh_pan_ps),
110184ab085aSmws 	    fp, "  Front Panel Reset Status: %u", h.smbh_pan_ps);
110284ab085aSmws }
110384ab085aSmws 
110484ab085aSmws static void
print_vprobe(smbios_hdl_t * shp,id_t id,FILE * fp)1105a448814aSRobert Mustacchi print_vprobe(smbios_hdl_t *shp, id_t id, FILE *fp)
1106a448814aSRobert Mustacchi {
1107a448814aSRobert Mustacchi 	smbios_vprobe_t vp;
1108a448814aSRobert Mustacchi 
1109a448814aSRobert Mustacchi 	if (smbios_info_vprobe(shp, id, &vp) != 0) {
1110a448814aSRobert Mustacchi 		smbios_warn(shp, "failed to read voltage probe information");
1111a448814aSRobert Mustacchi 		return;
1112a448814aSRobert Mustacchi 	}
1113a448814aSRobert Mustacchi 
1114a448814aSRobert Mustacchi 	oprintf(fp, "  Description: %s\n", vp.smbvp_description != NULL ?
1115a448814aSRobert Mustacchi 	    vp.smbvp_description : "unknown");
1116a448814aSRobert Mustacchi 	desc_printf(smbios_vprobe_loc_desc(vp.smbvp_location),
1117a448814aSRobert Mustacchi 	    fp, "  Location: %u", vp.smbvp_location);
1118a448814aSRobert Mustacchi 	desc_printf(smbios_vprobe_status_desc(vp.smbvp_status),
1119a448814aSRobert Mustacchi 	    fp, "  Status: %u", vp.smbvp_status);
1120a448814aSRobert Mustacchi 
1121a448814aSRobert Mustacchi 	if (vp.smbvp_maxval != SMB_PROBE_UNKNOWN_VALUE) {
1122a448814aSRobert Mustacchi 		oprintf(fp, "  Maximum Possible Voltage: %u mV\n",
1123a448814aSRobert Mustacchi 		    vp.smbvp_maxval);
1124a448814aSRobert Mustacchi 	} else {
1125a448814aSRobert Mustacchi 		oprintf(fp, "  Maximum Possible Voltage: unknown\n");
1126a448814aSRobert Mustacchi 	}
1127a448814aSRobert Mustacchi 
1128a448814aSRobert Mustacchi 	if (vp.smbvp_minval != SMB_PROBE_UNKNOWN_VALUE) {
1129a448814aSRobert Mustacchi 		oprintf(fp, "  Minimum Possible Voltage: %u mV\n",
1130a448814aSRobert Mustacchi 		    vp.smbvp_minval);
1131a448814aSRobert Mustacchi 	} else {
1132a448814aSRobert Mustacchi 		oprintf(fp, "  Minimum Possible Voltage: unknown\n");
1133a448814aSRobert Mustacchi 	}
1134a448814aSRobert Mustacchi 
1135a448814aSRobert Mustacchi 	if (vp.smbvp_resolution != SMB_PROBE_UNKNOWN_VALUE) {
1136a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Resolution: %u.%u mV\n",
1137a448814aSRobert Mustacchi 		    vp.smbvp_resolution / 10,
1138a448814aSRobert Mustacchi 		    vp.smbvp_resolution % 10);
1139a448814aSRobert Mustacchi 	} else {
1140a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Resolution: unknown\n");
1141a448814aSRobert Mustacchi 	}
1142a448814aSRobert Mustacchi 
1143a448814aSRobert Mustacchi 	if (vp.smbvp_tolerance != SMB_PROBE_UNKNOWN_VALUE) {
1144a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: +/-%u mV\n",
1145a448814aSRobert Mustacchi 		    vp.smbvp_tolerance);
1146a448814aSRobert Mustacchi 	} else {
1147a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: unknown\n");
1148a448814aSRobert Mustacchi 	}
1149a448814aSRobert Mustacchi 
1150a448814aSRobert Mustacchi 	if (vp.smbvp_accuracy != SMB_PROBE_UNKNOWN_VALUE) {
1151a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: +/-%u.%02u%%\n",
1152a448814aSRobert Mustacchi 		    vp.smbvp_accuracy / 100,
1153a448814aSRobert Mustacchi 		    vp.smbvp_accuracy % 100);
1154a448814aSRobert Mustacchi 	} else {
1155a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: unknown\n");
1156a448814aSRobert Mustacchi 	}
1157a448814aSRobert Mustacchi 
1158a448814aSRobert Mustacchi 	oprintf(fp, "  OEM- or BIOS- defined value: 0x%x\n", vp.smbvp_oem);
1159a448814aSRobert Mustacchi 
1160a448814aSRobert Mustacchi 	if (vp.smbvp_nominal != SMB_PROBE_UNKNOWN_VALUE) {
1161a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: %u mV\n", vp.smbvp_nominal);
1162a448814aSRobert Mustacchi 	} else {
1163a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: unknown\n");
1164a448814aSRobert Mustacchi 	}
1165a448814aSRobert Mustacchi }
1166a448814aSRobert Mustacchi 
1167a448814aSRobert Mustacchi static void
print_cooldev(smbios_hdl_t * shp,id_t id,FILE * fp)1168a448814aSRobert Mustacchi print_cooldev(smbios_hdl_t *shp, id_t id, FILE *fp)
1169a448814aSRobert Mustacchi {
1170a448814aSRobert Mustacchi 	smbios_cooldev_t cd;
1171a448814aSRobert Mustacchi 
1172a448814aSRobert Mustacchi 	if (smbios_info_cooldev(shp, id, &cd) != 0) {
1173a448814aSRobert Mustacchi 		smbios_warn(shp, "failed to read cooling device "
1174a448814aSRobert Mustacchi 		    "information");
1175a448814aSRobert Mustacchi 		return;
1176a448814aSRobert Mustacchi 	}
1177a448814aSRobert Mustacchi 
1178a448814aSRobert Mustacchi 	id_printf(fp, "  Temperature Probe Handle: ", cd.smbcd_tprobe);
1179a448814aSRobert Mustacchi 	desc_printf(smbios_cooldev_type_desc(cd.smbcd_type),
1180a448814aSRobert Mustacchi 	    fp, "  Device Type: %u", cd.smbcd_type);
1181a448814aSRobert Mustacchi 	desc_printf(smbios_cooldev_status_desc(cd.smbcd_status),
1182a448814aSRobert Mustacchi 	    fp, "  Status: %u", cd.smbcd_status);
1183a448814aSRobert Mustacchi 	oprintf(fp, "  Cooling Unit Group: %u\n", cd.smbcd_group);
1184a448814aSRobert Mustacchi 	oprintf(fp, "  OEM- or BIOS- defined data: 0x%x\n", cd.smbcd_oem);
1185a448814aSRobert Mustacchi 	if (cd.smbcd_nominal != SMB_PROBE_UNKNOWN_VALUE) {
1186a448814aSRobert Mustacchi 		oprintf(fp, "  Nominal Speed: %u RPM\n", cd.smbcd_nominal);
1187a448814aSRobert Mustacchi 	} else {
1188a448814aSRobert Mustacchi 		oprintf(fp, "  Nominal Speed: unknown\n");
1189a448814aSRobert Mustacchi 	}
1190a448814aSRobert Mustacchi 
1191a448814aSRobert Mustacchi 	if (cd.smbcd_descr != NULL && cd.smbcd_descr[0] != '\0') {
1192a448814aSRobert Mustacchi 		oprintf(fp, "  Description: %s\n", cd.smbcd_descr);
1193a448814aSRobert Mustacchi 	}
1194a448814aSRobert Mustacchi }
1195a448814aSRobert Mustacchi 
1196a448814aSRobert Mustacchi static void
print_tprobe(smbios_hdl_t * shp,id_t id,FILE * fp)1197a448814aSRobert Mustacchi print_tprobe(smbios_hdl_t *shp, id_t id, FILE *fp)
1198a448814aSRobert Mustacchi {
1199a448814aSRobert Mustacchi 	smbios_tprobe_t tp;
1200a448814aSRobert Mustacchi 
1201a448814aSRobert Mustacchi 	if (smbios_info_tprobe(shp, id, &tp) != 0) {
1202a448814aSRobert Mustacchi 		smbios_warn(shp, "failed to read temperature probe "
1203a448814aSRobert Mustacchi 		    "information");
1204a448814aSRobert Mustacchi 		return;
1205a448814aSRobert Mustacchi 	}
1206a448814aSRobert Mustacchi 
1207a448814aSRobert Mustacchi 	oprintf(fp, "  Description: %s\n", tp.smbtp_description != NULL ?
1208a448814aSRobert Mustacchi 	    tp.smbtp_description : "unknown");
1209a448814aSRobert Mustacchi 	desc_printf(smbios_tprobe_loc_desc(tp.smbtp_location),
1210a448814aSRobert Mustacchi 	    fp, "  Location: %u", tp.smbtp_location);
1211a448814aSRobert Mustacchi 	desc_printf(smbios_tprobe_status_desc(tp.smbtp_status),
1212a448814aSRobert Mustacchi 	    fp, "  Status: %u", tp.smbtp_status);
1213a448814aSRobert Mustacchi 
1214a448814aSRobert Mustacchi 	if (tp.smbtp_maxval != SMB_PROBE_UNKNOWN_VALUE) {
1215a448814aSRobert Mustacchi 		oprintf(fp, "  Maximum Possible Temperature: %u.%u C\n",
1216a448814aSRobert Mustacchi 		    tp.smbtp_maxval / 10, tp.smbtp_maxval % 10);
1217a448814aSRobert Mustacchi 	} else {
1218a448814aSRobert Mustacchi 		oprintf(fp, "  Maximum Possible Temperature: unknown\n");
1219a448814aSRobert Mustacchi 	}
1220a448814aSRobert Mustacchi 
1221a448814aSRobert Mustacchi 	if (tp.smbtp_minval != SMB_PROBE_UNKNOWN_VALUE) {
1222a448814aSRobert Mustacchi 		oprintf(fp, "  Minimum Possible Temperature: %u.%u C\n",
1223a448814aSRobert Mustacchi 		    tp.smbtp_minval / 10, tp.smbtp_minval % 10);
1224a448814aSRobert Mustacchi 	} else {
1225a448814aSRobert Mustacchi 		oprintf(fp, "  Minimum Possible Temperature: unknown\n");
1226a448814aSRobert Mustacchi 	}
1227a448814aSRobert Mustacchi 
1228a448814aSRobert Mustacchi 	if (tp.smbtp_resolution != SMB_PROBE_UNKNOWN_VALUE) {
1229a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Resolution: %u.%03u C\n",
1230a448814aSRobert Mustacchi 		    tp.smbtp_resolution / 1000,
1231a448814aSRobert Mustacchi 		    tp.smbtp_resolution % 1000);
1232a448814aSRobert Mustacchi 	} else {
1233a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Resolution: unknown\n");
1234a448814aSRobert Mustacchi 	}
1235a448814aSRobert Mustacchi 
1236a448814aSRobert Mustacchi 	if (tp.smbtp_tolerance != SMB_PROBE_UNKNOWN_VALUE) {
1237a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: +/-%u.%u C\n",
1238a448814aSRobert Mustacchi 		    tp.smbtp_tolerance / 10, tp.smbtp_tolerance % 10);
1239a448814aSRobert Mustacchi 	} else {
1240a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: unknown\n");
1241a448814aSRobert Mustacchi 	}
1242a448814aSRobert Mustacchi 
1243a448814aSRobert Mustacchi 	if (tp.smbtp_accuracy != SMB_PROBE_UNKNOWN_VALUE) {
1244a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: +/-%u.%02u%%\n",
1245a448814aSRobert Mustacchi 		    tp.smbtp_accuracy / 100,
1246a448814aSRobert Mustacchi 		    tp.smbtp_accuracy % 100);
1247a448814aSRobert Mustacchi 	} else {
1248a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: unknown\n");
1249a448814aSRobert Mustacchi 	}
1250a448814aSRobert Mustacchi 
1251a448814aSRobert Mustacchi 	oprintf(fp, "  OEM- or BIOS- defined value: 0x%x\n", tp.smbtp_oem);
1252a448814aSRobert Mustacchi 
1253a448814aSRobert Mustacchi 	if (tp.smbtp_nominal != SMB_PROBE_UNKNOWN_VALUE) {
1254a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: %u.%u C\n",
1255a448814aSRobert Mustacchi 		    tp.smbtp_nominal / 10, tp.smbtp_nominal % 10);
1256a448814aSRobert Mustacchi 	} else {
1257a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: unknown\n");
1258a448814aSRobert Mustacchi 	}
1259a448814aSRobert Mustacchi }
1260a448814aSRobert Mustacchi 
1261a448814aSRobert Mustacchi static void
print_iprobe(smbios_hdl_t * shp,id_t id,FILE * fp)1262a448814aSRobert Mustacchi print_iprobe(smbios_hdl_t *shp, id_t id, FILE *fp)
1263a448814aSRobert Mustacchi {
1264a448814aSRobert Mustacchi 	smbios_iprobe_t ip;
1265a448814aSRobert Mustacchi 
1266a448814aSRobert Mustacchi 	if (smbios_info_iprobe(shp, id, &ip) != 0) {
1267a448814aSRobert Mustacchi 		smbios_warn(shp, "failed to read current probe information");
1268a448814aSRobert Mustacchi 		return;
1269a448814aSRobert Mustacchi 	}
1270a448814aSRobert Mustacchi 
1271a448814aSRobert Mustacchi 	oprintf(fp, "  Description: %s\n", ip.smbip_description != NULL ?
1272a448814aSRobert Mustacchi 	    ip.smbip_description : "unknown");
1273a448814aSRobert Mustacchi 	desc_printf(smbios_iprobe_loc_desc(ip.smbip_location),
1274a448814aSRobert Mustacchi 	    fp, "  Location: %u", ip.smbip_location);
1275a448814aSRobert Mustacchi 	desc_printf(smbios_iprobe_status_desc(ip.smbip_status),
1276a448814aSRobert Mustacchi 	    fp, "  Status: %u", ip.smbip_status);
1277a448814aSRobert Mustacchi 
1278a448814aSRobert Mustacchi 	if (ip.smbip_maxval != SMB_PROBE_UNKNOWN_VALUE) {
1279a448814aSRobert Mustacchi 		oprintf(fp, "  Maximum Possible Current: %u mA\n",
1280a448814aSRobert Mustacchi 		    ip.smbip_maxval);
1281a448814aSRobert Mustacchi 	} else {
1282a448814aSRobert Mustacchi 		oprintf(fp, "  Maximum Possible Current: unknown\n");
1283a448814aSRobert Mustacchi 	}
1284a448814aSRobert Mustacchi 
1285a448814aSRobert Mustacchi 	if (ip.smbip_minval != SMB_PROBE_UNKNOWN_VALUE) {
1286a448814aSRobert Mustacchi 		oprintf(fp, "  Minimum Possible Current: %u mA\n",
1287a448814aSRobert Mustacchi 		    ip.smbip_minval);
1288a448814aSRobert Mustacchi 	} else {
1289a448814aSRobert Mustacchi 		oprintf(fp, "  Minimum Possible Current: unknown\n");
1290a448814aSRobert Mustacchi 	}
1291a448814aSRobert Mustacchi 
1292a448814aSRobert Mustacchi 	if (ip.smbip_resolution != SMB_PROBE_UNKNOWN_VALUE) {
1293a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Resolution: %u.%u mA\n",
1294a448814aSRobert Mustacchi 		    ip.smbip_resolution / 10,
1295a448814aSRobert Mustacchi 		    ip.smbip_resolution % 10);
1296a448814aSRobert Mustacchi 	} else {
1297a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Resolution: unknown\n");
1298a448814aSRobert Mustacchi 	}
1299a448814aSRobert Mustacchi 
1300a448814aSRobert Mustacchi 	if (ip.smbip_tolerance != SMB_PROBE_UNKNOWN_VALUE) {
1301a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: +/-%u mA\n",
1302a448814aSRobert Mustacchi 		    ip.smbip_tolerance);
1303a448814aSRobert Mustacchi 	} else {
1304a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: unknown\n");
1305a448814aSRobert Mustacchi 	}
1306a448814aSRobert Mustacchi 
1307a448814aSRobert Mustacchi 	if (ip.smbip_accuracy != SMB_PROBE_UNKNOWN_VALUE) {
1308a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: +/-%u.%02u%%\n",
1309a448814aSRobert Mustacchi 		    ip.smbip_accuracy / 100,
1310a448814aSRobert Mustacchi 		    ip.smbip_accuracy % 100);
1311a448814aSRobert Mustacchi 	} else {
1312a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: unknown\n");
1313a448814aSRobert Mustacchi 	}
1314a448814aSRobert Mustacchi 
1315a448814aSRobert Mustacchi 	oprintf(fp, "  OEM- or BIOS- defined value: 0x%x\n", ip.smbip_oem);
1316a448814aSRobert Mustacchi 
1317a448814aSRobert Mustacchi 	if (ip.smbip_nominal != SMB_PROBE_UNKNOWN_VALUE) {
1318a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: %u mA\n", ip.smbip_nominal);
1319a448814aSRobert Mustacchi 	} else {
1320a448814aSRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: unknown\n");
1321a448814aSRobert Mustacchi 	}
1322a448814aSRobert Mustacchi }
1323a448814aSRobert Mustacchi 
1324a448814aSRobert Mustacchi 
1325a448814aSRobert Mustacchi static void
print_boot(smbios_hdl_t * shp,FILE * fp)132684ab085aSmws print_boot(smbios_hdl_t *shp, FILE *fp)
132784ab085aSmws {
132884ab085aSmws 	smbios_boot_t b;
132984ab085aSmws 
133084ab085aSmws 	(void) smbios_info_boot(shp, &b);
133184ab085aSmws 
133284ab085aSmws 	desc_printf(smbios_boot_desc(b.smbt_status),
133384ab085aSmws 	    fp, "  Boot Status Code: 0x%x", b.smbt_status);
133484ab085aSmws 
133584ab085aSmws 	if (b.smbt_size != 0) {
133684ab085aSmws 		oprintf(fp, "  Boot Data (%lu bytes):\n", (ulong_t)b.smbt_size);
133784ab085aSmws 		print_bytes(b.smbt_data, b.smbt_size, fp);
133884ab085aSmws 	}
133984ab085aSmws }
134084ab085aSmws 
134184ab085aSmws static void
print_ipmi(smbios_hdl_t * shp,FILE * fp)134284ab085aSmws print_ipmi(smbios_hdl_t *shp, FILE *fp)
134384ab085aSmws {
134484ab085aSmws 	smbios_ipmi_t i;
134584ab085aSmws 
134684ab085aSmws 	(void) smbios_info_ipmi(shp, &i);
134784ab085aSmws 
134884ab085aSmws 	desc_printf(smbios_ipmi_type_desc(i.smbip_type),
134984ab085aSmws 	    fp, "  Type: %u", i.smbip_type);
135084ab085aSmws 
135184ab085aSmws 	oprintf(fp, "  BMC IPMI Version: %u.%u\n",
135284ab085aSmws 	    i.smbip_vers.smbv_major, i.smbip_vers.smbv_minor);
135384ab085aSmws 
135484ab085aSmws 	oprintf(fp, "  i2c Bus Slave Address: 0x%x\n", i.smbip_i2c);
135584ab085aSmws 	oprintf(fp, "  NV Storage Device Bus ID: 0x%x\n", i.smbip_bus);
135684ab085aSmws 	oprintf(fp, "  BMC Base Address: 0x%llx\n", (u_longlong_t)i.smbip_addr);
135784ab085aSmws 	oprintf(fp, "  Interrupt Number: %u\n", i.smbip_intr);
135884ab085aSmws 	oprintf(fp, "  Register Spacing: %u\n", i.smbip_regspacing);
135984ab085aSmws 
136084ab085aSmws 	flag_printf(fp, "Flags", i.smbip_flags, sizeof (i.smbip_flags) * NBBY,
136184ab085aSmws 	    smbios_ipmi_flag_name, smbios_ipmi_flag_desc);
136284ab085aSmws }
136384ab085aSmws 
1364074bb90dSTom Pothier static void
print_powersup(smbios_hdl_t * shp,id_t id,FILE * fp)136565fa020fSRobert Mustacchi print_powersup(smbios_hdl_t *shp, id_t id, FILE *fp)
136665fa020fSRobert Mustacchi {
136765fa020fSRobert Mustacchi 	smbios_powersup_t p;
136865fa020fSRobert Mustacchi 
136965fa020fSRobert Mustacchi 	if (smbios_info_powersup(shp, id, &p) != 0) {
137065fa020fSRobert Mustacchi 		smbios_warn(shp, "failed to read power supply information");
137165fa020fSRobert Mustacchi 		return;
137265fa020fSRobert Mustacchi 	}
137365fa020fSRobert Mustacchi 
137465fa020fSRobert Mustacchi 	oprintf(fp, "  Power Supply Group: %u\n", p.smbps_group);
137565fa020fSRobert Mustacchi 	if (p.smbps_maxout != 0x8000) {
137665fa020fSRobert Mustacchi 		oprintf(fp, "  Maximum Output: %llu mW\n", p.smbps_maxout);
137765fa020fSRobert Mustacchi 	} else {
137865fa020fSRobert Mustacchi 		oprintf(fp, "  Maximum Output: unknown\n");
137965fa020fSRobert Mustacchi 	}
138065fa020fSRobert Mustacchi 
138165fa020fSRobert Mustacchi 	flag_printf(fp, "Characteristics", p.smbps_flags,
138265fa020fSRobert Mustacchi 	    sizeof (p.smbps_flags) * NBBY, smbios_powersup_flag_name,
138365fa020fSRobert Mustacchi 	    smbios_powersup_flag_desc);
138465fa020fSRobert Mustacchi 
138565fa020fSRobert Mustacchi 	desc_printf(smbios_powersup_input_desc(p.smbps_ivrs),
138665fa020fSRobert Mustacchi 	    fp, "  Input Voltage Range Switching: %u", p.smbps_ivrs);
138765fa020fSRobert Mustacchi 	desc_printf(smbios_powersup_status_desc(p.smbps_status),
138865fa020fSRobert Mustacchi 	    fp, "  Status: %u", p.smbps_status);
138965fa020fSRobert Mustacchi 	desc_printf(smbios_powersup_type_desc(p.smbps_pstype),
139065fa020fSRobert Mustacchi 	    fp, "  Type: %u", p.smbps_pstype);
139165fa020fSRobert Mustacchi 
139265fa020fSRobert Mustacchi 	if (p.smbps_vprobe != 0xffff) {
139365fa020fSRobert Mustacchi 		oprintf(fp, "  Voltage Probe Handle: %lu\n", p.smbps_vprobe);
139465fa020fSRobert Mustacchi 	}
139565fa020fSRobert Mustacchi 
139665fa020fSRobert Mustacchi 	if (p.smbps_cooldev != 0xffff) {
139765fa020fSRobert Mustacchi 		oprintf(fp, "  Cooling Device Handle: %lu\n", p.smbps_cooldev);
139865fa020fSRobert Mustacchi 	}
139965fa020fSRobert Mustacchi 
140065fa020fSRobert Mustacchi 	if (p.smbps_iprobe != 0xffff) {
140165fa020fSRobert Mustacchi 		oprintf(fp, "  Current Probe Handle: %lu\n", p.smbps_iprobe);
140265fa020fSRobert Mustacchi 	}
140365fa020fSRobert Mustacchi }
140465fa020fSRobert Mustacchi 
140565fa020fSRobert Mustacchi static void
print_extprocessor(smbios_hdl_t * shp,id_t id,FILE * fp)1406074bb90dSTom Pothier print_extprocessor(smbios_hdl_t *shp, id_t id, FILE *fp)
1407074bb90dSTom Pothier {
1408074bb90dSTom Pothier 	int i;
1409074bb90dSTom Pothier 	smbios_processor_ext_t ep;
1410074bb90dSTom Pothier 
1411074bb90dSTom Pothier 	if (check_oem(shp) != 0)
1412074bb90dSTom Pothier 		return;
1413074bb90dSTom Pothier 
1414074bb90dSTom Pothier 	(void) smbios_info_extprocessor(shp, id, &ep);
1415074bb90dSTom Pothier 
1416074bb90dSTom Pothier 	oprintf(fp, "  Processor: %u\n", ep.smbpe_processor);
1417074bb90dSTom Pothier 	oprintf(fp, "  FRU: %u\n", ep.smbpe_fru);
1418074bb90dSTom Pothier 	oprintf(fp, "  Initial APIC ID count: %u\n\n", ep.smbpe_n);
1419074bb90dSTom Pothier 
1420074bb90dSTom Pothier 	for (i = 0; i < ep.smbpe_n; i++) {
1421074bb90dSTom Pothier 		oprintf(fp, "  Logical Strand %u: Initial APIC ID: %u\n", i,
1422074bb90dSTom Pothier 		    ep.smbpe_apicid[i]);
1423074bb90dSTom Pothier 	}
1424074bb90dSTom Pothier }
1425074bb90dSTom Pothier 
1426074bb90dSTom Pothier static void
print_extport(smbios_hdl_t * shp,id_t id,FILE * fp)142703f9f63dSTom Pothier print_extport(smbios_hdl_t *shp, id_t id, FILE *fp)
142803f9f63dSTom Pothier {
142903f9f63dSTom Pothier 	smbios_port_ext_t epo;
143003f9f63dSTom Pothier 
143103f9f63dSTom Pothier 	if (check_oem(shp) != 0)
143203f9f63dSTom Pothier 		return;
143303f9f63dSTom Pothier 
143403f9f63dSTom Pothier 	(void) smbios_info_extport(shp, id, &epo);
143503f9f63dSTom Pothier 
143603f9f63dSTom Pothier 	oprintf(fp, "  Chassis Handle: %u\n", epo.smbporte_chassis);
143703f9f63dSTom Pothier 	oprintf(fp, "  Port Connector Handle: %u\n", epo.smbporte_port);
143803f9f63dSTom Pothier 	oprintf(fp, "  Device Type: %u\n", epo.smbporte_dtype);
143903f9f63dSTom Pothier 	oprintf(fp, "  Device Handle: %u\n", epo.smbporte_devhdl);
144003f9f63dSTom Pothier 	oprintf(fp, "  PHY: %u\n", epo.smbporte_phy);
144103f9f63dSTom Pothier }
144203f9f63dSTom Pothier 
144303f9f63dSTom Pothier static void
print_pciexrc(smbios_hdl_t * shp,id_t id,FILE * fp)1444074bb90dSTom Pothier print_pciexrc(smbios_hdl_t *shp, id_t id, FILE *fp)
1445074bb90dSTom Pothier {
1446074bb90dSTom Pothier 	smbios_pciexrc_t pcie;
1447074bb90dSTom Pothier 
1448074bb90dSTom Pothier 	if (check_oem(shp) != 0)
1449074bb90dSTom Pothier 		return;
1450074bb90dSTom Pothier 
1451074bb90dSTom Pothier 	(void) smbios_info_pciexrc(shp, id, &pcie);
1452074bb90dSTom Pothier 
1453074bb90dSTom Pothier 	oprintf(fp, "  Component ID: %u\n", pcie.smbpcie_bb);
1454074bb90dSTom Pothier 	oprintf(fp, "  BDF: 0x%x\n", pcie.smbpcie_bdf);
1455074bb90dSTom Pothier }
1456074bb90dSTom Pothier 
1457074bb90dSTom Pothier static void
print_extmemarray(smbios_hdl_t * shp,id_t id,FILE * fp)1458074bb90dSTom Pothier print_extmemarray(smbios_hdl_t *shp, id_t id, FILE *fp)
1459074bb90dSTom Pothier {
1460074bb90dSTom Pothier 	smbios_memarray_ext_t em;
1461074bb90dSTom Pothier 
1462074bb90dSTom Pothier 	if (check_oem(shp) != 0)
1463074bb90dSTom Pothier 		return;
1464074bb90dSTom Pothier 
1465074bb90dSTom Pothier 	(void) smbios_info_extmemarray(shp, id, &em);
1466074bb90dSTom Pothier 
1467074bb90dSTom Pothier 	oprintf(fp, "  Physical Memory Array Handle: %u\n", em.smbmae_ma);
1468074bb90dSTom Pothier 	oprintf(fp, "  Component Parent Handle: %u\n", em.smbmae_comp);
1469074bb90dSTom Pothier 	oprintf(fp, "  BDF: 0x%x\n", em.smbmae_bdf);
1470074bb90dSTom Pothier }
1471074bb90dSTom Pothier 
1472074bb90dSTom Pothier static void
print_extmemdevice(smbios_hdl_t * shp,id_t id,FILE * fp)1473074bb90dSTom Pothier print_extmemdevice(smbios_hdl_t *shp, id_t id, FILE *fp)
1474074bb90dSTom Pothier {
1475074bb90dSTom Pothier 	int i;
1476074bb90dSTom Pothier 	smbios_memdevice_ext_t emd;
1477074bb90dSTom Pothier 
1478074bb90dSTom Pothier 	if (check_oem(shp) != 0)
1479074bb90dSTom Pothier 		return;
1480074bb90dSTom Pothier 
1481074bb90dSTom Pothier 	(void) smbios_info_extmemdevice(shp, id, &emd);
1482074bb90dSTom Pothier 
1483074bb90dSTom Pothier 	oprintf(fp, "  Memory Device Handle: %u\n", emd.smbmdeve_md);
1484074bb90dSTom Pothier 	oprintf(fp, "  DRAM Channel: %u\n", emd.smbmdeve_drch);
1485074bb90dSTom Pothier 	oprintf(fp, "  Number of Chip Selects: %u\n", emd.smbmdeve_ncs);
1486074bb90dSTom Pothier 
1487074bb90dSTom Pothier 	for (i = 0; i < emd.smbmdeve_ncs; i++) {
1488074bb90dSTom Pothier 		oprintf(fp, "  Chip Select: %u\n", emd.smbmdeve_cs[i]);
1489074bb90dSTom Pothier 	}
1490074bb90dSTom Pothier }
1491074bb90dSTom Pothier 
1492074bb90dSTom Pothier static int
print_struct(smbios_hdl_t * shp,const smbios_struct_t * sp,void * fp)149384ab085aSmws print_struct(smbios_hdl_t *shp, const smbios_struct_t *sp, void *fp)
149484ab085aSmws {
149584ab085aSmws 	smbios_info_t info;
149684ab085aSmws 	int hex = opt_x;
149784ab085aSmws 	const char *s;
149884ab085aSmws 
149984ab085aSmws 	if (opt_t != -1 && opt_t != sp->smbstr_type)
150084ab085aSmws 		return (0); /* skip struct if type doesn't match -t */
150184ab085aSmws 
150284ab085aSmws 	if (!opt_O && (sp->smbstr_type == SMB_TYPE_MEMCTL ||
150384ab085aSmws 	    sp->smbstr_type == SMB_TYPE_MEMMOD))
150484ab085aSmws 		return (0); /* skip struct if type is obsolete */
150584ab085aSmws 
150684ab085aSmws 	if (g_hdr++ == 0 || !opt_s)
150784ab085aSmws 		oprintf(fp, "%-5s %-4s %s\n", "ID", "SIZE", "TYPE");
150884ab085aSmws 
150984ab085aSmws 	oprintf(fp, "%-5u %-4lu",
151084ab085aSmws 	    (uint_t)sp->smbstr_id, (ulong_t)sp->smbstr_size);
151184ab085aSmws 
151284ab085aSmws 	if ((s = smbios_type_name(sp->smbstr_type)) != NULL)
151395ae7286SDan McDonald 		oprintf(fp, " %s (type %u)", s, sp->smbstr_type);
151484ab085aSmws 	else if (sp->smbstr_type > SMB_TYPE_OEM_LO &&
151584ab085aSmws 	    sp->smbstr_type < SMB_TYPE_OEM_HI)
151695ae7286SDan McDonald 		oprintf(fp, " %s+%u (type %u)", "SMB_TYPE_OEM_LO",
151795ae7286SDan McDonald 		    sp->smbstr_type - SMB_TYPE_OEM_LO, sp->smbstr_type);
151884ab085aSmws 	else
151984ab085aSmws 		oprintf(fp, " %u", sp->smbstr_type);
152084ab085aSmws 
152184ab085aSmws 	if ((s = smbios_type_desc(sp->smbstr_type)) != NULL)
152284ab085aSmws 		oprintf(fp, " (%s)\n", s);
152384ab085aSmws 	else
152484ab085aSmws 		oprintf(fp, "\n");
152584ab085aSmws 
152684ab085aSmws 	if (opt_s)
152784ab085aSmws 		return (0); /* only print header line if -s specified */
152884ab085aSmws 
152984ab085aSmws 	if (smbios_info_common(shp, sp->smbstr_id, &info) == 0) {
153084ab085aSmws 		oprintf(fp, "\n");
153184ab085aSmws 		print_common(&info, fp);
153284ab085aSmws 	}
153384ab085aSmws 
153484ab085aSmws 	switch (sp->smbstr_type) {
153584ab085aSmws 	case SMB_TYPE_BIOS:
153684ab085aSmws 		oprintf(fp, "\n");
153784ab085aSmws 		print_bios(shp, fp);
153884ab085aSmws 		break;
153984ab085aSmws 	case SMB_TYPE_SYSTEM:
154084ab085aSmws 		oprintf(fp, "\n");
154184ab085aSmws 		print_system(shp, fp);
154284ab085aSmws 		break;
154384ab085aSmws 	case SMB_TYPE_BASEBOARD:
154484ab085aSmws 		oprintf(fp, "\n");
154584ab085aSmws 		print_bboard(shp, sp->smbstr_id, fp);
154684ab085aSmws 		break;
154784ab085aSmws 	case SMB_TYPE_CHASSIS:
154884ab085aSmws 		oprintf(fp, "\n");
154984ab085aSmws 		print_chassis(shp, sp->smbstr_id, fp);
155084ab085aSmws 		break;
155184ab085aSmws 	case SMB_TYPE_PROCESSOR:
155284ab085aSmws 		oprintf(fp, "\n");
155384ab085aSmws 		print_processor(shp, sp->smbstr_id, fp);
155484ab085aSmws 		break;
155584ab085aSmws 	case SMB_TYPE_CACHE:
155684ab085aSmws 		oprintf(fp, "\n");
155784ab085aSmws 		print_cache(shp, sp->smbstr_id, fp);
155884ab085aSmws 		break;
155984ab085aSmws 	case SMB_TYPE_PORT:
156084ab085aSmws 		oprintf(fp, "\n");
156184ab085aSmws 		print_port(shp, sp->smbstr_id, fp);
156284ab085aSmws 		break;
156384ab085aSmws 	case SMB_TYPE_SLOT:
156484ab085aSmws 		oprintf(fp, "\n");
156584ab085aSmws 		print_slot(shp, sp->smbstr_id, fp);
156684ab085aSmws 		break;
156784ab085aSmws 	case SMB_TYPE_OBDEVS:
156884ab085aSmws 		oprintf(fp, "\n");
156984ab085aSmws 		print_obdevs(shp, sp->smbstr_id, fp);
157084ab085aSmws 		break;
157184ab085aSmws 	case SMB_TYPE_OEMSTR:
157284ab085aSmws 	case SMB_TYPE_SYSCONFSTR:
157384ab085aSmws 		oprintf(fp, "\n");
157484ab085aSmws 		print_strtab(shp, sp->smbstr_id, fp);
157584ab085aSmws 		break;
157684ab085aSmws 	case SMB_TYPE_LANG:
157784ab085aSmws 		oprintf(fp, "\n");
157884ab085aSmws 		print_lang(shp, sp->smbstr_id, fp);
157984ab085aSmws 		break;
158084ab085aSmws 	case SMB_TYPE_EVENTLOG:
158184ab085aSmws 		oprintf(fp, "\n");
158284ab085aSmws 		print_evlog(shp, sp->smbstr_id, fp);
158384ab085aSmws 		break;
158484ab085aSmws 	case SMB_TYPE_MEMARRAY:
158584ab085aSmws 		oprintf(fp, "\n");
158684ab085aSmws 		print_memarray(shp, sp->smbstr_id, fp);
158784ab085aSmws 		break;
158884ab085aSmws 	case SMB_TYPE_MEMDEVICE:
158984ab085aSmws 		oprintf(fp, "\n");
159084ab085aSmws 		print_memdevice(shp, sp->smbstr_id, fp);
159184ab085aSmws 		break;
159284ab085aSmws 	case SMB_TYPE_MEMARRAYMAP:
159384ab085aSmws 		oprintf(fp, "\n");
159484ab085aSmws 		print_memarrmap(shp, sp->smbstr_id, fp);
159584ab085aSmws 		break;
159684ab085aSmws 	case SMB_TYPE_MEMDEVICEMAP:
159784ab085aSmws 		oprintf(fp, "\n");
159884ab085aSmws 		print_memdevmap(shp, sp->smbstr_id, fp);
159984ab085aSmws 		break;
160084ab085aSmws 	case SMB_TYPE_SECURITY:
160184ab085aSmws 		oprintf(fp, "\n");
160284ab085aSmws 		print_hwsec(shp, fp);
160384ab085aSmws 		break;
1604a448814aSRobert Mustacchi 	case SMB_TYPE_VPROBE:
1605a448814aSRobert Mustacchi 		oprintf(fp, "\n");
1606a448814aSRobert Mustacchi 		print_vprobe(shp, sp->smbstr_id, fp);
1607a448814aSRobert Mustacchi 		break;
1608a448814aSRobert Mustacchi 	case SMB_TYPE_COOLDEV:
1609a448814aSRobert Mustacchi 		oprintf(fp, "\n");
1610a448814aSRobert Mustacchi 		print_cooldev(shp, sp->smbstr_id, fp);
1611a448814aSRobert Mustacchi 		break;
1612a448814aSRobert Mustacchi 	case SMB_TYPE_TPROBE:
1613a448814aSRobert Mustacchi 		oprintf(fp, "\n");
1614a448814aSRobert Mustacchi 		print_tprobe(shp, sp->smbstr_id, fp);
1615a448814aSRobert Mustacchi 		break;
1616a448814aSRobert Mustacchi 	case SMB_TYPE_IPROBE:
1617a448814aSRobert Mustacchi 		oprintf(fp, "\n");
1618a448814aSRobert Mustacchi 		print_iprobe(shp, sp->smbstr_id, fp);
1619a448814aSRobert Mustacchi 		break;
162084ab085aSmws 	case SMB_TYPE_BOOT:
162184ab085aSmws 		oprintf(fp, "\n");
162284ab085aSmws 		print_boot(shp, fp);
162384ab085aSmws 		break;
162484ab085aSmws 	case SMB_TYPE_IPMIDEV:
162584ab085aSmws 		oprintf(fp, "\n");
162684ab085aSmws 		print_ipmi(shp, fp);
162784ab085aSmws 		break;
162865fa020fSRobert Mustacchi 	case SMB_TYPE_POWERSUP:
162965fa020fSRobert Mustacchi 		oprintf(fp, "\n");
163065fa020fSRobert Mustacchi 		print_powersup(shp, sp->smbstr_id, fp);
163165fa020fSRobert Mustacchi 		break;
163203f9f63dSTom Pothier 	case SMB_TYPE_OBDEVEXT:
163303f9f63dSTom Pothier 		oprintf(fp, "\n");
163403f9f63dSTom Pothier 		print_obdevs_ext(shp, sp->smbstr_id, fp);
163503f9f63dSTom Pothier 		break;
1636074bb90dSTom Pothier 	case SUN_OEM_EXT_PROCESSOR:
1637074bb90dSTom Pothier 		oprintf(fp, "\n");
1638074bb90dSTom Pothier 		print_extprocessor(shp, sp->smbstr_id, fp);
1639074bb90dSTom Pothier 		break;
164003f9f63dSTom Pothier 	case SUN_OEM_EXT_PORT:
164103f9f63dSTom Pothier 		oprintf(fp, "\n");
164203f9f63dSTom Pothier 		print_extport(shp, sp->smbstr_id, fp);
164303f9f63dSTom Pothier 		break;
1644074bb90dSTom Pothier 	case SUN_OEM_PCIEXRC:
1645074bb90dSTom Pothier 		oprintf(fp, "\n");
1646074bb90dSTom Pothier 		print_pciexrc(shp, sp->smbstr_id, fp);
1647074bb90dSTom Pothier 		break;
1648074bb90dSTom Pothier 	case SUN_OEM_EXT_MEMARRAY:
1649074bb90dSTom Pothier 		oprintf(fp, "\n");
1650074bb90dSTom Pothier 		print_extmemarray(shp, sp->smbstr_id, fp);
1651074bb90dSTom Pothier 		break;
1652074bb90dSTom Pothier 	case SUN_OEM_EXT_MEMDEVICE:
1653074bb90dSTom Pothier 		oprintf(fp, "\n");
1654074bb90dSTom Pothier 		print_extmemdevice(shp, sp->smbstr_id, fp);
1655074bb90dSTom Pothier 		break;
165684ab085aSmws 	default:
165784ab085aSmws 		hex++;
165884ab085aSmws 	}
165984ab085aSmws 
166084ab085aSmws 	if (hex)
166184ab085aSmws 		print_bytes(sp->smbstr_data, sp->smbstr_size, fp);
166284ab085aSmws 	else
166384ab085aSmws 		oprintf(fp, "\n");
166484ab085aSmws 
166584ab085aSmws 	return (0);
166684ab085aSmws }
166784ab085aSmws 
166884ab085aSmws static uint16_t
getu16(const char * name,const char * s)166984ab085aSmws getu16(const char *name, const char *s)
167084ab085aSmws {
167184ab085aSmws 	u_longlong_t val;
167284ab085aSmws 	char *p;
167384ab085aSmws 
167484ab085aSmws 	errno = 0;
167584ab085aSmws 	val = strtoull(s, &p, 0);
167684ab085aSmws 
167784ab085aSmws 	if (errno != 0 || p == s || *p != '\0' || val > UINT16_MAX) {
167884ab085aSmws 		(void) fprintf(stderr, "%s: invalid %s argument -- %s\n",
167984ab085aSmws 		    g_pname, name, s);
168084ab085aSmws 		exit(SMBIOS_USAGE);
168184ab085aSmws 	}
168284ab085aSmws 
168384ab085aSmws 	return ((uint16_t)val);
168484ab085aSmws }
168584ab085aSmws 
168684ab085aSmws static uint16_t
getstype(const char * name,const char * s)168784ab085aSmws getstype(const char *name, const char *s)
168884ab085aSmws {
168984ab085aSmws 	const char *ts;
169084ab085aSmws 	uint16_t t;
169184ab085aSmws 
169284ab085aSmws 	for (t = 0; t < SMB_TYPE_OEM_LO; t++) {
169384ab085aSmws 		if ((ts = smbios_type_name(t)) != NULL && strcmp(s, ts) == 0)
169484ab085aSmws 			return (t);
169584ab085aSmws 	}
169684ab085aSmws 
169784ab085aSmws 	(void) fprintf(stderr, "%s: invalid %s argument -- %s\n",
169884ab085aSmws 	    g_pname, name, s);
169984ab085aSmws 
170084ab085aSmws 	exit(SMBIOS_USAGE);
170184ab085aSmws 	/*NOTREACHED*/
170284ab085aSmws }
170384ab085aSmws 
170484ab085aSmws static int
usage(FILE * fp)170584ab085aSmws usage(FILE *fp)
170684ab085aSmws {
170784ab085aSmws 	(void) fprintf(fp, "Usage: %s "
170884ab085aSmws 	    "[-BeOsx] [-i id] [-t type] [-w file] [file]\n\n", g_pname);
170984ab085aSmws 
171084ab085aSmws 	(void) fprintf(fp,
171184ab085aSmws 	    "\t-B disable header validation for broken BIOSes\n"
171284ab085aSmws 	    "\t-e display SMBIOS entry point information\n"
171384ab085aSmws 	    "\t-i display only the specified structure\n"
171484ab085aSmws 	    "\t-O display obsolete structure types\n"
171584ab085aSmws 	    "\t-s display only a summary of structure identifiers and types\n"
171684ab085aSmws 	    "\t-t display only the specified structure type\n"
171784ab085aSmws 	    "\t-w write the raw data to the specified file\n"
171884ab085aSmws 	    "\t-x display raw data for structures\n");
171984ab085aSmws 
172084ab085aSmws 	return (SMBIOS_USAGE);
172184ab085aSmws }
172284ab085aSmws 
172384ab085aSmws int
main(int argc,char * argv[])172484ab085aSmws main(int argc, char *argv[])
172584ab085aSmws {
172684ab085aSmws 	const char *ifile = NULL;
172784ab085aSmws 	const char *ofile = NULL;
172884ab085aSmws 	int oflags = 0;
172984ab085aSmws 
173084ab085aSmws 	smbios_hdl_t *shp;
173184ab085aSmws 	smbios_struct_t s;
173284ab085aSmws 	int err, fd, c;
173384ab085aSmws 	char *p;
173484ab085aSmws 
173584ab085aSmws 	if ((p = strrchr(argv[0], '/')) == NULL)
173684ab085aSmws 		g_pname = argv[0];
173784ab085aSmws 	else
173884ab085aSmws 		g_pname = p + 1;
173984ab085aSmws 
174084ab085aSmws 	while (optind < argc) {
174184ab085aSmws 		while ((c = getopt(argc, argv, "Bei:Ost:w:xZ")) != EOF) {
174284ab085aSmws 			switch (c) {
174384ab085aSmws 			case 'B':
174484ab085aSmws 				oflags |= SMB_O_NOCKSUM | SMB_O_NOVERS;
174584ab085aSmws 				break;
174684ab085aSmws 			case 'e':
174784ab085aSmws 				opt_e++;
174884ab085aSmws 				break;
174984ab085aSmws 			case 'i':
175084ab085aSmws 				opt_i = getu16("struct ID", optarg);
175184ab085aSmws 				break;
175284ab085aSmws 			case 'O':
175384ab085aSmws 				opt_O++;
175484ab085aSmws 				break;
175584ab085aSmws 			case 's':
175684ab085aSmws 				opt_s++;
175784ab085aSmws 				break;
175884ab085aSmws 			case 't':
175984ab085aSmws 				if (isdigit(optarg[0]))
176084ab085aSmws 					opt_t = getu16("struct type", optarg);
176184ab085aSmws 				else
176284ab085aSmws 					opt_t = getstype("struct type", optarg);
176384ab085aSmws 				break;
176484ab085aSmws 			case 'w':
176584ab085aSmws 				ofile = optarg;
176684ab085aSmws 				break;
176784ab085aSmws 			case 'x':
176884ab085aSmws 				opt_x++;
176984ab085aSmws 				break;
177084ab085aSmws 			case 'Z':
177184ab085aSmws 				oflags |= SMB_O_ZIDS; /* undocumented */
177284ab085aSmws 				break;
177384ab085aSmws 			default:
177484ab085aSmws 				return (usage(stderr));
177584ab085aSmws 			}
177684ab085aSmws 		}
177784ab085aSmws 
177884ab085aSmws 		if (optind < argc) {
177984ab085aSmws 			if (ifile != NULL) {
178084ab085aSmws 				(void) fprintf(stderr, "%s: illegal "
178184ab085aSmws 				    "argument -- %s\n", g_pname, argv[optind]);
178284ab085aSmws 				return (SMBIOS_USAGE);
178384ab085aSmws 			}
178484ab085aSmws 			ifile = argv[optind++];
178584ab085aSmws 		}
178684ab085aSmws 	}
178784ab085aSmws 
178884ab085aSmws 	if ((shp = smbios_open(ifile, SMB_VERSION, oflags, &err)) == NULL) {
178984ab085aSmws 		(void) fprintf(stderr, "%s: failed to load SMBIOS: %s\n",
179084ab085aSmws 		    g_pname, smbios_errmsg(err));
179184ab085aSmws 		return (SMBIOS_ERROR);
179284ab085aSmws 	}
179384ab085aSmws 
1794dabec466SJonathan Matthew 	if (opt_i == -1 && opt_t == -1 && opt_e == 0 &&
1795dabec466SJonathan Matthew 	    smbios_truncated(shp))
1796dabec466SJonathan Matthew 		(void) fprintf(stderr, "%s: SMBIOS table is truncated\n",
1797dabec466SJonathan Matthew 		    g_pname);
1798dabec466SJonathan Matthew 
179984ab085aSmws 	if (ofile != NULL) {
180084ab085aSmws 		if ((fd = open(ofile, O_WRONLY|O_CREAT|O_TRUNC, 0666)) == -1) {
180184ab085aSmws 			(void) fprintf(stderr, "%s: failed to open %s: %s\n",
180284ab085aSmws 			    g_pname, ofile, strerror(errno));
180384ab085aSmws 			err = SMBIOS_ERROR;
180484ab085aSmws 		} else if (smbios_write(shp, fd) != 0) {
180584ab085aSmws 			(void) fprintf(stderr, "%s: failed to write %s: %s\n",
180684ab085aSmws 			    g_pname, ofile, smbios_errmsg(smbios_errno(shp)));
180784ab085aSmws 			err = SMBIOS_ERROR;
180884ab085aSmws 		}
180984ab085aSmws 		smbios_close(shp);
181084ab085aSmws 		return (err);
181184ab085aSmws 	}
181284ab085aSmws 
181384ab085aSmws 	if (opt_e) {
181484ab085aSmws 		print_smbios(shp, stdout);
181584ab085aSmws 		smbios_close(shp);
181684ab085aSmws 		return (SMBIOS_SUCCESS);
181784ab085aSmws 	}
181884ab085aSmws 
181984ab085aSmws 	if (opt_O && (opt_i != -1 || opt_t != -1))
182084ab085aSmws 		opt_O++; /* -i or -t imply displaying obsolete records */
182184ab085aSmws 
182284ab085aSmws 	if (opt_i != -1)
182384ab085aSmws 		err = smbios_lookup_id(shp, opt_i, &s);
182484ab085aSmws 	else
182584ab085aSmws 		err = smbios_iter(shp, print_struct, stdout);
182684ab085aSmws 
182784ab085aSmws 	if (err != 0) {
182884ab085aSmws 		(void) fprintf(stderr, "%s: failed to access SMBIOS: %s\n",
182984ab085aSmws 		    g_pname, smbios_errmsg(smbios_errno(shp)));
183084ab085aSmws 		smbios_close(shp);
183184ab085aSmws 		return (SMBIOS_ERROR);
183284ab085aSmws 	}
183384ab085aSmws 
183484ab085aSmws 	if (opt_i != -1)
183584ab085aSmws 		(void) print_struct(shp, &s, stdout);
183684ab085aSmws 
183784ab085aSmws 	smbios_close(shp);
183884ab085aSmws 	return (SMBIOS_SUCCESS);
183984ab085aSmws }
1840