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
oprintf(FILE * fp,const char * format,...)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
desc_printf(const char * d,FILE * fp,const char * format,...)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
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))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
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))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
id_printf(FILE * fp,const char * s,id_t id)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
check_oem(smbios_hdl_t * shp)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
print_smbios(smbios_hdl_t * shp,FILE * fp)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
print_common(const smbios_info_t * ip,FILE * fp)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
print_bios(smbios_hdl_t * shp,FILE * fp)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
print_system(smbios_hdl_t * shp,FILE * fp)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
print_bboard(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_chassis(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_processor(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_cache(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_port(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_slot(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_obdevs_ext(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_obdevs(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_strtab(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_lang(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_evlog(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_bytes(const uint8_t * data,size_t size,FILE * fp)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
print_memarray(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_memdevice(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_memarrmap(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_memdevmap(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_hwsec(smbios_hdl_t * shp,FILE * fp)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
print_boot(smbios_hdl_t * shp,FILE * fp)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
print_ipmi(smbios_hdl_t * shp,FILE * fp)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
print_extprocessor(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_extport(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_pciexrc(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_extmemarray(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_extmemdevice(smbios_hdl_t * shp,id_t id,FILE * fp)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
print_struct(smbios_hdl_t * shp,const smbios_struct_t * sp,void * fp)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
getu16(const char * name,const char * s)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
getstype(const char * name,const char * s)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
usage(FILE * fp)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
main(int argc,char * argv[])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