xref: /illumos-gate/usr/src/cmd/smbios/smbios.c (revision 482d9a1a850e3b2f04f2b4c7bad55b93f23e5c8d)
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.
24e9cac61dSRobert Mustacchi  * Copyright (c) 2017, Joyent, Inc.
25064d431aSRobert Mustacchi  * Copyright 2024 Oxide Computer Company
2603f9f63dSTom Pothier  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
2784ab085aSmws  * Use is subject to license terms.
2884ab085aSmws  */
2984ab085aSmws 
3084ab085aSmws #include <sys/sysmacros.h>
3184ab085aSmws #include <sys/param.h>
326bc074b1SRobert Mustacchi #include <sys/bitext.h>
3343379a28SAndy Fiddaman #include <sys/hexdump.h>
3484ab085aSmws 
3584ab085aSmws #include <smbios.h>
3684ab085aSmws #include <alloca.h>
3784ab085aSmws #include <limits.h>
3884ab085aSmws #include <unistd.h>
3984ab085aSmws #include <strings.h>
4084ab085aSmws #include <stdlib.h>
4184ab085aSmws #include <stdarg.h>
4284ab085aSmws #include <stdio.h>
4384ab085aSmws #include <fcntl.h>
4484ab085aSmws #include <errno.h>
4584ab085aSmws #include <ctype.h>
461566bc34SRobert Mustacchi #include <libjedec.h>
4784ab085aSmws 
4884ab085aSmws #define	SMBIOS_SUCCESS	0
4984ab085aSmws #define	SMBIOS_ERROR	1
5084ab085aSmws #define	SMBIOS_USAGE	2
5184ab085aSmws 
5284ab085aSmws static const char *g_pname;
5384ab085aSmws static int g_hdr;
5484ab085aSmws 
5584ab085aSmws static int opt_e;
5684ab085aSmws static int opt_i = -1;
5784ab085aSmws static int opt_O;
5884ab085aSmws static int opt_s;
5984ab085aSmws static int opt_t = -1;
6084ab085aSmws static int opt_x;
6184ab085aSmws 
621566bc34SRobert Mustacchi static boolean_t
smbios_vergteq(smbios_version_t * v,uint_t major,uint_t minor)631566bc34SRobert Mustacchi smbios_vergteq(smbios_version_t *v, uint_t major, uint_t minor)
641566bc34SRobert Mustacchi {
651566bc34SRobert Mustacchi 	if (v->smbv_major > major)
661566bc34SRobert Mustacchi 		return (B_TRUE);
671566bc34SRobert Mustacchi 	if (v->smbv_major == major &&
681566bc34SRobert Mustacchi 	    v->smbv_minor >= minor)
691566bc34SRobert Mustacchi 		return (B_TRUE);
701566bc34SRobert Mustacchi 	return (B_FALSE);
711566bc34SRobert Mustacchi }
721566bc34SRobert Mustacchi 
7343379a28SAndy Fiddaman static void __PRINTFLIKE(2)
smbios_warn(smbios_hdl_t * shp,const char * format,...)7432ece1f9SRobert Mustacchi smbios_warn(smbios_hdl_t *shp, const char *format, ...)
7532ece1f9SRobert Mustacchi {
7632ece1f9SRobert Mustacchi 	va_list ap;
7732ece1f9SRobert Mustacchi 
7832ece1f9SRobert Mustacchi 	va_start(ap, format);
7932ece1f9SRobert Mustacchi 	(void) vfprintf(stderr, format, ap);
8032ece1f9SRobert Mustacchi 	va_end(ap);
8132ece1f9SRobert Mustacchi 
8232ece1f9SRobert Mustacchi 	if (shp != NULL) {
8332ece1f9SRobert Mustacchi 		(void) fprintf(stderr, ": %s",
8432ece1f9SRobert Mustacchi 		    smbios_errmsg(smbios_errno(shp)));
8532ece1f9SRobert Mustacchi 	}
8632ece1f9SRobert Mustacchi 
8732ece1f9SRobert Mustacchi 	(void) fprintf(stderr, "\n");
8832ece1f9SRobert Mustacchi }
8932ece1f9SRobert Mustacchi 
9043379a28SAndy Fiddaman static void __PRINTFLIKE(2)
oprintf(FILE * fp,const char * format,...)9184ab085aSmws oprintf(FILE *fp, const char *format, ...)
9284ab085aSmws {
9384ab085aSmws 	va_list ap;
9484ab085aSmws 
9584ab085aSmws 	va_start(ap, format);
9684ab085aSmws 	(void) vfprintf(fp, format, ap);
9784ab085aSmws 	va_end(ap);
9884ab085aSmws }
9984ab085aSmws 
10043379a28SAndy Fiddaman static void __PRINTFLIKE(3)
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 
1741566bc34SRobert Mustacchi static void
jedec_print(FILE * fp,const char * desc,uint_t id)1751566bc34SRobert Mustacchi jedec_print(FILE *fp, const char *desc, uint_t id)
1761566bc34SRobert Mustacchi {
1771566bc34SRobert Mustacchi 	const char *name;
1781566bc34SRobert Mustacchi 	uint_t cont, vendor;
1791566bc34SRobert Mustacchi 
1809f9cceb6SRobert Mustacchi 	/*
1819f9cceb6SRobert Mustacchi 	 * SMBIOS encodes data in the way that the underlying memory standard
1829f9cceb6SRobert Mustacchi 	 * does. In this case, the upper byte indicates the vendor that we care
1839f9cceb6SRobert Mustacchi 	 * about while the lower byte indicates the number of continuations that
1849f9cceb6SRobert Mustacchi 	 * are needed. libjedec indexes this based on zero (e.g. table 1 is zero
1859f9cceb6SRobert Mustacchi 	 * continuations), which is how the spec encodes it. We add one so that
1869f9cceb6SRobert Mustacchi 	 * we can match how the spec describes it.
1879f9cceb6SRobert Mustacchi 	 */
1889f9cceb6SRobert Mustacchi 	vendor = id >> 8;
1899f9cceb6SRobert Mustacchi 	cont = id & 0x7f;
1901566bc34SRobert Mustacchi 	name = libjedec_vendor_string(cont, vendor);
1911566bc34SRobert Mustacchi 	if (name == NULL) {
1929f9cceb6SRobert Mustacchi 		oprintf(fp, "  %s: Bank: 0x%x Vendor: 0x%x\n", desc, cont + 1,
1939f9cceb6SRobert Mustacchi 		    vendor);
1941566bc34SRobert Mustacchi 	} else {
1959f9cceb6SRobert Mustacchi 		oprintf(fp, "  %s: Bank: 0x%x Vendor: 0x%x (%s)\n", desc,
1969f9cceb6SRobert Mustacchi 		    cont + 1, vendor, name);
1971566bc34SRobert Mustacchi 	}
1981566bc34SRobert Mustacchi }
1991566bc34SRobert Mustacchi 
200176a9270SRobert Mustacchi /*
2016bc074b1SRobert Mustacchi  * Convert an SMBIOS encoded JEDEDC component revision into its actual form. In
2026bc074b1SRobert Mustacchi  * general, JEDEC revisions are single byte values; however, the SMBIOS fields
2036bc074b1SRobert Mustacchi  * are two bytes wide. The byte that we care about is the "first" byte which
2046bc074b1SRobert Mustacchi  * translates into the upper bits here. The revision is binary coded decimal
2056bc074b1SRobert Mustacchi  * (BCD) represented with each nibble as major.minor. The major is the upper
2066bc074b1SRobert Mustacchi  * nibble and the minor is the lower one.
2076bc074b1SRobert Mustacchi  */
2086bc074b1SRobert Mustacchi static void
jedec_rev_print(FILE * fp,const char * desc,uint16_t raw_rev)2096bc074b1SRobert Mustacchi jedec_rev_print(FILE *fp, const char *desc, uint16_t raw_rev)
2106bc074b1SRobert Mustacchi {
2116bc074b1SRobert Mustacchi 	uint8_t rev = (uint8_t)bitx16(raw_rev, 15, 8);
2126bc074b1SRobert Mustacchi 	uint8_t maj = bitx8(rev, 7, 4);
2136bc074b1SRobert Mustacchi 	uint8_t min = bitx8(rev, 3, 0);
2146bc074b1SRobert Mustacchi 	oprintf(fp, "  %s: %x.%x\n", desc, maj, min);
2156bc074b1SRobert Mustacchi }
2166bc074b1SRobert Mustacchi 
2176bc074b1SRobert Mustacchi /*
218176a9270SRobert Mustacchi  * Print a 128-bit data as a series of 16 hex digits.
219176a9270SRobert Mustacchi  */
220176a9270SRobert Mustacchi static void
u128_print(FILE * fp,const char * desc,const uint8_t * data)221176a9270SRobert Mustacchi u128_print(FILE *fp, const char *desc, const uint8_t *data)
222176a9270SRobert Mustacchi {
223176a9270SRobert Mustacchi 	uint_t i;
224176a9270SRobert Mustacchi 
225176a9270SRobert Mustacchi 	oprintf(fp, "%s: ", desc);
226176a9270SRobert Mustacchi 	for (i = 0; i < 16; i++) {
227176a9270SRobert Mustacchi 		oprintf(fp, " %02x", data[i]);
228176a9270SRobert Mustacchi 	}
229176a9270SRobert Mustacchi 	oprintf(fp, "\n");
230176a9270SRobert Mustacchi }
231176a9270SRobert Mustacchi 
2320f56e145SRobert Mustacchi /*
2330f56e145SRobert Mustacchi  * Print a string that came from an SMBIOS table. We do this character by
2340f56e145SRobert Mustacchi  * character so we can potentially escape strings.
2350f56e145SRobert Mustacchi  */
2360f56e145SRobert Mustacchi static void
str_print_label(FILE * fp,const char * header,const char * str,boolean_t label)2370f56e145SRobert Mustacchi str_print_label(FILE *fp, const char *header, const char *str, boolean_t label)
2380f56e145SRobert Mustacchi {
2390f56e145SRobert Mustacchi 	const char *c;
2400f56e145SRobert Mustacchi 
2410f56e145SRobert Mustacchi 	oprintf(fp, header);
2420f56e145SRobert Mustacchi 	if (label) {
2430f56e145SRobert Mustacchi 		oprintf(fp, ": ");
2440f56e145SRobert Mustacchi 	}
2450f56e145SRobert Mustacchi 
2460f56e145SRobert Mustacchi 	for (c = str; *c != '\0'; c++) {
2470f56e145SRobert Mustacchi 		if (isprint(*c)) {
2480f56e145SRobert Mustacchi 			oprintf(fp, "%c", *c);
2490f56e145SRobert Mustacchi 		} else {
2500f56e145SRobert Mustacchi 			oprintf(fp, "\\x%02x", *c);
2510f56e145SRobert Mustacchi 		}
2520f56e145SRobert Mustacchi 	}
2530f56e145SRobert Mustacchi 
2540f56e145SRobert Mustacchi 	oprintf(fp, "\n");
2550f56e145SRobert Mustacchi }
2560f56e145SRobert Mustacchi 
2570f56e145SRobert Mustacchi static void
str_print_nolabel(FILE * fp,const char * ws,const char * str)2580f56e145SRobert Mustacchi str_print_nolabel(FILE *fp, const char *ws, const char *str)
2590f56e145SRobert Mustacchi {
2600f56e145SRobert Mustacchi 	return (str_print_label(fp, ws, str, B_FALSE));
2610f56e145SRobert Mustacchi }
2620f56e145SRobert Mustacchi 
2630f56e145SRobert Mustacchi static void
str_print(FILE * fp,const char * header,const char * str)2640f56e145SRobert Mustacchi str_print(FILE *fp, const char *header, const char *str)
2650f56e145SRobert Mustacchi {
2660f56e145SRobert Mustacchi 	return (str_print_label(fp, header, str, B_TRUE));
2670f56e145SRobert Mustacchi }
2680f56e145SRobert Mustacchi 
26903f9f63dSTom Pothier static int
check_oem(smbios_hdl_t * shp)27003f9f63dSTom Pothier check_oem(smbios_hdl_t *shp)
27103f9f63dSTom Pothier {
27203f9f63dSTom Pothier 	int i;
27303f9f63dSTom Pothier 	int cnt;
27403f9f63dSTom Pothier 	int rv;
27503f9f63dSTom Pothier 	id_t oem_id;
27603f9f63dSTom Pothier 	smbios_struct_t s;
27703f9f63dSTom Pothier 	const char **oem_str;
27803f9f63dSTom Pothier 
27903f9f63dSTom Pothier 	rv = smbios_lookup_type(shp, SMB_TYPE_OEMSTR, &s);
28003f9f63dSTom Pothier 	if (rv != 0) {
28103f9f63dSTom Pothier 		return (-1);
28203f9f63dSTom Pothier 	}
28303f9f63dSTom Pothier 
28403f9f63dSTom Pothier 	oem_id = s.smbstr_id;
28503f9f63dSTom Pothier 
28603f9f63dSTom Pothier 	cnt = smbios_info_strtab(shp, oem_id, 0, NULL);
28703f9f63dSTom Pothier 	if (cnt > 0) {
28803f9f63dSTom Pothier 		oem_str =  alloca(sizeof (char *) * cnt);
28903f9f63dSTom Pothier 		(void) smbios_info_strtab(shp, oem_id, cnt, oem_str);
29003f9f63dSTom Pothier 
29103f9f63dSTom Pothier 		for (i = 0; i < cnt; i++) {
29203f9f63dSTom Pothier 			if (strncmp(oem_str[i], SMB_PRMS1,
29303f9f63dSTom Pothier 			    strlen(SMB_PRMS1) + 1) == 0) {
29403f9f63dSTom Pothier 				return (0);
29503f9f63dSTom Pothier 			}
29603f9f63dSTom Pothier 		}
29703f9f63dSTom Pothier 	}
29803f9f63dSTom Pothier 
29903f9f63dSTom Pothier 	return (-1);
30003f9f63dSTom Pothier }
30103f9f63dSTom Pothier 
30284ab085aSmws static void
print_smbios_21(smbios_21_entry_t * ep,FILE * fp)3031951a933SToomas Soome print_smbios_21(smbios_21_entry_t *ep, FILE *fp)
30484ab085aSmws {
30584ab085aSmws 	int i;
30684ab085aSmws 
30784ab085aSmws 	oprintf(fp, "Entry Point Anchor Tag: %*.*s\n",
3081951a933SToomas Soome 	    (int)sizeof (ep->smbe_eanchor), (int)sizeof (ep->smbe_eanchor),
3091951a933SToomas Soome 	    ep->smbe_eanchor);
31084ab085aSmws 
3111951a933SToomas Soome 	oprintf(fp, "Entry Point Checksum: 0x%x\n", ep->smbe_ecksum);
3121951a933SToomas Soome 	oprintf(fp, "Entry Point Length: %u\n", ep->smbe_elen);
31384ab085aSmws 	oprintf(fp, "Entry Point Version: %u.%u\n",
3141951a933SToomas Soome 	    ep->smbe_major, ep->smbe_minor);
3151951a933SToomas Soome 	oprintf(fp, "Max Structure Size: %u\n", ep->smbe_maxssize);
3161951a933SToomas Soome 	oprintf(fp, "Entry Point Revision: 0x%x\n", ep->smbe_revision);
31784ab085aSmws 
31884ab085aSmws 	oprintf(fp, "Entry Point Revision Data:");
3191951a933SToomas Soome 	for (i = 0; i < sizeof (ep->smbe_format); i++)
3201951a933SToomas Soome 		oprintf(fp, " 0x%02x", ep->smbe_format[i]);
32184ab085aSmws 	oprintf(fp, "\n");
32284ab085aSmws 
32384ab085aSmws 	oprintf(fp, "Intermediate Anchor Tag: %*.*s\n",
3241951a933SToomas Soome 	    (int)sizeof (ep->smbe_ianchor), (int)sizeof (ep->smbe_ianchor),
3251951a933SToomas Soome 	    ep->smbe_ianchor);
32684ab085aSmws 
3271951a933SToomas Soome 	oprintf(fp, "Intermediate Checksum: 0x%x\n", ep->smbe_icksum);
3281951a933SToomas Soome 	oprintf(fp, "Structure Table Length: %u\n", ep->smbe_stlen);
3291951a933SToomas Soome 	oprintf(fp, "Structure Table Address: 0x%x\n", ep->smbe_staddr);
3301951a933SToomas Soome 	oprintf(fp, "Structure Table Entries: %u\n", ep->smbe_stnum);
3311951a933SToomas Soome 	oprintf(fp, "DMI BCD Revision: 0x%x\n", ep->smbe_bcdrev);
3321951a933SToomas Soome }
3331951a933SToomas Soome 
3341951a933SToomas Soome static void
print_smbios_30(smbios_30_entry_t * ep,FILE * fp)3351951a933SToomas Soome print_smbios_30(smbios_30_entry_t *ep, FILE *fp)
3361951a933SToomas Soome {
3371951a933SToomas Soome 	oprintf(fp, "Entry Point Anchor Tag: %*.*s\n",
3381951a933SToomas Soome 	    (int)sizeof (ep->smbe_eanchor), (int)sizeof (ep->smbe_eanchor),
3391951a933SToomas Soome 	    ep->smbe_eanchor);
3401951a933SToomas Soome 
3411951a933SToomas Soome 	oprintf(fp, "Entry Point Checksum: 0x%x\n", ep->smbe_ecksum);
3421951a933SToomas Soome 	oprintf(fp, "Entry Point Length: %u\n", ep->smbe_elen);
3431951a933SToomas Soome 	oprintf(fp, "SMBIOS Version: %u.%u\n",
3441951a933SToomas Soome 	    ep->smbe_major, ep->smbe_minor);
3451951a933SToomas Soome 	oprintf(fp, "SMBIOS DocRev: 0x%x\n", ep->smbe_docrev);
3461951a933SToomas Soome 	oprintf(fp, "Entry Point Revision: 0x%x\n", ep->smbe_revision);
3471951a933SToomas Soome 
3481951a933SToomas Soome 	oprintf(fp, "Structure Table Length: %u\n", ep->smbe_stlen);
3491951a933SToomas Soome 	oprintf(fp, "Structure Table Address: 0x%" PRIx64 "\n",
3501951a933SToomas Soome 	    ep->smbe_staddr);
3511951a933SToomas Soome }
3521951a933SToomas Soome 
3531951a933SToomas Soome static void
print_smbios(smbios_hdl_t * shp,FILE * fp)3541951a933SToomas Soome print_smbios(smbios_hdl_t *shp, FILE *fp)
3551951a933SToomas Soome {
3561951a933SToomas Soome 	smbios_entry_t ep;
3571951a933SToomas Soome 
3581951a933SToomas Soome 	switch (smbios_info_smbios(shp, &ep)) {
3591951a933SToomas Soome 	case SMBIOS_ENTRY_POINT_21:
3601951a933SToomas Soome 		print_smbios_21(&ep.ep21, fp);
3611951a933SToomas Soome 		break;
3621951a933SToomas Soome 	case SMBIOS_ENTRY_POINT_30:
3631951a933SToomas Soome 		print_smbios_30(&ep.ep30, fp);
3641951a933SToomas Soome 		break;
3651951a933SToomas Soome 	}
36684ab085aSmws }
36784ab085aSmws 
36884ab085aSmws static void
print_common(const smbios_info_t * ip,FILE * fp)36984ab085aSmws print_common(const smbios_info_t *ip, FILE *fp)
37084ab085aSmws {
37184ab085aSmws 	if (ip->smbi_manufacturer[0] != '\0')
3720f56e145SRobert Mustacchi 		str_print(fp, "  Manufacturer", ip->smbi_manufacturer);
37384ab085aSmws 	if (ip->smbi_product[0] != '\0')
3740f56e145SRobert Mustacchi 		str_print(fp, "  Product", ip->smbi_product);
37584ab085aSmws 	if (ip->smbi_version[0] != '\0')
3760f56e145SRobert Mustacchi 		str_print(fp, "  Version", ip->smbi_version);
37784ab085aSmws 	if (ip->smbi_serial[0] != '\0')
3780f56e145SRobert Mustacchi 		str_print(fp, "  Serial Number", ip->smbi_serial);
37984ab085aSmws 	if (ip->smbi_asset[0] != '\0')
3800f56e145SRobert Mustacchi 		str_print(fp, "  Asset Tag", ip->smbi_asset);
38184ab085aSmws 	if (ip->smbi_location[0] != '\0')
3820f56e145SRobert Mustacchi 		str_print(fp, "  Location Tag", ip->smbi_location);
38384ab085aSmws 	if (ip->smbi_part[0] != '\0')
3840f56e145SRobert Mustacchi 		str_print(fp, "  Part Number", ip->smbi_part);
38584ab085aSmws }
38684ab085aSmws 
38784ab085aSmws static void
print_bios(smbios_hdl_t * shp,FILE * fp)38884ab085aSmws print_bios(smbios_hdl_t *shp, FILE *fp)
38984ab085aSmws {
39084ab085aSmws 	smbios_bios_t b;
39184ab085aSmws 
392a7aaa513SRobert Mustacchi 	if (smbios_info_bios(shp, &b) == -1) {
393a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read BIOS information");
394a7aaa513SRobert Mustacchi 		return;
395a7aaa513SRobert Mustacchi 	}
39684ab085aSmws 
3970f56e145SRobert Mustacchi 	str_print(fp, "  Vendor", b.smbb_vendor);
3980f56e145SRobert Mustacchi 	str_print(fp, "  Version String", b.smbb_version);
3990f56e145SRobert Mustacchi 	str_print(fp, "  Release Date", b.smbb_reldate);
40084ab085aSmws 	oprintf(fp, "  Address Segment: 0x%x\n", b.smbb_segment);
401e5cce96fSRobert Mustacchi 	oprintf(fp, "  ROM Size: %" PRIu64 " bytes\n", b.smbb_extromsize);
40284ab085aSmws 	oprintf(fp, "  Image Size: %u bytes\n", b.smbb_runsize);
40384ab085aSmws 
40484ab085aSmws 	flag64_printf(fp, "Characteristics",
40584ab085aSmws 	    b.smbb_cflags, sizeof (b.smbb_cflags) * NBBY,
40684ab085aSmws 	    smbios_bios_flag_name, smbios_bios_flag_desc);
40784ab085aSmws 
40884ab085aSmws 	if (b.smbb_nxcflags > SMB_BIOSXB_1) {
40984ab085aSmws 		flag_printf(fp, "Characteristics Extension Byte 1",
41084ab085aSmws 		    b.smbb_xcflags[SMB_BIOSXB_1],
41184ab085aSmws 		    sizeof (b.smbb_xcflags[SMB_BIOSXB_1]) * NBBY,
41284ab085aSmws 		    smbios_bios_xb1_name, smbios_bios_xb1_desc);
41384ab085aSmws 	}
41484ab085aSmws 
41584ab085aSmws 	if (b.smbb_nxcflags > SMB_BIOSXB_2) {
41684ab085aSmws 		flag_printf(fp, "Characteristics Extension Byte 2",
41784ab085aSmws 		    b.smbb_xcflags[SMB_BIOSXB_2],
41884ab085aSmws 		    sizeof (b.smbb_xcflags[SMB_BIOSXB_2]) * NBBY,
41984ab085aSmws 		    smbios_bios_xb2_name, smbios_bios_xb2_desc);
42084ab085aSmws 	}
42184ab085aSmws 
42284ab085aSmws 	if (b.smbb_nxcflags > SMB_BIOSXB_BIOS_MIN) {
42384ab085aSmws 		oprintf(fp, "  Version Number: %u.%u\n",
42484ab085aSmws 		    b.smbb_biosv.smbv_major, b.smbb_biosv.smbv_minor);
42584ab085aSmws 	}
42684ab085aSmws 
427f44a1392SRobert Mustacchi 	/*
428f44a1392SRobert Mustacchi 	 * If the major and minor versions are 0xff then that indicates that the
429f44a1392SRobert Mustacchi 	 * embedded controller does not exist.
430f44a1392SRobert Mustacchi 	 */
431f44a1392SRobert Mustacchi 	if (b.smbb_nxcflags > SMB_BIOSXB_ECFW_MIN &&
432f44a1392SRobert Mustacchi 	    b.smbb_ecfwv.smbv_major != 0xff &&
433f44a1392SRobert Mustacchi 	    b.smbb_ecfwv.smbv_minor != 0xff) {
43484ab085aSmws 		oprintf(fp, "  Embedded Ctlr Firmware Version Number: %u.%u\n",
43584ab085aSmws 		    b.smbb_ecfwv.smbv_major, b.smbb_ecfwv.smbv_minor);
43684ab085aSmws 	}
43784ab085aSmws }
43884ab085aSmws 
43984ab085aSmws static void
print_system(smbios_hdl_t * shp,FILE * fp)44084ab085aSmws print_system(smbios_hdl_t *shp, FILE *fp)
44184ab085aSmws {
44284ab085aSmws 	smbios_system_t s;
44384ab085aSmws 	uint_t i;
44484ab085aSmws 
445a7aaa513SRobert Mustacchi 	if (smbios_info_system(shp, &s) == -1) {
446a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read system information");
447a7aaa513SRobert Mustacchi 		return;
448a7aaa513SRobert Mustacchi 	}
44984ab085aSmws 
45084ab085aSmws 	oprintf(fp, "  UUID: ");
45184ab085aSmws 	for (i = 0; i < s.smbs_uuidlen; i++) {
45284ab085aSmws 		oprintf(fp, "%02x", s.smbs_uuid[i]);
45384ab085aSmws 		if (i == 3 || i == 5 || i == 7 || i == 9)
45484ab085aSmws 			oprintf(fp, "-");
45584ab085aSmws 	}
45684ab085aSmws 	oprintf(fp, "\n");
45784ab085aSmws 
45884ab085aSmws 	desc_printf(smbios_system_wakeup_desc(s.smbs_wakeup),
45984ab085aSmws 	    fp, "  Wake-Up Event: 0x%x", s.smbs_wakeup);
46084ab085aSmws 
4610f56e145SRobert Mustacchi 	str_print(fp, "  SKU Number", s.smbs_sku);
4620f56e145SRobert Mustacchi 	str_print(fp, "  Family", s.smbs_family);
46384ab085aSmws }
46484ab085aSmws 
46584ab085aSmws static void
print_bboard(smbios_hdl_t * shp,id_t id,FILE * fp)46684ab085aSmws print_bboard(smbios_hdl_t *shp, id_t id, FILE *fp)
46784ab085aSmws {
46884ab085aSmws 	smbios_bboard_t b;
469074bb90dSTom Pothier 	int chdl_cnt;
47084ab085aSmws 
471a7aaa513SRobert Mustacchi 	if (smbios_info_bboard(shp, id, &b) != 0) {
472a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read baseboard information");
473a7aaa513SRobert Mustacchi 		return;
474a7aaa513SRobert Mustacchi 	}
47584ab085aSmws 
47684ab085aSmws 	oprintf(fp, "  Chassis: %u\n", (uint_t)b.smbb_chassis);
47784ab085aSmws 
47884ab085aSmws 	flag_printf(fp, "Flags", b.smbb_flags, sizeof (b.smbb_flags) * NBBY,
47984ab085aSmws 	    smbios_bboard_flag_name, smbios_bboard_flag_desc);
48084ab085aSmws 
48184ab085aSmws 	desc_printf(smbios_bboard_type_desc(b.smbb_type),
48284ab085aSmws 	    fp, "  Board Type: 0x%x", b.smbb_type);
483074bb90dSTom Pothier 
484074bb90dSTom Pothier 	chdl_cnt = b.smbb_contn;
485074bb90dSTom Pothier 	if (chdl_cnt != 0) {
486074bb90dSTom Pothier 		id_t *chdl;
487074bb90dSTom Pothier 		uint16_t hdl;
488074bb90dSTom Pothier 		int i, n, cnt;
489074bb90dSTom Pothier 
490074bb90dSTom Pothier 		chdl = alloca(chdl_cnt * sizeof (id_t));
491074bb90dSTom Pothier 		cnt = smbios_info_contains(shp, id, chdl_cnt, chdl);
492074bb90dSTom Pothier 		if (cnt > SMB_CONT_MAX)
493074bb90dSTom Pothier 			return;
494074bb90dSTom Pothier 		n = MIN(chdl_cnt, cnt);
495074bb90dSTom Pothier 
496074bb90dSTom Pothier 		oprintf(fp, "\n");
497074bb90dSTom Pothier 		for (i = 0; i < n; i++) {
498074bb90dSTom Pothier 			hdl = (uint16_t)chdl[i];
499074bb90dSTom Pothier 			oprintf(fp, "  Contained Handle: %u\n", hdl);
500074bb90dSTom Pothier 		}
501074bb90dSTom Pothier 	}
50284ab085aSmws }
50384ab085aSmws 
50484ab085aSmws static void
print_chassis(smbios_hdl_t * shp,id_t id,FILE * fp)50584ab085aSmws print_chassis(smbios_hdl_t *shp, id_t id, FILE *fp)
50684ab085aSmws {
50784ab085aSmws 	smbios_chassis_t c;
508d53cdfabSRobert Mustacchi 	smbios_chassis_entry_t *elts;
509d53cdfabSRobert Mustacchi 	uint_t nelts, i;
51084ab085aSmws 
511a7aaa513SRobert Mustacchi 	if (smbios_info_chassis(shp, id, &c) != 0) {
512a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read chassis information");
513a7aaa513SRobert Mustacchi 		return;
514a7aaa513SRobert Mustacchi 	}
51584ab085aSmws 
51684ab085aSmws 	oprintf(fp, "  OEM Data: 0x%x\n", c.smbc_oemdata);
5170f56e145SRobert Mustacchi 	str_print(fp, "  SKU Number",
518d54e4739SYuri Pankov 	    c.smbc_sku[0] == '\0' ? "<unknown>" : c.smbc_sku);
51984ab085aSmws 	oprintf(fp, "  Lock Present: %s\n", c.smbc_lock ? "Y" : "N");
52084ab085aSmws 
52184ab085aSmws 	desc_printf(smbios_chassis_type_desc(c.smbc_type),
52284ab085aSmws 	    fp, "  Chassis Type: 0x%x", c.smbc_type);
52384ab085aSmws 
52484ab085aSmws 	desc_printf(smbios_chassis_state_desc(c.smbc_bustate),
52584ab085aSmws 	    fp, "  Boot-Up State: 0x%x", c.smbc_bustate);
52684ab085aSmws 
52784ab085aSmws 	desc_printf(smbios_chassis_state_desc(c.smbc_psstate),
52884ab085aSmws 	    fp, "  Power Supply State: 0x%x", c.smbc_psstate);
52984ab085aSmws 
53084ab085aSmws 	desc_printf(smbios_chassis_state_desc(c.smbc_thstate),
53184ab085aSmws 	    fp, "  Thermal State: 0x%x", c.smbc_thstate);
53284ab085aSmws 
53384ab085aSmws 	oprintf(fp, "  Chassis Height: %uu\n", c.smbc_uheight);
53484ab085aSmws 	oprintf(fp, "  Power Cords: %u\n", c.smbc_cords);
535074bb90dSTom Pothier 
536d53cdfabSRobert Mustacchi 	oprintf(fp, "  Element Records: %u\n", c.smbc_elems);
537074bb90dSTom Pothier 
538d53cdfabSRobert Mustacchi 	if (c.smbc_elems == 0) {
539074bb90dSTom Pothier 		return;
540d53cdfabSRobert Mustacchi 	}
541d53cdfabSRobert Mustacchi 
542d53cdfabSRobert Mustacchi 	if (smbios_info_chassis_elts(shp, id, &nelts, &elts) != 0) {
543d53cdfabSRobert Mustacchi 		smbios_warn(shp, "failed to read chassis elements");
544d53cdfabSRobert Mustacchi 		return;
545d53cdfabSRobert Mustacchi 	}
546074bb90dSTom Pothier 
547074bb90dSTom Pothier 	oprintf(fp, "\n");
548d53cdfabSRobert Mustacchi 
549d53cdfabSRobert Mustacchi 	for (i = 0; i < nelts; i++) {
550d53cdfabSRobert Mustacchi 		switch (elts[i].smbce_type) {
551d53cdfabSRobert Mustacchi 		case SMB_CELT_BBOARD:
552d53cdfabSRobert Mustacchi 			desc_printf(smbios_bboard_type_desc(elts[i].smbce_elt),
553d53cdfabSRobert Mustacchi 			    fp, "  Contained SMBIOS Base Board Type: 0x%x",
554d53cdfabSRobert Mustacchi 			    elts[i].smbce_elt);
555d53cdfabSRobert Mustacchi 			break;
556d53cdfabSRobert Mustacchi 		case SMB_CELT_SMBIOS:
557d53cdfabSRobert Mustacchi 			desc_printf(smbios_type_name(elts[i].smbce_elt), fp,
558074bb90dSTom Pothier 			    "  Contained SMBIOS structure Type: %u",
559d53cdfabSRobert Mustacchi 			    elts[i].smbce_elt);
560d53cdfabSRobert Mustacchi 			break;
561d53cdfabSRobert Mustacchi 		default:
562d53cdfabSRobert Mustacchi 			oprintf(fp, "  Unknown contained Type: %u/%u\n",
563d53cdfabSRobert Mustacchi 			    elts[i].smbce_type, elts[i].smbce_elt);
564d53cdfabSRobert Mustacchi 			break;
565074bb90dSTom Pothier 		}
566d53cdfabSRobert Mustacchi 		oprintf(fp, "    Minimum number: %u\n", elts[i].smbce_min);
567d53cdfabSRobert Mustacchi 		oprintf(fp, "    Maximum number: %u\n", elts[i].smbce_max);
568074bb90dSTom Pothier 	}
56984ab085aSmws }
57084ab085aSmws 
57184ab085aSmws static void
print_processor(smbios_hdl_t * shp,id_t id,FILE * fp)57284ab085aSmws print_processor(smbios_hdl_t *shp, id_t id, FILE *fp)
57384ab085aSmws {
57484ab085aSmws 	smbios_processor_t p;
57584ab085aSmws 	uint_t status;
57684ab085aSmws 
577a7aaa513SRobert Mustacchi 	if (smbios_info_processor(shp, id, &p) != 0) {
578a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read processor information");
579a7aaa513SRobert Mustacchi 		return;
580a7aaa513SRobert Mustacchi 	}
58184ab085aSmws 	status = SMB_PRSTATUS_STATUS(p.smbp_status);
58284ab085aSmws 
58384ab085aSmws 	desc_printf(smbios_processor_family_desc(p.smbp_family),
58484ab085aSmws 	    fp, "  Family: %u", p.smbp_family);
58584ab085aSmws 
58684ab085aSmws 	oprintf(fp, "  CPUID: 0x%llx\n", (u_longlong_t)p.smbp_cpuid);
58784ab085aSmws 
58884ab085aSmws 	desc_printf(smbios_processor_type_desc(p.smbp_type),
58984ab085aSmws 	    fp, "  Type: %u", p.smbp_type);
59084ab085aSmws 
59184ab085aSmws 	desc_printf(smbios_processor_upgrade_desc(p.smbp_upgrade),
59284ab085aSmws 	    fp, "  Socket Upgrade: %u", p.smbp_upgrade);
59384ab085aSmws 
59484ab085aSmws 	oprintf(fp, "  Socket Status: %s\n",
59584ab085aSmws 	    SMB_PRSTATUS_PRESENT(p.smbp_status) ?
59684ab085aSmws 	    "Populated" : "Not Populated");
59784ab085aSmws 
59884ab085aSmws 	desc_printf(smbios_processor_status_desc(status),
59984ab085aSmws 	    fp, "  Processor Status: %u", status);
60084ab085aSmws 
60184ab085aSmws 	if (SMB_PRV_LEGACY(p.smbp_voltage)) {
60284ab085aSmws 		oprintf(fp, "  Supported Voltages:");
60384ab085aSmws 		switch (p.smbp_voltage) {
60484ab085aSmws 		case SMB_PRV_5V:
60584ab085aSmws 			oprintf(fp, " 5.0V");
60684ab085aSmws 			break;
60784ab085aSmws 		case SMB_PRV_33V:
60884ab085aSmws 			oprintf(fp, " 3.3V");
60984ab085aSmws 			break;
61084ab085aSmws 		case SMB_PRV_29V:
61184ab085aSmws 			oprintf(fp, " 2.9V");
61284ab085aSmws 			break;
61384ab085aSmws 		}
61484ab085aSmws 		oprintf(fp, "\n");
61584ab085aSmws 	} else {
61684ab085aSmws 		oprintf(fp, "  Supported Voltages: %.1fV\n",
61784ab085aSmws 		    (float)SMB_PRV_VOLTAGE(p.smbp_voltage) / 10);
61884ab085aSmws 	}
61984ab085aSmws 
6206734c4b0SRobert Mustacchi 	if (p.smbp_corecount != 0) {
6214e901881SDale Ghent 		oprintf(fp, "  Core Count: %u\n", p.smbp_corecount);
6226734c4b0SRobert Mustacchi 	} else {
6234e901881SDale Ghent 		oprintf(fp, "  Core Count: Unknown\n");
6246734c4b0SRobert Mustacchi 	}
6254e901881SDale Ghent 
6266734c4b0SRobert Mustacchi 	if (p.smbp_coresenabled != 0) {
627176a9270SRobert Mustacchi 		oprintf(fp, "  Cores Enabled: %u\n", p.smbp_coresenabled);
6286734c4b0SRobert Mustacchi 	} else {
6294e901881SDale Ghent 		oprintf(fp, "  Cores Enabled: Unknown\n");
6306734c4b0SRobert Mustacchi 	}
6314e901881SDale Ghent 
6326734c4b0SRobert Mustacchi 	if (p.smbp_threadcount != 0) {
633176a9270SRobert Mustacchi 		oprintf(fp, "  Thread Count: %u\n", p.smbp_threadcount);
6346734c4b0SRobert Mustacchi 	} else {
6354e901881SDale Ghent 		oprintf(fp, "  Thread Count: Unknown\n");
6366734c4b0SRobert Mustacchi 	}
6374e901881SDale Ghent 
6384e901881SDale Ghent 	if (p.smbp_cflags) {
6394e901881SDale Ghent 		flag_printf(fp, "Processor Characteristics",
6404e901881SDale Ghent 		    p.smbp_cflags, sizeof (p.smbp_cflags) * NBBY,
6414e901881SDale Ghent 		    smbios_processor_core_flag_name,
6424e901881SDale Ghent 		    smbios_processor_core_flag_desc);
6434e901881SDale Ghent 	}
6444e901881SDale Ghent 
64584ab085aSmws 	if (p.smbp_clkspeed != 0)
64684ab085aSmws 		oprintf(fp, "  External Clock Speed: %uMHz\n", p.smbp_clkspeed);
64784ab085aSmws 	else
64884ab085aSmws 		oprintf(fp, "  External Clock Speed: Unknown\n");
64984ab085aSmws 
65084ab085aSmws 	if (p.smbp_maxspeed != 0)
65184ab085aSmws 		oprintf(fp, "  Maximum Speed: %uMHz\n", p.smbp_maxspeed);
65284ab085aSmws 	else
65384ab085aSmws 		oprintf(fp, "  Maximum Speed: Unknown\n");
65484ab085aSmws 
65584ab085aSmws 	if (p.smbp_curspeed != 0)
65684ab085aSmws 		oprintf(fp, "  Current Speed: %uMHz\n", p.smbp_curspeed);
65784ab085aSmws 	else
65884ab085aSmws 		oprintf(fp, "  Current Speed: Unknown\n");
65984ab085aSmws 
66032ece1f9SRobert Mustacchi 	id_printf(fp, "  L1 Cache Handle: ", p.smbp_l1cache);
66132ece1f9SRobert Mustacchi 	id_printf(fp, "  L2 Cache Handle: ", p.smbp_l2cache);
66232ece1f9SRobert Mustacchi 	id_printf(fp, "  L3 Cache Handle: ", p.smbp_l3cache);
6631d1fc316SRobert Mustacchi 
6641d1fc316SRobert Mustacchi 	if (p.smbp_threadsenabled != 0) {
6651d1fc316SRobert Mustacchi 		oprintf(fp, "  Threads Enabled: %u\n", p.smbp_threadsenabled);
6661d1fc316SRobert Mustacchi 	} else {
6671d1fc316SRobert Mustacchi 		oprintf(fp, "  Threads Enabled: Unknown\n");
6681d1fc316SRobert Mustacchi 	}
66949f72993SRobert Mustacchi 
67049f72993SRobert Mustacchi 	/*
67149f72993SRobert Mustacchi 	 * The Socket Type string overlaps with the upgrade string. Only print
67249f72993SRobert Mustacchi 	 * something if we have a valid value.
67349f72993SRobert Mustacchi 	 */
67449f72993SRobert Mustacchi 	if (*p.smbp_socktype != '\0') {
67549f72993SRobert Mustacchi 		str_print(fp, "  Socket Type", p.smbp_socktype);
67649f72993SRobert Mustacchi 	}
67784ab085aSmws }
67884ab085aSmws 
67984ab085aSmws static void
print_cache(smbios_hdl_t * shp,id_t id,FILE * fp)68084ab085aSmws print_cache(smbios_hdl_t *shp, id_t id, FILE *fp)
68184ab085aSmws {
68284ab085aSmws 	smbios_cache_t c;
68384ab085aSmws 
684a7aaa513SRobert Mustacchi 	if (smbios_info_cache(shp, id, &c) != 0) {
685a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read cache information");
686a7aaa513SRobert Mustacchi 		return;
687a7aaa513SRobert Mustacchi 	}
68884ab085aSmws 
68984ab085aSmws 	oprintf(fp, "  Level: %u\n", c.smba_level);
690e5cce96fSRobert Mustacchi 	oprintf(fp, "  Maximum Installed Size: %" PRIu64 " bytes\n",
691e5cce96fSRobert Mustacchi 	    c.smba_maxsize2);
69284ab085aSmws 
693e5cce96fSRobert Mustacchi 	if (c.smba_size2 != 0) {
694e5cce96fSRobert Mustacchi 		oprintf(fp, "  Installed Size: %" PRIu64 " bytes\n",
695e5cce96fSRobert Mustacchi 		    c.smba_size2);
696e5cce96fSRobert Mustacchi 	} else {
69784ab085aSmws 		oprintf(fp, "  Installed Size: Not Installed\n");
698e5cce96fSRobert Mustacchi 	}
69984ab085aSmws 
70084ab085aSmws 	if (c.smba_speed != 0)
70184ab085aSmws 		oprintf(fp, "  Speed: %uns\n", c.smba_speed);
70284ab085aSmws 	else
70384ab085aSmws 		oprintf(fp, "  Speed: Unknown\n");
70484ab085aSmws 
70584ab085aSmws 	flag_printf(fp, "Supported SRAM Types",
70684ab085aSmws 	    c.smba_stype, sizeof (c.smba_stype) * NBBY,
70784ab085aSmws 	    smbios_cache_ctype_name, smbios_cache_ctype_desc);
70884ab085aSmws 
70984ab085aSmws 	desc_printf(smbios_cache_ctype_desc(c.smba_ctype),
71084ab085aSmws 	    fp, "  Current SRAM Type: 0x%x", c.smba_ctype);
71184ab085aSmws 
71284ab085aSmws 	desc_printf(smbios_cache_ecc_desc(c.smba_etype),
71384ab085aSmws 	    fp, "  Error Correction Type: %u", c.smba_etype);
71484ab085aSmws 
71584ab085aSmws 	desc_printf(smbios_cache_logical_desc(c.smba_ltype),
71684ab085aSmws 	    fp, "  Logical Cache Type: %u", c.smba_ltype);
71784ab085aSmws 
71884ab085aSmws 	desc_printf(smbios_cache_assoc_desc(c.smba_assoc),
71984ab085aSmws 	    fp, "  Associativity: %u", c.smba_assoc);
72084ab085aSmws 
72184ab085aSmws 	desc_printf(smbios_cache_mode_desc(c.smba_mode),
72284ab085aSmws 	    fp, "  Mode: %u", c.smba_mode);
72384ab085aSmws 
72484ab085aSmws 	desc_printf(smbios_cache_loc_desc(c.smba_location),
72584ab085aSmws 	    fp, "  Location: %u", c.smba_location);
72684ab085aSmws 
72784ab085aSmws 	flag_printf(fp, "Flags", c.smba_flags, sizeof (c.smba_flags) * NBBY,
72884ab085aSmws 	    smbios_cache_flag_name, smbios_cache_flag_desc);
72984ab085aSmws }
73084ab085aSmws 
73184ab085aSmws static void
print_port(smbios_hdl_t * shp,id_t id,FILE * fp)73284ab085aSmws print_port(smbios_hdl_t *shp, id_t id, FILE *fp)
73384ab085aSmws {
73484ab085aSmws 	smbios_port_t p;
73584ab085aSmws 
736a7aaa513SRobert Mustacchi 	if (smbios_info_port(shp, id, &p) != 0) {
737a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read port information");
738a7aaa513SRobert Mustacchi 		return;
739a7aaa513SRobert Mustacchi 	}
74084ab085aSmws 
7410f56e145SRobert Mustacchi 	str_print(fp, "  Internal Reference Designator", p.smbo_iref);
7420f56e145SRobert Mustacchi 	str_print(fp, "  External Reference Designator", p.smbo_eref);
74384ab085aSmws 
74484ab085aSmws 	desc_printf(smbios_port_conn_desc(p.smbo_itype),
74584ab085aSmws 	    fp, "  Internal Connector Type: %u", p.smbo_itype);
74684ab085aSmws 
74784ab085aSmws 	desc_printf(smbios_port_conn_desc(p.smbo_etype),
74884ab085aSmws 	    fp, "  External Connector Type: %u", p.smbo_etype);
74984ab085aSmws 
75084ab085aSmws 	desc_printf(smbios_port_type_desc(p.smbo_ptype),
75184ab085aSmws 	    fp, "  Port Type: %u", p.smbo_ptype);
75284ab085aSmws }
75384ab085aSmws 
75484ab085aSmws static void
print_slot(smbios_hdl_t * shp,id_t id,FILE * fp)75584ab085aSmws print_slot(smbios_hdl_t *shp, id_t id, FILE *fp)
75684ab085aSmws {
75784ab085aSmws 	smbios_slot_t s;
7581951a933SToomas Soome 	smbios_version_t v;
75984ab085aSmws 
760a7aaa513SRobert Mustacchi 	if (smbios_info_slot(shp, id, &s) != 0) {
761a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read slot information");
762a7aaa513SRobert Mustacchi 		return;
763a7aaa513SRobert Mustacchi 	}
7641951a933SToomas Soome 	smbios_info_smbios_version(shp, &v);
76584ab085aSmws 
7660f56e145SRobert Mustacchi 	str_print(fp, "  Reference Designator", s.smbl_name);
76784ab085aSmws 	oprintf(fp, "  Slot ID: 0x%x\n", s.smbl_id);
76884ab085aSmws 
76984ab085aSmws 	desc_printf(smbios_slot_type_desc(s.smbl_type),
77084ab085aSmws 	    fp, "  Type: 0x%x", s.smbl_type);
77184ab085aSmws 
77284ab085aSmws 	desc_printf(smbios_slot_width_desc(s.smbl_width),
77384ab085aSmws 	    fp, "  Width: 0x%x", s.smbl_width);
77484ab085aSmws 
77584ab085aSmws 	desc_printf(smbios_slot_usage_desc(s.smbl_usage),
77684ab085aSmws 	    fp, "  Usage: 0x%x", s.smbl_usage);
77784ab085aSmws 
77884ab085aSmws 	desc_printf(smbios_slot_length_desc(s.smbl_length),
77984ab085aSmws 	    fp, "  Length: 0x%x", s.smbl_length);
78084ab085aSmws 
78184ab085aSmws 	flag_printf(fp, "Slot Characteristics 1",
78284ab085aSmws 	    s.smbl_ch1, sizeof (s.smbl_ch1) * NBBY,
78384ab085aSmws 	    smbios_slot_ch1_name, smbios_slot_ch1_desc);
78484ab085aSmws 
78584ab085aSmws 	flag_printf(fp, "Slot Characteristics 2",
78684ab085aSmws 	    s.smbl_ch2, sizeof (s.smbl_ch2) * NBBY,
78784ab085aSmws 	    smbios_slot_ch2_name, smbios_slot_ch2_desc);
78803f9f63dSTom Pothier 
7891566bc34SRobert Mustacchi 	if (check_oem(shp) != 0 && !smbios_vergteq(&v, 2, 6))
79003f9f63dSTom Pothier 		return;
79103f9f63dSTom Pothier 
79203f9f63dSTom Pothier 	oprintf(fp, "  Segment Group: %u\n", s.smbl_sg);
79303f9f63dSTom Pothier 	oprintf(fp, "  Bus Number: %u\n", s.smbl_bus);
7941566bc34SRobert Mustacchi 	oprintf(fp, "  Device/Function Number: %u/%u\n", s.smbl_df >> 3,
7951566bc34SRobert Mustacchi 	    s.smbl_df & 0x7);
7961566bc34SRobert Mustacchi 
7971566bc34SRobert Mustacchi 	if (s.smbl_dbw != 0) {
7981566bc34SRobert Mustacchi 		oprintf(fp, "  Data Bus Width: %d\n", s.smbl_dbw);
7991566bc34SRobert Mustacchi 	}
8001566bc34SRobert Mustacchi 
8011566bc34SRobert Mustacchi 	if (s.smbl_npeers > 0) {
8021566bc34SRobert Mustacchi 		smbios_slot_peer_t *peer;
8031566bc34SRobert Mustacchi 		uint_t i, npeers;
8041566bc34SRobert Mustacchi 
8051566bc34SRobert Mustacchi 		if (smbios_info_slot_peers(shp, id, &npeers, &peer) != 0) {
8061566bc34SRobert Mustacchi 			smbios_warn(shp, "failed to read slot peer "
8071566bc34SRobert Mustacchi 			    "information");
8081566bc34SRobert Mustacchi 			return;
8091566bc34SRobert Mustacchi 		}
8101566bc34SRobert Mustacchi 
8111566bc34SRobert Mustacchi 		for (i = 0; i < npeers; i++) {
8121566bc34SRobert Mustacchi 			oprintf(fp, "  Slot Peer %u:\n", i);
8131566bc34SRobert Mustacchi 			oprintf(fp, "    Segment group: %u\n",
8141566bc34SRobert Mustacchi 			    peer[i].smblp_group);
815c6795799SRobert Mustacchi 			oprintf(fp, "    Bus/Device/Function: %u/%u/%u\n",
8161566bc34SRobert Mustacchi 			    peer[i].smblp_bus, peer[i].smblp_device,
8171566bc34SRobert Mustacchi 			    peer[i].smblp_function);
8181566bc34SRobert Mustacchi 			oprintf(fp, "    Electrical width: %u\n",
8191566bc34SRobert Mustacchi 			    peer[i].smblp_data_width);
8201566bc34SRobert Mustacchi 		}
8211566bc34SRobert Mustacchi 
8221566bc34SRobert Mustacchi 		smbios_info_slot_peers_free(shp, npeers, peer);
8231566bc34SRobert Mustacchi 	}
824c6795799SRobert Mustacchi 
825c6795799SRobert Mustacchi 	if (s.smbl_info != 0) {
826c6795799SRobert Mustacchi 		if (s.smbl_type >= SMB_SLT_PCIE &&
827c6795799SRobert Mustacchi 		    s.smbl_type <= SMB_SLT_PCIEG6P) {
828c6795799SRobert Mustacchi 			oprintf(fp, "  PCIe Generation: %d\n", s.smbl_info);
829c6795799SRobert Mustacchi 		} else {
830c6795799SRobert Mustacchi 			oprintf(fp, "  Slot Type: 0x%x\n", s.smbl_info);
831c6795799SRobert Mustacchi 		}
832c6795799SRobert Mustacchi 	}
833c6795799SRobert Mustacchi 
834c6795799SRobert Mustacchi 	if (s.smbl_pwidth != 0) {
835c6795799SRobert Mustacchi 		desc_printf(smbios_slot_width_desc(s.smbl_pwidth),
836c6795799SRobert Mustacchi 		    fp, "  Physical Width: 0x%x", s.smbl_pwidth);
837c6795799SRobert Mustacchi 	}
838c6795799SRobert Mustacchi 
839c6795799SRobert Mustacchi 	if (s.smbl_pitch != 0) {
840c6795799SRobert Mustacchi 		oprintf(fp, "  Slot Pitch: %u.%u mm\n", s.smbl_pitch / 100,
841c6795799SRobert Mustacchi 		    s.smbl_pitch % 100);
842c6795799SRobert Mustacchi 	}
843d53cdfabSRobert Mustacchi 
844d53cdfabSRobert Mustacchi 	/*
845d53cdfabSRobert Mustacchi 	 * The slot height was introduced in SMBIOS 3.5. However, a value of
846d53cdfabSRobert Mustacchi 	 * zero here does not mean that it is unknown, but rather that the
847d53cdfabSRobert Mustacchi 	 * concept is not applicable. Therefore we cannot use a standard check
848d53cdfabSRobert Mustacchi 	 * against zero for this and instead use the version.
849d53cdfabSRobert Mustacchi 	 */
850d53cdfabSRobert Mustacchi 	if (smbios_vergteq(&v, 3, 5)) {
851d53cdfabSRobert Mustacchi 		desc_printf(smbios_slot_height_desc(s.smbl_height), fp,
852d53cdfabSRobert Mustacchi 		    "  Height: 0x%x", s.smbl_height);
853d53cdfabSRobert Mustacchi 	} else {
854d53cdfabSRobert Mustacchi 		oprintf(fp, "  Height:  unknown\n");
855d53cdfabSRobert Mustacchi 	}
85603f9f63dSTom Pothier }
85703f9f63dSTom Pothier 
85803f9f63dSTom Pothier static void
print_obdevs_ext(smbios_hdl_t * shp,id_t id,FILE * fp)85903f9f63dSTom Pothier print_obdevs_ext(smbios_hdl_t *shp, id_t id, FILE *fp)
86003f9f63dSTom Pothier {
8616734c4b0SRobert Mustacchi 	boolean_t enabled;
86203f9f63dSTom Pothier 	smbios_obdev_ext_t oe;
8636734c4b0SRobert Mustacchi 	const char *type;
86403f9f63dSTom Pothier 
865a7aaa513SRobert Mustacchi 	if (smbios_info_obdevs_ext(shp, id, &oe) != 0) {
866a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read extended on-board devices "
867a7aaa513SRobert Mustacchi 		    "information");
868a7aaa513SRobert Mustacchi 		return;
869a7aaa513SRobert Mustacchi 	}
87003f9f63dSTom Pothier 
8716734c4b0SRobert Mustacchi 	/*
8726734c4b0SRobert Mustacchi 	 * Bit 7 is always whether or not the device is enabled while bits 0:6
8736734c4b0SRobert Mustacchi 	 * are the actual device type.
8746734c4b0SRobert Mustacchi 	 */
8756734c4b0SRobert Mustacchi 	enabled = oe.smboe_dtype >> 7;
876d53cdfabSRobert Mustacchi 	type = smbios_onboard_ext_type_desc(oe.smboe_dtype & 0x7f);
8776734c4b0SRobert Mustacchi 
8780f56e145SRobert Mustacchi 	str_print(fp, "  Reference Designator", oe.smboe_name);
8796734c4b0SRobert Mustacchi 	oprintf(fp, "  Device Enabled: %s\n", enabled == B_TRUE ? "true" :
8806734c4b0SRobert Mustacchi 	    "false");
8816734c4b0SRobert Mustacchi 	oprintf(fp, "  Device Type: %s\n", type);
88203f9f63dSTom Pothier 	oprintf(fp, "  Device Type Instance: %u\n", oe.smboe_dti);
88303f9f63dSTom Pothier 	oprintf(fp, "  Segment Group Number: %u\n", oe.smboe_sg);
88403f9f63dSTom Pothier 	oprintf(fp, "  Bus Number: %u\n", oe.smboe_bus);
88503f9f63dSTom Pothier 	oprintf(fp, "  Device/Function Number: %u\n", oe.smboe_df);
88684ab085aSmws }
88784ab085aSmws 
88884ab085aSmws static void
print_obdevs(smbios_hdl_t * shp,id_t id,FILE * fp)88984ab085aSmws print_obdevs(smbios_hdl_t *shp, id_t id, FILE *fp)
89084ab085aSmws {
89184ab085aSmws 	smbios_obdev_t *argv;
89284ab085aSmws 	int i, argc;
89384ab085aSmws 
89484ab085aSmws 	if ((argc = smbios_info_obdevs(shp, id, 0, NULL)) > 0) {
89584ab085aSmws 		argv = alloca(sizeof (smbios_obdev_t) * argc);
896a7aaa513SRobert Mustacchi 		if (smbios_info_obdevs(shp, id, argc, argv) == -1) {
897a7aaa513SRobert Mustacchi 			smbios_warn(shp, "failed to read on-board device "
898a7aaa513SRobert Mustacchi 			    "information");
899a7aaa513SRobert Mustacchi 			return;
900a7aaa513SRobert Mustacchi 		}
90184ab085aSmws 		for (i = 0; i < argc; i++)
9020f56e145SRobert Mustacchi 			str_print_nolabel(fp, "  ", argv[i].smbd_name);
90384ab085aSmws 	}
90484ab085aSmws }
90584ab085aSmws 
90684ab085aSmws static void
print_strtab(smbios_hdl_t * shp,id_t id,FILE * fp)90784ab085aSmws print_strtab(smbios_hdl_t *shp, id_t id, FILE *fp)
90884ab085aSmws {
90984ab085aSmws 	const char **argv;
91084ab085aSmws 	int i, argc;
91184ab085aSmws 
91284ab085aSmws 	if ((argc = smbios_info_strtab(shp, id, 0, NULL)) > 0) {
91384ab085aSmws 		argv = alloca(sizeof (char *) * argc);
914a7aaa513SRobert Mustacchi 		if (smbios_info_strtab(shp, id, argc, argv) == -1) {
915a7aaa513SRobert Mustacchi 			smbios_warn(shp, "failed to read string table "
916a7aaa513SRobert Mustacchi 			    "information");
917a7aaa513SRobert Mustacchi 			return;
918a7aaa513SRobert Mustacchi 		}
91984ab085aSmws 		for (i = 0; i < argc; i++)
9200f56e145SRobert Mustacchi 			str_print_nolabel(fp, "  ", argv[i]);
92184ab085aSmws 	}
92284ab085aSmws }
92384ab085aSmws 
92484ab085aSmws static void
print_lang(smbios_hdl_t * shp,id_t id,FILE * fp)92584ab085aSmws print_lang(smbios_hdl_t *shp, id_t id, FILE *fp)
92684ab085aSmws {
92784ab085aSmws 	smbios_lang_t l;
92884ab085aSmws 
929a7aaa513SRobert Mustacchi 	if (smbios_info_lang(shp, &l) == -1) {
930a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read language information");
931a7aaa513SRobert Mustacchi 		return;
932a7aaa513SRobert Mustacchi 	}
93384ab085aSmws 
9340f56e145SRobert Mustacchi 	str_print(fp, "  Current Language", l.smbla_cur);
93584ab085aSmws 	oprintf(fp, "  Language String Format: %u\n", l.smbla_fmt);
93684ab085aSmws 	oprintf(fp, "  Number of Installed Languages: %u\n", l.smbla_num);
93784ab085aSmws 	oprintf(fp, "  Installed Languages:\n");
93884ab085aSmws 
93984ab085aSmws 	print_strtab(shp, id, fp);
94084ab085aSmws }
94184ab085aSmws 
94284ab085aSmws /*ARGSUSED*/
94384ab085aSmws static void
print_evlog(smbios_hdl_t * shp,id_t id,FILE * fp)94484ab085aSmws print_evlog(smbios_hdl_t *shp, id_t id, FILE *fp)
94584ab085aSmws {
94684ab085aSmws 	smbios_evlog_t ev;
94784ab085aSmws 	uint32_t i;
94884ab085aSmws 
949a7aaa513SRobert Mustacchi 	if (smbios_info_eventlog(shp, &ev) == -1) {
950a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read event log information");
951a7aaa513SRobert Mustacchi 		return;
952a7aaa513SRobert Mustacchi 	}
95384ab085aSmws 
95484ab085aSmws 	oprintf(fp, "  Log Area Size: %lu bytes\n", (ulong_t)ev.smbev_size);
95584ab085aSmws 	oprintf(fp, "  Header Offset: %lu\n", (ulong_t)ev.smbev_hdr);
95684ab085aSmws 	oprintf(fp, "  Data Offset: %lu\n", (ulong_t)ev.smbev_data);
95784ab085aSmws 
95884ab085aSmws 	desc_printf(smbios_evlog_method_desc(ev.smbev_method),
95984ab085aSmws 	    fp, "  Data Access Method: %u", ev.smbev_method);
96084ab085aSmws 
96184ab085aSmws 	flag_printf(fp, "Log Flags",
96284ab085aSmws 	    ev.smbev_flags, sizeof (ev.smbev_flags) * NBBY,
96384ab085aSmws 	    smbios_evlog_flag_name, smbios_evlog_flag_desc);
96484ab085aSmws 
96584ab085aSmws 	desc_printf(smbios_evlog_format_desc(ev.smbev_format),
96684ab085aSmws 	    fp, "  Log Header Format: %u", ev.smbev_format);
96784ab085aSmws 
96884ab085aSmws 	oprintf(fp, "  Update Token: 0x%x\n", ev.smbev_token);
96984ab085aSmws 	oprintf(fp, "  Data Access Address: ");
97084ab085aSmws 
97184ab085aSmws 	switch (ev.smbev_method) {
97284ab085aSmws 	case SMB_EVM_1x1i_1x1d:
97384ab085aSmws 	case SMB_EVM_2x1i_1x1d:
97484ab085aSmws 	case SMB_EVM_1x2i_1x1d:
97584ab085aSmws 		oprintf(fp, "Index Address 0x%x, Data Address 0x%x\n",
97684ab085aSmws 		    ev.smbev_addr.eva_io.evi_iaddr,
97784ab085aSmws 		    ev.smbev_addr.eva_io.evi_daddr);
97884ab085aSmws 		break;
97984ab085aSmws 	case SMB_EVM_GPNV:
98084ab085aSmws 		oprintf(fp, "0x%x\n", ev.smbev_addr.eva_gpnv);
98184ab085aSmws 		break;
98284ab085aSmws 	default:
98384ab085aSmws 		oprintf(fp, "0x%x\n", ev.smbev_addr.eva_addr);
98484ab085aSmws 	}
98584ab085aSmws 
98684ab085aSmws 	oprintf(fp, "  Type Descriptors:\n");
98784ab085aSmws 
98884ab085aSmws 	for (i = 0; i < ev.smbev_typec; i++) {
98984ab085aSmws 		oprintf(fp, "  %u: Log Type 0x%x, Data Type 0x%x\n", i,
99084ab085aSmws 		    ev.smbev_typev[i].smbevt_ltype,
99184ab085aSmws 		    ev.smbev_typev[i].smbevt_dtype);
99284ab085aSmws 	}
99384ab085aSmws }
99484ab085aSmws 
99584ab085aSmws static void
print_bytes(const uint8_t * data,size_t size,FILE * fp)99684ab085aSmws print_bytes(const uint8_t *data, size_t size, FILE *fp)
99784ab085aSmws {
99843379a28SAndy Fiddaman 	hexdump_t h;
99984ab085aSmws 
100043379a28SAndy Fiddaman 	hexdump_init(&h);
100143379a28SAndy Fiddaman 	hexdump_set_grouping(&h, 4);
100243379a28SAndy Fiddaman 	hexdump_set_indent(&h, 2);
100384ab085aSmws 
100443379a28SAndy Fiddaman 	(void) fprintf(fp, "\n");
100543379a28SAndy Fiddaman 	(void) hexdump_fileh(&h, data, size, HDF_DEFAULT, fp);
100643379a28SAndy Fiddaman 	(void) fprintf(fp, "\n");
100784ab085aSmws 
100843379a28SAndy Fiddaman 	hexdump_fini(&h);
100984ab085aSmws }
101084ab085aSmws 
101184ab085aSmws static void
print_memarray(smbios_hdl_t * shp,id_t id,FILE * fp)101284ab085aSmws print_memarray(smbios_hdl_t *shp, id_t id, FILE *fp)
101384ab085aSmws {
101484ab085aSmws 	smbios_memarray_t ma;
101584ab085aSmws 
1016a7aaa513SRobert Mustacchi 	if (smbios_info_memarray(shp, id, &ma) != 0) {
1017a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read memarray information");
1018a7aaa513SRobert Mustacchi 		return;
1019a7aaa513SRobert Mustacchi 	}
102084ab085aSmws 
102184ab085aSmws 	desc_printf(smbios_memarray_loc_desc(ma.smbma_location),
102284ab085aSmws 	    fp, "  Location: %u", ma.smbma_location);
102384ab085aSmws 
102484ab085aSmws 	desc_printf(smbios_memarray_use_desc(ma.smbma_use),
102584ab085aSmws 	    fp, "  Use: %u", ma.smbma_use);
102684ab085aSmws 
102784ab085aSmws 	desc_printf(smbios_memarray_ecc_desc(ma.smbma_ecc),
102884ab085aSmws 	    fp, "  ECC: %u", ma.smbma_ecc);
102984ab085aSmws 
103084ab085aSmws 	oprintf(fp, "  Number of Slots/Sockets: %u\n", ma.smbma_ndevs);
103184ab085aSmws 	id_printf(fp, "  Memory Error Data: ", ma.smbma_err);
103284ab085aSmws 	oprintf(fp, "  Max Capacity: %llu bytes\n",
103384ab085aSmws 	    (u_longlong_t)ma.smbma_size);
103484ab085aSmws }
103584ab085aSmws 
103684ab085aSmws static void
print_memdevice(smbios_hdl_t * shp,id_t id,FILE * fp)103784ab085aSmws print_memdevice(smbios_hdl_t *shp, id_t id, FILE *fp)
103884ab085aSmws {
103984ab085aSmws 	smbios_memdevice_t md;
104084ab085aSmws 
1041a7aaa513SRobert Mustacchi 	if (smbios_info_memdevice(shp, id, &md) != 0) {
1042a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read memory device information");
1043a7aaa513SRobert Mustacchi 		return;
1044a7aaa513SRobert Mustacchi 	}
104584ab085aSmws 
104684ab085aSmws 	id_printf(fp, "  Physical Memory Array: ", md.smbmd_array);
104784ab085aSmws 	id_printf(fp, "  Memory Error Data: ", md.smbmd_error);
104884ab085aSmws 
104984ab085aSmws 	if (md.smbmd_twidth != -1u)
105084ab085aSmws 		oprintf(fp, "  Total Width: %u bits\n", md.smbmd_twidth);
105184ab085aSmws 	else
105284ab085aSmws 		oprintf(fp, "  Total Width: Unknown\n");
105384ab085aSmws 
105484ab085aSmws 	if (md.smbmd_dwidth != -1u)
105584ab085aSmws 		oprintf(fp, "  Data Width: %u bits\n", md.smbmd_dwidth);
105684ab085aSmws 	else
105784ab085aSmws 		oprintf(fp, "  Data Width: Unknown\n");
105884ab085aSmws 
105984ab085aSmws 	switch (md.smbmd_size) {
106084ab085aSmws 	case -1ull:
106184ab085aSmws 		oprintf(fp, "  Size: Unknown\n");
106284ab085aSmws 		break;
106384ab085aSmws 	case 0:
106484ab085aSmws 		oprintf(fp, "  Size: Not Populated\n");
106584ab085aSmws 		break;
106684ab085aSmws 	default:
106784ab085aSmws 		oprintf(fp, "  Size: %llu bytes\n",
106884ab085aSmws 		    (u_longlong_t)md.smbmd_size);
106984ab085aSmws 	}
107084ab085aSmws 
107184ab085aSmws 	desc_printf(smbios_memdevice_form_desc(md.smbmd_form),
107284ab085aSmws 	    fp, "  Form Factor: %u", md.smbmd_form);
107384ab085aSmws 
107484ab085aSmws 	if (md.smbmd_set == 0)
107584ab085aSmws 		oprintf(fp, "  Set: None\n");
107684ab085aSmws 	else if (md.smbmd_set == (uint8_t)-1u)
107784ab085aSmws 		oprintf(fp, "  Set: Unknown\n");
107884ab085aSmws 	else
107984ab085aSmws 		oprintf(fp, "  Set: %u\n", md.smbmd_set);
108084ab085aSmws 
10814e901881SDale Ghent 	if (md.smbmd_rank != 0) {
10824e901881SDale Ghent 		desc_printf(smbios_memdevice_rank_desc(md.smbmd_rank),
10834e901881SDale Ghent 		    fp, "  Rank: %u", md.smbmd_rank);
10844e901881SDale Ghent 	} else {
10854e901881SDale Ghent 		oprintf(fp, "  Rank: Unknown\n");
10864e901881SDale Ghent 	}
10874e901881SDale Ghent 
108884ab085aSmws 	desc_printf(smbios_memdevice_type_desc(md.smbmd_type),
108984ab085aSmws 	    fp, "  Memory Type: %u", md.smbmd_type);
109084ab085aSmws 
109184ab085aSmws 	flag_printf(fp, "Flags", md.smbmd_flags, sizeof (md.smbmd_flags) * NBBY,
109284ab085aSmws 	    smbios_memdevice_flag_name, smbios_memdevice_flag_desc);
109384ab085aSmws 
1094176a9270SRobert Mustacchi 	if (md.smbmd_extspeed != 0) {
1095176a9270SRobert Mustacchi 		oprintf(fp, "  Speed: %" PRIu64 " MT/s\n", md.smbmd_extspeed);
1096176a9270SRobert Mustacchi 	} else {
109784ab085aSmws 		oprintf(fp, "  Speed: Unknown\n");
1098176a9270SRobert Mustacchi 	}
109984ab085aSmws 
1100176a9270SRobert Mustacchi 	if (md.smbmd_extclkspeed != 0) {
1101176a9270SRobert Mustacchi 		oprintf(fp, "  Configured Speed: %" PRIu64 " MT/s\n",
1102176a9270SRobert Mustacchi 		    md.smbmd_extclkspeed);
1103176a9270SRobert Mustacchi 	} else {
11044e901881SDale Ghent 		oprintf(fp, "  Configured Speed: Unknown\n");
1105176a9270SRobert Mustacchi 	}
11064e901881SDale Ghent 
11070f56e145SRobert Mustacchi 	str_print(fp, "  Device Locator", md.smbmd_dloc);
11080f56e145SRobert Mustacchi 	str_print(fp, "  Bank Locator", md.smbmd_bloc);
11094e901881SDale Ghent 
11104e901881SDale Ghent 	if (md.smbmd_minvolt != 0) {
11114e901881SDale Ghent 		oprintf(fp, "  Minimum Voltage: %.2fV\n",
11124e901881SDale Ghent 		    md.smbmd_minvolt / 1000.0);
11134e901881SDale Ghent 	} else {
11144e901881SDale Ghent 		oprintf(fp, "  Minimum Voltage: Unknown\n");
11154e901881SDale Ghent 	}
11164e901881SDale Ghent 
11174e901881SDale Ghent 	if (md.smbmd_maxvolt != 0) {
11184e901881SDale Ghent 		oprintf(fp, "  Maximum Voltage: %.2fV\n",
11194e901881SDale Ghent 		    md.smbmd_maxvolt / 1000.0);
11204e901881SDale Ghent 	} else {
11214e901881SDale Ghent 		oprintf(fp, "  Maximum Voltage: Unknown\n");
11224e901881SDale Ghent 	}
11234e901881SDale Ghent 
11244e901881SDale Ghent 	if (md.smbmd_confvolt != 0) {
11254e901881SDale Ghent 		oprintf(fp, "  Configured Voltage: %.2fV\n",
11264e901881SDale Ghent 		    md.smbmd_confvolt / 1000.0);
11274e901881SDale Ghent 	} else {
11284e901881SDale Ghent 		oprintf(fp, "  Configured Voltage: Unknown\n");
11294e901881SDale Ghent 	}
11301566bc34SRobert Mustacchi 
11311566bc34SRobert Mustacchi 	if (md.smbmd_memtech != 0) {
11321566bc34SRobert Mustacchi 		desc_printf(smbios_memdevice_memtech_desc(md.smbmd_memtech),
11331566bc34SRobert Mustacchi 		    fp, "  Memory Technology: %u", md.smbmd_memtech);
11341566bc34SRobert Mustacchi 	}
11351566bc34SRobert Mustacchi 
11361566bc34SRobert Mustacchi 	if (md.smbmd_opcap_flags != 0) {
11371566bc34SRobert Mustacchi 		flag_printf(fp, "Operating Mode Capabilities",
11381566bc34SRobert Mustacchi 		    md.smbmd_opcap_flags, sizeof (md.smbmd_opcap_flags) * NBBY,
11391566bc34SRobert Mustacchi 		    smbios_memdevice_op_capab_name,
11401566bc34SRobert Mustacchi 		    smbios_memdevice_op_capab_desc);
11411566bc34SRobert Mustacchi 	}
11421566bc34SRobert Mustacchi 
11431566bc34SRobert Mustacchi 	if (md.smbmd_firmware_rev[0] != '\0') {
11440f56e145SRobert Mustacchi 		str_print(fp, "  Firmware Revision", md.smbmd_firmware_rev);
11451566bc34SRobert Mustacchi 	}
11461566bc34SRobert Mustacchi 
11476bc074b1SRobert Mustacchi 	if (md.smbmd_modmfg_id != SMB_MD_MFG_UNKNOWN) {
11481566bc34SRobert Mustacchi 		jedec_print(fp, "Module Manufacturer ID", md.smbmd_modmfg_id);
11491566bc34SRobert Mustacchi 	}
11501566bc34SRobert Mustacchi 
11511566bc34SRobert Mustacchi 	if (md.smbmd_modprod_id  != 0) {
11521566bc34SRobert Mustacchi 		jedec_print(fp, "Module Product ID", md.smbmd_modprod_id);
11531566bc34SRobert Mustacchi 	}
11541566bc34SRobert Mustacchi 
11556bc074b1SRobert Mustacchi 	if (md.smbmd_cntrlmfg_id != SMB_MD_MFG_UNKNOWN) {
11561566bc34SRobert Mustacchi 		jedec_print(fp, "Memory Subsystem Controller Manufacturer ID",
11571566bc34SRobert Mustacchi 		    md.smbmd_cntrlmfg_id);
11581566bc34SRobert Mustacchi 	}
11591566bc34SRobert Mustacchi 
11601566bc34SRobert Mustacchi 	if (md.smbmd_cntrlprod_id != 0) {
11611566bc34SRobert Mustacchi 		jedec_print(fp, "Memory Subsystem Controller Product ID",
11621566bc34SRobert Mustacchi 		    md.smbmd_cntrlprod_id);
11631566bc34SRobert Mustacchi 	}
11641566bc34SRobert Mustacchi 
11651566bc34SRobert Mustacchi 	if (md.smbmd_nvsize == UINT64_MAX) {
11661566bc34SRobert Mustacchi 		oprintf(fp, "  Non-volatile Size: Unknown\n");
11671566bc34SRobert Mustacchi 	} else if (md.smbmd_nvsize != 0) {
11681566bc34SRobert Mustacchi 		oprintf(fp, "  Non-volatile Size: %llu bytes\n",
11691566bc34SRobert Mustacchi 		    (u_longlong_t)md.smbmd_nvsize);
11701566bc34SRobert Mustacchi 	}
11711566bc34SRobert Mustacchi 
11721566bc34SRobert Mustacchi 	if (md.smbmd_volatile_size == UINT64_MAX) {
11731566bc34SRobert Mustacchi 		oprintf(fp, "  Volatile Size: Unknown\n");
11741566bc34SRobert Mustacchi 	} else if (md.smbmd_volatile_size != 0) {
11751566bc34SRobert Mustacchi 		oprintf(fp, "  Volatile Size: %llu bytes\n",
11761566bc34SRobert Mustacchi 		    (u_longlong_t)md.smbmd_volatile_size);
11771566bc34SRobert Mustacchi 	}
11781566bc34SRobert Mustacchi 
11791566bc34SRobert Mustacchi 	if (md.smbmd_cache_size == UINT64_MAX) {
11801566bc34SRobert Mustacchi 		oprintf(fp, "  Cache Size: Unknown\n");
11811566bc34SRobert Mustacchi 	} else if (md.smbmd_cache_size != 0) {
11821566bc34SRobert Mustacchi 		oprintf(fp, "  Cache Size: %llu bytes\n",
11831566bc34SRobert Mustacchi 		    (u_longlong_t)md.smbmd_cache_size);
11841566bc34SRobert Mustacchi 	}
11851566bc34SRobert Mustacchi 
11861566bc34SRobert Mustacchi 	if (md.smbmd_logical_size == UINT64_MAX) {
11871566bc34SRobert Mustacchi 		oprintf(fp, "  Logical Size: Unknown\n");
11881566bc34SRobert Mustacchi 	} else if (md.smbmd_logical_size != 0) {
11891566bc34SRobert Mustacchi 		oprintf(fp, "  Logical Size: %llu bytes\n",
11901566bc34SRobert Mustacchi 		    (u_longlong_t)md.smbmd_logical_size);
11911566bc34SRobert Mustacchi 	}
11926bc074b1SRobert Mustacchi 
11936bc074b1SRobert Mustacchi 	if (md.smbmd_pmic0_mfgid != SMB_MD_MFG_UNKNOWN) {
11946bc074b1SRobert Mustacchi 		jedec_print(fp, "PMIC0 Manufacturer ID", md.smbmd_pmic0_mfgid);
11956bc074b1SRobert Mustacchi 	}
11966bc074b1SRobert Mustacchi 
11976bc074b1SRobert Mustacchi 	if (md.smbmd_pmic0_rev != SMB_MD_REV_UNKNOWN) {
11986bc074b1SRobert Mustacchi 		jedec_rev_print(fp, "PMIC0 Revision", md.smbmd_pmic0_rev);
11996bc074b1SRobert Mustacchi 	}
12006bc074b1SRobert Mustacchi 
12016bc074b1SRobert Mustacchi 	if (md.smbmd_rcd_mfgid != SMB_MD_MFG_UNKNOWN) {
12026bc074b1SRobert Mustacchi 		jedec_print(fp, "RCD Manufacturer ID", md.smbmd_rcd_mfgid);
12036bc074b1SRobert Mustacchi 	}
12046bc074b1SRobert Mustacchi 
12056bc074b1SRobert Mustacchi 	if (md.smbmd_rcd_rev != SMB_MD_REV_UNKNOWN) {
12066bc074b1SRobert Mustacchi 		jedec_rev_print(fp, "RCD Revision", md.smbmd_rcd_rev);
12076bc074b1SRobert Mustacchi 	}
120884ab085aSmws }
120984ab085aSmws 
121084ab085aSmws static void
print_memarrmap(smbios_hdl_t * shp,id_t id,FILE * fp)121184ab085aSmws print_memarrmap(smbios_hdl_t *shp, id_t id, FILE *fp)
121284ab085aSmws {
121384ab085aSmws 	smbios_memarrmap_t ma;
121484ab085aSmws 
1215a7aaa513SRobert Mustacchi 	if (smbios_info_memarrmap(shp, id, &ma) != 0) {
1216a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read memory array map information");
1217a7aaa513SRobert Mustacchi 		return;
1218a7aaa513SRobert Mustacchi 	}
121984ab085aSmws 
122084ab085aSmws 	id_printf(fp, "  Physical Memory Array: ", ma.smbmam_array);
122184ab085aSmws 	oprintf(fp, "  Devices per Row: %u\n", ma.smbmam_width);
122284ab085aSmws 
122384ab085aSmws 	oprintf(fp, "  Physical Address: 0x%llx\n  Size: %llu bytes\n",
122484ab085aSmws 	    (u_longlong_t)ma.smbmam_addr, (u_longlong_t)ma.smbmam_size);
122584ab085aSmws }
122684ab085aSmws 
122784ab085aSmws static void
print_memdevmap(smbios_hdl_t * shp,id_t id,FILE * fp)122884ab085aSmws print_memdevmap(smbios_hdl_t *shp, id_t id, FILE *fp)
122984ab085aSmws {
123084ab085aSmws 	smbios_memdevmap_t md;
123184ab085aSmws 
1232a7aaa513SRobert Mustacchi 	if (smbios_info_memdevmap(shp, id, &md) != 0) {
1233a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read memory device map "
1234a7aaa513SRobert Mustacchi 		    "information");
1235a7aaa513SRobert Mustacchi 		return;
1236a7aaa513SRobert Mustacchi 	}
123784ab085aSmws 
123884ab085aSmws 	id_printf(fp, "  Memory Device: ", md.smbmdm_device);
123984ab085aSmws 	id_printf(fp, "  Memory Array Mapped Address: ", md.smbmdm_arrmap);
124084ab085aSmws 
124184ab085aSmws 	oprintf(fp, "  Physical Address: 0x%llx\n  Size: %llu bytes\n",
124284ab085aSmws 	    (u_longlong_t)md.smbmdm_addr, (u_longlong_t)md.smbmdm_size);
124384ab085aSmws 
124484ab085aSmws 	oprintf(fp, "  Partition Row Position: %u\n", md.smbmdm_rpos);
124584ab085aSmws 	oprintf(fp, "  Interleave Position: %u\n", md.smbmdm_ipos);
124684ab085aSmws 	oprintf(fp, "  Interleave Data Depth: %u\n", md.smbmdm_idepth);
124784ab085aSmws }
124884ab085aSmws 
124984ab085aSmws static void
print_hwsec(smbios_hdl_t * shp,FILE * fp)125084ab085aSmws print_hwsec(smbios_hdl_t *shp, FILE *fp)
125184ab085aSmws {
125284ab085aSmws 	smbios_hwsec_t h;
125384ab085aSmws 
1254a7aaa513SRobert Mustacchi 	if (smbios_info_hwsec(shp, &h) == -1) {
1255a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read hwsec information");
1256a7aaa513SRobert Mustacchi 		return;
1257a7aaa513SRobert Mustacchi 	}
125884ab085aSmws 
125984ab085aSmws 	desc_printf(smbios_hwsec_desc(h.smbh_pwr_ps),
126084ab085aSmws 	    fp, "  Power-On Password Status: %u", h.smbh_pwr_ps);
126184ab085aSmws 	desc_printf(smbios_hwsec_desc(h.smbh_kbd_ps),
126284ab085aSmws 	    fp, "  Keyboard Password Status: %u", h.smbh_kbd_ps);
126384ab085aSmws 	desc_printf(smbios_hwsec_desc(h.smbh_adm_ps),
126484ab085aSmws 	    fp, "  Administrator Password Status: %u", h.smbh_adm_ps);
126584ab085aSmws 	desc_printf(smbios_hwsec_desc(h.smbh_pan_ps),
126684ab085aSmws 	    fp, "  Front Panel Reset Status: %u", h.smbh_pan_ps);
126784ab085aSmws }
126884ab085aSmws 
126984ab085aSmws static void
print_vprobe(smbios_hdl_t * shp,id_t id,FILE * fp)1270f44a1392SRobert Mustacchi print_vprobe(smbios_hdl_t *shp, id_t id, FILE *fp)
1271f44a1392SRobert Mustacchi {
1272f44a1392SRobert Mustacchi 	smbios_vprobe_t vp;
1273f44a1392SRobert Mustacchi 
1274f44a1392SRobert Mustacchi 	if (smbios_info_vprobe(shp, id, &vp) != 0) {
1275f44a1392SRobert Mustacchi 		smbios_warn(shp, "failed to read voltage probe information");
1276f44a1392SRobert Mustacchi 		return;
1277f44a1392SRobert Mustacchi 	}
1278f44a1392SRobert Mustacchi 
12790f56e145SRobert Mustacchi 	str_print(fp, "  Description", vp.smbvp_description != NULL ?
1280f44a1392SRobert Mustacchi 	    vp.smbvp_description : "unknown");
1281f44a1392SRobert Mustacchi 	desc_printf(smbios_vprobe_loc_desc(vp.smbvp_location),
1282f44a1392SRobert Mustacchi 	    fp, "  Location: %u", vp.smbvp_location);
1283f44a1392SRobert Mustacchi 	desc_printf(smbios_vprobe_status_desc(vp.smbvp_status),
1284f44a1392SRobert Mustacchi 	    fp, "  Status: %u", vp.smbvp_status);
1285f44a1392SRobert Mustacchi 
1286f44a1392SRobert Mustacchi 	if (vp.smbvp_maxval != SMB_PROBE_UNKNOWN_VALUE) {
1287f44a1392SRobert Mustacchi 		oprintf(fp, "  Maximum Possible Voltage: %u mV\n",
1288f44a1392SRobert Mustacchi 		    vp.smbvp_maxval);
1289f44a1392SRobert Mustacchi 	} else {
1290f44a1392SRobert Mustacchi 		oprintf(fp, "  Maximum Possible Voltage: unknown\n");
1291f44a1392SRobert Mustacchi 	}
1292f44a1392SRobert Mustacchi 
1293f44a1392SRobert Mustacchi 	if (vp.smbvp_minval != SMB_PROBE_UNKNOWN_VALUE) {
1294f44a1392SRobert Mustacchi 		oprintf(fp, "  Minimum Possible Voltage: %u mV\n",
1295f44a1392SRobert Mustacchi 		    vp.smbvp_minval);
1296f44a1392SRobert Mustacchi 	} else {
1297f44a1392SRobert Mustacchi 		oprintf(fp, "  Minimum Possible Voltage: unknown\n");
1298f44a1392SRobert Mustacchi 	}
1299f44a1392SRobert Mustacchi 
1300f44a1392SRobert Mustacchi 	if (vp.smbvp_resolution != SMB_PROBE_UNKNOWN_VALUE) {
1301f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Resolution: %u.%u mV\n",
1302f44a1392SRobert Mustacchi 		    vp.smbvp_resolution / 10,
1303f44a1392SRobert Mustacchi 		    vp.smbvp_resolution % 10);
1304f44a1392SRobert Mustacchi 	} else {
1305f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Resolution: unknown\n");
1306f44a1392SRobert Mustacchi 	}
1307f44a1392SRobert Mustacchi 
1308f44a1392SRobert Mustacchi 	if (vp.smbvp_tolerance != SMB_PROBE_UNKNOWN_VALUE) {
1309f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: +/-%u mV\n",
1310f44a1392SRobert Mustacchi 		    vp.smbvp_tolerance);
1311f44a1392SRobert Mustacchi 	} else {
1312f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: unknown\n");
1313f44a1392SRobert Mustacchi 	}
1314f44a1392SRobert Mustacchi 
1315f44a1392SRobert Mustacchi 	if (vp.smbvp_accuracy != SMB_PROBE_UNKNOWN_VALUE) {
1316f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: +/-%u.%02u%%\n",
1317f44a1392SRobert Mustacchi 		    vp.smbvp_accuracy / 100,
1318f44a1392SRobert Mustacchi 		    vp.smbvp_accuracy % 100);
1319f44a1392SRobert Mustacchi 	} else {
1320f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: unknown\n");
1321f44a1392SRobert Mustacchi 	}
1322f44a1392SRobert Mustacchi 
1323f44a1392SRobert Mustacchi 	oprintf(fp, "  OEM- or BIOS- defined value: 0x%x\n", vp.smbvp_oem);
1324f44a1392SRobert Mustacchi 
1325f44a1392SRobert Mustacchi 	if (vp.smbvp_nominal != SMB_PROBE_UNKNOWN_VALUE) {
1326f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: %u mV\n", vp.smbvp_nominal);
1327f44a1392SRobert Mustacchi 	} else {
1328f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: unknown\n");
1329f44a1392SRobert Mustacchi 	}
1330f44a1392SRobert Mustacchi }
1331f44a1392SRobert Mustacchi 
1332f44a1392SRobert Mustacchi static void
print_cooldev(smbios_hdl_t * shp,id_t id,FILE * fp)1333f44a1392SRobert Mustacchi print_cooldev(smbios_hdl_t *shp, id_t id, FILE *fp)
1334f44a1392SRobert Mustacchi {
1335f44a1392SRobert Mustacchi 	smbios_cooldev_t cd;
1336f44a1392SRobert Mustacchi 
1337f44a1392SRobert Mustacchi 	if (smbios_info_cooldev(shp, id, &cd) != 0) {
1338f44a1392SRobert Mustacchi 		smbios_warn(shp, "failed to read cooling device "
1339f44a1392SRobert Mustacchi 		    "information");
1340f44a1392SRobert Mustacchi 		return;
1341f44a1392SRobert Mustacchi 	}
1342f44a1392SRobert Mustacchi 
1343f44a1392SRobert Mustacchi 	id_printf(fp, "  Temperature Probe Handle: ", cd.smbcd_tprobe);
1344f44a1392SRobert Mustacchi 	desc_printf(smbios_cooldev_type_desc(cd.smbcd_type),
1345f44a1392SRobert Mustacchi 	    fp, "  Device Type: %u", cd.smbcd_type);
1346f44a1392SRobert Mustacchi 	desc_printf(smbios_cooldev_status_desc(cd.smbcd_status),
1347f44a1392SRobert Mustacchi 	    fp, "  Status: %u", cd.smbcd_status);
1348f44a1392SRobert Mustacchi 	oprintf(fp, "  Cooling Unit Group: %u\n", cd.smbcd_group);
1349f44a1392SRobert Mustacchi 	oprintf(fp, "  OEM- or BIOS- defined data: 0x%x\n", cd.smbcd_oem);
1350f44a1392SRobert Mustacchi 	if (cd.smbcd_nominal != SMB_PROBE_UNKNOWN_VALUE) {
1351f44a1392SRobert Mustacchi 		oprintf(fp, "  Nominal Speed: %u RPM\n", cd.smbcd_nominal);
1352f44a1392SRobert Mustacchi 	} else {
1353f44a1392SRobert Mustacchi 		oprintf(fp, "  Nominal Speed: unknown\n");
1354f44a1392SRobert Mustacchi 	}
1355f44a1392SRobert Mustacchi 
1356f44a1392SRobert Mustacchi 	if (cd.smbcd_descr != NULL && cd.smbcd_descr[0] != '\0') {
13570f56e145SRobert Mustacchi 		str_print(fp, "  Description", cd.smbcd_descr);
1358f44a1392SRobert Mustacchi 	}
1359f44a1392SRobert Mustacchi }
1360f44a1392SRobert Mustacchi 
1361f44a1392SRobert Mustacchi static void
print_tprobe(smbios_hdl_t * shp,id_t id,FILE * fp)1362f44a1392SRobert Mustacchi print_tprobe(smbios_hdl_t *shp, id_t id, FILE *fp)
1363f44a1392SRobert Mustacchi {
1364f44a1392SRobert Mustacchi 	smbios_tprobe_t tp;
1365f44a1392SRobert Mustacchi 
1366f44a1392SRobert Mustacchi 	if (smbios_info_tprobe(shp, id, &tp) != 0) {
1367f44a1392SRobert Mustacchi 		smbios_warn(shp, "failed to read temperature probe "
1368f44a1392SRobert Mustacchi 		    "information");
1369f44a1392SRobert Mustacchi 		return;
1370f44a1392SRobert Mustacchi 	}
1371f44a1392SRobert Mustacchi 
13720f56e145SRobert Mustacchi 	str_print(fp, "  Description", tp.smbtp_description != NULL ?
1373f44a1392SRobert Mustacchi 	    tp.smbtp_description : "unknown");
1374f44a1392SRobert Mustacchi 	desc_printf(smbios_tprobe_loc_desc(tp.smbtp_location),
1375f44a1392SRobert Mustacchi 	    fp, "  Location: %u", tp.smbtp_location);
1376f44a1392SRobert Mustacchi 	desc_printf(smbios_tprobe_status_desc(tp.smbtp_status),
1377f44a1392SRobert Mustacchi 	    fp, "  Status: %u", tp.smbtp_status);
1378f44a1392SRobert Mustacchi 
1379f44a1392SRobert Mustacchi 	if (tp.smbtp_maxval != SMB_PROBE_UNKNOWN_VALUE) {
1380f44a1392SRobert Mustacchi 		oprintf(fp, "  Maximum Possible Temperature: %u.%u C\n",
1381f44a1392SRobert Mustacchi 		    tp.smbtp_maxval / 10, tp.smbtp_maxval % 10);
1382f44a1392SRobert Mustacchi 	} else {
1383f44a1392SRobert Mustacchi 		oprintf(fp, "  Maximum Possible Temperature: unknown\n");
1384f44a1392SRobert Mustacchi 	}
1385f44a1392SRobert Mustacchi 
1386f44a1392SRobert Mustacchi 	if (tp.smbtp_minval != SMB_PROBE_UNKNOWN_VALUE) {
1387f44a1392SRobert Mustacchi 		oprintf(fp, "  Minimum Possible Temperature: %u.%u C\n",
1388f44a1392SRobert Mustacchi 		    tp.smbtp_minval / 10, tp.smbtp_minval % 10);
1389f44a1392SRobert Mustacchi 	} else {
1390f44a1392SRobert Mustacchi 		oprintf(fp, "  Minimum Possible Temperature: unknown\n");
1391f44a1392SRobert Mustacchi 	}
1392f44a1392SRobert Mustacchi 
1393f44a1392SRobert Mustacchi 	if (tp.smbtp_resolution != SMB_PROBE_UNKNOWN_VALUE) {
1394f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Resolution: %u.%03u C\n",
1395f44a1392SRobert Mustacchi 		    tp.smbtp_resolution / 1000,
1396f44a1392SRobert Mustacchi 		    tp.smbtp_resolution % 1000);
1397f44a1392SRobert Mustacchi 	} else {
1398f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Resolution: unknown\n");
1399f44a1392SRobert Mustacchi 	}
1400f44a1392SRobert Mustacchi 
1401f44a1392SRobert Mustacchi 	if (tp.smbtp_tolerance != SMB_PROBE_UNKNOWN_VALUE) {
1402f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: +/-%u.%u C\n",
1403f44a1392SRobert Mustacchi 		    tp.smbtp_tolerance / 10, tp.smbtp_tolerance % 10);
1404f44a1392SRobert Mustacchi 	} else {
1405f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: unknown\n");
1406f44a1392SRobert Mustacchi 	}
1407f44a1392SRobert Mustacchi 
1408f44a1392SRobert Mustacchi 	if (tp.smbtp_accuracy != SMB_PROBE_UNKNOWN_VALUE) {
1409f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: +/-%u.%02u%%\n",
1410f44a1392SRobert Mustacchi 		    tp.smbtp_accuracy / 100,
1411f44a1392SRobert Mustacchi 		    tp.smbtp_accuracy % 100);
1412f44a1392SRobert Mustacchi 	} else {
1413f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: unknown\n");
1414f44a1392SRobert Mustacchi 	}
1415f44a1392SRobert Mustacchi 
1416f44a1392SRobert Mustacchi 	oprintf(fp, "  OEM- or BIOS- defined value: 0x%x\n", tp.smbtp_oem);
1417f44a1392SRobert Mustacchi 
1418f44a1392SRobert Mustacchi 	if (tp.smbtp_nominal != SMB_PROBE_UNKNOWN_VALUE) {
1419f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: %u.%u C\n",
1420f44a1392SRobert Mustacchi 		    tp.smbtp_nominal / 10, tp.smbtp_nominal % 10);
1421f44a1392SRobert Mustacchi 	} else {
1422f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: unknown\n");
1423f44a1392SRobert Mustacchi 	}
1424f44a1392SRobert Mustacchi }
1425f44a1392SRobert Mustacchi 
1426f44a1392SRobert Mustacchi static void
print_iprobe(smbios_hdl_t * shp,id_t id,FILE * fp)1427f44a1392SRobert Mustacchi print_iprobe(smbios_hdl_t *shp, id_t id, FILE *fp)
1428f44a1392SRobert Mustacchi {
1429f44a1392SRobert Mustacchi 	smbios_iprobe_t ip;
1430f44a1392SRobert Mustacchi 
1431f44a1392SRobert Mustacchi 	if (smbios_info_iprobe(shp, id, &ip) != 0) {
1432f44a1392SRobert Mustacchi 		smbios_warn(shp, "failed to read current probe information");
1433f44a1392SRobert Mustacchi 		return;
1434f44a1392SRobert Mustacchi 	}
1435f44a1392SRobert Mustacchi 
14360f56e145SRobert Mustacchi 	str_print(fp, "  Description", ip.smbip_description != NULL ?
1437f44a1392SRobert Mustacchi 	    ip.smbip_description : "unknown");
1438f44a1392SRobert Mustacchi 	desc_printf(smbios_iprobe_loc_desc(ip.smbip_location),
1439f44a1392SRobert Mustacchi 	    fp, "  Location: %u", ip.smbip_location);
1440f44a1392SRobert Mustacchi 	desc_printf(smbios_iprobe_status_desc(ip.smbip_status),
1441f44a1392SRobert Mustacchi 	    fp, "  Status: %u", ip.smbip_status);
1442f44a1392SRobert Mustacchi 
1443f44a1392SRobert Mustacchi 	if (ip.smbip_maxval != SMB_PROBE_UNKNOWN_VALUE) {
1444f44a1392SRobert Mustacchi 		oprintf(fp, "  Maximum Possible Current: %u mA\n",
1445f44a1392SRobert Mustacchi 		    ip.smbip_maxval);
1446f44a1392SRobert Mustacchi 	} else {
1447f44a1392SRobert Mustacchi 		oprintf(fp, "  Maximum Possible Current: unknown\n");
1448f44a1392SRobert Mustacchi 	}
1449f44a1392SRobert Mustacchi 
1450f44a1392SRobert Mustacchi 	if (ip.smbip_minval != SMB_PROBE_UNKNOWN_VALUE) {
1451f44a1392SRobert Mustacchi 		oprintf(fp, "  Minimum Possible Current: %u mA\n",
1452f44a1392SRobert Mustacchi 		    ip.smbip_minval);
1453f44a1392SRobert Mustacchi 	} else {
1454f44a1392SRobert Mustacchi 		oprintf(fp, "  Minimum Possible Current: unknown\n");
1455f44a1392SRobert Mustacchi 	}
1456f44a1392SRobert Mustacchi 
1457f44a1392SRobert Mustacchi 	if (ip.smbip_resolution != SMB_PROBE_UNKNOWN_VALUE) {
1458f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Resolution: %u.%u mA\n",
1459f44a1392SRobert Mustacchi 		    ip.smbip_resolution / 10,
1460f44a1392SRobert Mustacchi 		    ip.smbip_resolution % 10);
1461f44a1392SRobert Mustacchi 	} else {
1462f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Resolution: unknown\n");
1463f44a1392SRobert Mustacchi 	}
1464f44a1392SRobert Mustacchi 
1465f44a1392SRobert Mustacchi 	if (ip.smbip_tolerance != SMB_PROBE_UNKNOWN_VALUE) {
1466f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: +/-%u mA\n",
1467f44a1392SRobert Mustacchi 		    ip.smbip_tolerance);
1468f44a1392SRobert Mustacchi 	} else {
1469f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Tolerance: unknown\n");
1470f44a1392SRobert Mustacchi 	}
1471f44a1392SRobert Mustacchi 
1472f44a1392SRobert Mustacchi 	if (ip.smbip_accuracy != SMB_PROBE_UNKNOWN_VALUE) {
1473f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: +/-%u.%02u%%\n",
1474f44a1392SRobert Mustacchi 		    ip.smbip_accuracy / 100,
1475f44a1392SRobert Mustacchi 		    ip.smbip_accuracy % 100);
1476f44a1392SRobert Mustacchi 	} else {
1477f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Accuracy: unknown\n");
1478f44a1392SRobert Mustacchi 	}
1479f44a1392SRobert Mustacchi 
1480f44a1392SRobert Mustacchi 	oprintf(fp, "  OEM- or BIOS- defined value: 0x%x\n", ip.smbip_oem);
1481f44a1392SRobert Mustacchi 
1482f44a1392SRobert Mustacchi 	if (ip.smbip_nominal != SMB_PROBE_UNKNOWN_VALUE) {
1483f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: %u mA\n", ip.smbip_nominal);
1484f44a1392SRobert Mustacchi 	} else {
1485f44a1392SRobert Mustacchi 		oprintf(fp, "  Probe Nominal Value: unknown\n");
1486f44a1392SRobert Mustacchi 	}
1487f44a1392SRobert Mustacchi }
1488f44a1392SRobert Mustacchi 
1489f44a1392SRobert Mustacchi static void
print_boot(smbios_hdl_t * shp,FILE * fp)149084ab085aSmws print_boot(smbios_hdl_t *shp, FILE *fp)
149184ab085aSmws {
149284ab085aSmws 	smbios_boot_t b;
149384ab085aSmws 
1494a7aaa513SRobert Mustacchi 	if (smbios_info_boot(shp, &b) == -1) {
1495a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read boot information");
1496a7aaa513SRobert Mustacchi 		return;
1497a7aaa513SRobert Mustacchi 	}
149884ab085aSmws 
149984ab085aSmws 	desc_printf(smbios_boot_desc(b.smbt_status),
150084ab085aSmws 	    fp, "  Boot Status Code: 0x%x", b.smbt_status);
150184ab085aSmws 
150284ab085aSmws 	if (b.smbt_size != 0) {
150384ab085aSmws 		oprintf(fp, "  Boot Data (%lu bytes):\n", (ulong_t)b.smbt_size);
150484ab085aSmws 		print_bytes(b.smbt_data, b.smbt_size, fp);
150584ab085aSmws 	}
150684ab085aSmws }
150784ab085aSmws 
150884ab085aSmws static void
print_ipmi(smbios_hdl_t * shp,FILE * fp)150984ab085aSmws print_ipmi(smbios_hdl_t *shp, FILE *fp)
151084ab085aSmws {
151184ab085aSmws 	smbios_ipmi_t i;
151284ab085aSmws 
1513a7aaa513SRobert Mustacchi 	if (smbios_info_ipmi(shp, &i) == -1) {
1514a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read ipmi information");
1515a7aaa513SRobert Mustacchi 		return;
1516a7aaa513SRobert Mustacchi 	}
151784ab085aSmws 
151884ab085aSmws 	desc_printf(smbios_ipmi_type_desc(i.smbip_type),
151984ab085aSmws 	    fp, "  Type: %u", i.smbip_type);
152084ab085aSmws 
152184ab085aSmws 	oprintf(fp, "  BMC IPMI Version: %u.%u\n",
152284ab085aSmws 	    i.smbip_vers.smbv_major, i.smbip_vers.smbv_minor);
152384ab085aSmws 
152484ab085aSmws 	oprintf(fp, "  i2c Bus Slave Address: 0x%x\n", i.smbip_i2c);
152584ab085aSmws 	oprintf(fp, "  NV Storage Device Bus ID: 0x%x\n", i.smbip_bus);
152684ab085aSmws 	oprintf(fp, "  BMC Base Address: 0x%llx\n", (u_longlong_t)i.smbip_addr);
152784ab085aSmws 	oprintf(fp, "  Interrupt Number: %u\n", i.smbip_intr);
152884ab085aSmws 	oprintf(fp, "  Register Spacing: %u\n", i.smbip_regspacing);
152984ab085aSmws 
153084ab085aSmws 	flag_printf(fp, "Flags", i.smbip_flags, sizeof (i.smbip_flags) * NBBY,
153184ab085aSmws 	    smbios_ipmi_flag_name, smbios_ipmi_flag_desc);
153284ab085aSmws }
153384ab085aSmws 
1534074bb90dSTom Pothier static void
print_powersup(smbios_hdl_t * shp,id_t id,FILE * fp)153532ece1f9SRobert Mustacchi print_powersup(smbios_hdl_t *shp, id_t id, FILE *fp)
153632ece1f9SRobert Mustacchi {
153732ece1f9SRobert Mustacchi 	smbios_powersup_t p;
153832ece1f9SRobert Mustacchi 
153932ece1f9SRobert Mustacchi 	if (smbios_info_powersup(shp, id, &p) != 0) {
154032ece1f9SRobert Mustacchi 		smbios_warn(shp, "failed to read power supply information");
154132ece1f9SRobert Mustacchi 		return;
154232ece1f9SRobert Mustacchi 	}
154332ece1f9SRobert Mustacchi 
154432ece1f9SRobert Mustacchi 	oprintf(fp, "  Power Supply Group: %u\n", p.smbps_group);
154532ece1f9SRobert Mustacchi 	if (p.smbps_maxout != 0x8000) {
1546*482d9a1aSRichard Lowe 		oprintf(fp, "  Maximum Output: %" PRIu64 " mW\n",
1547*482d9a1aSRichard Lowe 		    p.smbps_maxout);
154832ece1f9SRobert Mustacchi 	} else {
154932ece1f9SRobert Mustacchi 		oprintf(fp, "  Maximum Output: unknown\n");
155032ece1f9SRobert Mustacchi 	}
155132ece1f9SRobert Mustacchi 
155232ece1f9SRobert Mustacchi 	flag_printf(fp, "Characteristics", p.smbps_flags,
155332ece1f9SRobert Mustacchi 	    sizeof (p.smbps_flags) * NBBY, smbios_powersup_flag_name,
155432ece1f9SRobert Mustacchi 	    smbios_powersup_flag_desc);
155532ece1f9SRobert Mustacchi 
155632ece1f9SRobert Mustacchi 	desc_printf(smbios_powersup_input_desc(p.smbps_ivrs),
155732ece1f9SRobert Mustacchi 	    fp, "  Input Voltage Range Switching: %u", p.smbps_ivrs);
155832ece1f9SRobert Mustacchi 	desc_printf(smbios_powersup_status_desc(p.smbps_status),
155932ece1f9SRobert Mustacchi 	    fp, "  Status: %u", p.smbps_status);
156032ece1f9SRobert Mustacchi 	desc_printf(smbios_powersup_type_desc(p.smbps_pstype),
156132ece1f9SRobert Mustacchi 	    fp, "  Type: %u", p.smbps_pstype);
156232ece1f9SRobert Mustacchi 
156332ece1f9SRobert Mustacchi 	if (p.smbps_vprobe != 0xffff) {
1564*482d9a1aSRichard Lowe 		oprintf(fp, "  Voltage Probe Handle: %" _PRIuID "\n",
1565*482d9a1aSRichard Lowe 		    p.smbps_vprobe);
156632ece1f9SRobert Mustacchi 	}
156732ece1f9SRobert Mustacchi 
156832ece1f9SRobert Mustacchi 	if (p.smbps_cooldev != 0xffff) {
1569*482d9a1aSRichard Lowe 		oprintf(fp, "  Cooling Device Handle: %" _PRIuID "\n",
1570*482d9a1aSRichard Lowe 		    p.smbps_cooldev);
157132ece1f9SRobert Mustacchi 	}
157232ece1f9SRobert Mustacchi 
157332ece1f9SRobert Mustacchi 	if (p.smbps_iprobe != 0xffff) {
1574*482d9a1aSRichard Lowe 		oprintf(fp, "  Current Probe Handle: %" _PRIuID "\n",
1575*482d9a1aSRichard Lowe 		    p.smbps_iprobe);
157632ece1f9SRobert Mustacchi 	}
157732ece1f9SRobert Mustacchi }
157832ece1f9SRobert Mustacchi 
157932ece1f9SRobert Mustacchi static void
print_addinfo(smbios_hdl_t * shp,id_t id,FILE * fp)1580064d431aSRobert Mustacchi print_addinfo(smbios_hdl_t *shp, id_t id, FILE *fp)
1581064d431aSRobert Mustacchi {
1582064d431aSRobert Mustacchi 	uint_t nents, i;
1583064d431aSRobert Mustacchi 
1584064d431aSRobert Mustacchi 	if (smbios_info_addinfo_nents(shp, id, &nents) != 0) {
1585064d431aSRobert Mustacchi 		smbios_warn(shp, "failed to read additional information");
1586064d431aSRobert Mustacchi 		return;
1587064d431aSRobert Mustacchi 	}
1588064d431aSRobert Mustacchi 
1589064d431aSRobert Mustacchi 	oprintf(fp, "  Number of Additional Information Entries: %u\n", nents);
1590064d431aSRobert Mustacchi 	for (i = 0; i < nents; i++) {
1591064d431aSRobert Mustacchi 		smbios_addinfo_ent_t *ent;
1592064d431aSRobert Mustacchi 
1593064d431aSRobert Mustacchi 		oprintf(fp, "  Additional Information Entry %u\n", i);
1594064d431aSRobert Mustacchi 		if (smbios_info_addinfo_ent(shp, id, i, &ent) != 0) {
1595064d431aSRobert Mustacchi 			smbios_warn(shp, "failed to read additional "
1596064d431aSRobert Mustacchi 			    "information entry %u", i);
1597064d431aSRobert Mustacchi 			continue;
1598064d431aSRobert Mustacchi 		}
1599064d431aSRobert Mustacchi 
1600*482d9a1aSRichard Lowe 		oprintf(fp, "    Referenced handle: %" _PRIuID "\n",
1601*482d9a1aSRichard Lowe 		    ent->smbai_ref);
1602064d431aSRobert Mustacchi 		oprintf(fp, "    Handle offset: %u\n", ent->smbai_ref_off);
1603064d431aSRobert Mustacchi 		if (ent->smbai_str != NULL) {
1604064d431aSRobert Mustacchi 			str_print(fp, "    Information String", ent->smbai_str);
1605064d431aSRobert Mustacchi 		}
1606064d431aSRobert Mustacchi 
1607064d431aSRobert Mustacchi 		/*
1608064d431aSRobert Mustacchi 		 * As of SMBIOS 3.7, there are no extra data entries strictly
1609064d431aSRobert Mustacchi 		 * defined in the spec, but there may be something. If we find
1610064d431aSRobert Mustacchi 		 * something that's a standard integer size, then we'll
1611064d431aSRobert Mustacchi 		 * interpret it and print it as a hex value. In theory this is
1612064d431aSRobert Mustacchi 		 * supposed to refer back to some field, but hard to say how
1613064d431aSRobert Mustacchi 		 * this'll actually be used. The first time we encountered it
1614064d431aSRobert Mustacchi 		 * was just an additional string entry.
1615064d431aSRobert Mustacchi 		 */
1616064d431aSRobert Mustacchi 		if (ent->smbai_dlen > 0) {
1617064d431aSRobert Mustacchi 			oprintf(fp, "    Data Length: %u\n", ent->smbai_dlen);
1618064d431aSRobert Mustacchi 			switch (ent->smbai_dlen) {
1619064d431aSRobert Mustacchi 			case 1:
1620064d431aSRobert Mustacchi 				oprintf(fp, "    Data: 0x%x\n",
1621064d431aSRobert Mustacchi 				    *(uint8_t *)ent->smbai_data);
1622064d431aSRobert Mustacchi 				break;
1623064d431aSRobert Mustacchi 			case 2:
1624064d431aSRobert Mustacchi 				oprintf(fp, "    Data: 0x%x\n",
1625064d431aSRobert Mustacchi 				    *(uint16_t *)ent->smbai_data);
1626064d431aSRobert Mustacchi 				break;
1627064d431aSRobert Mustacchi 			case 4:
1628064d431aSRobert Mustacchi 				oprintf(fp, "    Data: 0x%x\n",
1629064d431aSRobert Mustacchi 				    *(uint32_t *)ent->smbai_data);
1630064d431aSRobert Mustacchi 				break;
1631064d431aSRobert Mustacchi 			case 8:
1632*482d9a1aSRichard Lowe 				oprintf(fp, "    Data: 0x%" PRIx64 "\n",
1633064d431aSRobert Mustacchi 				    *(uint64_t *)ent->smbai_data);
1634064d431aSRobert Mustacchi 				break;
1635064d431aSRobert Mustacchi 			default:
1636064d431aSRobert Mustacchi 				break;
1637064d431aSRobert Mustacchi 			}
1638064d431aSRobert Mustacchi 		}
1639064d431aSRobert Mustacchi 
1640064d431aSRobert Mustacchi 		smbios_info_addinfo_ent_free(shp, ent);
1641064d431aSRobert Mustacchi 	}
1642064d431aSRobert Mustacchi }
1643064d431aSRobert Mustacchi 
1644064d431aSRobert Mustacchi 
1645064d431aSRobert Mustacchi static void
print_processor_info_riscv(smbios_hdl_t * shp,id_t id,FILE * fp)1646176a9270SRobert Mustacchi print_processor_info_riscv(smbios_hdl_t *shp, id_t id, FILE *fp)
1647176a9270SRobert Mustacchi {
1648176a9270SRobert Mustacchi 	smbios_processor_info_riscv_t rv;
1649176a9270SRobert Mustacchi 
1650176a9270SRobert Mustacchi 	if (smbios_info_processor_riscv(shp, id, &rv) != 0) {
1651176a9270SRobert Mustacchi 		smbios_warn(shp, "failed to read RISC-V specific processor "
1652176a9270SRobert Mustacchi 		    "information");
1653176a9270SRobert Mustacchi 		return;
1654176a9270SRobert Mustacchi 	}
1655176a9270SRobert Mustacchi 
1656176a9270SRobert Mustacchi 	if (rv.smbpirv_boothart != 0) {
1657176a9270SRobert Mustacchi 		oprintf(fp, "    Boot Hart\n");
1658176a9270SRobert Mustacchi 	}
1659176a9270SRobert Mustacchi 	u128_print(fp, "    Hart ID", rv.smbpirv_hartid);
1660176a9270SRobert Mustacchi 	u128_print(fp, "    Vendor ID", rv.smbpirv_vendid);
1661176a9270SRobert Mustacchi 	u128_print(fp, "    Architecture ID", rv.smbpirv_archid);
1662176a9270SRobert Mustacchi 	u128_print(fp, "    Implementation ID", rv.smbpirv_machid);
1663176a9270SRobert Mustacchi 	flag64_printf(fp, "  ISA", rv.smbpirv_isa,
1664176a9270SRobert Mustacchi 	    sizeof (rv.smbpirv_isa) * NBBY, smbios_riscv_isa_name,
1665176a9270SRobert Mustacchi 	    smbios_riscv_isa_desc);
1666176a9270SRobert Mustacchi 	flag_printf(fp, "  Privilege Levels", rv.smbpirv_privlvl,
1667176a9270SRobert Mustacchi 	    sizeof (rv.smbpirv_privlvl) * NBBY, smbios_riscv_priv_name,
1668176a9270SRobert Mustacchi 	    smbios_riscv_priv_desc);
1669176a9270SRobert Mustacchi 	u128_print(fp, "    Machine Exception Trap Delegation",
1670176a9270SRobert Mustacchi 	    rv.smbpirv_metdi);
1671176a9270SRobert Mustacchi 	u128_print(fp, "    Machine Interrupt Trap Delegation",
1672176a9270SRobert Mustacchi 	    rv.smbpirv_mitdi);
1673176a9270SRobert Mustacchi 	desc_printf(smbios_riscv_width_desc(rv.smbpirv_xlen),
1674176a9270SRobert Mustacchi 	    fp, "    Register Width: 0x%x", rv.smbpirv_xlen);
1675176a9270SRobert Mustacchi 	desc_printf(smbios_riscv_width_desc(rv.smbpirv_mxlen),
1676176a9270SRobert Mustacchi 	    fp, "    M-Mode Register Width: 0x%x", rv.smbpirv_mxlen);
1677176a9270SRobert Mustacchi 	desc_printf(smbios_riscv_width_desc(rv.smbpirv_sxlen),
1678176a9270SRobert Mustacchi 	    fp, "    S-Mode Register Width: 0x%x", rv.smbpirv_sxlen);
1679176a9270SRobert Mustacchi 	desc_printf(smbios_riscv_width_desc(rv.smbpirv_uxlen),
1680176a9270SRobert Mustacchi 	    fp, "    U-Mode Register Width: 0x%x", rv.smbpirv_uxlen);
1681176a9270SRobert Mustacchi }
1682176a9270SRobert Mustacchi 
1683176a9270SRobert Mustacchi static void
print_processor_info(smbios_hdl_t * shp,id_t id,FILE * fp)1684176a9270SRobert Mustacchi print_processor_info(smbios_hdl_t *shp, id_t id, FILE *fp)
1685176a9270SRobert Mustacchi {
1686176a9270SRobert Mustacchi 	smbios_processor_info_t p;
1687176a9270SRobert Mustacchi 
1688176a9270SRobert Mustacchi 	if (smbios_info_processor_info(shp, id, &p) != 0) {
1689176a9270SRobert Mustacchi 		smbios_warn(shp, "failed to read processor additional "
1690176a9270SRobert Mustacchi 		    "information");
1691176a9270SRobert Mustacchi 		return;
1692176a9270SRobert Mustacchi 	}
1693176a9270SRobert Mustacchi 
1694176a9270SRobert Mustacchi 	id_printf(fp, "  Processor Handle: ", p.smbpi_processor);
1695176a9270SRobert Mustacchi 	desc_printf(smbios_processor_info_type_desc(p.smbpi_ptype),
1696176a9270SRobert Mustacchi 	    fp, "  Processor Type: %u", p.smbpi_ptype);
1697176a9270SRobert Mustacchi 
1698176a9270SRobert Mustacchi 	switch (p.smbpi_ptype) {
1699176a9270SRobert Mustacchi 	case SMB_PROCINFO_T_RV32:
1700176a9270SRobert Mustacchi 	case SMB_PROCINFO_T_RV64:
1701176a9270SRobert Mustacchi 	case SMB_PROCINFO_T_RV128:
1702176a9270SRobert Mustacchi 		oprintf(fp, "  RISC-V Additional Processor Information:\n");
1703176a9270SRobert Mustacchi 		print_processor_info_riscv(shp, id, fp);
1704176a9270SRobert Mustacchi 		break;
1705176a9270SRobert Mustacchi 	default:
1706176a9270SRobert Mustacchi 		break;
1707176a9270SRobert Mustacchi 	}
1708176a9270SRobert Mustacchi }
1709176a9270SRobert Mustacchi 
1710176a9270SRobert Mustacchi static void
print_battery(smbios_hdl_t * shp,id_t id,FILE * fp)171146782190SRobert Mustacchi print_battery(smbios_hdl_t *shp, id_t id, FILE *fp)
171246782190SRobert Mustacchi {
171346782190SRobert Mustacchi 	smbios_battery_t bat;
171446782190SRobert Mustacchi 
171546782190SRobert Mustacchi 	if (smbios_info_battery(shp, id, &bat) != 0) {
171646782190SRobert Mustacchi 		smbios_warn(shp, "failed to read battery information");
171746782190SRobert Mustacchi 		return;
171846782190SRobert Mustacchi 	}
171946782190SRobert Mustacchi 
172046782190SRobert Mustacchi 	if (bat.smbb_date != NULL) {
17210f56e145SRobert Mustacchi 		str_print(fp, "  Manufacture Date", bat.smbb_date);
172246782190SRobert Mustacchi 	}
172346782190SRobert Mustacchi 
172446782190SRobert Mustacchi 	if (bat.smbb_serial != NULL) {
17250f56e145SRobert Mustacchi 		str_print(fp, "  Serial Number", bat.smbb_serial);
172646782190SRobert Mustacchi 	}
172746782190SRobert Mustacchi 
172846782190SRobert Mustacchi 	if (bat.smbb_chem != SMB_BDC_UNKNOWN) {
172946782190SRobert Mustacchi 		desc_printf(smbios_battery_chem_desc(bat.smbb_chem),
173046782190SRobert Mustacchi 		    fp, "  Battery Chemistry: 0x%x", bat.smbb_chem);
173146782190SRobert Mustacchi 	}
173246782190SRobert Mustacchi 
173346782190SRobert Mustacchi 	if (bat.smbb_cap != 0) {
173446782190SRobert Mustacchi 		oprintf(fp, "  Design Capacity: %u mWh\n", bat.smbb_cap);
173546782190SRobert Mustacchi 	} else {
173646782190SRobert Mustacchi 		oprintf(fp, "  Design Capacity: unknown\n");
173746782190SRobert Mustacchi 	}
173846782190SRobert Mustacchi 
173946782190SRobert Mustacchi 	if (bat.smbb_volt != 0) {
174046782190SRobert Mustacchi 		oprintf(fp, "  Design Voltage: %u mV\n", bat.smbb_volt);
174146782190SRobert Mustacchi 	} else {
174246782190SRobert Mustacchi 		oprintf(fp, "  Design Voltage: unknown\n");
174346782190SRobert Mustacchi 	}
174446782190SRobert Mustacchi 
17450f56e145SRobert Mustacchi 	str_print(fp, "  SBDS Version Number", bat.smbb_version);
174646782190SRobert Mustacchi 	if (bat.smbb_err != UINT8_MAX) {
174746782190SRobert Mustacchi 		oprintf(fp, "  Maximum Error: %u\n", bat.smbb_err);
174846782190SRobert Mustacchi 	} else {
174943379a28SAndy Fiddaman 		oprintf(fp, "  Maximum Error: unknown\n");
175046782190SRobert Mustacchi 	}
175146782190SRobert Mustacchi 	oprintf(fp, "  SBDS Serial Number: %04x\n", bat.smbb_ssn);
175246782190SRobert Mustacchi 	oprintf(fp, "  SBDS Manufacture Date: %u-%02u-%02u\n", bat.smbb_syear,
175346782190SRobert Mustacchi 	    bat.smbb_smonth, bat.smbb_sday);
17540f56e145SRobert Mustacchi 	str_print(fp, "  SBDS Device Chemistry", bat.smbb_schem);
175546782190SRobert Mustacchi 	oprintf(fp, "  OEM-specific Information: 0x%08x\n", bat.smbb_oemdata);
175646782190SRobert Mustacchi }
175746782190SRobert Mustacchi 
175846782190SRobert Mustacchi static void
print_pointdev(smbios_hdl_t * shp,id_t id,FILE * fp)17591d77dcdaSRobert Mustacchi print_pointdev(smbios_hdl_t *shp, id_t id, FILE *fp)
17601d77dcdaSRobert Mustacchi {
17611d77dcdaSRobert Mustacchi 	smbios_pointdev_t pd;
17621d77dcdaSRobert Mustacchi 
17631d77dcdaSRobert Mustacchi 	if (smbios_info_pointdev(shp, id, &pd) != 0) {
17641d77dcdaSRobert Mustacchi 		smbios_warn(shp, "failed to read pointer device information");
17651d77dcdaSRobert Mustacchi 		return;
17661d77dcdaSRobert Mustacchi 	}
17671d77dcdaSRobert Mustacchi 
17681d77dcdaSRobert Mustacchi 	desc_printf(smbios_pointdev_type_desc(pd.smbpd_type),
17691d77dcdaSRobert Mustacchi 	    fp, "  Type: %u", pd.smbpd_type);
17701d77dcdaSRobert Mustacchi 	desc_printf(smbios_pointdev_iface_desc(pd.smbpd_iface),
17711d77dcdaSRobert Mustacchi 	    fp, "  Interface: %u", pd.smbpd_iface);
17721d77dcdaSRobert Mustacchi 	oprintf(fp, "  Buttons: %u\n", pd.smbpd_nbuttons);
17731d77dcdaSRobert Mustacchi }
17741d77dcdaSRobert Mustacchi 
17751d77dcdaSRobert Mustacchi static void
print_extprocessor(smbios_hdl_t * shp,id_t id,FILE * fp)1776074bb90dSTom Pothier print_extprocessor(smbios_hdl_t *shp, id_t id, FILE *fp)
1777074bb90dSTom Pothier {
1778074bb90dSTom Pothier 	int i;
1779074bb90dSTom Pothier 	smbios_processor_ext_t ep;
1780074bb90dSTom Pothier 
1781074bb90dSTom Pothier 	if (check_oem(shp) != 0)
1782074bb90dSTom Pothier 		return;
1783074bb90dSTom Pothier 
1784a7aaa513SRobert Mustacchi 	if (smbios_info_extprocessor(shp, id, &ep) != 0) {
1785a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read extended processor "
1786a7aaa513SRobert Mustacchi 		    "information");
1787a7aaa513SRobert Mustacchi 		return;
1788a7aaa513SRobert Mustacchi 	}
1789074bb90dSTom Pothier 
1790074bb90dSTom Pothier 	oprintf(fp, "  Processor: %u\n", ep.smbpe_processor);
1791074bb90dSTom Pothier 	oprintf(fp, "  FRU: %u\n", ep.smbpe_fru);
1792074bb90dSTom Pothier 	oprintf(fp, "  Initial APIC ID count: %u\n\n", ep.smbpe_n);
1793074bb90dSTom Pothier 
1794074bb90dSTom Pothier 	for (i = 0; i < ep.smbpe_n; i++) {
1795074bb90dSTom Pothier 		oprintf(fp, "  Logical Strand %u: Initial APIC ID: %u\n", i,
1796074bb90dSTom Pothier 		    ep.smbpe_apicid[i]);
1797074bb90dSTom Pothier 	}
1798074bb90dSTom Pothier }
1799074bb90dSTom Pothier 
1800074bb90dSTom Pothier static void
print_extport(smbios_hdl_t * shp,id_t id,FILE * fp)180103f9f63dSTom Pothier print_extport(smbios_hdl_t *shp, id_t id, FILE *fp)
180203f9f63dSTom Pothier {
180303f9f63dSTom Pothier 	smbios_port_ext_t epo;
180403f9f63dSTom Pothier 
180503f9f63dSTom Pothier 	if (check_oem(shp) != 0)
180603f9f63dSTom Pothier 		return;
180703f9f63dSTom Pothier 
1808a7aaa513SRobert Mustacchi 	if (smbios_info_extport(shp, id, &epo) != 0) {
1809a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read extended port information");
1810a7aaa513SRobert Mustacchi 		return;
1811a7aaa513SRobert Mustacchi 	}
181203f9f63dSTom Pothier 
181303f9f63dSTom Pothier 	oprintf(fp, "  Chassis Handle: %u\n", epo.smbporte_chassis);
181403f9f63dSTom Pothier 	oprintf(fp, "  Port Connector Handle: %u\n", epo.smbporte_port);
181503f9f63dSTom Pothier 	oprintf(fp, "  Device Type: %u\n", epo.smbporte_dtype);
181603f9f63dSTom Pothier 	oprintf(fp, "  Device Handle: %u\n", epo.smbporte_devhdl);
181703f9f63dSTom Pothier 	oprintf(fp, "  PHY: %u\n", epo.smbporte_phy);
181803f9f63dSTom Pothier }
181903f9f63dSTom Pothier 
182003f9f63dSTom Pothier static void
print_pciexrc(smbios_hdl_t * shp,id_t id,FILE * fp)1821074bb90dSTom Pothier print_pciexrc(smbios_hdl_t *shp, id_t id, FILE *fp)
1822074bb90dSTom Pothier {
1823074bb90dSTom Pothier 	smbios_pciexrc_t pcie;
1824074bb90dSTom Pothier 
1825074bb90dSTom Pothier 	if (check_oem(shp) != 0)
1826074bb90dSTom Pothier 		return;
1827074bb90dSTom Pothier 
1828a7aaa513SRobert Mustacchi 	if (smbios_info_pciexrc(shp, id, &pcie) != 0) {
1829a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read pciexrc information");
1830a7aaa513SRobert Mustacchi 		return;
1831a7aaa513SRobert Mustacchi 	}
1832074bb90dSTom Pothier 
1833074bb90dSTom Pothier 	oprintf(fp, "  Component ID: %u\n", pcie.smbpcie_bb);
1834074bb90dSTom Pothier 	oprintf(fp, "  BDF: 0x%x\n", pcie.smbpcie_bdf);
1835074bb90dSTom Pothier }
1836074bb90dSTom Pothier 
1837074bb90dSTom Pothier static void
print_extmemarray(smbios_hdl_t * shp,id_t id,FILE * fp)1838074bb90dSTom Pothier print_extmemarray(smbios_hdl_t *shp, id_t id, FILE *fp)
1839074bb90dSTom Pothier {
1840074bb90dSTom Pothier 	smbios_memarray_ext_t em;
1841074bb90dSTom Pothier 
1842074bb90dSTom Pothier 	if (check_oem(shp) != 0)
1843074bb90dSTom Pothier 		return;
1844074bb90dSTom Pothier 
1845a7aaa513SRobert Mustacchi 	if (smbios_info_extmemarray(shp, id, &em) != 0) {
1846a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read extmemarray information");
1847a7aaa513SRobert Mustacchi 		return;
1848a7aaa513SRobert Mustacchi 	}
1849074bb90dSTom Pothier 
1850074bb90dSTom Pothier 	oprintf(fp, "  Physical Memory Array Handle: %u\n", em.smbmae_ma);
1851074bb90dSTom Pothier 	oprintf(fp, "  Component Parent Handle: %u\n", em.smbmae_comp);
1852074bb90dSTom Pothier 	oprintf(fp, "  BDF: 0x%x\n", em.smbmae_bdf);
1853074bb90dSTom Pothier }
1854074bb90dSTom Pothier 
1855074bb90dSTom Pothier static void
print_extmemdevice(smbios_hdl_t * shp,id_t id,FILE * fp)1856074bb90dSTom Pothier print_extmemdevice(smbios_hdl_t *shp, id_t id, FILE *fp)
1857074bb90dSTom Pothier {
1858679a141eSToomas Soome 	uint_t i, ncs;
1859679a141eSToomas Soome 	uint8_t *cs;
1860074bb90dSTom Pothier 	smbios_memdevice_ext_t emd;
1861074bb90dSTom Pothier 
1862074bb90dSTom Pothier 	if (check_oem(shp) != 0)
1863074bb90dSTom Pothier 		return;
1864074bb90dSTom Pothier 
1865a7aaa513SRobert Mustacchi 	if (smbios_info_extmemdevice(shp, id, &emd) != 0) {
1866a7aaa513SRobert Mustacchi 		smbios_warn(shp, "failed to read extmemdevice information");
1867a7aaa513SRobert Mustacchi 		return;
1868a7aaa513SRobert Mustacchi 	}
1869074bb90dSTom Pothier 
1870074bb90dSTom Pothier 	oprintf(fp, "  Memory Device Handle: %u\n", emd.smbmdeve_md);
1871074bb90dSTom Pothier 	oprintf(fp, "  DRAM Channel: %u\n", emd.smbmdeve_drch);
1872074bb90dSTom Pothier 	oprintf(fp, "  Number of Chip Selects: %u\n", emd.smbmdeve_ncs);
1873074bb90dSTom Pothier 
1874679a141eSToomas Soome 	if (emd.smbmdeve_ncs == 0)
1875679a141eSToomas Soome 		return;
1876679a141eSToomas Soome 
1877679a141eSToomas Soome 	if (smbios_info_extmemdevice_cs(shp, id, &ncs, &cs) != 0) {
1878679a141eSToomas Soome 		smbios_warn(shp, "failed to read extmemdevice cs information");
1879679a141eSToomas Soome 		return;
1880074bb90dSTom Pothier 	}
1881679a141eSToomas Soome 
1882679a141eSToomas Soome 	for (i = 0; i < ncs; i++) {
1883679a141eSToomas Soome 		oprintf(fp, "  Chip Select: %u\n", cs[i]);
1884679a141eSToomas Soome 	}
1885679a141eSToomas Soome 	smbios_info_extmemdevice_cs_free(shp, ncs, cs);
1886074bb90dSTom Pothier }
1887074bb90dSTom Pothier 
1888d53cdfabSRobert Mustacchi static void
print_strprop_info(smbios_hdl_t * shp,id_t id,FILE * fp)1889d53cdfabSRobert Mustacchi print_strprop_info(smbios_hdl_t *shp, id_t id, FILE *fp)
1890d53cdfabSRobert Mustacchi {
1891d53cdfabSRobert Mustacchi 	smbios_strprop_t prop;
1892d53cdfabSRobert Mustacchi 
1893d53cdfabSRobert Mustacchi 	if (smbios_info_strprop(shp, id, &prop) != 0) {
1894d53cdfabSRobert Mustacchi 		smbios_warn(shp, "failed to read string property information");
1895d53cdfabSRobert Mustacchi 		return;
1896d53cdfabSRobert Mustacchi 	}
1897d53cdfabSRobert Mustacchi 
1898d53cdfabSRobert Mustacchi 	desc_printf(smbios_strprop_id_desc(prop.smbsp_prop_id), fp,
1899d53cdfabSRobert Mustacchi 	    "  Property ID: %u", prop.smbsp_prop_id);
1900d53cdfabSRobert Mustacchi 	if (prop.smbsp_prop_val != NULL) {
1901d53cdfabSRobert Mustacchi 		str_print(fp, "  Property Value", prop.smbsp_prop_val);
1902d53cdfabSRobert Mustacchi 	}
1903d53cdfabSRobert Mustacchi 	id_printf(fp, "  Parent Handle: ", prop.smbsp_parent);
1904d53cdfabSRobert Mustacchi }
1905d53cdfabSRobert Mustacchi 
1906d53cdfabSRobert Mustacchi static void
print_fwinfo(smbios_hdl_t * shp,id_t id,FILE * fp)1907d53cdfabSRobert Mustacchi print_fwinfo(smbios_hdl_t *shp, id_t id, FILE *fp)
1908d53cdfabSRobert Mustacchi {
1909d53cdfabSRobert Mustacchi 	smbios_fwinfo_t fw;
1910d53cdfabSRobert Mustacchi 	smbios_fwinfo_comp_t *comps;
1911d53cdfabSRobert Mustacchi 	uint_t ncomps, i;
1912d53cdfabSRobert Mustacchi 
1913d53cdfabSRobert Mustacchi 	if (smbios_info_fwinfo(shp, id, &fw) != 0) {
1914d53cdfabSRobert Mustacchi 		smbios_warn(shp, "failed to read firmware inventory");
1915d53cdfabSRobert Mustacchi 		return;
1916d53cdfabSRobert Mustacchi 	}
1917d53cdfabSRobert Mustacchi 
1918d53cdfabSRobert Mustacchi 	str_print(fp, "  Component Name", fw.smbfw_name);
1919d53cdfabSRobert Mustacchi 	str_print(fp, "  ID", fw.smbfw_id);
1920d53cdfabSRobert Mustacchi 	str_print(fp, "  Release Date", fw.smbfw_reldate);
1921d53cdfabSRobert Mustacchi 	str_print(fp, "  Lowest Supported Version", fw.smbfw_lsv);
1922d53cdfabSRobert Mustacchi 	desc_printf(smbios_fwinfo_vers_desc(fw.smbfw_vers_fmt), fp,
1923d53cdfabSRobert Mustacchi 	    "  Version Format: %u", fw.smbfw_vers_fmt);
1924d53cdfabSRobert Mustacchi 	desc_printf(smbios_fwinfo_id_desc(fw.smbfw_id_fmt), fp,
1925d53cdfabSRobert Mustacchi 	    "  ID Format: %u", fw.smbfw_id_fmt);
1926d53cdfabSRobert Mustacchi 	if (fw.smbfw_imgsz != UINT64_MAX) {
1927d53cdfabSRobert Mustacchi 		oprintf(fp, "  Image Size: %" PRIu64 "\n", fw.smbfw_imgsz);
1928d53cdfabSRobert Mustacchi 	} else {
1929d53cdfabSRobert Mustacchi 		oprintf(fp, "  Image Size: unknown\n");
1930d53cdfabSRobert Mustacchi 	}
1931d53cdfabSRobert Mustacchi 
1932d53cdfabSRobert Mustacchi 	flag_printf(fp, "Characteristics", fw.smbfw_chars,
1933d53cdfabSRobert Mustacchi 	    sizeof (fw.smbfw_chars) * NBBY, smbios_fwinfo_ch_name,
1934d53cdfabSRobert Mustacchi 	    smbios_fwinfo_ch_desc);
1935d53cdfabSRobert Mustacchi 
1936d53cdfabSRobert Mustacchi 	desc_printf(smbios_fwinfo_state_desc(fw.smbfw_state), fp, "  State: %u",
1937d53cdfabSRobert Mustacchi 	    fw.smbfw_state);
1938d53cdfabSRobert Mustacchi 	oprintf(fp, "  Number of Associated Components: %u\n",
1939d53cdfabSRobert Mustacchi 	    fw.smbfw_ncomps);
1940d53cdfabSRobert Mustacchi 
1941d53cdfabSRobert Mustacchi 	if (fw.smbfw_ncomps == 0)
1942d53cdfabSRobert Mustacchi 		return;
1943d53cdfabSRobert Mustacchi 
1944d53cdfabSRobert Mustacchi 	if (smbios_info_fwinfo_comps(shp, id, &ncomps, &comps) == -1) {
1945d53cdfabSRobert Mustacchi 		smbios_warn(shp, "failed to read firmware inventory "
1946d53cdfabSRobert Mustacchi 		    "components");
1947d53cdfabSRobert Mustacchi 		return;
1948d53cdfabSRobert Mustacchi 	}
1949d53cdfabSRobert Mustacchi 
1950d53cdfabSRobert Mustacchi 	oprintf(fp, "\n  Component Handles:\n");
1951d53cdfabSRobert Mustacchi 	for (i = 0; i < ncomps; i++) {
1952*482d9a1aSRichard Lowe 		oprintf(fp, "    %" _PRIdID "\n", comps[i].smbfwe_id);
1953d53cdfabSRobert Mustacchi 	}
1954d53cdfabSRobert Mustacchi }
1955d53cdfabSRobert Mustacchi 
1956074bb90dSTom Pothier static int
print_struct(smbios_hdl_t * shp,const smbios_struct_t * sp,void * fp)195784ab085aSmws print_struct(smbios_hdl_t *shp, const smbios_struct_t *sp, void *fp)
195884ab085aSmws {
195984ab085aSmws 	smbios_info_t info;
196084ab085aSmws 	int hex = opt_x;
196184ab085aSmws 	const char *s;
196284ab085aSmws 
196384ab085aSmws 	if (opt_t != -1 && opt_t != sp->smbstr_type)
196484ab085aSmws 		return (0); /* skip struct if type doesn't match -t */
196584ab085aSmws 
196684ab085aSmws 	if (!opt_O && (sp->smbstr_type == SMB_TYPE_MEMCTL ||
196784ab085aSmws 	    sp->smbstr_type == SMB_TYPE_MEMMOD))
196884ab085aSmws 		return (0); /* skip struct if type is obsolete */
196984ab085aSmws 
197084ab085aSmws 	if (g_hdr++ == 0 || !opt_s)
197184ab085aSmws 		oprintf(fp, "%-5s %-4s %s\n", "ID", "SIZE", "TYPE");
197284ab085aSmws 
197384ab085aSmws 	oprintf(fp, "%-5u %-4lu",
197484ab085aSmws 	    (uint_t)sp->smbstr_id, (ulong_t)sp->smbstr_size);
197584ab085aSmws 
197684ab085aSmws 	if ((s = smbios_type_name(sp->smbstr_type)) != NULL)
197795ae7286SDan McDonald 		oprintf(fp, " %s (type %u)", s, sp->smbstr_type);
197884ab085aSmws 	else if (sp->smbstr_type > SMB_TYPE_OEM_LO &&
197984ab085aSmws 	    sp->smbstr_type < SMB_TYPE_OEM_HI)
198095ae7286SDan McDonald 		oprintf(fp, " %s+%u (type %u)", "SMB_TYPE_OEM_LO",
198195ae7286SDan McDonald 		    sp->smbstr_type - SMB_TYPE_OEM_LO, sp->smbstr_type);
198284ab085aSmws 	else
198384ab085aSmws 		oprintf(fp, " %u", sp->smbstr_type);
198484ab085aSmws 
198584ab085aSmws 	if ((s = smbios_type_desc(sp->smbstr_type)) != NULL)
198684ab085aSmws 		oprintf(fp, " (%s)\n", s);
198784ab085aSmws 	else
198884ab085aSmws 		oprintf(fp, "\n");
198984ab085aSmws 
199084ab085aSmws 	if (opt_s)
199184ab085aSmws 		return (0); /* only print header line if -s specified */
199284ab085aSmws 
199384ab085aSmws 	if (smbios_info_common(shp, sp->smbstr_id, &info) == 0) {
199484ab085aSmws 		oprintf(fp, "\n");
199584ab085aSmws 		print_common(&info, fp);
199684ab085aSmws 	}
199784ab085aSmws 
199884ab085aSmws 	switch (sp->smbstr_type) {
199984ab085aSmws 	case SMB_TYPE_BIOS:
200084ab085aSmws 		oprintf(fp, "\n");
200184ab085aSmws 		print_bios(shp, fp);
200284ab085aSmws 		break;
200384ab085aSmws 	case SMB_TYPE_SYSTEM:
200484ab085aSmws 		oprintf(fp, "\n");
200584ab085aSmws 		print_system(shp, fp);
200684ab085aSmws 		break;
200784ab085aSmws 	case SMB_TYPE_BASEBOARD:
200884ab085aSmws 		oprintf(fp, "\n");
200984ab085aSmws 		print_bboard(shp, sp->smbstr_id, fp);
201084ab085aSmws 		break;
201184ab085aSmws 	case SMB_TYPE_CHASSIS:
201284ab085aSmws 		oprintf(fp, "\n");
201384ab085aSmws 		print_chassis(shp, sp->smbstr_id, fp);
201484ab085aSmws 		break;
201584ab085aSmws 	case SMB_TYPE_PROCESSOR:
201684ab085aSmws 		oprintf(fp, "\n");
201784ab085aSmws 		print_processor(shp, sp->smbstr_id, fp);
201884ab085aSmws 		break;
201984ab085aSmws 	case SMB_TYPE_CACHE:
202084ab085aSmws 		oprintf(fp, "\n");
202184ab085aSmws 		print_cache(shp, sp->smbstr_id, fp);
202284ab085aSmws 		break;
202384ab085aSmws 	case SMB_TYPE_PORT:
202484ab085aSmws 		oprintf(fp, "\n");
202584ab085aSmws 		print_port(shp, sp->smbstr_id, fp);
202684ab085aSmws 		break;
202784ab085aSmws 	case SMB_TYPE_SLOT:
202884ab085aSmws 		oprintf(fp, "\n");
202984ab085aSmws 		print_slot(shp, sp->smbstr_id, fp);
203084ab085aSmws 		break;
203184ab085aSmws 	case SMB_TYPE_OBDEVS:
203284ab085aSmws 		oprintf(fp, "\n");
203384ab085aSmws 		print_obdevs(shp, sp->smbstr_id, fp);
203484ab085aSmws 		break;
203584ab085aSmws 	case SMB_TYPE_OEMSTR:
203684ab085aSmws 	case SMB_TYPE_SYSCONFSTR:
203784ab085aSmws 		oprintf(fp, "\n");
203884ab085aSmws 		print_strtab(shp, sp->smbstr_id, fp);
203984ab085aSmws 		break;
204084ab085aSmws 	case SMB_TYPE_LANG:
204184ab085aSmws 		oprintf(fp, "\n");
204284ab085aSmws 		print_lang(shp, sp->smbstr_id, fp);
204384ab085aSmws 		break;
204484ab085aSmws 	case SMB_TYPE_EVENTLOG:
204584ab085aSmws 		oprintf(fp, "\n");
204684ab085aSmws 		print_evlog(shp, sp->smbstr_id, fp);
204784ab085aSmws 		break;
204884ab085aSmws 	case SMB_TYPE_MEMARRAY:
204984ab085aSmws 		oprintf(fp, "\n");
205084ab085aSmws 		print_memarray(shp, sp->smbstr_id, fp);
205184ab085aSmws 		break;
205284ab085aSmws 	case SMB_TYPE_MEMDEVICE:
205384ab085aSmws 		oprintf(fp, "\n");
205484ab085aSmws 		print_memdevice(shp, sp->smbstr_id, fp);
205584ab085aSmws 		break;
205684ab085aSmws 	case SMB_TYPE_MEMARRAYMAP:
205784ab085aSmws 		oprintf(fp, "\n");
205884ab085aSmws 		print_memarrmap(shp, sp->smbstr_id, fp);
205984ab085aSmws 		break;
206084ab085aSmws 	case SMB_TYPE_MEMDEVICEMAP:
206184ab085aSmws 		oprintf(fp, "\n");
206284ab085aSmws 		print_memdevmap(shp, sp->smbstr_id, fp);
206384ab085aSmws 		break;
206446782190SRobert Mustacchi 	case SMB_TYPE_BATTERY:
206546782190SRobert Mustacchi 		oprintf(fp, "\n");
206646782190SRobert Mustacchi 		print_battery(shp, sp->smbstr_id, fp);
206746782190SRobert Mustacchi 		break;
20681d77dcdaSRobert Mustacchi 	case SMB_TYPE_POINTDEV:
20691d77dcdaSRobert Mustacchi 		oprintf(fp, "\n");
20701d77dcdaSRobert Mustacchi 		print_pointdev(shp, sp->smbstr_id, fp);
20711d77dcdaSRobert Mustacchi 		break;
207284ab085aSmws 	case SMB_TYPE_SECURITY:
207384ab085aSmws 		oprintf(fp, "\n");
207484ab085aSmws 		print_hwsec(shp, fp);
207584ab085aSmws 		break;
2076f44a1392SRobert Mustacchi 	case SMB_TYPE_VPROBE:
2077f44a1392SRobert Mustacchi 		oprintf(fp, "\n");
2078f44a1392SRobert Mustacchi 		print_vprobe(shp, sp->smbstr_id, fp);
2079f44a1392SRobert Mustacchi 		break;
2080f44a1392SRobert Mustacchi 	case SMB_TYPE_COOLDEV:
2081f44a1392SRobert Mustacchi 		oprintf(fp, "\n");
2082f44a1392SRobert Mustacchi 		print_cooldev(shp, sp->smbstr_id, fp);
2083f44a1392SRobert Mustacchi 		break;
2084f44a1392SRobert Mustacchi 	case SMB_TYPE_TPROBE:
2085f44a1392SRobert Mustacchi 		oprintf(fp, "\n");
2086f44a1392SRobert Mustacchi 		print_tprobe(shp, sp->smbstr_id, fp);
2087f44a1392SRobert Mustacchi 		break;
2088f44a1392SRobert Mustacchi 	case SMB_TYPE_IPROBE:
2089f44a1392SRobert Mustacchi 		oprintf(fp, "\n");
2090f44a1392SRobert Mustacchi 		print_iprobe(shp, sp->smbstr_id, fp);
2091f44a1392SRobert Mustacchi 		break;
209284ab085aSmws 	case SMB_TYPE_BOOT:
209384ab085aSmws 		oprintf(fp, "\n");
209484ab085aSmws 		print_boot(shp, fp);
209584ab085aSmws 		break;
209684ab085aSmws 	case SMB_TYPE_IPMIDEV:
209784ab085aSmws 		oprintf(fp, "\n");
209884ab085aSmws 		print_ipmi(shp, fp);
209984ab085aSmws 		break;
210032ece1f9SRobert Mustacchi 	case SMB_TYPE_POWERSUP:
210132ece1f9SRobert Mustacchi 		oprintf(fp, "\n");
210232ece1f9SRobert Mustacchi 		print_powersup(shp, sp->smbstr_id, fp);
210332ece1f9SRobert Mustacchi 		break;
2104064d431aSRobert Mustacchi 	case SMB_TYPE_ADDINFO:
2105064d431aSRobert Mustacchi 		oprintf(fp, "\n");
2106064d431aSRobert Mustacchi 		print_addinfo(shp, sp->smbstr_id, fp);
2107064d431aSRobert Mustacchi 		break;
210803f9f63dSTom Pothier 	case SMB_TYPE_OBDEVEXT:
210903f9f63dSTom Pothier 		oprintf(fp, "\n");
211003f9f63dSTom Pothier 		print_obdevs_ext(shp, sp->smbstr_id, fp);
211103f9f63dSTom Pothier 		break;
2112176a9270SRobert Mustacchi 	case SMB_TYPE_PROCESSOR_INFO:
2113176a9270SRobert Mustacchi 		oprintf(fp, "\n");
2114176a9270SRobert Mustacchi 		print_processor_info(shp, sp->smbstr_id, fp);
2115176a9270SRobert Mustacchi 		break;
2116d53cdfabSRobert Mustacchi 	case SMB_TYPE_STRPROP:
2117d53cdfabSRobert Mustacchi 		oprintf(fp, "\n");
2118d53cdfabSRobert Mustacchi 		print_strprop_info(shp, sp->smbstr_id, fp);
2119d53cdfabSRobert Mustacchi 		break;
2120d53cdfabSRobert Mustacchi 	case SMB_TYPE_FWINFO:
2121d53cdfabSRobert Mustacchi 		oprintf(fp, "\n");
2122d53cdfabSRobert Mustacchi 		print_fwinfo(shp, sp->smbstr_id, fp);
2123d53cdfabSRobert Mustacchi 		break;
2124074bb90dSTom Pothier 	case SUN_OEM_EXT_PROCESSOR:
2125074bb90dSTom Pothier 		oprintf(fp, "\n");
2126074bb90dSTom Pothier 		print_extprocessor(shp, sp->smbstr_id, fp);
2127074bb90dSTom Pothier 		break;
212803f9f63dSTom Pothier 	case SUN_OEM_EXT_PORT:
212903f9f63dSTom Pothier 		oprintf(fp, "\n");
213003f9f63dSTom Pothier 		print_extport(shp, sp->smbstr_id, fp);
213103f9f63dSTom Pothier 		break;
2132074bb90dSTom Pothier 	case SUN_OEM_PCIEXRC:
2133074bb90dSTom Pothier 		oprintf(fp, "\n");
2134074bb90dSTom Pothier 		print_pciexrc(shp, sp->smbstr_id, fp);
2135074bb90dSTom Pothier 		break;
2136074bb90dSTom Pothier 	case SUN_OEM_EXT_MEMARRAY:
2137074bb90dSTom Pothier 		oprintf(fp, "\n");
2138074bb90dSTom Pothier 		print_extmemarray(shp, sp->smbstr_id, fp);
2139074bb90dSTom Pothier 		break;
2140074bb90dSTom Pothier 	case SUN_OEM_EXT_MEMDEVICE:
2141074bb90dSTom Pothier 		oprintf(fp, "\n");
2142074bb90dSTom Pothier 		print_extmemdevice(shp, sp->smbstr_id, fp);
2143074bb90dSTom Pothier 		break;
214484ab085aSmws 	default:
214584ab085aSmws 		hex++;
214684ab085aSmws 	}
214784ab085aSmws 
214884ab085aSmws 	if (hex)
214984ab085aSmws 		print_bytes(sp->smbstr_data, sp->smbstr_size, fp);
215084ab085aSmws 	else
215184ab085aSmws 		oprintf(fp, "\n");
215284ab085aSmws 
215384ab085aSmws 	return (0);
215484ab085aSmws }
215584ab085aSmws 
215684ab085aSmws static uint16_t
getu16(const char * name,const char * s)215784ab085aSmws getu16(const char *name, const char *s)
215884ab085aSmws {
215984ab085aSmws 	u_longlong_t val;
216084ab085aSmws 	char *p;
216184ab085aSmws 
216284ab085aSmws 	errno = 0;
216384ab085aSmws 	val = strtoull(s, &p, 0);
216484ab085aSmws 
216584ab085aSmws 	if (errno != 0 || p == s || *p != '\0' || val > UINT16_MAX) {
216684ab085aSmws 		(void) fprintf(stderr, "%s: invalid %s argument -- %s\n",
216784ab085aSmws 		    g_pname, name, s);
216884ab085aSmws 		exit(SMBIOS_USAGE);
216984ab085aSmws 	}
217084ab085aSmws 
217184ab085aSmws 	return ((uint16_t)val);
217284ab085aSmws }
217384ab085aSmws 
217484ab085aSmws static uint16_t
getstype(const char * name,const char * s)217584ab085aSmws getstype(const char *name, const char *s)
217684ab085aSmws {
217784ab085aSmws 	const char *ts;
217884ab085aSmws 	uint16_t t;
217984ab085aSmws 
218084ab085aSmws 	for (t = 0; t < SMB_TYPE_OEM_LO; t++) {
218184ab085aSmws 		if ((ts = smbios_type_name(t)) != NULL && strcmp(s, ts) == 0)
218284ab085aSmws 			return (t);
218384ab085aSmws 	}
218484ab085aSmws 
218584ab085aSmws 	(void) fprintf(stderr, "%s: invalid %s argument -- %s\n",
218684ab085aSmws 	    g_pname, name, s);
218784ab085aSmws 
218884ab085aSmws 	exit(SMBIOS_USAGE);
218984ab085aSmws 	/*NOTREACHED*/
219084ab085aSmws }
219184ab085aSmws 
219284ab085aSmws static int
usage(FILE * fp)219384ab085aSmws usage(FILE *fp)
219484ab085aSmws {
219584ab085aSmws 	(void) fprintf(fp, "Usage: %s "
219684ab085aSmws 	    "[-BeOsx] [-i id] [-t type] [-w file] [file]\n\n", g_pname);
219784ab085aSmws 
219884ab085aSmws 	(void) fprintf(fp,
219984ab085aSmws 	    "\t-B disable header validation for broken BIOSes\n"
220084ab085aSmws 	    "\t-e display SMBIOS entry point information\n"
220184ab085aSmws 	    "\t-i display only the specified structure\n"
220284ab085aSmws 	    "\t-O display obsolete structure types\n"
220384ab085aSmws 	    "\t-s display only a summary of structure identifiers and types\n"
220484ab085aSmws 	    "\t-t display only the specified structure type\n"
220584ab085aSmws 	    "\t-w write the raw data to the specified file\n"
220684ab085aSmws 	    "\t-x display raw data for structures\n");
220784ab085aSmws 
220884ab085aSmws 	return (SMBIOS_USAGE);
220984ab085aSmws }
221084ab085aSmws 
221184ab085aSmws int
main(int argc,char * argv[])221284ab085aSmws main(int argc, char *argv[])
221384ab085aSmws {
221484ab085aSmws 	const char *ifile = NULL;
221584ab085aSmws 	const char *ofile = NULL;
221684ab085aSmws 	int oflags = 0;
221784ab085aSmws 
221884ab085aSmws 	smbios_hdl_t *shp;
221984ab085aSmws 	smbios_struct_t s;
222084ab085aSmws 	int err, fd, c;
222184ab085aSmws 	char *p;
222284ab085aSmws 
222384ab085aSmws 	if ((p = strrchr(argv[0], '/')) == NULL)
222484ab085aSmws 		g_pname = argv[0];
222584ab085aSmws 	else
222684ab085aSmws 		g_pname = p + 1;
222784ab085aSmws 
222884ab085aSmws 	while (optind < argc) {
222984ab085aSmws 		while ((c = getopt(argc, argv, "Bei:Ost:w:xZ")) != EOF) {
223084ab085aSmws 			switch (c) {
223184ab085aSmws 			case 'B':
223284ab085aSmws 				oflags |= SMB_O_NOCKSUM | SMB_O_NOVERS;
223384ab085aSmws 				break;
223484ab085aSmws 			case 'e':
223584ab085aSmws 				opt_e++;
223684ab085aSmws 				break;
223784ab085aSmws 			case 'i':
223884ab085aSmws 				opt_i = getu16("struct ID", optarg);
223984ab085aSmws 				break;
224084ab085aSmws 			case 'O':
224184ab085aSmws 				opt_O++;
224284ab085aSmws 				break;
224384ab085aSmws 			case 's':
224484ab085aSmws 				opt_s++;
224584ab085aSmws 				break;
224684ab085aSmws 			case 't':
224784ab085aSmws 				if (isdigit(optarg[0]))
224884ab085aSmws 					opt_t = getu16("struct type", optarg);
224984ab085aSmws 				else
225084ab085aSmws 					opt_t = getstype("struct type", optarg);
225184ab085aSmws 				break;
225284ab085aSmws 			case 'w':
225384ab085aSmws 				ofile = optarg;
225484ab085aSmws 				break;
225584ab085aSmws 			case 'x':
225684ab085aSmws 				opt_x++;
225784ab085aSmws 				break;
225884ab085aSmws 			case 'Z':
225984ab085aSmws 				oflags |= SMB_O_ZIDS; /* undocumented */
226084ab085aSmws 				break;
226184ab085aSmws 			default:
226284ab085aSmws 				return (usage(stderr));
226384ab085aSmws 			}
226484ab085aSmws 		}
226584ab085aSmws 
226684ab085aSmws 		if (optind < argc) {
226784ab085aSmws 			if (ifile != NULL) {
226884ab085aSmws 				(void) fprintf(stderr, "%s: illegal "
226984ab085aSmws 				    "argument -- %s\n", g_pname, argv[optind]);
227084ab085aSmws 				return (SMBIOS_USAGE);
227184ab085aSmws 			}
227284ab085aSmws 			ifile = argv[optind++];
227384ab085aSmws 		}
227484ab085aSmws 	}
227584ab085aSmws 
227684ab085aSmws 	if ((shp = smbios_open(ifile, SMB_VERSION, oflags, &err)) == NULL) {
227784ab085aSmws 		(void) fprintf(stderr, "%s: failed to load SMBIOS: %s\n",
227884ab085aSmws 		    g_pname, smbios_errmsg(err));
227984ab085aSmws 		return (SMBIOS_ERROR);
228084ab085aSmws 	}
228184ab085aSmws 
2282516627f3SJonathan Matthew 	if (opt_i == -1 && opt_t == -1 && opt_e == 0 &&
2283516627f3SJonathan Matthew 	    smbios_truncated(shp))
2284516627f3SJonathan Matthew 		(void) fprintf(stderr, "%s: SMBIOS table is truncated\n",
2285516627f3SJonathan Matthew 		    g_pname);
2286516627f3SJonathan Matthew 
228784ab085aSmws 	if (ofile != NULL) {
228884ab085aSmws 		if ((fd = open(ofile, O_WRONLY|O_CREAT|O_TRUNC, 0666)) == -1) {
228984ab085aSmws 			(void) fprintf(stderr, "%s: failed to open %s: %s\n",
229084ab085aSmws 			    g_pname, ofile, strerror(errno));
229184ab085aSmws 			err = SMBIOS_ERROR;
229284ab085aSmws 		} else if (smbios_write(shp, fd) != 0) {
229384ab085aSmws 			(void) fprintf(stderr, "%s: failed to write %s: %s\n",
229484ab085aSmws 			    g_pname, ofile, smbios_errmsg(smbios_errno(shp)));
229584ab085aSmws 			err = SMBIOS_ERROR;
229684ab085aSmws 		}
229784ab085aSmws 		smbios_close(shp);
229884ab085aSmws 		return (err);
229984ab085aSmws 	}
230084ab085aSmws 
230184ab085aSmws 	if (opt_e) {
230284ab085aSmws 		print_smbios(shp, stdout);
230384ab085aSmws 		smbios_close(shp);
230484ab085aSmws 		return (SMBIOS_SUCCESS);
230584ab085aSmws 	}
230684ab085aSmws 
230784ab085aSmws 	if (opt_O && (opt_i != -1 || opt_t != -1))
230884ab085aSmws 		opt_O++; /* -i or -t imply displaying obsolete records */
230984ab085aSmws 
231084ab085aSmws 	if (opt_i != -1)
231184ab085aSmws 		err = smbios_lookup_id(shp, opt_i, &s);
231284ab085aSmws 	else
231384ab085aSmws 		err = smbios_iter(shp, print_struct, stdout);
231484ab085aSmws 
231584ab085aSmws 	if (err != 0) {
231684ab085aSmws 		(void) fprintf(stderr, "%s: failed to access SMBIOS: %s\n",
231784ab085aSmws 		    g_pname, smbios_errmsg(smbios_errno(shp)));
231884ab085aSmws 		smbios_close(shp);
231984ab085aSmws 		return (SMBIOS_ERROR);
232084ab085aSmws 	}
232184ab085aSmws 
232284ab085aSmws 	if (opt_i != -1)
232384ab085aSmws 		(void) print_struct(shp, &s, stdout);
232484ab085aSmws 
232584ab085aSmws 	smbios_close(shp);
232684ab085aSmws 	return (SMBIOS_SUCCESS);
232784ab085aSmws }
2328