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