xref: /titanic_52/usr/src/cmd/smbios/smbios.c (revision f89940742f5d14dde79b69b98a414dd7b7f585c7)
1 /*
2  * CDDL HEADER START
3  *
4  * The contents of this file are subject to the terms of the
5  * Common Development and Distribution License (the "License").
6  * You may not use this file except in compliance with the License.
7  *
8  * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
9  * or http://www.opensolaris.org/os/licensing.
10  * See the License for the specific language governing permissions
11  * and limitations under the License.
12  *
13  * When distributing Covered Code, include this CDDL HEADER in each
14  * file and include the License file at usr/src/OPENSOLARIS.LICENSE.
15  * If applicable, add the following below this CDDL HEADER, with the
16  * fields enclosed by brackets "[]" replaced with your own identifying
17  * information: Portions Copyright [yyyy] [name of copyright owner]
18  *
19  * CDDL HEADER END
20  */
21 
22 /*
23  * Copyright 2010 Sun Microsystems, Inc.  All rights reserved.
24  * Use is subject to license terms.
25  */
26 
27 #include <sys/sysmacros.h>
28 #include <sys/param.h>
29 
30 #include <smbios.h>
31 #include <alloca.h>
32 #include <limits.h>
33 #include <unistd.h>
34 #include <strings.h>
35 #include <stdlib.h>
36 #include <stdarg.h>
37 #include <stdio.h>
38 #include <fcntl.h>
39 #include <errno.h>
40 #include <ctype.h>
41 
42 #define	SMBIOS_SUCCESS	0
43 #define	SMBIOS_ERROR	1
44 #define	SMBIOS_USAGE	2
45 
46 static const char *g_pname;
47 static int g_hdr;
48 
49 static int opt_e;
50 static int opt_i = -1;
51 static int opt_O;
52 static int opt_s;
53 static int opt_t = -1;
54 static int opt_x;
55 
56 /*PRINTFLIKE2*/
57 static void
58 oprintf(FILE *fp, const char *format, ...)
59 {
60 	va_list ap;
61 
62 	va_start(ap, format);
63 	(void) vfprintf(fp, format, ap);
64 	va_end(ap);
65 }
66 
67 /*PRINTFLIKE3*/
68 static void
69 desc_printf(const char *d, FILE *fp, const char *format, ...)
70 {
71 	va_list ap;
72 
73 	va_start(ap, format);
74 	(void) vfprintf(fp, format, ap);
75 	va_end(ap);
76 
77 	if (d != NULL)
78 		(void) fprintf(fp, " (%s)\n", d);
79 	else
80 		(void) fprintf(fp, "\n");
81 }
82 
83 static void
84 flag_printf(FILE *fp, const char *s, uint_t flags, size_t bits,
85     const char *(*flag_name)(uint_t), const char *(*flag_desc)(uint_t))
86 {
87 	size_t i;
88 
89 	oprintf(fp, "  %s: 0x%x\n", s, flags);
90 
91 	for (i = 0; i < bits; i++) {
92 		uint_t f = 1 << i;
93 		const char *n;
94 
95 		if (!(flags & f))
96 			continue;
97 
98 		if ((n = flag_name(f)) != NULL)
99 			desc_printf(flag_desc(f), fp, "\t%s", n);
100 		else
101 			desc_printf(flag_desc(f), fp, "\t0x%x", f);
102 	}
103 }
104 
105 static void
106 flag64_printf(FILE *fp, const char *s, uint64_t flags, size_t bits,
107     const char *(*flag_name)(uint64_t), const char *(*flag_desc)(uint64_t))
108 {
109 	size_t i;
110 
111 	oprintf(fp, "  %s: 0x%llx\n", s, (u_longlong_t)flags);
112 
113 	for (i = 0; i < bits; i++) {
114 		u_longlong_t f = 1ULL << i;
115 		const char *n;
116 
117 		if (!(flags & f))
118 			continue;
119 
120 		if ((n = flag_name(f)) != NULL)
121 			desc_printf(flag_desc(f), fp, "\t%s", n);
122 		else
123 			desc_printf(flag_desc(f), fp, "\t0x%llx", f);
124 	}
125 }
126 
127 static void
128 id_printf(FILE *fp, const char *s, id_t id)
129 {
130 	switch (id) {
131 	case SMB_ID_NONE:
132 		oprintf(fp, "%sNone\n", s);
133 		break;
134 	case SMB_ID_NOTSUP:
135 		oprintf(fp, "%sNot Supported\n", s);
136 		break;
137 	default:
138 		oprintf(fp, "%s%u\n", s, (uint_t)id);
139 	}
140 }
141 
142 static int
143 check_oem(smbios_hdl_t *shp)
144 {
145 	int i;
146 	int cnt;
147 	int rv;
148 	id_t oem_id;
149 	smbios_struct_t s;
150 	const char **oem_str;
151 
152 	rv = smbios_lookup_type(shp, SMB_TYPE_OEMSTR, &s);
153 	if (rv != 0) {
154 		return (-1);
155 	}
156 
157 	oem_id = s.smbstr_id;
158 
159 	cnt = smbios_info_strtab(shp, oem_id, 0, NULL);
160 	if (cnt > 0) {
161 		oem_str =  alloca(sizeof (char *) * cnt);
162 		(void) smbios_info_strtab(shp, oem_id, cnt, oem_str);
163 
164 		for (i = 0; i < cnt; i++) {
165 			if (strncmp(oem_str[i], SMB_PRMS1,
166 			    strlen(SMB_PRMS1) + 1) == 0) {
167 				return (0);
168 			}
169 		}
170 	}
171 
172 	return (-1);
173 }
174 
175 static void
176 print_smbios(smbios_hdl_t *shp, FILE *fp)
177 {
178 	smbios_entry_t ep;
179 	int i;
180 
181 	smbios_info_smbios(shp, &ep);
182 
183 	oprintf(fp, "Entry Point Anchor Tag: %*.*s\n",
184 	    (int)sizeof (ep.smbe_eanchor), (int)sizeof (ep.smbe_eanchor),
185 	    ep.smbe_eanchor);
186 
187 	oprintf(fp, "Entry Point Checksum: 0x%x\n", ep.smbe_ecksum);
188 	oprintf(fp, "Entry Point Length: %u\n", ep.smbe_elen);
189 	oprintf(fp, "Entry Point Version: %u.%u\n",
190 	    ep.smbe_major, ep.smbe_minor);
191 	oprintf(fp, "Max Structure Size: %u\n", ep.smbe_maxssize);
192 	oprintf(fp, "Entry Point Revision: 0x%x\n", ep.smbe_revision);
193 
194 	oprintf(fp, "Entry Point Revision Data:");
195 	for (i = 0; i < sizeof (ep.smbe_format); i++)
196 		oprintf(fp, " 0x%02x", ep.smbe_format[i]);
197 	oprintf(fp, "\n");
198 
199 	oprintf(fp, "Intermediate Anchor Tag: %*.*s\n",
200 	    (int)sizeof (ep.smbe_ianchor), (int)sizeof (ep.smbe_ianchor),
201 	    ep.smbe_ianchor);
202 
203 	oprintf(fp, "Intermediate Checksum: 0x%x\n", ep.smbe_icksum);
204 	oprintf(fp, "Structure Table Length: %u\n", ep.smbe_stlen);
205 	oprintf(fp, "Structure Table Address: 0x%x\n", ep.smbe_staddr);
206 	oprintf(fp, "Structure Table Entries: %u\n", ep.smbe_stnum);
207 	oprintf(fp, "DMI BCD Revision: 0x%x\n", ep.smbe_bcdrev);
208 }
209 
210 static void
211 print_common(const smbios_info_t *ip, FILE *fp)
212 {
213 	if (ip->smbi_manufacturer[0] != '\0')
214 		oprintf(fp, "  Manufacturer: %s\n", ip->smbi_manufacturer);
215 	if (ip->smbi_product[0] != '\0')
216 		oprintf(fp, "  Product: %s\n", ip->smbi_product);
217 	if (ip->smbi_version[0] != '\0')
218 		oprintf(fp, "  Version: %s\n", ip->smbi_version);
219 	if (ip->smbi_serial[0] != '\0')
220 		oprintf(fp, "  Serial Number: %s\n", ip->smbi_serial);
221 	if (ip->smbi_asset[0] != '\0')
222 		oprintf(fp, "  Asset Tag: %s\n", ip->smbi_asset);
223 	if (ip->smbi_location[0] != '\0')
224 		oprintf(fp, "  Location Tag: %s\n", ip->smbi_location);
225 	if (ip->smbi_part[0] != '\0')
226 		oprintf(fp, "  Part Number: %s\n", ip->smbi_part);
227 }
228 
229 static void
230 print_bios(smbios_hdl_t *shp, FILE *fp)
231 {
232 	smbios_bios_t b;
233 
234 	(void) smbios_info_bios(shp, &b);
235 
236 	oprintf(fp, "  Vendor: %s\n", b.smbb_vendor);
237 	oprintf(fp, "  Version String: %s\n", b.smbb_version);
238 	oprintf(fp, "  Release Date: %s\n", b.smbb_reldate);
239 	oprintf(fp, "  Address Segment: 0x%x\n", b.smbb_segment);
240 	oprintf(fp, "  ROM Size: %u bytes\n", b.smbb_romsize);
241 	oprintf(fp, "  Image Size: %u bytes\n", b.smbb_runsize);
242 
243 	flag64_printf(fp, "Characteristics",
244 	    b.smbb_cflags, sizeof (b.smbb_cflags) * NBBY,
245 	    smbios_bios_flag_name, smbios_bios_flag_desc);
246 
247 	if (b.smbb_nxcflags > SMB_BIOSXB_1) {
248 		flag_printf(fp, "Characteristics Extension Byte 1",
249 		    b.smbb_xcflags[SMB_BIOSXB_1],
250 		    sizeof (b.smbb_xcflags[SMB_BIOSXB_1]) * NBBY,
251 		    smbios_bios_xb1_name, smbios_bios_xb1_desc);
252 	}
253 
254 	if (b.smbb_nxcflags > SMB_BIOSXB_2) {
255 		flag_printf(fp, "Characteristics Extension Byte 2",
256 		    b.smbb_xcflags[SMB_BIOSXB_2],
257 		    sizeof (b.smbb_xcflags[SMB_BIOSXB_2]) * NBBY,
258 		    smbios_bios_xb2_name, smbios_bios_xb2_desc);
259 	}
260 
261 	if (b.smbb_nxcflags > SMB_BIOSXB_BIOS_MIN) {
262 		oprintf(fp, "  Version Number: %u.%u\n",
263 		    b.smbb_biosv.smbv_major, b.smbb_biosv.smbv_minor);
264 	}
265 
266 	if (b.smbb_nxcflags > SMB_BIOSXB_ECFW_MIN) {
267 		oprintf(fp, "  Embedded Ctlr Firmware Version Number: %u.%u\n",
268 		    b.smbb_ecfwv.smbv_major, b.smbb_ecfwv.smbv_minor);
269 	}
270 }
271 
272 static void
273 print_system(smbios_hdl_t *shp, FILE *fp)
274 {
275 	smbios_system_t s;
276 	uint_t i;
277 
278 	(void) smbios_info_system(shp, &s);
279 
280 	oprintf(fp, "  UUID: ");
281 	for (i = 0; i < s.smbs_uuidlen; i++) {
282 		oprintf(fp, "%02x", s.smbs_uuid[i]);
283 		if (i == 3 || i == 5 || i == 7 || i == 9)
284 			oprintf(fp, "-");
285 	}
286 	oprintf(fp, "\n");
287 
288 	desc_printf(smbios_system_wakeup_desc(s.smbs_wakeup),
289 	    fp, "  Wake-Up Event: 0x%x", s.smbs_wakeup);
290 
291 	oprintf(fp, "  SKU Number: %s\n", s.smbs_sku);
292 	oprintf(fp, "  Family: %s\n", s.smbs_family);
293 }
294 
295 static void
296 print_bboard(smbios_hdl_t *shp, id_t id, FILE *fp)
297 {
298 	smbios_bboard_t b;
299 	int chdl_cnt;
300 
301 	(void) smbios_info_bboard(shp, id, &b);
302 
303 	oprintf(fp, "  Chassis: %u\n", (uint_t)b.smbb_chassis);
304 
305 	flag_printf(fp, "Flags", b.smbb_flags, sizeof (b.smbb_flags) * NBBY,
306 	    smbios_bboard_flag_name, smbios_bboard_flag_desc);
307 
308 	desc_printf(smbios_bboard_type_desc(b.smbb_type),
309 	    fp, "  Board Type: 0x%x", b.smbb_type);
310 
311 	chdl_cnt = b.smbb_contn;
312 	if (chdl_cnt != 0) {
313 		id_t *chdl;
314 		uint16_t hdl;
315 		int i, n, cnt;
316 
317 		chdl = alloca(chdl_cnt * sizeof (id_t));
318 		cnt = smbios_info_contains(shp, id, chdl_cnt, chdl);
319 		if (cnt > SMB_CONT_MAX)
320 			return;
321 		n = MIN(chdl_cnt, cnt);
322 
323 		oprintf(fp, "\n");
324 		for (i = 0; i < n; i++) {
325 			hdl = (uint16_t)chdl[i];
326 			oprintf(fp, "  Contained Handle: %u\n", hdl);
327 		}
328 	}
329 }
330 
331 static void
332 print_chassis(smbios_hdl_t *shp, id_t id, FILE *fp)
333 {
334 	smbios_chassis_t c;
335 	int elem_cnt;
336 
337 	(void) smbios_info_chassis(shp, id, &c);
338 
339 	oprintf(fp, "  OEM Data: 0x%x\n", c.smbc_oemdata);
340 	oprintf(fp, "  Lock Present: %s\n", c.smbc_lock ? "Y" : "N");
341 
342 	desc_printf(smbios_chassis_type_desc(c.smbc_type),
343 	    fp, "  Chassis Type: 0x%x", c.smbc_type);
344 
345 	desc_printf(smbios_chassis_state_desc(c.smbc_bustate),
346 	    fp, "  Boot-Up State: 0x%x", c.smbc_bustate);
347 
348 	desc_printf(smbios_chassis_state_desc(c.smbc_psstate),
349 	    fp, "  Power Supply State: 0x%x", c.smbc_psstate);
350 
351 	desc_printf(smbios_chassis_state_desc(c.smbc_thstate),
352 	    fp, "  Thermal State: 0x%x", c.smbc_thstate);
353 
354 	oprintf(fp, "  Chassis Height: %uu\n", c.smbc_uheight);
355 	oprintf(fp, "  Power Cords: %u\n", c.smbc_cords);
356 
357 	elem_cnt = c.smbc_elems;
358 	oprintf(fp, "  Element Records: %u\n", elem_cnt);
359 
360 	if (elem_cnt > 0) {
361 		id_t *elems;
362 		uint8_t type;
363 		int i, n, cnt;
364 
365 		elems = alloca(c.smbc_elems * sizeof (id_t));
366 		cnt = smbios_info_contains(shp, id, elem_cnt, elems);
367 		if (cnt > SMB_CONT_MAX)
368 			return;
369 		n = MIN(elem_cnt, cnt);
370 
371 		oprintf(fp, "\n");
372 		for (i = 0; i < n; i++) {
373 			type = (uint8_t)elems[i];
374 			if (type & 0x80) {
375 				/* SMBIOS structrure Type */
376 				desc_printf(smbios_type_name(type & 0x7f), fp,
377 				    "  Contained SMBIOS structure Type: %u",
378 				    type & 0x80);
379 			} else {
380 				/* SMBIOS Base Board Type */
381 				desc_printf(smbios_bboard_type_desc(type), fp,
382 				    "  Contained SMBIOS Base Board Type: 0x%x",
383 				    type);
384 			}
385 		}
386 	}
387 }
388 
389 static void
390 print_processor(smbios_hdl_t *shp, id_t id, FILE *fp)
391 {
392 	smbios_processor_t p;
393 	uint_t status;
394 
395 	(void) smbios_info_processor(shp, id, &p);
396 	status = SMB_PRSTATUS_STATUS(p.smbp_status);
397 
398 	desc_printf(smbios_processor_family_desc(p.smbp_family),
399 	    fp, "  Family: %u", p.smbp_family);
400 
401 	oprintf(fp, "  CPUID: 0x%llx\n", (u_longlong_t)p.smbp_cpuid);
402 
403 	desc_printf(smbios_processor_type_desc(p.smbp_type),
404 	    fp, "  Type: %u", p.smbp_type);
405 
406 	desc_printf(smbios_processor_upgrade_desc(p.smbp_upgrade),
407 	    fp, "  Socket Upgrade: %u", p.smbp_upgrade);
408 
409 	oprintf(fp, "  Socket Status: %s\n",
410 	    SMB_PRSTATUS_PRESENT(p.smbp_status) ?
411 	    "Populated" : "Not Populated");
412 
413 	desc_printf(smbios_processor_status_desc(status),
414 	    fp, "  Processor Status: %u", status);
415 
416 	if (SMB_PRV_LEGACY(p.smbp_voltage)) {
417 		oprintf(fp, "  Supported Voltages:");
418 		switch (p.smbp_voltage) {
419 		case SMB_PRV_5V:
420 			oprintf(fp, " 5.0V");
421 			break;
422 		case SMB_PRV_33V:
423 			oprintf(fp, " 3.3V");
424 			break;
425 		case SMB_PRV_29V:
426 			oprintf(fp, " 2.9V");
427 			break;
428 		}
429 		oprintf(fp, "\n");
430 	} else {
431 		oprintf(fp, "  Supported Voltages: %.1fV\n",
432 		    (float)SMB_PRV_VOLTAGE(p.smbp_voltage) / 10);
433 	}
434 
435 	if (p.smbp_clkspeed != 0)
436 		oprintf(fp, "  External Clock Speed: %uMHz\n", p.smbp_clkspeed);
437 	else
438 		oprintf(fp, "  External Clock Speed: Unknown\n");
439 
440 	if (p.smbp_maxspeed != 0)
441 		oprintf(fp, "  Maximum Speed: %uMHz\n", p.smbp_maxspeed);
442 	else
443 		oprintf(fp, "  Maximum Speed: Unknown\n");
444 
445 	if (p.smbp_curspeed != 0)
446 		oprintf(fp, "  Current Speed: %uMHz\n", p.smbp_curspeed);
447 	else
448 		oprintf(fp, "  Current Speed: Unknown\n");
449 
450 	id_printf(fp, "  L1 Cache: ", p.smbp_l1cache);
451 	id_printf(fp, "  L2 Cache: ", p.smbp_l2cache);
452 	id_printf(fp, "  L3 Cache: ", p.smbp_l3cache);
453 }
454 
455 static void
456 print_cache(smbios_hdl_t *shp, id_t id, FILE *fp)
457 {
458 	smbios_cache_t c;
459 
460 	(void) smbios_info_cache(shp, id, &c);
461 
462 	oprintf(fp, "  Level: %u\n", c.smba_level);
463 	oprintf(fp, "  Maximum Installed Size: %u bytes\n", c.smba_maxsize);
464 
465 	if (c.smba_size != 0)
466 		oprintf(fp, "  Installed Size: %u bytes\n", c.smba_size);
467 	else
468 		oprintf(fp, "  Installed Size: Not Installed\n");
469 
470 	if (c.smba_speed != 0)
471 		oprintf(fp, "  Speed: %uns\n", c.smba_speed);
472 	else
473 		oprintf(fp, "  Speed: Unknown\n");
474 
475 	flag_printf(fp, "Supported SRAM Types",
476 	    c.smba_stype, sizeof (c.smba_stype) * NBBY,
477 	    smbios_cache_ctype_name, smbios_cache_ctype_desc);
478 
479 	desc_printf(smbios_cache_ctype_desc(c.smba_ctype),
480 	    fp, "  Current SRAM Type: 0x%x", c.smba_ctype);
481 
482 	desc_printf(smbios_cache_ecc_desc(c.smba_etype),
483 	    fp, "  Error Correction Type: %u", c.smba_etype);
484 
485 	desc_printf(smbios_cache_logical_desc(c.smba_ltype),
486 	    fp, "  Logical Cache Type: %u", c.smba_ltype);
487 
488 	desc_printf(smbios_cache_assoc_desc(c.smba_assoc),
489 	    fp, "  Associativity: %u", c.smba_assoc);
490 
491 	desc_printf(smbios_cache_mode_desc(c.smba_mode),
492 	    fp, "  Mode: %u", c.smba_mode);
493 
494 	desc_printf(smbios_cache_loc_desc(c.smba_location),
495 	    fp, "  Location: %u", c.smba_location);
496 
497 	flag_printf(fp, "Flags", c.smba_flags, sizeof (c.smba_flags) * NBBY,
498 	    smbios_cache_flag_name, smbios_cache_flag_desc);
499 }
500 
501 static void
502 print_port(smbios_hdl_t *shp, id_t id, FILE *fp)
503 {
504 	smbios_port_t p;
505 
506 	(void) smbios_info_port(shp, id, &p);
507 
508 	oprintf(fp, "  Internal Reference Designator: %s\n", p.smbo_iref);
509 	oprintf(fp, "  External Reference Designator: %s\n", p.smbo_eref);
510 
511 	desc_printf(smbios_port_conn_desc(p.smbo_itype),
512 	    fp, "  Internal Connector Type: %u", p.smbo_itype);
513 
514 	desc_printf(smbios_port_conn_desc(p.smbo_etype),
515 	    fp, "  External Connector Type: %u", p.smbo_etype);
516 
517 	desc_printf(smbios_port_type_desc(p.smbo_ptype),
518 	    fp, "  Port Type: %u", p.smbo_ptype);
519 }
520 
521 static void
522 print_slot(smbios_hdl_t *shp, id_t id, FILE *fp)
523 {
524 	smbios_slot_t s;
525 	smbios_entry_t e;
526 
527 	(void) smbios_info_slot(shp, id, &s);
528 	(void) smbios_info_smbios(shp, &e);
529 
530 	oprintf(fp, "  Reference Designator: %s\n", s.smbl_name);
531 	oprintf(fp, "  Slot ID: 0x%x\n", s.smbl_id);
532 
533 	desc_printf(smbios_slot_type_desc(s.smbl_type),
534 	    fp, "  Type: 0x%x", s.smbl_type);
535 
536 	desc_printf(smbios_slot_width_desc(s.smbl_width),
537 	    fp, "  Width: 0x%x", s.smbl_width);
538 
539 	desc_printf(smbios_slot_usage_desc(s.smbl_usage),
540 	    fp, "  Usage: 0x%x", s.smbl_usage);
541 
542 	desc_printf(smbios_slot_length_desc(s.smbl_length),
543 	    fp, "  Length: 0x%x", s.smbl_length);
544 
545 	flag_printf(fp, "Slot Characteristics 1",
546 	    s.smbl_ch1, sizeof (s.smbl_ch1) * NBBY,
547 	    smbios_slot_ch1_name, smbios_slot_ch1_desc);
548 
549 	flag_printf(fp, "Slot Characteristics 2",
550 	    s.smbl_ch2, sizeof (s.smbl_ch2) * NBBY,
551 	    smbios_slot_ch2_name, smbios_slot_ch2_desc);
552 
553 	if (check_oem(shp) != 0 && (e.smbe_major < 2 || e.smbe_minor < 6))
554 		return;
555 
556 	oprintf(fp, "  Segment Group: %u\n", s.smbl_sg);
557 	oprintf(fp, "  Bus Number: %u\n", s.smbl_bus);
558 	oprintf(fp, "  Device/Function Number: %u\n", s.smbl_df);
559 }
560 
561 static void
562 print_obdevs_ext(smbios_hdl_t *shp, id_t id, FILE *fp)
563 {
564 	smbios_obdev_ext_t oe;
565 
566 	(void) smbios_info_obdevs_ext(shp, id, &oe);
567 
568 	oprintf(fp, "  Reference Designator: %s\n", oe.smboe_name);
569 	oprintf(fp, "  Device Type: %u\n", oe.smboe_dtype);
570 	oprintf(fp, "  Device Type Instance: %u\n", oe.smboe_dti);
571 	oprintf(fp, "  Segment Group Number: %u\n", oe.smboe_sg);
572 	oprintf(fp, "  Bus Number: %u\n", oe.smboe_bus);
573 	oprintf(fp, "  Device/Function Number: %u\n", oe.smboe_df);
574 }
575 
576 static void
577 print_obdevs(smbios_hdl_t *shp, id_t id, FILE *fp)
578 {
579 	smbios_obdev_t *argv;
580 	int i, argc;
581 
582 	if ((argc = smbios_info_obdevs(shp, id, 0, NULL)) > 0) {
583 		argv = alloca(sizeof (smbios_obdev_t) * argc);
584 		(void) smbios_info_obdevs(shp, id, argc, argv);
585 		for (i = 0; i < argc; i++)
586 			oprintf(fp, "  %s\n", argv[i].smbd_name);
587 	}
588 }
589 
590 static void
591 print_strtab(smbios_hdl_t *shp, id_t id, FILE *fp)
592 {
593 	const char **argv;
594 	int i, argc;
595 
596 	if ((argc = smbios_info_strtab(shp, id, 0, NULL)) > 0) {
597 		argv = alloca(sizeof (char *) * argc);
598 		(void) smbios_info_strtab(shp, id, argc, argv);
599 		for (i = 0; i < argc; i++)
600 			oprintf(fp, "  %s\n", argv[i]);
601 	}
602 }
603 
604 static void
605 print_lang(smbios_hdl_t *shp, id_t id, FILE *fp)
606 {
607 	smbios_lang_t l;
608 
609 	(void) smbios_info_lang(shp, &l);
610 
611 	oprintf(fp, "  Current Language: %s\n", l.smbla_cur);
612 	oprintf(fp, "  Language String Format: %u\n", l.smbla_fmt);
613 	oprintf(fp, "  Number of Installed Languages: %u\n", l.smbla_num);
614 	oprintf(fp, "  Installed Languages:\n");
615 
616 	print_strtab(shp, id, fp);
617 }
618 
619 /*ARGSUSED*/
620 static void
621 print_evlog(smbios_hdl_t *shp, id_t id, FILE *fp)
622 {
623 	smbios_evlog_t ev;
624 	uint32_t i;
625 
626 	(void) smbios_info_eventlog(shp, &ev);
627 
628 	oprintf(fp, "  Log Area Size: %lu bytes\n", (ulong_t)ev.smbev_size);
629 	oprintf(fp, "  Header Offset: %lu\n", (ulong_t)ev.smbev_hdr);
630 	oprintf(fp, "  Data Offset: %lu\n", (ulong_t)ev.smbev_data);
631 
632 	desc_printf(smbios_evlog_method_desc(ev.smbev_method),
633 	    fp, "  Data Access Method: %u", ev.smbev_method);
634 
635 	flag_printf(fp, "Log Flags",
636 	    ev.smbev_flags, sizeof (ev.smbev_flags) * NBBY,
637 	    smbios_evlog_flag_name, smbios_evlog_flag_desc);
638 
639 	desc_printf(smbios_evlog_format_desc(ev.smbev_format),
640 	    fp, "  Log Header Format: %u", ev.smbev_format);
641 
642 	oprintf(fp, "  Update Token: 0x%x\n", ev.smbev_token);
643 	oprintf(fp, "  Data Access Address: ");
644 
645 	switch (ev.smbev_method) {
646 	case SMB_EVM_1x1i_1x1d:
647 	case SMB_EVM_2x1i_1x1d:
648 	case SMB_EVM_1x2i_1x1d:
649 		oprintf(fp, "Index Address 0x%x, Data Address 0x%x\n",
650 		    ev.smbev_addr.eva_io.evi_iaddr,
651 		    ev.smbev_addr.eva_io.evi_daddr);
652 		break;
653 	case SMB_EVM_GPNV:
654 		oprintf(fp, "0x%x\n", ev.smbev_addr.eva_gpnv);
655 		break;
656 	default:
657 		oprintf(fp, "0x%x\n", ev.smbev_addr.eva_addr);
658 	}
659 
660 	oprintf(fp, "  Type Descriptors:\n");
661 
662 	for (i = 0; i < ev.smbev_typec; i++) {
663 		oprintf(fp, "  %u: Log Type 0x%x, Data Type 0x%x\n", i,
664 		    ev.smbev_typev[i].smbevt_ltype,
665 		    ev.smbev_typev[i].smbevt_dtype);
666 	}
667 }
668 
669 static void
670 print_bytes(const uint8_t *data, size_t size, FILE *fp)
671 {
672 	size_t row, rows = P2ROUNDUP(size, 16) / 16;
673 	size_t col, cols;
674 
675 	char buf[17];
676 	uint8_t x;
677 
678 	oprintf(fp, "\n  offset:   0 1 2 3  4 5 6 7  8 9 a b  c d e f  "
679 	    "0123456789abcdef\n");
680 
681 	for (row = 0; row < rows; row++) {
682 		oprintf(fp, "    %#4lx: ", (ulong_t)row * 16);
683 		cols = MIN(size - row * 16, 16);
684 
685 		for (col = 0; col < cols; col++) {
686 			if (col % 4 == 0)
687 				oprintf(fp, " ");
688 			x = *data++;
689 			oprintf(fp, "%02x", x);
690 			buf[col] = x <= ' ' || x > '~' ? '.' : x;
691 		}
692 
693 		for (; col < 16; col++) {
694 			if (col % 4 == 0)
695 				oprintf(fp, " ");
696 			oprintf(fp, "  ");
697 			buf[col] = ' ';
698 		}
699 
700 		buf[col] = '\0';
701 		oprintf(fp, "  %s\n", buf);
702 	}
703 
704 	oprintf(fp, "\n");
705 }
706 
707 static void
708 print_memarray(smbios_hdl_t *shp, id_t id, FILE *fp)
709 {
710 	smbios_memarray_t ma;
711 
712 	(void) smbios_info_memarray(shp, id, &ma);
713 
714 	desc_printf(smbios_memarray_loc_desc(ma.smbma_location),
715 	    fp, "  Location: %u", ma.smbma_location);
716 
717 	desc_printf(smbios_memarray_use_desc(ma.smbma_use),
718 	    fp, "  Use: %u", ma.smbma_use);
719 
720 	desc_printf(smbios_memarray_ecc_desc(ma.smbma_ecc),
721 	    fp, "  ECC: %u", ma.smbma_ecc);
722 
723 	oprintf(fp, "  Number of Slots/Sockets: %u\n", ma.smbma_ndevs);
724 	id_printf(fp, "  Memory Error Data: ", ma.smbma_err);
725 	oprintf(fp, "  Max Capacity: %llu bytes\n",
726 	    (u_longlong_t)ma.smbma_size);
727 }
728 
729 static void
730 print_memdevice(smbios_hdl_t *shp, id_t id, FILE *fp)
731 {
732 	smbios_memdevice_t md;
733 
734 	(void) smbios_info_memdevice(shp, id, &md);
735 
736 	id_printf(fp, "  Physical Memory Array: ", md.smbmd_array);
737 	id_printf(fp, "  Memory Error Data: ", md.smbmd_error);
738 
739 	if (md.smbmd_twidth != -1u)
740 		oprintf(fp, "  Total Width: %u bits\n", md.smbmd_twidth);
741 	else
742 		oprintf(fp, "  Total Width: Unknown\n");
743 
744 	if (md.smbmd_dwidth != -1u)
745 		oprintf(fp, "  Data Width: %u bits\n", md.smbmd_dwidth);
746 	else
747 		oprintf(fp, "  Data Width: Unknown\n");
748 
749 	switch (md.smbmd_size) {
750 	case -1ull:
751 		oprintf(fp, "  Size: Unknown\n");
752 		break;
753 	case 0:
754 		oprintf(fp, "  Size: Not Populated\n");
755 		break;
756 	default:
757 		oprintf(fp, "  Size: %llu bytes\n",
758 		    (u_longlong_t)md.smbmd_size);
759 	}
760 
761 	desc_printf(smbios_memdevice_form_desc(md.smbmd_form),
762 	    fp, "  Form Factor: %u", md.smbmd_form);
763 
764 	if (md.smbmd_set == 0)
765 		oprintf(fp, "  Set: None\n");
766 	else if (md.smbmd_set == (uint8_t)-1u)
767 		oprintf(fp, "  Set: Unknown\n");
768 	else
769 		oprintf(fp, "  Set: %u\n", md.smbmd_set);
770 
771 	desc_printf(smbios_memdevice_type_desc(md.smbmd_type),
772 	    fp, "  Memory Type: %u", md.smbmd_type);
773 
774 	flag_printf(fp, "Flags", md.smbmd_flags, sizeof (md.smbmd_flags) * NBBY,
775 	    smbios_memdevice_flag_name, smbios_memdevice_flag_desc);
776 
777 	if (md.smbmd_speed != 0)
778 		oprintf(fp, "  Speed: %uns\n", md.smbmd_speed);
779 	else
780 		oprintf(fp, "  Speed: Unknown\n");
781 
782 	oprintf(fp, "  Device Locator: %s\n", md.smbmd_dloc);
783 	oprintf(fp, "  Bank Locator: %s\n", md.smbmd_bloc);
784 }
785 
786 static void
787 print_memarrmap(smbios_hdl_t *shp, id_t id, FILE *fp)
788 {
789 	smbios_memarrmap_t ma;
790 
791 	(void) smbios_info_memarrmap(shp, id, &ma);
792 
793 	id_printf(fp, "  Physical Memory Array: ", ma.smbmam_array);
794 	oprintf(fp, "  Devices per Row: %u\n", ma.smbmam_width);
795 
796 	oprintf(fp, "  Physical Address: 0x%llx\n  Size: %llu bytes\n",
797 	    (u_longlong_t)ma.smbmam_addr, (u_longlong_t)ma.smbmam_size);
798 }
799 
800 static void
801 print_memdevmap(smbios_hdl_t *shp, id_t id, FILE *fp)
802 {
803 	smbios_memdevmap_t md;
804 
805 	(void) smbios_info_memdevmap(shp, id, &md);
806 
807 	id_printf(fp, "  Memory Device: ", md.smbmdm_device);
808 	id_printf(fp, "  Memory Array Mapped Address: ", md.smbmdm_arrmap);
809 
810 	oprintf(fp, "  Physical Address: 0x%llx\n  Size: %llu bytes\n",
811 	    (u_longlong_t)md.smbmdm_addr, (u_longlong_t)md.smbmdm_size);
812 
813 	oprintf(fp, "  Partition Row Position: %u\n", md.smbmdm_rpos);
814 	oprintf(fp, "  Interleave Position: %u\n", md.smbmdm_ipos);
815 	oprintf(fp, "  Interleave Data Depth: %u\n", md.smbmdm_idepth);
816 }
817 
818 static void
819 print_hwsec(smbios_hdl_t *shp, FILE *fp)
820 {
821 	smbios_hwsec_t h;
822 
823 	(void) smbios_info_hwsec(shp, &h);
824 
825 	desc_printf(smbios_hwsec_desc(h.smbh_pwr_ps),
826 	    fp, "  Power-On Password Status: %u", h.smbh_pwr_ps);
827 	desc_printf(smbios_hwsec_desc(h.smbh_kbd_ps),
828 	    fp, "  Keyboard Password Status: %u", h.smbh_kbd_ps);
829 	desc_printf(smbios_hwsec_desc(h.smbh_adm_ps),
830 	    fp, "  Administrator Password Status: %u", h.smbh_adm_ps);
831 	desc_printf(smbios_hwsec_desc(h.smbh_pan_ps),
832 	    fp, "  Front Panel Reset Status: %u", h.smbh_pan_ps);
833 }
834 
835 static void
836 print_boot(smbios_hdl_t *shp, FILE *fp)
837 {
838 	smbios_boot_t b;
839 
840 	(void) smbios_info_boot(shp, &b);
841 
842 	desc_printf(smbios_boot_desc(b.smbt_status),
843 	    fp, "  Boot Status Code: 0x%x", b.smbt_status);
844 
845 	if (b.smbt_size != 0) {
846 		oprintf(fp, "  Boot Data (%lu bytes):\n", (ulong_t)b.smbt_size);
847 		print_bytes(b.smbt_data, b.smbt_size, fp);
848 	}
849 }
850 
851 static void
852 print_ipmi(smbios_hdl_t *shp, FILE *fp)
853 {
854 	smbios_ipmi_t i;
855 
856 	(void) smbios_info_ipmi(shp, &i);
857 
858 	desc_printf(smbios_ipmi_type_desc(i.smbip_type),
859 	    fp, "  Type: %u", i.smbip_type);
860 
861 	oprintf(fp, "  BMC IPMI Version: %u.%u\n",
862 	    i.smbip_vers.smbv_major, i.smbip_vers.smbv_minor);
863 
864 	oprintf(fp, "  i2c Bus Slave Address: 0x%x\n", i.smbip_i2c);
865 	oprintf(fp, "  NV Storage Device Bus ID: 0x%x\n", i.smbip_bus);
866 	oprintf(fp, "  BMC Base Address: 0x%llx\n", (u_longlong_t)i.smbip_addr);
867 	oprintf(fp, "  Interrupt Number: %u\n", i.smbip_intr);
868 	oprintf(fp, "  Register Spacing: %u\n", i.smbip_regspacing);
869 
870 	flag_printf(fp, "Flags", i.smbip_flags, sizeof (i.smbip_flags) * NBBY,
871 	    smbios_ipmi_flag_name, smbios_ipmi_flag_desc);
872 }
873 
874 static void
875 print_extprocessor(smbios_hdl_t *shp, id_t id, FILE *fp)
876 {
877 	int i;
878 	smbios_processor_ext_t ep;
879 
880 	if (check_oem(shp) != 0)
881 		return;
882 
883 	(void) smbios_info_extprocessor(shp, id, &ep);
884 
885 	oprintf(fp, "  Processor: %u\n", ep.smbpe_processor);
886 	oprintf(fp, "  FRU: %u\n", ep.smbpe_fru);
887 	oprintf(fp, "  Initial APIC ID count: %u\n\n", ep.smbpe_n);
888 
889 	for (i = 0; i < ep.smbpe_n; i++) {
890 		oprintf(fp, "  Logical Strand %u: Initial APIC ID: %u\n", i,
891 		    ep.smbpe_apicid[i]);
892 	}
893 }
894 
895 static void
896 print_extport(smbios_hdl_t *shp, id_t id, FILE *fp)
897 {
898 	smbios_port_ext_t epo;
899 
900 	if (check_oem(shp) != 0)
901 		return;
902 
903 	(void) smbios_info_extport(shp, id, &epo);
904 
905 	oprintf(fp, "  Chassis Handle: %u\n", epo.smbporte_chassis);
906 	oprintf(fp, "  Port Connector Handle: %u\n", epo.smbporte_port);
907 	oprintf(fp, "  Device Type: %u\n", epo.smbporte_dtype);
908 	oprintf(fp, "  Device Handle: %u\n", epo.smbporte_devhdl);
909 	oprintf(fp, "  PHY: %u\n", epo.smbporte_phy);
910 }
911 
912 static void
913 print_pciexrc(smbios_hdl_t *shp, id_t id, FILE *fp)
914 {
915 	smbios_pciexrc_t pcie;
916 
917 	if (check_oem(shp) != 0)
918 		return;
919 
920 	(void) smbios_info_pciexrc(shp, id, &pcie);
921 
922 	oprintf(fp, "  Component ID: %u\n", pcie.smbpcie_bb);
923 	oprintf(fp, "  BDF: 0x%x\n", pcie.smbpcie_bdf);
924 }
925 
926 static void
927 print_extmemarray(smbios_hdl_t *shp, id_t id, FILE *fp)
928 {
929 	smbios_memarray_ext_t em;
930 
931 	if (check_oem(shp) != 0)
932 		return;
933 
934 	(void) smbios_info_extmemarray(shp, id, &em);
935 
936 	oprintf(fp, "  Physical Memory Array Handle: %u\n", em.smbmae_ma);
937 	oprintf(fp, "  Component Parent Handle: %u\n", em.smbmae_comp);
938 	oprintf(fp, "  BDF: 0x%x\n", em.smbmae_bdf);
939 }
940 
941 static void
942 print_extmemdevice(smbios_hdl_t *shp, id_t id, FILE *fp)
943 {
944 	int i;
945 	smbios_memdevice_ext_t emd;
946 
947 	if (check_oem(shp) != 0)
948 		return;
949 
950 	(void) smbios_info_extmemdevice(shp, id, &emd);
951 
952 	oprintf(fp, "  Memory Device Handle: %u\n", emd.smbmdeve_md);
953 	oprintf(fp, "  DRAM Channel: %u\n", emd.smbmdeve_drch);
954 	oprintf(fp, "  Number of Chip Selects: %u\n", emd.smbmdeve_ncs);
955 
956 	for (i = 0; i < emd.smbmdeve_ncs; i++) {
957 		oprintf(fp, "  Chip Select: %u\n", emd.smbmdeve_cs[i]);
958 	}
959 }
960 
961 static int
962 print_struct(smbios_hdl_t *shp, const smbios_struct_t *sp, void *fp)
963 {
964 	smbios_info_t info;
965 	int hex = opt_x;
966 	const char *s;
967 
968 	if (opt_t != -1 && opt_t != sp->smbstr_type)
969 		return (0); /* skip struct if type doesn't match -t */
970 
971 	if (!opt_O && (sp->smbstr_type == SMB_TYPE_MEMCTL ||
972 	    sp->smbstr_type == SMB_TYPE_MEMMOD))
973 		return (0); /* skip struct if type is obsolete */
974 
975 	if (g_hdr++ == 0 || !opt_s)
976 		oprintf(fp, "%-5s %-4s %s\n", "ID", "SIZE", "TYPE");
977 
978 	oprintf(fp, "%-5u %-4lu",
979 	    (uint_t)sp->smbstr_id, (ulong_t)sp->smbstr_size);
980 
981 	if ((s = smbios_type_name(sp->smbstr_type)) != NULL)
982 		oprintf(fp, " %s", s);
983 	else if (sp->smbstr_type > SMB_TYPE_OEM_LO &&
984 	    sp->smbstr_type < SMB_TYPE_OEM_HI)
985 		oprintf(fp, " %s+%u", "SMB_TYPE_OEM_LO",
986 		    sp->smbstr_type - SMB_TYPE_OEM_LO);
987 	else
988 		oprintf(fp, " %u", sp->smbstr_type);
989 
990 	if ((s = smbios_type_desc(sp->smbstr_type)) != NULL)
991 		oprintf(fp, " (%s)\n", s);
992 	else
993 		oprintf(fp, "\n");
994 
995 	if (opt_s)
996 		return (0); /* only print header line if -s specified */
997 
998 	if (smbios_info_common(shp, sp->smbstr_id, &info) == 0) {
999 		oprintf(fp, "\n");
1000 		print_common(&info, fp);
1001 	}
1002 
1003 	switch (sp->smbstr_type) {
1004 	case SMB_TYPE_BIOS:
1005 		oprintf(fp, "\n");
1006 		print_bios(shp, fp);
1007 		break;
1008 	case SMB_TYPE_SYSTEM:
1009 		oprintf(fp, "\n");
1010 		print_system(shp, fp);
1011 		break;
1012 	case SMB_TYPE_BASEBOARD:
1013 		oprintf(fp, "\n");
1014 		print_bboard(shp, sp->smbstr_id, fp);
1015 		break;
1016 	case SMB_TYPE_CHASSIS:
1017 		oprintf(fp, "\n");
1018 		print_chassis(shp, sp->smbstr_id, fp);
1019 		break;
1020 	case SMB_TYPE_PROCESSOR:
1021 		oprintf(fp, "\n");
1022 		print_processor(shp, sp->smbstr_id, fp);
1023 		break;
1024 	case SMB_TYPE_CACHE:
1025 		oprintf(fp, "\n");
1026 		print_cache(shp, sp->smbstr_id, fp);
1027 		break;
1028 	case SMB_TYPE_PORT:
1029 		oprintf(fp, "\n");
1030 		print_port(shp, sp->smbstr_id, fp);
1031 		break;
1032 	case SMB_TYPE_SLOT:
1033 		oprintf(fp, "\n");
1034 		print_slot(shp, sp->smbstr_id, fp);
1035 		break;
1036 	case SMB_TYPE_OBDEVS:
1037 		oprintf(fp, "\n");
1038 		print_obdevs(shp, sp->smbstr_id, fp);
1039 		break;
1040 	case SMB_TYPE_OEMSTR:
1041 	case SMB_TYPE_SYSCONFSTR:
1042 		oprintf(fp, "\n");
1043 		print_strtab(shp, sp->smbstr_id, fp);
1044 		break;
1045 	case SMB_TYPE_LANG:
1046 		oprintf(fp, "\n");
1047 		print_lang(shp, sp->smbstr_id, fp);
1048 		break;
1049 	case SMB_TYPE_EVENTLOG:
1050 		oprintf(fp, "\n");
1051 		print_evlog(shp, sp->smbstr_id, fp);
1052 		break;
1053 	case SMB_TYPE_MEMARRAY:
1054 		oprintf(fp, "\n");
1055 		print_memarray(shp, sp->smbstr_id, fp);
1056 		break;
1057 	case SMB_TYPE_MEMDEVICE:
1058 		oprintf(fp, "\n");
1059 		print_memdevice(shp, sp->smbstr_id, fp);
1060 		break;
1061 	case SMB_TYPE_MEMARRAYMAP:
1062 		oprintf(fp, "\n");
1063 		print_memarrmap(shp, sp->smbstr_id, fp);
1064 		break;
1065 	case SMB_TYPE_MEMDEVICEMAP:
1066 		oprintf(fp, "\n");
1067 		print_memdevmap(shp, sp->smbstr_id, fp);
1068 		break;
1069 	case SMB_TYPE_SECURITY:
1070 		oprintf(fp, "\n");
1071 		print_hwsec(shp, fp);
1072 		break;
1073 	case SMB_TYPE_BOOT:
1074 		oprintf(fp, "\n");
1075 		print_boot(shp, fp);
1076 		break;
1077 	case SMB_TYPE_IPMIDEV:
1078 		oprintf(fp, "\n");
1079 		print_ipmi(shp, fp);
1080 		break;
1081 	case SMB_TYPE_OBDEVEXT:
1082 		oprintf(fp, "\n");
1083 		print_obdevs_ext(shp, sp->smbstr_id, fp);
1084 		break;
1085 	case SUN_OEM_EXT_PROCESSOR:
1086 		oprintf(fp, "\n");
1087 		print_extprocessor(shp, sp->smbstr_id, fp);
1088 		break;
1089 	case SUN_OEM_EXT_PORT:
1090 		oprintf(fp, "\n");
1091 		print_extport(shp, sp->smbstr_id, fp);
1092 		break;
1093 	case SUN_OEM_PCIEXRC:
1094 		oprintf(fp, "\n");
1095 		print_pciexrc(shp, sp->smbstr_id, fp);
1096 		break;
1097 	case SUN_OEM_EXT_MEMARRAY:
1098 		oprintf(fp, "\n");
1099 		print_extmemarray(shp, sp->smbstr_id, fp);
1100 		break;
1101 	case SUN_OEM_EXT_MEMDEVICE:
1102 		oprintf(fp, "\n");
1103 		print_extmemdevice(shp, sp->smbstr_id, fp);
1104 		break;
1105 	default:
1106 		hex++;
1107 	}
1108 
1109 	if (hex)
1110 		print_bytes(sp->smbstr_data, sp->smbstr_size, fp);
1111 	else
1112 		oprintf(fp, "\n");
1113 
1114 	return (0);
1115 }
1116 
1117 static uint16_t
1118 getu16(const char *name, const char *s)
1119 {
1120 	u_longlong_t val;
1121 	char *p;
1122 
1123 	errno = 0;
1124 	val = strtoull(s, &p, 0);
1125 
1126 	if (errno != 0 || p == s || *p != '\0' || val > UINT16_MAX) {
1127 		(void) fprintf(stderr, "%s: invalid %s argument -- %s\n",
1128 		    g_pname, name, s);
1129 		exit(SMBIOS_USAGE);
1130 	}
1131 
1132 	return ((uint16_t)val);
1133 }
1134 
1135 static uint16_t
1136 getstype(const char *name, const char *s)
1137 {
1138 	const char *ts;
1139 	uint16_t t;
1140 
1141 	for (t = 0; t < SMB_TYPE_OEM_LO; t++) {
1142 		if ((ts = smbios_type_name(t)) != NULL && strcmp(s, ts) == 0)
1143 			return (t);
1144 	}
1145 
1146 	(void) fprintf(stderr, "%s: invalid %s argument -- %s\n",
1147 	    g_pname, name, s);
1148 
1149 	exit(SMBIOS_USAGE);
1150 	/*NOTREACHED*/
1151 }
1152 
1153 static int
1154 usage(FILE *fp)
1155 {
1156 	(void) fprintf(fp, "Usage: %s "
1157 	    "[-BeOsx] [-i id] [-t type] [-w file] [file]\n\n", g_pname);
1158 
1159 	(void) fprintf(fp,
1160 	    "\t-B disable header validation for broken BIOSes\n"
1161 	    "\t-e display SMBIOS entry point information\n"
1162 	    "\t-i display only the specified structure\n"
1163 	    "\t-O display obsolete structure types\n"
1164 	    "\t-s display only a summary of structure identifiers and types\n"
1165 	    "\t-t display only the specified structure type\n"
1166 	    "\t-w write the raw data to the specified file\n"
1167 	    "\t-x display raw data for structures\n");
1168 
1169 	return (SMBIOS_USAGE);
1170 }
1171 
1172 int
1173 main(int argc, char *argv[])
1174 {
1175 	const char *ifile = NULL;
1176 	const char *ofile = NULL;
1177 	int oflags = 0;
1178 
1179 	smbios_hdl_t *shp;
1180 	smbios_struct_t s;
1181 	int err, fd, c;
1182 	char *p;
1183 
1184 	if ((p = strrchr(argv[0], '/')) == NULL)
1185 		g_pname = argv[0];
1186 	else
1187 		g_pname = p + 1;
1188 
1189 	while (optind < argc) {
1190 		while ((c = getopt(argc, argv, "Bei:Ost:w:xZ")) != EOF) {
1191 			switch (c) {
1192 			case 'B':
1193 				oflags |= SMB_O_NOCKSUM | SMB_O_NOVERS;
1194 				break;
1195 			case 'e':
1196 				opt_e++;
1197 				break;
1198 			case 'i':
1199 				opt_i = getu16("struct ID", optarg);
1200 				break;
1201 			case 'O':
1202 				opt_O++;
1203 				break;
1204 			case 's':
1205 				opt_s++;
1206 				break;
1207 			case 't':
1208 				if (isdigit(optarg[0]))
1209 					opt_t = getu16("struct type", optarg);
1210 				else
1211 					opt_t = getstype("struct type", optarg);
1212 				break;
1213 			case 'w':
1214 				ofile = optarg;
1215 				break;
1216 			case 'x':
1217 				opt_x++;
1218 				break;
1219 			case 'Z':
1220 				oflags |= SMB_O_ZIDS; /* undocumented */
1221 				break;
1222 			default:
1223 				return (usage(stderr));
1224 			}
1225 		}
1226 
1227 		if (optind < argc) {
1228 			if (ifile != NULL) {
1229 				(void) fprintf(stderr, "%s: illegal "
1230 				    "argument -- %s\n", g_pname, argv[optind]);
1231 				return (SMBIOS_USAGE);
1232 			}
1233 			ifile = argv[optind++];
1234 		}
1235 	}
1236 
1237 	if ((shp = smbios_open(ifile, SMB_VERSION, oflags, &err)) == NULL) {
1238 		(void) fprintf(stderr, "%s: failed to load SMBIOS: %s\n",
1239 		    g_pname, smbios_errmsg(err));
1240 		return (SMBIOS_ERROR);
1241 	}
1242 
1243 	if (ofile != NULL) {
1244 		if ((fd = open(ofile, O_WRONLY|O_CREAT|O_TRUNC, 0666)) == -1) {
1245 			(void) fprintf(stderr, "%s: failed to open %s: %s\n",
1246 			    g_pname, ofile, strerror(errno));
1247 			err = SMBIOS_ERROR;
1248 		} else if (smbios_write(shp, fd) != 0) {
1249 			(void) fprintf(stderr, "%s: failed to write %s: %s\n",
1250 			    g_pname, ofile, smbios_errmsg(smbios_errno(shp)));
1251 			err = SMBIOS_ERROR;
1252 		}
1253 		smbios_close(shp);
1254 		return (err);
1255 	}
1256 
1257 	if (opt_e) {
1258 		print_smbios(shp, stdout);
1259 		smbios_close(shp);
1260 		return (SMBIOS_SUCCESS);
1261 	}
1262 
1263 	if (opt_O && (opt_i != -1 || opt_t != -1))
1264 		opt_O++; /* -i or -t imply displaying obsolete records */
1265 
1266 	if (opt_i != -1)
1267 		err = smbios_lookup_id(shp, opt_i, &s);
1268 	else
1269 		err = smbios_iter(shp, print_struct, stdout);
1270 
1271 	if (err != 0) {
1272 		(void) fprintf(stderr, "%s: failed to access SMBIOS: %s\n",
1273 		    g_pname, smbios_errmsg(smbios_errno(shp)));
1274 		smbios_close(shp);
1275 		return (SMBIOS_ERROR);
1276 	}
1277 
1278 	if (opt_i != -1)
1279 		(void) print_struct(shp, &s, stdout);
1280 
1281 	smbios_close(shp);
1282 	return (SMBIOS_SUCCESS);
1283 }
1284