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 2015 OmniTI Computer Consulting, Inc. All rights reserved. 24 * Copyright (c) 2017, Joyent, Inc. 25 * Copyright 2010 Sun Microsystems, Inc. All rights reserved. 26 * Use is subject to license terms. 27 */ 28 29 #include <sys/sysmacros.h> 30 #include <sys/param.h> 31 32 #include <smbios.h> 33 #include <alloca.h> 34 #include <limits.h> 35 #include <unistd.h> 36 #include <strings.h> 37 #include <stdlib.h> 38 #include <stdarg.h> 39 #include <stdio.h> 40 #include <fcntl.h> 41 #include <errno.h> 42 #include <ctype.h> 43 #include <libjedec.h> 44 45 #define SMBIOS_SUCCESS 0 46 #define SMBIOS_ERROR 1 47 #define SMBIOS_USAGE 2 48 49 static const char *g_pname; 50 static int g_hdr; 51 52 static int opt_e; 53 static int opt_i = -1; 54 static int opt_O; 55 static int opt_s; 56 static int opt_t = -1; 57 static int opt_x; 58 59 static boolean_t 60 smbios_vergteq(smbios_version_t *v, uint_t major, uint_t minor) 61 { 62 if (v->smbv_major > major) 63 return (B_TRUE); 64 if (v->smbv_major == major && 65 v->smbv_minor >= minor) 66 return (B_TRUE); 67 return (B_FALSE); 68 } 69 70 /*PRINTFLIKE2*/ 71 static void 72 smbios_warn(smbios_hdl_t *shp, const char *format, ...) 73 { 74 va_list ap; 75 76 va_start(ap, format); 77 (void) vfprintf(stderr, format, ap); 78 va_end(ap); 79 80 if (shp != NULL) { 81 (void) fprintf(stderr, ": %s", 82 smbios_errmsg(smbios_errno(shp))); 83 } 84 85 (void) fprintf(stderr, "\n"); 86 } 87 88 /*PRINTFLIKE2*/ 89 static void 90 oprintf(FILE *fp, const char *format, ...) 91 { 92 va_list ap; 93 94 va_start(ap, format); 95 (void) vfprintf(fp, format, ap); 96 va_end(ap); 97 } 98 99 /*PRINTFLIKE3*/ 100 static void 101 desc_printf(const char *d, FILE *fp, const char *format, ...) 102 { 103 va_list ap; 104 105 va_start(ap, format); 106 (void) vfprintf(fp, format, ap); 107 va_end(ap); 108 109 if (d != NULL) 110 (void) fprintf(fp, " (%s)\n", d); 111 else 112 (void) fprintf(fp, "\n"); 113 } 114 115 static void 116 flag_printf(FILE *fp, const char *s, uint_t flags, size_t bits, 117 const char *(*flag_name)(uint_t), const char *(*flag_desc)(uint_t)) 118 { 119 size_t i; 120 121 oprintf(fp, " %s: 0x%x\n", s, flags); 122 123 for (i = 0; i < bits; i++) { 124 uint_t f = 1 << i; 125 const char *n; 126 127 if (!(flags & f)) 128 continue; 129 130 if ((n = flag_name(f)) != NULL) 131 desc_printf(flag_desc(f), fp, "\t%s", n); 132 else 133 desc_printf(flag_desc(f), fp, "\t0x%x", f); 134 } 135 } 136 137 static void 138 flag64_printf(FILE *fp, const char *s, uint64_t flags, size_t bits, 139 const char *(*flag_name)(uint64_t), const char *(*flag_desc)(uint64_t)) 140 { 141 size_t i; 142 143 oprintf(fp, " %s: 0x%llx\n", s, (u_longlong_t)flags); 144 145 for (i = 0; i < bits; i++) { 146 u_longlong_t f = 1ULL << i; 147 const char *n; 148 149 if (!(flags & f)) 150 continue; 151 152 if ((n = flag_name(f)) != NULL) 153 desc_printf(flag_desc(f), fp, "\t%s", n); 154 else 155 desc_printf(flag_desc(f), fp, "\t0x%llx", f); 156 } 157 } 158 159 static void 160 id_printf(FILE *fp, const char *s, id_t id) 161 { 162 switch (id) { 163 case SMB_ID_NONE: 164 oprintf(fp, "%sNone\n", s); 165 break; 166 case SMB_ID_NOTSUP: 167 oprintf(fp, "%sNot Supported\n", s); 168 break; 169 default: 170 oprintf(fp, "%s%u\n", s, (uint_t)id); 171 } 172 } 173 174 static void 175 jedec_print(FILE *fp, const char *desc, uint_t id) 176 { 177 const char *name; 178 uint_t cont, vendor; 179 180 vendor = id & 0xff; 181 cont = (id >> 8) & 0xff; 182 name = libjedec_vendor_string(cont, vendor); 183 if (name == NULL) { 184 oprintf(fp, " %s: 0x%x\n", desc, id); 185 } else { 186 oprintf(fp, " %s: 0x%x (%s)\n", desc, id, name); 187 } 188 } 189 190 /* 191 * Print a 128-bit data as a series of 16 hex digits. 192 */ 193 static void 194 u128_print(FILE *fp, const char *desc, const uint8_t *data) 195 { 196 uint_t i; 197 198 oprintf(fp, "%s: ", desc); 199 for (i = 0; i < 16; i++) { 200 oprintf(fp, " %02x", data[i]); 201 } 202 oprintf(fp, "\n"); 203 } 204 205 static int 206 check_oem(smbios_hdl_t *shp) 207 { 208 int i; 209 int cnt; 210 int rv; 211 id_t oem_id; 212 smbios_struct_t s; 213 const char **oem_str; 214 215 rv = smbios_lookup_type(shp, SMB_TYPE_OEMSTR, &s); 216 if (rv != 0) { 217 return (-1); 218 } 219 220 oem_id = s.smbstr_id; 221 222 cnt = smbios_info_strtab(shp, oem_id, 0, NULL); 223 if (cnt > 0) { 224 oem_str = alloca(sizeof (char *) * cnt); 225 (void) smbios_info_strtab(shp, oem_id, cnt, oem_str); 226 227 for (i = 0; i < cnt; i++) { 228 if (strncmp(oem_str[i], SMB_PRMS1, 229 strlen(SMB_PRMS1) + 1) == 0) { 230 return (0); 231 } 232 } 233 } 234 235 return (-1); 236 } 237 238 static void 239 print_smbios_21(smbios_21_entry_t *ep, FILE *fp) 240 { 241 int i; 242 243 oprintf(fp, "Entry Point Anchor Tag: %*.*s\n", 244 (int)sizeof (ep->smbe_eanchor), (int)sizeof (ep->smbe_eanchor), 245 ep->smbe_eanchor); 246 247 oprintf(fp, "Entry Point Checksum: 0x%x\n", ep->smbe_ecksum); 248 oprintf(fp, "Entry Point Length: %u\n", ep->smbe_elen); 249 oprintf(fp, "Entry Point Version: %u.%u\n", 250 ep->smbe_major, ep->smbe_minor); 251 oprintf(fp, "Max Structure Size: %u\n", ep->smbe_maxssize); 252 oprintf(fp, "Entry Point Revision: 0x%x\n", ep->smbe_revision); 253 254 oprintf(fp, "Entry Point Revision Data:"); 255 for (i = 0; i < sizeof (ep->smbe_format); i++) 256 oprintf(fp, " 0x%02x", ep->smbe_format[i]); 257 oprintf(fp, "\n"); 258 259 oprintf(fp, "Intermediate Anchor Tag: %*.*s\n", 260 (int)sizeof (ep->smbe_ianchor), (int)sizeof (ep->smbe_ianchor), 261 ep->smbe_ianchor); 262 263 oprintf(fp, "Intermediate Checksum: 0x%x\n", ep->smbe_icksum); 264 oprintf(fp, "Structure Table Length: %u\n", ep->smbe_stlen); 265 oprintf(fp, "Structure Table Address: 0x%x\n", ep->smbe_staddr); 266 oprintf(fp, "Structure Table Entries: %u\n", ep->smbe_stnum); 267 oprintf(fp, "DMI BCD Revision: 0x%x\n", ep->smbe_bcdrev); 268 } 269 270 static void 271 print_smbios_30(smbios_30_entry_t *ep, FILE *fp) 272 { 273 oprintf(fp, "Entry Point Anchor Tag: %*.*s\n", 274 (int)sizeof (ep->smbe_eanchor), (int)sizeof (ep->smbe_eanchor), 275 ep->smbe_eanchor); 276 277 oprintf(fp, "Entry Point Checksum: 0x%x\n", ep->smbe_ecksum); 278 oprintf(fp, "Entry Point Length: %u\n", ep->smbe_elen); 279 oprintf(fp, "SMBIOS Version: %u.%u\n", 280 ep->smbe_major, ep->smbe_minor); 281 oprintf(fp, "SMBIOS DocRev: 0x%x\n", ep->smbe_docrev); 282 oprintf(fp, "Entry Point Revision: 0x%x\n", ep->smbe_revision); 283 284 oprintf(fp, "Structure Table Length: %u\n", ep->smbe_stlen); 285 oprintf(fp, "Structure Table Address: 0x%" PRIx64 "\n", 286 ep->smbe_staddr); 287 } 288 289 static void 290 print_smbios(smbios_hdl_t *shp, FILE *fp) 291 { 292 smbios_entry_t ep; 293 294 switch (smbios_info_smbios(shp, &ep)) { 295 case SMBIOS_ENTRY_POINT_21: 296 print_smbios_21(&ep.ep21, fp); 297 break; 298 case SMBIOS_ENTRY_POINT_30: 299 print_smbios_30(&ep.ep30, fp); 300 break; 301 } 302 } 303 304 static void 305 print_common(const smbios_info_t *ip, FILE *fp) 306 { 307 if (ip->smbi_manufacturer[0] != '\0') 308 oprintf(fp, " Manufacturer: %s\n", ip->smbi_manufacturer); 309 if (ip->smbi_product[0] != '\0') 310 oprintf(fp, " Product: %s\n", ip->smbi_product); 311 if (ip->smbi_version[0] != '\0') 312 oprintf(fp, " Version: %s\n", ip->smbi_version); 313 if (ip->smbi_serial[0] != '\0') 314 oprintf(fp, " Serial Number: %s\n", ip->smbi_serial); 315 if (ip->smbi_asset[0] != '\0') 316 oprintf(fp, " Asset Tag: %s\n", ip->smbi_asset); 317 if (ip->smbi_location[0] != '\0') 318 oprintf(fp, " Location Tag: %s\n", ip->smbi_location); 319 if (ip->smbi_part[0] != '\0') 320 oprintf(fp, " Part Number: %s\n", ip->smbi_part); 321 } 322 323 static void 324 print_bios(smbios_hdl_t *shp, FILE *fp) 325 { 326 smbios_bios_t b; 327 328 (void) smbios_info_bios(shp, &b); 329 330 oprintf(fp, " Vendor: %s\n", b.smbb_vendor); 331 oprintf(fp, " Version String: %s\n", b.smbb_version); 332 oprintf(fp, " Release Date: %s\n", b.smbb_reldate); 333 oprintf(fp, " Address Segment: 0x%x\n", b.smbb_segment); 334 oprintf(fp, " ROM Size: %" PRIu64 " bytes\n", b.smbb_extromsize); 335 oprintf(fp, " Image Size: %u bytes\n", b.smbb_runsize); 336 337 flag64_printf(fp, "Characteristics", 338 b.smbb_cflags, sizeof (b.smbb_cflags) * NBBY, 339 smbios_bios_flag_name, smbios_bios_flag_desc); 340 341 if (b.smbb_nxcflags > SMB_BIOSXB_1) { 342 flag_printf(fp, "Characteristics Extension Byte 1", 343 b.smbb_xcflags[SMB_BIOSXB_1], 344 sizeof (b.smbb_xcflags[SMB_BIOSXB_1]) * NBBY, 345 smbios_bios_xb1_name, smbios_bios_xb1_desc); 346 } 347 348 if (b.smbb_nxcflags > SMB_BIOSXB_2) { 349 flag_printf(fp, "Characteristics Extension Byte 2", 350 b.smbb_xcflags[SMB_BIOSXB_2], 351 sizeof (b.smbb_xcflags[SMB_BIOSXB_2]) * NBBY, 352 smbios_bios_xb2_name, smbios_bios_xb2_desc); 353 } 354 355 if (b.smbb_nxcflags > SMB_BIOSXB_BIOS_MIN) { 356 oprintf(fp, " Version Number: %u.%u\n", 357 b.smbb_biosv.smbv_major, b.smbb_biosv.smbv_minor); 358 } 359 360 /* 361 * If the major and minor versions are 0xff then that indicates that the 362 * embedded controller does not exist. 363 */ 364 if (b.smbb_nxcflags > SMB_BIOSXB_ECFW_MIN && 365 b.smbb_ecfwv.smbv_major != 0xff && 366 b.smbb_ecfwv.smbv_minor != 0xff) { 367 oprintf(fp, " Embedded Ctlr Firmware Version Number: %u.%u\n", 368 b.smbb_ecfwv.smbv_major, b.smbb_ecfwv.smbv_minor); 369 } 370 } 371 372 static void 373 print_system(smbios_hdl_t *shp, FILE *fp) 374 { 375 smbios_system_t s; 376 uint_t i; 377 378 (void) smbios_info_system(shp, &s); 379 380 oprintf(fp, " UUID: "); 381 for (i = 0; i < s.smbs_uuidlen; i++) { 382 oprintf(fp, "%02x", s.smbs_uuid[i]); 383 if (i == 3 || i == 5 || i == 7 || i == 9) 384 oprintf(fp, "-"); 385 } 386 oprintf(fp, "\n"); 387 388 desc_printf(smbios_system_wakeup_desc(s.smbs_wakeup), 389 fp, " Wake-Up Event: 0x%x", s.smbs_wakeup); 390 391 oprintf(fp, " SKU Number: %s\n", s.smbs_sku); 392 oprintf(fp, " Family: %s\n", s.smbs_family); 393 } 394 395 static void 396 print_bboard(smbios_hdl_t *shp, id_t id, FILE *fp) 397 { 398 smbios_bboard_t b; 399 int chdl_cnt; 400 401 (void) smbios_info_bboard(shp, id, &b); 402 403 oprintf(fp, " Chassis: %u\n", (uint_t)b.smbb_chassis); 404 405 flag_printf(fp, "Flags", b.smbb_flags, sizeof (b.smbb_flags) * NBBY, 406 smbios_bboard_flag_name, smbios_bboard_flag_desc); 407 408 desc_printf(smbios_bboard_type_desc(b.smbb_type), 409 fp, " Board Type: 0x%x", b.smbb_type); 410 411 chdl_cnt = b.smbb_contn; 412 if (chdl_cnt != 0) { 413 id_t *chdl; 414 uint16_t hdl; 415 int i, n, cnt; 416 417 chdl = alloca(chdl_cnt * sizeof (id_t)); 418 cnt = smbios_info_contains(shp, id, chdl_cnt, chdl); 419 if (cnt > SMB_CONT_MAX) 420 return; 421 n = MIN(chdl_cnt, cnt); 422 423 oprintf(fp, "\n"); 424 for (i = 0; i < n; i++) { 425 hdl = (uint16_t)chdl[i]; 426 oprintf(fp, " Contained Handle: %u\n", hdl); 427 } 428 } 429 } 430 431 static void 432 print_chassis(smbios_hdl_t *shp, id_t id, FILE *fp) 433 { 434 smbios_chassis_t c; 435 int elem_cnt; 436 437 (void) smbios_info_chassis(shp, id, &c); 438 439 oprintf(fp, " OEM Data: 0x%x\n", c.smbc_oemdata); 440 oprintf(fp, " SKU number: %s\n", 441 c.smbc_sku[0] == '\0' ? "<unknown>" : c.smbc_sku); 442 oprintf(fp, " Lock Present: %s\n", c.smbc_lock ? "Y" : "N"); 443 444 desc_printf(smbios_chassis_type_desc(c.smbc_type), 445 fp, " Chassis Type: 0x%x", c.smbc_type); 446 447 desc_printf(smbios_chassis_state_desc(c.smbc_bustate), 448 fp, " Boot-Up State: 0x%x", c.smbc_bustate); 449 450 desc_printf(smbios_chassis_state_desc(c.smbc_psstate), 451 fp, " Power Supply State: 0x%x", c.smbc_psstate); 452 453 desc_printf(smbios_chassis_state_desc(c.smbc_thstate), 454 fp, " Thermal State: 0x%x", c.smbc_thstate); 455 456 oprintf(fp, " Chassis Height: %uu\n", c.smbc_uheight); 457 oprintf(fp, " Power Cords: %u\n", c.smbc_cords); 458 459 elem_cnt = c.smbc_elems; 460 oprintf(fp, " Element Records: %u\n", elem_cnt); 461 462 if (elem_cnt > 0) { 463 id_t *elems; 464 uint8_t type; 465 int i, n, cnt; 466 467 elems = alloca(c.smbc_elems * sizeof (id_t)); 468 cnt = smbios_info_contains(shp, id, elem_cnt, elems); 469 if (cnt > SMB_CONT_MAX) 470 return; 471 n = MIN(elem_cnt, cnt); 472 473 oprintf(fp, "\n"); 474 for (i = 0; i < n; i++) { 475 type = (uint8_t)elems[i]; 476 if (type & 0x80) { 477 /* SMBIOS structrure Type */ 478 desc_printf(smbios_type_name(type & 0x7f), fp, 479 " Contained SMBIOS structure Type: %u", 480 type & 0x80); 481 } else { 482 /* SMBIOS Base Board Type */ 483 desc_printf(smbios_bboard_type_desc(type), fp, 484 " Contained SMBIOS Base Board Type: 0x%x", 485 type); 486 } 487 } 488 } 489 } 490 491 static void 492 print_processor(smbios_hdl_t *shp, id_t id, FILE *fp) 493 { 494 smbios_processor_t p; 495 uint_t status; 496 497 (void) smbios_info_processor(shp, id, &p); 498 status = SMB_PRSTATUS_STATUS(p.smbp_status); 499 500 desc_printf(smbios_processor_family_desc(p.smbp_family), 501 fp, " Family: %u", p.smbp_family); 502 503 oprintf(fp, " CPUID: 0x%llx\n", (u_longlong_t)p.smbp_cpuid); 504 505 desc_printf(smbios_processor_type_desc(p.smbp_type), 506 fp, " Type: %u", p.smbp_type); 507 508 desc_printf(smbios_processor_upgrade_desc(p.smbp_upgrade), 509 fp, " Socket Upgrade: %u", p.smbp_upgrade); 510 511 oprintf(fp, " Socket Status: %s\n", 512 SMB_PRSTATUS_PRESENT(p.smbp_status) ? 513 "Populated" : "Not Populated"); 514 515 desc_printf(smbios_processor_status_desc(status), 516 fp, " Processor Status: %u", status); 517 518 if (SMB_PRV_LEGACY(p.smbp_voltage)) { 519 oprintf(fp, " Supported Voltages:"); 520 switch (p.smbp_voltage) { 521 case SMB_PRV_5V: 522 oprintf(fp, " 5.0V"); 523 break; 524 case SMB_PRV_33V: 525 oprintf(fp, " 3.3V"); 526 break; 527 case SMB_PRV_29V: 528 oprintf(fp, " 2.9V"); 529 break; 530 } 531 oprintf(fp, "\n"); 532 } else { 533 oprintf(fp, " Supported Voltages: %.1fV\n", 534 (float)SMB_PRV_VOLTAGE(p.smbp_voltage) / 10); 535 } 536 537 if (p.smbp_corecount != 0) { 538 oprintf(fp, " Core Count: %u\n", p.smbp_corecount); 539 } else { 540 oprintf(fp, " Core Count: Unknown\n"); 541 } 542 543 if (p.smbp_coresenabled != 0) { 544 oprintf(fp, " Cores Enabled: %u\n", p.smbp_coresenabled); 545 } else { 546 oprintf(fp, " Cores Enabled: Unknown\n"); 547 } 548 549 if (p.smbp_threadcount != 0) { 550 oprintf(fp, " Thread Count: %u\n", p.smbp_threadcount); 551 } else { 552 oprintf(fp, " Thread Count: Unknown\n"); 553 } 554 555 if (p.smbp_cflags) { 556 flag_printf(fp, "Processor Characteristics", 557 p.smbp_cflags, sizeof (p.smbp_cflags) * NBBY, 558 smbios_processor_core_flag_name, 559 smbios_processor_core_flag_desc); 560 } 561 562 if (p.smbp_clkspeed != 0) 563 oprintf(fp, " External Clock Speed: %uMHz\n", p.smbp_clkspeed); 564 else 565 oprintf(fp, " External Clock Speed: Unknown\n"); 566 567 if (p.smbp_maxspeed != 0) 568 oprintf(fp, " Maximum Speed: %uMHz\n", p.smbp_maxspeed); 569 else 570 oprintf(fp, " Maximum Speed: Unknown\n"); 571 572 if (p.smbp_curspeed != 0) 573 oprintf(fp, " Current Speed: %uMHz\n", p.smbp_curspeed); 574 else 575 oprintf(fp, " Current Speed: Unknown\n"); 576 577 id_printf(fp, " L1 Cache Handle: ", p.smbp_l1cache); 578 id_printf(fp, " L2 Cache Handle: ", p.smbp_l2cache); 579 id_printf(fp, " L3 Cache Handle: ", p.smbp_l3cache); 580 } 581 582 static void 583 print_cache(smbios_hdl_t *shp, id_t id, FILE *fp) 584 { 585 smbios_cache_t c; 586 587 (void) smbios_info_cache(shp, id, &c); 588 589 oprintf(fp, " Level: %u\n", c.smba_level); 590 oprintf(fp, " Maximum Installed Size: %" PRIu64 " bytes\n", 591 c.smba_maxsize2); 592 593 if (c.smba_size2 != 0) { 594 oprintf(fp, " Installed Size: %" PRIu64 " bytes\n", 595 c.smba_size2); 596 } else { 597 oprintf(fp, " Installed Size: Not Installed\n"); 598 } 599 600 if (c.smba_speed != 0) 601 oprintf(fp, " Speed: %uns\n", c.smba_speed); 602 else 603 oprintf(fp, " Speed: Unknown\n"); 604 605 flag_printf(fp, "Supported SRAM Types", 606 c.smba_stype, sizeof (c.smba_stype) * NBBY, 607 smbios_cache_ctype_name, smbios_cache_ctype_desc); 608 609 desc_printf(smbios_cache_ctype_desc(c.smba_ctype), 610 fp, " Current SRAM Type: 0x%x", c.smba_ctype); 611 612 desc_printf(smbios_cache_ecc_desc(c.smba_etype), 613 fp, " Error Correction Type: %u", c.smba_etype); 614 615 desc_printf(smbios_cache_logical_desc(c.smba_ltype), 616 fp, " Logical Cache Type: %u", c.smba_ltype); 617 618 desc_printf(smbios_cache_assoc_desc(c.smba_assoc), 619 fp, " Associativity: %u", c.smba_assoc); 620 621 desc_printf(smbios_cache_mode_desc(c.smba_mode), 622 fp, " Mode: %u", c.smba_mode); 623 624 desc_printf(smbios_cache_loc_desc(c.smba_location), 625 fp, " Location: %u", c.smba_location); 626 627 flag_printf(fp, "Flags", c.smba_flags, sizeof (c.smba_flags) * NBBY, 628 smbios_cache_flag_name, smbios_cache_flag_desc); 629 } 630 631 static void 632 print_port(smbios_hdl_t *shp, id_t id, FILE *fp) 633 { 634 smbios_port_t p; 635 636 (void) smbios_info_port(shp, id, &p); 637 638 oprintf(fp, " Internal Reference Designator: %s\n", p.smbo_iref); 639 oprintf(fp, " External Reference Designator: %s\n", p.smbo_eref); 640 641 desc_printf(smbios_port_conn_desc(p.smbo_itype), 642 fp, " Internal Connector Type: %u", p.smbo_itype); 643 644 desc_printf(smbios_port_conn_desc(p.smbo_etype), 645 fp, " External Connector Type: %u", p.smbo_etype); 646 647 desc_printf(smbios_port_type_desc(p.smbo_ptype), 648 fp, " Port Type: %u", p.smbo_ptype); 649 } 650 651 static void 652 print_slot(smbios_hdl_t *shp, id_t id, FILE *fp) 653 { 654 smbios_slot_t s; 655 smbios_version_t v; 656 657 (void) smbios_info_slot(shp, id, &s); 658 smbios_info_smbios_version(shp, &v); 659 660 oprintf(fp, " Reference Designator: %s\n", s.smbl_name); 661 oprintf(fp, " Slot ID: 0x%x\n", s.smbl_id); 662 663 desc_printf(smbios_slot_type_desc(s.smbl_type), 664 fp, " Type: 0x%x", s.smbl_type); 665 666 desc_printf(smbios_slot_width_desc(s.smbl_width), 667 fp, " Width: 0x%x", s.smbl_width); 668 669 desc_printf(smbios_slot_usage_desc(s.smbl_usage), 670 fp, " Usage: 0x%x", s.smbl_usage); 671 672 desc_printf(smbios_slot_length_desc(s.smbl_length), 673 fp, " Length: 0x%x", s.smbl_length); 674 675 flag_printf(fp, "Slot Characteristics 1", 676 s.smbl_ch1, sizeof (s.smbl_ch1) * NBBY, 677 smbios_slot_ch1_name, smbios_slot_ch1_desc); 678 679 flag_printf(fp, "Slot Characteristics 2", 680 s.smbl_ch2, sizeof (s.smbl_ch2) * NBBY, 681 smbios_slot_ch2_name, smbios_slot_ch2_desc); 682 683 if (check_oem(shp) != 0 && !smbios_vergteq(&v, 2, 6)) 684 return; 685 686 oprintf(fp, " Segment Group: %u\n", s.smbl_sg); 687 oprintf(fp, " Bus Number: %u\n", s.smbl_bus); 688 oprintf(fp, " Device/Function Number: %u/%u\n", s.smbl_df >> 3, 689 s.smbl_df & 0x7); 690 691 if (s.smbl_dbw != 0) { 692 oprintf(fp, " Data Bus Width: %d\n", s.smbl_dbw); 693 } 694 695 if (s.smbl_npeers > 0) { 696 smbios_slot_peer_t *peer; 697 uint_t i, npeers; 698 699 if (smbios_info_slot_peers(shp, id, &npeers, &peer) != 0) { 700 smbios_warn(shp, "failed to read slot peer " 701 "information"); 702 return; 703 } 704 705 for (i = 0; i < npeers; i++) { 706 oprintf(fp, " Slot Peer %u:\n", i); 707 oprintf(fp, " Segment group: %u\n", 708 peer[i].smblp_group); 709 oprintf(fp, " Bus/Device/Function: %u/%u/%u", 710 peer[i].smblp_bus, peer[i].smblp_device, 711 peer[i].smblp_function); 712 oprintf(fp, " Electrical width: %u\n", 713 peer[i].smblp_data_width); 714 } 715 716 smbios_info_slot_peers_free(shp, npeers, peer); 717 } 718 } 719 720 static void 721 print_obdevs_ext(smbios_hdl_t *shp, id_t id, FILE *fp) 722 { 723 boolean_t enabled; 724 smbios_obdev_ext_t oe; 725 const char *type; 726 727 (void) smbios_info_obdevs_ext(shp, id, &oe); 728 729 /* 730 * Bit 7 is always whether or not the device is enabled while bits 0:6 731 * are the actual device type. 732 */ 733 enabled = oe.smboe_dtype >> 7; 734 type = smbios_onboard_type_desc(oe.smboe_dtype & 0x7f); 735 736 oprintf(fp, " Reference Designator: %s\n", oe.smboe_name); 737 oprintf(fp, " Device Enabled: %s\n", enabled == B_TRUE ? "true" : 738 "false"); 739 oprintf(fp, " Device Type: %s\n", type); 740 oprintf(fp, " Device Type Instance: %u\n", oe.smboe_dti); 741 oprintf(fp, " Segment Group Number: %u\n", oe.smboe_sg); 742 oprintf(fp, " Bus Number: %u\n", oe.smboe_bus); 743 oprintf(fp, " Device/Function Number: %u\n", oe.smboe_df); 744 } 745 746 static void 747 print_obdevs(smbios_hdl_t *shp, id_t id, FILE *fp) 748 { 749 smbios_obdev_t *argv; 750 int i, argc; 751 752 if ((argc = smbios_info_obdevs(shp, id, 0, NULL)) > 0) { 753 argv = alloca(sizeof (smbios_obdev_t) * argc); 754 (void) smbios_info_obdevs(shp, id, argc, argv); 755 for (i = 0; i < argc; i++) 756 oprintf(fp, " %s\n", argv[i].smbd_name); 757 } 758 } 759 760 static void 761 print_strtab(smbios_hdl_t *shp, id_t id, FILE *fp) 762 { 763 const char **argv; 764 int i, argc; 765 766 if ((argc = smbios_info_strtab(shp, id, 0, NULL)) > 0) { 767 argv = alloca(sizeof (char *) * argc); 768 (void) smbios_info_strtab(shp, id, argc, argv); 769 for (i = 0; i < argc; i++) 770 oprintf(fp, " %s\n", argv[i]); 771 } 772 } 773 774 static void 775 print_lang(smbios_hdl_t *shp, id_t id, FILE *fp) 776 { 777 smbios_lang_t l; 778 779 (void) smbios_info_lang(shp, &l); 780 781 oprintf(fp, " Current Language: %s\n", l.smbla_cur); 782 oprintf(fp, " Language String Format: %u\n", l.smbla_fmt); 783 oprintf(fp, " Number of Installed Languages: %u\n", l.smbla_num); 784 oprintf(fp, " Installed Languages:\n"); 785 786 print_strtab(shp, id, fp); 787 } 788 789 /*ARGSUSED*/ 790 static void 791 print_evlog(smbios_hdl_t *shp, id_t id, FILE *fp) 792 { 793 smbios_evlog_t ev; 794 uint32_t i; 795 796 (void) smbios_info_eventlog(shp, &ev); 797 798 oprintf(fp, " Log Area Size: %lu bytes\n", (ulong_t)ev.smbev_size); 799 oprintf(fp, " Header Offset: %lu\n", (ulong_t)ev.smbev_hdr); 800 oprintf(fp, " Data Offset: %lu\n", (ulong_t)ev.smbev_data); 801 802 desc_printf(smbios_evlog_method_desc(ev.smbev_method), 803 fp, " Data Access Method: %u", ev.smbev_method); 804 805 flag_printf(fp, "Log Flags", 806 ev.smbev_flags, sizeof (ev.smbev_flags) * NBBY, 807 smbios_evlog_flag_name, smbios_evlog_flag_desc); 808 809 desc_printf(smbios_evlog_format_desc(ev.smbev_format), 810 fp, " Log Header Format: %u", ev.smbev_format); 811 812 oprintf(fp, " Update Token: 0x%x\n", ev.smbev_token); 813 oprintf(fp, " Data Access Address: "); 814 815 switch (ev.smbev_method) { 816 case SMB_EVM_1x1i_1x1d: 817 case SMB_EVM_2x1i_1x1d: 818 case SMB_EVM_1x2i_1x1d: 819 oprintf(fp, "Index Address 0x%x, Data Address 0x%x\n", 820 ev.smbev_addr.eva_io.evi_iaddr, 821 ev.smbev_addr.eva_io.evi_daddr); 822 break; 823 case SMB_EVM_GPNV: 824 oprintf(fp, "0x%x\n", ev.smbev_addr.eva_gpnv); 825 break; 826 default: 827 oprintf(fp, "0x%x\n", ev.smbev_addr.eva_addr); 828 } 829 830 oprintf(fp, " Type Descriptors:\n"); 831 832 for (i = 0; i < ev.smbev_typec; i++) { 833 oprintf(fp, " %u: Log Type 0x%x, Data Type 0x%x\n", i, 834 ev.smbev_typev[i].smbevt_ltype, 835 ev.smbev_typev[i].smbevt_dtype); 836 } 837 } 838 839 static void 840 print_bytes(const uint8_t *data, size_t size, FILE *fp) 841 { 842 size_t row, rows = P2ROUNDUP(size, 16) / 16; 843 size_t col, cols; 844 845 char buf[17]; 846 uint8_t x; 847 848 oprintf(fp, "\n offset: 0 1 2 3 4 5 6 7 8 9 a b c d e f " 849 "0123456789abcdef\n"); 850 851 for (row = 0; row < rows; row++) { 852 oprintf(fp, " %#6lx: ", (ulong_t)row * 16); 853 cols = MIN(size - row * 16, 16); 854 855 for (col = 0; col < cols; col++) { 856 if (col % 4 == 0) 857 oprintf(fp, " "); 858 x = *data++; 859 oprintf(fp, "%02x", x); 860 buf[col] = x <= ' ' || x > '~' ? '.' : x; 861 } 862 863 for (; col < 16; col++) { 864 if (col % 4 == 0) 865 oprintf(fp, " "); 866 oprintf(fp, " "); 867 buf[col] = ' '; 868 } 869 870 buf[col] = '\0'; 871 oprintf(fp, " %s\n", buf); 872 } 873 874 oprintf(fp, "\n"); 875 } 876 877 static void 878 print_memarray(smbios_hdl_t *shp, id_t id, FILE *fp) 879 { 880 smbios_memarray_t ma; 881 882 (void) smbios_info_memarray(shp, id, &ma); 883 884 desc_printf(smbios_memarray_loc_desc(ma.smbma_location), 885 fp, " Location: %u", ma.smbma_location); 886 887 desc_printf(smbios_memarray_use_desc(ma.smbma_use), 888 fp, " Use: %u", ma.smbma_use); 889 890 desc_printf(smbios_memarray_ecc_desc(ma.smbma_ecc), 891 fp, " ECC: %u", ma.smbma_ecc); 892 893 oprintf(fp, " Number of Slots/Sockets: %u\n", ma.smbma_ndevs); 894 id_printf(fp, " Memory Error Data: ", ma.smbma_err); 895 oprintf(fp, " Max Capacity: %llu bytes\n", 896 (u_longlong_t)ma.smbma_size); 897 } 898 899 static void 900 print_memdevice(smbios_hdl_t *shp, id_t id, FILE *fp) 901 { 902 smbios_memdevice_t md; 903 904 (void) smbios_info_memdevice(shp, id, &md); 905 906 id_printf(fp, " Physical Memory Array: ", md.smbmd_array); 907 id_printf(fp, " Memory Error Data: ", md.smbmd_error); 908 909 if (md.smbmd_twidth != -1u) 910 oprintf(fp, " Total Width: %u bits\n", md.smbmd_twidth); 911 else 912 oprintf(fp, " Total Width: Unknown\n"); 913 914 if (md.smbmd_dwidth != -1u) 915 oprintf(fp, " Data Width: %u bits\n", md.smbmd_dwidth); 916 else 917 oprintf(fp, " Data Width: Unknown\n"); 918 919 switch (md.smbmd_size) { 920 case -1ull: 921 oprintf(fp, " Size: Unknown\n"); 922 break; 923 case 0: 924 oprintf(fp, " Size: Not Populated\n"); 925 break; 926 default: 927 oprintf(fp, " Size: %llu bytes\n", 928 (u_longlong_t)md.smbmd_size); 929 } 930 931 desc_printf(smbios_memdevice_form_desc(md.smbmd_form), 932 fp, " Form Factor: %u", md.smbmd_form); 933 934 if (md.smbmd_set == 0) 935 oprintf(fp, " Set: None\n"); 936 else if (md.smbmd_set == (uint8_t)-1u) 937 oprintf(fp, " Set: Unknown\n"); 938 else 939 oprintf(fp, " Set: %u\n", md.smbmd_set); 940 941 if (md.smbmd_rank != 0) { 942 desc_printf(smbios_memdevice_rank_desc(md.smbmd_rank), 943 fp, " Rank: %u", md.smbmd_rank); 944 } else { 945 oprintf(fp, " Rank: Unknown\n"); 946 } 947 948 desc_printf(smbios_memdevice_type_desc(md.smbmd_type), 949 fp, " Memory Type: %u", md.smbmd_type); 950 951 flag_printf(fp, "Flags", md.smbmd_flags, sizeof (md.smbmd_flags) * NBBY, 952 smbios_memdevice_flag_name, smbios_memdevice_flag_desc); 953 954 if (md.smbmd_extspeed != 0) { 955 oprintf(fp, " Speed: %" PRIu64 " MT/s\n", md.smbmd_extspeed); 956 } else { 957 oprintf(fp, " Speed: Unknown\n"); 958 } 959 960 if (md.smbmd_extclkspeed != 0) { 961 oprintf(fp, " Configured Speed: %" PRIu64 " MT/s\n", 962 md.smbmd_extclkspeed); 963 } else { 964 oprintf(fp, " Configured Speed: Unknown\n"); 965 } 966 967 oprintf(fp, " Device Locator: %s\n", md.smbmd_dloc); 968 oprintf(fp, " Bank Locator: %s\n", md.smbmd_bloc); 969 970 if (md.smbmd_minvolt != 0) { 971 oprintf(fp, " Minimum Voltage: %.2fV\n", 972 md.smbmd_minvolt / 1000.0); 973 } else { 974 oprintf(fp, " Minimum Voltage: Unknown\n"); 975 } 976 977 if (md.smbmd_maxvolt != 0) { 978 oprintf(fp, " Maximum Voltage: %.2fV\n", 979 md.smbmd_maxvolt / 1000.0); 980 } else { 981 oprintf(fp, " Maximum Voltage: Unknown\n"); 982 } 983 984 if (md.smbmd_confvolt != 0) { 985 oprintf(fp, " Configured Voltage: %.2fV\n", 986 md.smbmd_confvolt / 1000.0); 987 } else { 988 oprintf(fp, " Configured Voltage: Unknown\n"); 989 } 990 991 if (md.smbmd_memtech != 0) { 992 desc_printf(smbios_memdevice_memtech_desc(md.smbmd_memtech), 993 fp, " Memory Technology: %u", md.smbmd_memtech); 994 } 995 996 if (md.smbmd_opcap_flags != 0) { 997 flag_printf(fp, " Operating Mode Capabilities", 998 md.smbmd_opcap_flags, sizeof (md.smbmd_opcap_flags) * NBBY, 999 smbios_memdevice_op_capab_name, 1000 smbios_memdevice_op_capab_desc); 1001 } 1002 1003 if (md.smbmd_firmware_rev[0] != '\0') { 1004 oprintf(fp, " Firmware Revision: %s\n", md.smbmd_firmware_rev); 1005 } 1006 1007 if (md.smbmd_modmfg_id != 0) { 1008 jedec_print(fp, "Module Manufacturer ID", md.smbmd_modmfg_id); 1009 } 1010 1011 if (md.smbmd_modprod_id != 0) { 1012 jedec_print(fp, "Module Product ID", md.smbmd_modprod_id); 1013 } 1014 1015 if (md.smbmd_cntrlmfg_id != 0) { 1016 jedec_print(fp, "Memory Subsystem Controller Manufacturer ID", 1017 md.smbmd_cntrlmfg_id); 1018 } 1019 1020 if (md.smbmd_cntrlprod_id != 0) { 1021 jedec_print(fp, "Memory Subsystem Controller Product ID", 1022 md.smbmd_cntrlprod_id); 1023 } 1024 1025 if (md.smbmd_nvsize == UINT64_MAX) { 1026 oprintf(fp, " Non-volatile Size: Unknown\n"); 1027 } else if (md.smbmd_nvsize != 0) { 1028 oprintf(fp, " Non-volatile Size: %llu bytes\n", 1029 (u_longlong_t)md.smbmd_nvsize); 1030 } 1031 1032 if (md.smbmd_volatile_size == UINT64_MAX) { 1033 oprintf(fp, " Volatile Size: Unknown\n"); 1034 } else if (md.smbmd_volatile_size != 0) { 1035 oprintf(fp, " Volatile Size: %llu bytes\n", 1036 (u_longlong_t)md.smbmd_volatile_size); 1037 } 1038 1039 if (md.smbmd_cache_size == UINT64_MAX) { 1040 oprintf(fp, " Cache Size: Unknown\n"); 1041 } else if (md.smbmd_cache_size != 0) { 1042 oprintf(fp, " Cache Size: %llu bytes\n", 1043 (u_longlong_t)md.smbmd_cache_size); 1044 } 1045 1046 if (md.smbmd_logical_size == UINT64_MAX) { 1047 oprintf(fp, " Logical Size: Unknown\n"); 1048 } else if (md.smbmd_logical_size != 0) { 1049 oprintf(fp, " Logical Size: %llu bytes\n", 1050 (u_longlong_t)md.smbmd_logical_size); 1051 } 1052 } 1053 1054 static void 1055 print_memarrmap(smbios_hdl_t *shp, id_t id, FILE *fp) 1056 { 1057 smbios_memarrmap_t ma; 1058 1059 (void) smbios_info_memarrmap(shp, id, &ma); 1060 1061 id_printf(fp, " Physical Memory Array: ", ma.smbmam_array); 1062 oprintf(fp, " Devices per Row: %u\n", ma.smbmam_width); 1063 1064 oprintf(fp, " Physical Address: 0x%llx\n Size: %llu bytes\n", 1065 (u_longlong_t)ma.smbmam_addr, (u_longlong_t)ma.smbmam_size); 1066 } 1067 1068 static void 1069 print_memdevmap(smbios_hdl_t *shp, id_t id, FILE *fp) 1070 { 1071 smbios_memdevmap_t md; 1072 1073 (void) smbios_info_memdevmap(shp, id, &md); 1074 1075 id_printf(fp, " Memory Device: ", md.smbmdm_device); 1076 id_printf(fp, " Memory Array Mapped Address: ", md.smbmdm_arrmap); 1077 1078 oprintf(fp, " Physical Address: 0x%llx\n Size: %llu bytes\n", 1079 (u_longlong_t)md.smbmdm_addr, (u_longlong_t)md.smbmdm_size); 1080 1081 oprintf(fp, " Partition Row Position: %u\n", md.smbmdm_rpos); 1082 oprintf(fp, " Interleave Position: %u\n", md.smbmdm_ipos); 1083 oprintf(fp, " Interleave Data Depth: %u\n", md.smbmdm_idepth); 1084 } 1085 1086 static void 1087 print_hwsec(smbios_hdl_t *shp, FILE *fp) 1088 { 1089 smbios_hwsec_t h; 1090 1091 (void) smbios_info_hwsec(shp, &h); 1092 1093 desc_printf(smbios_hwsec_desc(h.smbh_pwr_ps), 1094 fp, " Power-On Password Status: %u", h.smbh_pwr_ps); 1095 desc_printf(smbios_hwsec_desc(h.smbh_kbd_ps), 1096 fp, " Keyboard Password Status: %u", h.smbh_kbd_ps); 1097 desc_printf(smbios_hwsec_desc(h.smbh_adm_ps), 1098 fp, " Administrator Password Status: %u", h.smbh_adm_ps); 1099 desc_printf(smbios_hwsec_desc(h.smbh_pan_ps), 1100 fp, " Front Panel Reset Status: %u", h.smbh_pan_ps); 1101 } 1102 1103 static void 1104 print_vprobe(smbios_hdl_t *shp, id_t id, FILE *fp) 1105 { 1106 smbios_vprobe_t vp; 1107 1108 if (smbios_info_vprobe(shp, id, &vp) != 0) { 1109 smbios_warn(shp, "failed to read voltage probe information"); 1110 return; 1111 } 1112 1113 oprintf(fp, " Description: %s\n", vp.smbvp_description != NULL ? 1114 vp.smbvp_description : "unknown"); 1115 desc_printf(smbios_vprobe_loc_desc(vp.smbvp_location), 1116 fp, " Location: %u", vp.smbvp_location); 1117 desc_printf(smbios_vprobe_status_desc(vp.smbvp_status), 1118 fp, " Status: %u", vp.smbvp_status); 1119 1120 if (vp.smbvp_maxval != SMB_PROBE_UNKNOWN_VALUE) { 1121 oprintf(fp, " Maximum Possible Voltage: %u mV\n", 1122 vp.smbvp_maxval); 1123 } else { 1124 oprintf(fp, " Maximum Possible Voltage: unknown\n"); 1125 } 1126 1127 if (vp.smbvp_minval != SMB_PROBE_UNKNOWN_VALUE) { 1128 oprintf(fp, " Minimum Possible Voltage: %u mV\n", 1129 vp.smbvp_minval); 1130 } else { 1131 oprintf(fp, " Minimum Possible Voltage: unknown\n"); 1132 } 1133 1134 if (vp.smbvp_resolution != SMB_PROBE_UNKNOWN_VALUE) { 1135 oprintf(fp, " Probe Resolution: %u.%u mV\n", 1136 vp.smbvp_resolution / 10, 1137 vp.smbvp_resolution % 10); 1138 } else { 1139 oprintf(fp, " Probe Resolution: unknown\n"); 1140 } 1141 1142 if (vp.smbvp_tolerance != SMB_PROBE_UNKNOWN_VALUE) { 1143 oprintf(fp, " Probe Tolerance: +/-%u mV\n", 1144 vp.smbvp_tolerance); 1145 } else { 1146 oprintf(fp, " Probe Tolerance: unknown\n"); 1147 } 1148 1149 if (vp.smbvp_accuracy != SMB_PROBE_UNKNOWN_VALUE) { 1150 oprintf(fp, " Probe Accuracy: +/-%u.%02u%%\n", 1151 vp.smbvp_accuracy / 100, 1152 vp.smbvp_accuracy % 100); 1153 } else { 1154 oprintf(fp, " Probe Accuracy: unknown\n"); 1155 } 1156 1157 oprintf(fp, " OEM- or BIOS- defined value: 0x%x\n", vp.smbvp_oem); 1158 1159 if (vp.smbvp_nominal != SMB_PROBE_UNKNOWN_VALUE) { 1160 oprintf(fp, " Probe Nominal Value: %u mV\n", vp.smbvp_nominal); 1161 } else { 1162 oprintf(fp, " Probe Nominal Value: unknown\n"); 1163 } 1164 } 1165 1166 static void 1167 print_cooldev(smbios_hdl_t *shp, id_t id, FILE *fp) 1168 { 1169 smbios_cooldev_t cd; 1170 1171 if (smbios_info_cooldev(shp, id, &cd) != 0) { 1172 smbios_warn(shp, "failed to read cooling device " 1173 "information"); 1174 return; 1175 } 1176 1177 id_printf(fp, " Temperature Probe Handle: ", cd.smbcd_tprobe); 1178 desc_printf(smbios_cooldev_type_desc(cd.smbcd_type), 1179 fp, " Device Type: %u", cd.smbcd_type); 1180 desc_printf(smbios_cooldev_status_desc(cd.smbcd_status), 1181 fp, " Status: %u", cd.smbcd_status); 1182 oprintf(fp, " Cooling Unit Group: %u\n", cd.smbcd_group); 1183 oprintf(fp, " OEM- or BIOS- defined data: 0x%x\n", cd.smbcd_oem); 1184 if (cd.smbcd_nominal != SMB_PROBE_UNKNOWN_VALUE) { 1185 oprintf(fp, " Nominal Speed: %u RPM\n", cd.smbcd_nominal); 1186 } else { 1187 oprintf(fp, " Nominal Speed: unknown\n"); 1188 } 1189 1190 if (cd.smbcd_descr != NULL && cd.smbcd_descr[0] != '\0') { 1191 oprintf(fp, " Description: %s\n", cd.smbcd_descr); 1192 } 1193 } 1194 1195 static void 1196 print_tprobe(smbios_hdl_t *shp, id_t id, FILE *fp) 1197 { 1198 smbios_tprobe_t tp; 1199 1200 if (smbios_info_tprobe(shp, id, &tp) != 0) { 1201 smbios_warn(shp, "failed to read temperature probe " 1202 "information"); 1203 return; 1204 } 1205 1206 oprintf(fp, " Description: %s\n", tp.smbtp_description != NULL ? 1207 tp.smbtp_description : "unknown"); 1208 desc_printf(smbios_tprobe_loc_desc(tp.smbtp_location), 1209 fp, " Location: %u", tp.smbtp_location); 1210 desc_printf(smbios_tprobe_status_desc(tp.smbtp_status), 1211 fp, " Status: %u", tp.smbtp_status); 1212 1213 if (tp.smbtp_maxval != SMB_PROBE_UNKNOWN_VALUE) { 1214 oprintf(fp, " Maximum Possible Temperature: %u.%u C\n", 1215 tp.smbtp_maxval / 10, tp.smbtp_maxval % 10); 1216 } else { 1217 oprintf(fp, " Maximum Possible Temperature: unknown\n"); 1218 } 1219 1220 if (tp.smbtp_minval != SMB_PROBE_UNKNOWN_VALUE) { 1221 oprintf(fp, " Minimum Possible Temperature: %u.%u C\n", 1222 tp.smbtp_minval / 10, tp.smbtp_minval % 10); 1223 } else { 1224 oprintf(fp, " Minimum Possible Temperature: unknown\n"); 1225 } 1226 1227 if (tp.smbtp_resolution != SMB_PROBE_UNKNOWN_VALUE) { 1228 oprintf(fp, " Probe Resolution: %u.%03u C\n", 1229 tp.smbtp_resolution / 1000, 1230 tp.smbtp_resolution % 1000); 1231 } else { 1232 oprintf(fp, " Probe Resolution: unknown\n"); 1233 } 1234 1235 if (tp.smbtp_tolerance != SMB_PROBE_UNKNOWN_VALUE) { 1236 oprintf(fp, " Probe Tolerance: +/-%u.%u C\n", 1237 tp.smbtp_tolerance / 10, tp.smbtp_tolerance % 10); 1238 } else { 1239 oprintf(fp, " Probe Tolerance: unknown\n"); 1240 } 1241 1242 if (tp.smbtp_accuracy != SMB_PROBE_UNKNOWN_VALUE) { 1243 oprintf(fp, " Probe Accuracy: +/-%u.%02u%%\n", 1244 tp.smbtp_accuracy / 100, 1245 tp.smbtp_accuracy % 100); 1246 } else { 1247 oprintf(fp, " Probe Accuracy: unknown\n"); 1248 } 1249 1250 oprintf(fp, " OEM- or BIOS- defined value: 0x%x\n", tp.smbtp_oem); 1251 1252 if (tp.smbtp_nominal != SMB_PROBE_UNKNOWN_VALUE) { 1253 oprintf(fp, " Probe Nominal Value: %u.%u C\n", 1254 tp.smbtp_nominal / 10, tp.smbtp_nominal % 10); 1255 } else { 1256 oprintf(fp, " Probe Nominal Value: unknown\n"); 1257 } 1258 } 1259 1260 static void 1261 print_iprobe(smbios_hdl_t *shp, id_t id, FILE *fp) 1262 { 1263 smbios_iprobe_t ip; 1264 1265 if (smbios_info_iprobe(shp, id, &ip) != 0) { 1266 smbios_warn(shp, "failed to read current probe information"); 1267 return; 1268 } 1269 1270 oprintf(fp, " Description: %s\n", ip.smbip_description != NULL ? 1271 ip.smbip_description : "unknown"); 1272 desc_printf(smbios_iprobe_loc_desc(ip.smbip_location), 1273 fp, " Location: %u", ip.smbip_location); 1274 desc_printf(smbios_iprobe_status_desc(ip.smbip_status), 1275 fp, " Status: %u", ip.smbip_status); 1276 1277 if (ip.smbip_maxval != SMB_PROBE_UNKNOWN_VALUE) { 1278 oprintf(fp, " Maximum Possible Current: %u mA\n", 1279 ip.smbip_maxval); 1280 } else { 1281 oprintf(fp, " Maximum Possible Current: unknown\n"); 1282 } 1283 1284 if (ip.smbip_minval != SMB_PROBE_UNKNOWN_VALUE) { 1285 oprintf(fp, " Minimum Possible Current: %u mA\n", 1286 ip.smbip_minval); 1287 } else { 1288 oprintf(fp, " Minimum Possible Current: unknown\n"); 1289 } 1290 1291 if (ip.smbip_resolution != SMB_PROBE_UNKNOWN_VALUE) { 1292 oprintf(fp, " Probe Resolution: %u.%u mA\n", 1293 ip.smbip_resolution / 10, 1294 ip.smbip_resolution % 10); 1295 } else { 1296 oprintf(fp, " Probe Resolution: unknown\n"); 1297 } 1298 1299 if (ip.smbip_tolerance != SMB_PROBE_UNKNOWN_VALUE) { 1300 oprintf(fp, " Probe Tolerance: +/-%u mA\n", 1301 ip.smbip_tolerance); 1302 } else { 1303 oprintf(fp, " Probe Tolerance: unknown\n"); 1304 } 1305 1306 if (ip.smbip_accuracy != SMB_PROBE_UNKNOWN_VALUE) { 1307 oprintf(fp, " Probe Accuracy: +/-%u.%02u%%\n", 1308 ip.smbip_accuracy / 100, 1309 ip.smbip_accuracy % 100); 1310 } else { 1311 oprintf(fp, " Probe Accuracy: unknown\n"); 1312 } 1313 1314 oprintf(fp, " OEM- or BIOS- defined value: 0x%x\n", ip.smbip_oem); 1315 1316 if (ip.smbip_nominal != SMB_PROBE_UNKNOWN_VALUE) { 1317 oprintf(fp, " Probe Nominal Value: %u mA\n", ip.smbip_nominal); 1318 } else { 1319 oprintf(fp, " Probe Nominal Value: unknown\n"); 1320 } 1321 } 1322 1323 1324 static void 1325 print_boot(smbios_hdl_t *shp, FILE *fp) 1326 { 1327 smbios_boot_t b; 1328 1329 (void) smbios_info_boot(shp, &b); 1330 1331 desc_printf(smbios_boot_desc(b.smbt_status), 1332 fp, " Boot Status Code: 0x%x", b.smbt_status); 1333 1334 if (b.smbt_size != 0) { 1335 oprintf(fp, " Boot Data (%lu bytes):\n", (ulong_t)b.smbt_size); 1336 print_bytes(b.smbt_data, b.smbt_size, fp); 1337 } 1338 } 1339 1340 static void 1341 print_ipmi(smbios_hdl_t *shp, FILE *fp) 1342 { 1343 smbios_ipmi_t i; 1344 1345 (void) smbios_info_ipmi(shp, &i); 1346 1347 desc_printf(smbios_ipmi_type_desc(i.smbip_type), 1348 fp, " Type: %u", i.smbip_type); 1349 1350 oprintf(fp, " BMC IPMI Version: %u.%u\n", 1351 i.smbip_vers.smbv_major, i.smbip_vers.smbv_minor); 1352 1353 oprintf(fp, " i2c Bus Slave Address: 0x%x\n", i.smbip_i2c); 1354 oprintf(fp, " NV Storage Device Bus ID: 0x%x\n", i.smbip_bus); 1355 oprintf(fp, " BMC Base Address: 0x%llx\n", (u_longlong_t)i.smbip_addr); 1356 oprintf(fp, " Interrupt Number: %u\n", i.smbip_intr); 1357 oprintf(fp, " Register Spacing: %u\n", i.smbip_regspacing); 1358 1359 flag_printf(fp, "Flags", i.smbip_flags, sizeof (i.smbip_flags) * NBBY, 1360 smbios_ipmi_flag_name, smbios_ipmi_flag_desc); 1361 } 1362 1363 static void 1364 print_powersup(smbios_hdl_t *shp, id_t id, FILE *fp) 1365 { 1366 smbios_powersup_t p; 1367 1368 if (smbios_info_powersup(shp, id, &p) != 0) { 1369 smbios_warn(shp, "failed to read power supply information"); 1370 return; 1371 } 1372 1373 oprintf(fp, " Power Supply Group: %u\n", p.smbps_group); 1374 if (p.smbps_maxout != 0x8000) { 1375 oprintf(fp, " Maximum Output: %llu mW\n", p.smbps_maxout); 1376 } else { 1377 oprintf(fp, " Maximum Output: unknown\n"); 1378 } 1379 1380 flag_printf(fp, "Characteristics", p.smbps_flags, 1381 sizeof (p.smbps_flags) * NBBY, smbios_powersup_flag_name, 1382 smbios_powersup_flag_desc); 1383 1384 desc_printf(smbios_powersup_input_desc(p.smbps_ivrs), 1385 fp, " Input Voltage Range Switching: %u", p.smbps_ivrs); 1386 desc_printf(smbios_powersup_status_desc(p.smbps_status), 1387 fp, " Status: %u", p.smbps_status); 1388 desc_printf(smbios_powersup_type_desc(p.smbps_pstype), 1389 fp, " Type: %u", p.smbps_pstype); 1390 1391 if (p.smbps_vprobe != 0xffff) { 1392 oprintf(fp, " Voltage Probe Handle: %lu\n", p.smbps_vprobe); 1393 } 1394 1395 if (p.smbps_cooldev != 0xffff) { 1396 oprintf(fp, " Cooling Device Handle: %lu\n", p.smbps_cooldev); 1397 } 1398 1399 if (p.smbps_iprobe != 0xffff) { 1400 oprintf(fp, " Current Probe Handle: %lu\n", p.smbps_iprobe); 1401 } 1402 } 1403 1404 static void 1405 print_processor_info_riscv(smbios_hdl_t *shp, id_t id, FILE *fp) 1406 { 1407 smbios_processor_info_riscv_t rv; 1408 1409 if (smbios_info_processor_riscv(shp, id, &rv) != 0) { 1410 smbios_warn(shp, "failed to read RISC-V specific processor " 1411 "information"); 1412 return; 1413 } 1414 1415 if (rv.smbpirv_boothart != 0) { 1416 oprintf(fp, " Boot Hart\n"); 1417 } 1418 u128_print(fp, " Hart ID", rv.smbpirv_hartid); 1419 u128_print(fp, " Vendor ID", rv.smbpirv_vendid); 1420 u128_print(fp, " Architecture ID", rv.smbpirv_archid); 1421 u128_print(fp, " Implementation ID", rv.smbpirv_machid); 1422 flag64_printf(fp, " ISA", rv.smbpirv_isa, 1423 sizeof (rv.smbpirv_isa) * NBBY, smbios_riscv_isa_name, 1424 smbios_riscv_isa_desc); 1425 flag_printf(fp, " Privilege Levels", rv.smbpirv_privlvl, 1426 sizeof (rv.smbpirv_privlvl) * NBBY, smbios_riscv_priv_name, 1427 smbios_riscv_priv_desc); 1428 u128_print(fp, " Machine Exception Trap Delegation", 1429 rv.smbpirv_metdi); 1430 u128_print(fp, " Machine Interrupt Trap Delegation", 1431 rv.smbpirv_mitdi); 1432 desc_printf(smbios_riscv_width_desc(rv.smbpirv_xlen), 1433 fp, " Register Width: 0x%x", rv.smbpirv_xlen); 1434 desc_printf(smbios_riscv_width_desc(rv.smbpirv_mxlen), 1435 fp, " M-Mode Register Width: 0x%x", rv.smbpirv_mxlen); 1436 desc_printf(smbios_riscv_width_desc(rv.smbpirv_sxlen), 1437 fp, " S-Mode Register Width: 0x%x", rv.smbpirv_sxlen); 1438 desc_printf(smbios_riscv_width_desc(rv.smbpirv_uxlen), 1439 fp, " U-Mode Register Width: 0x%x", rv.smbpirv_uxlen); 1440 } 1441 1442 static void 1443 print_processor_info(smbios_hdl_t *shp, id_t id, FILE *fp) 1444 { 1445 smbios_processor_info_t p; 1446 1447 if (smbios_info_processor_info(shp, id, &p) != 0) { 1448 smbios_warn(shp, "failed to read processor additional " 1449 "information"); 1450 return; 1451 } 1452 1453 id_printf(fp, " Processor Handle: ", p.smbpi_processor); 1454 desc_printf(smbios_processor_info_type_desc(p.smbpi_ptype), 1455 fp, " Processor Type: %u", p.smbpi_ptype); 1456 1457 switch (p.smbpi_ptype) { 1458 case SMB_PROCINFO_T_RV32: 1459 case SMB_PROCINFO_T_RV64: 1460 case SMB_PROCINFO_T_RV128: 1461 oprintf(fp, " RISC-V Additional Processor Information:\n"); 1462 print_processor_info_riscv(shp, id, fp); 1463 break; 1464 default: 1465 break; 1466 } 1467 } 1468 1469 static void 1470 print_pointdev(smbios_hdl_t *shp, id_t id, FILE *fp) 1471 { 1472 smbios_pointdev_t pd; 1473 1474 if (smbios_info_pointdev(shp, id, &pd) != 0) { 1475 smbios_warn(shp, "failed to read pointer device information"); 1476 return; 1477 } 1478 1479 desc_printf(smbios_pointdev_type_desc(pd.smbpd_type), 1480 fp, " Type: %u", pd.smbpd_type); 1481 desc_printf(smbios_pointdev_iface_desc(pd.smbpd_iface), 1482 fp, " Interface: %u", pd.smbpd_iface); 1483 oprintf(fp, " Buttons: %u\n", pd.smbpd_nbuttons); 1484 } 1485 1486 static void 1487 print_extprocessor(smbios_hdl_t *shp, id_t id, FILE *fp) 1488 { 1489 int i; 1490 smbios_processor_ext_t ep; 1491 1492 if (check_oem(shp) != 0) 1493 return; 1494 1495 (void) smbios_info_extprocessor(shp, id, &ep); 1496 1497 oprintf(fp, " Processor: %u\n", ep.smbpe_processor); 1498 oprintf(fp, " FRU: %u\n", ep.smbpe_fru); 1499 oprintf(fp, " Initial APIC ID count: %u\n\n", ep.smbpe_n); 1500 1501 for (i = 0; i < ep.smbpe_n; i++) { 1502 oprintf(fp, " Logical Strand %u: Initial APIC ID: %u\n", i, 1503 ep.smbpe_apicid[i]); 1504 } 1505 } 1506 1507 static void 1508 print_extport(smbios_hdl_t *shp, id_t id, FILE *fp) 1509 { 1510 smbios_port_ext_t epo; 1511 1512 if (check_oem(shp) != 0) 1513 return; 1514 1515 (void) smbios_info_extport(shp, id, &epo); 1516 1517 oprintf(fp, " Chassis Handle: %u\n", epo.smbporte_chassis); 1518 oprintf(fp, " Port Connector Handle: %u\n", epo.smbporte_port); 1519 oprintf(fp, " Device Type: %u\n", epo.smbporte_dtype); 1520 oprintf(fp, " Device Handle: %u\n", epo.smbporte_devhdl); 1521 oprintf(fp, " PHY: %u\n", epo.smbporte_phy); 1522 } 1523 1524 static void 1525 print_pciexrc(smbios_hdl_t *shp, id_t id, FILE *fp) 1526 { 1527 smbios_pciexrc_t pcie; 1528 1529 if (check_oem(shp) != 0) 1530 return; 1531 1532 (void) smbios_info_pciexrc(shp, id, &pcie); 1533 1534 oprintf(fp, " Component ID: %u\n", pcie.smbpcie_bb); 1535 oprintf(fp, " BDF: 0x%x\n", pcie.smbpcie_bdf); 1536 } 1537 1538 static void 1539 print_extmemarray(smbios_hdl_t *shp, id_t id, FILE *fp) 1540 { 1541 smbios_memarray_ext_t em; 1542 1543 if (check_oem(shp) != 0) 1544 return; 1545 1546 (void) smbios_info_extmemarray(shp, id, &em); 1547 1548 oprintf(fp, " Physical Memory Array Handle: %u\n", em.smbmae_ma); 1549 oprintf(fp, " Component Parent Handle: %u\n", em.smbmae_comp); 1550 oprintf(fp, " BDF: 0x%x\n", em.smbmae_bdf); 1551 } 1552 1553 static void 1554 print_extmemdevice(smbios_hdl_t *shp, id_t id, FILE *fp) 1555 { 1556 int i; 1557 smbios_memdevice_ext_t emd; 1558 1559 if (check_oem(shp) != 0) 1560 return; 1561 1562 (void) smbios_info_extmemdevice(shp, id, &emd); 1563 1564 oprintf(fp, " Memory Device Handle: %u\n", emd.smbmdeve_md); 1565 oprintf(fp, " DRAM Channel: %u\n", emd.smbmdeve_drch); 1566 oprintf(fp, " Number of Chip Selects: %u\n", emd.smbmdeve_ncs); 1567 1568 for (i = 0; i < emd.smbmdeve_ncs; i++) { 1569 oprintf(fp, " Chip Select: %u\n", emd.smbmdeve_cs[i]); 1570 } 1571 } 1572 1573 static int 1574 print_struct(smbios_hdl_t *shp, const smbios_struct_t *sp, void *fp) 1575 { 1576 smbios_info_t info; 1577 int hex = opt_x; 1578 const char *s; 1579 1580 if (opt_t != -1 && opt_t != sp->smbstr_type) 1581 return (0); /* skip struct if type doesn't match -t */ 1582 1583 if (!opt_O && (sp->smbstr_type == SMB_TYPE_MEMCTL || 1584 sp->smbstr_type == SMB_TYPE_MEMMOD)) 1585 return (0); /* skip struct if type is obsolete */ 1586 1587 if (g_hdr++ == 0 || !opt_s) 1588 oprintf(fp, "%-5s %-4s %s\n", "ID", "SIZE", "TYPE"); 1589 1590 oprintf(fp, "%-5u %-4lu", 1591 (uint_t)sp->smbstr_id, (ulong_t)sp->smbstr_size); 1592 1593 if ((s = smbios_type_name(sp->smbstr_type)) != NULL) 1594 oprintf(fp, " %s (type %u)", s, sp->smbstr_type); 1595 else if (sp->smbstr_type > SMB_TYPE_OEM_LO && 1596 sp->smbstr_type < SMB_TYPE_OEM_HI) 1597 oprintf(fp, " %s+%u (type %u)", "SMB_TYPE_OEM_LO", 1598 sp->smbstr_type - SMB_TYPE_OEM_LO, sp->smbstr_type); 1599 else 1600 oprintf(fp, " %u", sp->smbstr_type); 1601 1602 if ((s = smbios_type_desc(sp->smbstr_type)) != NULL) 1603 oprintf(fp, " (%s)\n", s); 1604 else 1605 oprintf(fp, "\n"); 1606 1607 if (opt_s) 1608 return (0); /* only print header line if -s specified */ 1609 1610 if (smbios_info_common(shp, sp->smbstr_id, &info) == 0) { 1611 oprintf(fp, "\n"); 1612 print_common(&info, fp); 1613 } 1614 1615 switch (sp->smbstr_type) { 1616 case SMB_TYPE_BIOS: 1617 oprintf(fp, "\n"); 1618 print_bios(shp, fp); 1619 break; 1620 case SMB_TYPE_SYSTEM: 1621 oprintf(fp, "\n"); 1622 print_system(shp, fp); 1623 break; 1624 case SMB_TYPE_BASEBOARD: 1625 oprintf(fp, "\n"); 1626 print_bboard(shp, sp->smbstr_id, fp); 1627 break; 1628 case SMB_TYPE_CHASSIS: 1629 oprintf(fp, "\n"); 1630 print_chassis(shp, sp->smbstr_id, fp); 1631 break; 1632 case SMB_TYPE_PROCESSOR: 1633 oprintf(fp, "\n"); 1634 print_processor(shp, sp->smbstr_id, fp); 1635 break; 1636 case SMB_TYPE_CACHE: 1637 oprintf(fp, "\n"); 1638 print_cache(shp, sp->smbstr_id, fp); 1639 break; 1640 case SMB_TYPE_PORT: 1641 oprintf(fp, "\n"); 1642 print_port(shp, sp->smbstr_id, fp); 1643 break; 1644 case SMB_TYPE_SLOT: 1645 oprintf(fp, "\n"); 1646 print_slot(shp, sp->smbstr_id, fp); 1647 break; 1648 case SMB_TYPE_OBDEVS: 1649 oprintf(fp, "\n"); 1650 print_obdevs(shp, sp->smbstr_id, fp); 1651 break; 1652 case SMB_TYPE_OEMSTR: 1653 case SMB_TYPE_SYSCONFSTR: 1654 oprintf(fp, "\n"); 1655 print_strtab(shp, sp->smbstr_id, fp); 1656 break; 1657 case SMB_TYPE_LANG: 1658 oprintf(fp, "\n"); 1659 print_lang(shp, sp->smbstr_id, fp); 1660 break; 1661 case SMB_TYPE_EVENTLOG: 1662 oprintf(fp, "\n"); 1663 print_evlog(shp, sp->smbstr_id, fp); 1664 break; 1665 case SMB_TYPE_MEMARRAY: 1666 oprintf(fp, "\n"); 1667 print_memarray(shp, sp->smbstr_id, fp); 1668 break; 1669 case SMB_TYPE_MEMDEVICE: 1670 oprintf(fp, "\n"); 1671 print_memdevice(shp, sp->smbstr_id, fp); 1672 break; 1673 case SMB_TYPE_MEMARRAYMAP: 1674 oprintf(fp, "\n"); 1675 print_memarrmap(shp, sp->smbstr_id, fp); 1676 break; 1677 case SMB_TYPE_MEMDEVICEMAP: 1678 oprintf(fp, "\n"); 1679 print_memdevmap(shp, sp->smbstr_id, fp); 1680 break; 1681 case SMB_TYPE_POINTDEV: 1682 oprintf(fp, "\n"); 1683 print_pointdev(shp, sp->smbstr_id, fp); 1684 break; 1685 case SMB_TYPE_SECURITY: 1686 oprintf(fp, "\n"); 1687 print_hwsec(shp, fp); 1688 break; 1689 case SMB_TYPE_VPROBE: 1690 oprintf(fp, "\n"); 1691 print_vprobe(shp, sp->smbstr_id, fp); 1692 break; 1693 case SMB_TYPE_COOLDEV: 1694 oprintf(fp, "\n"); 1695 print_cooldev(shp, sp->smbstr_id, fp); 1696 break; 1697 case SMB_TYPE_TPROBE: 1698 oprintf(fp, "\n"); 1699 print_tprobe(shp, sp->smbstr_id, fp); 1700 break; 1701 case SMB_TYPE_IPROBE: 1702 oprintf(fp, "\n"); 1703 print_iprobe(shp, sp->smbstr_id, fp); 1704 break; 1705 case SMB_TYPE_BOOT: 1706 oprintf(fp, "\n"); 1707 print_boot(shp, fp); 1708 break; 1709 case SMB_TYPE_IPMIDEV: 1710 oprintf(fp, "\n"); 1711 print_ipmi(shp, fp); 1712 break; 1713 case SMB_TYPE_POWERSUP: 1714 oprintf(fp, "\n"); 1715 print_powersup(shp, sp->smbstr_id, fp); 1716 break; 1717 case SMB_TYPE_OBDEVEXT: 1718 oprintf(fp, "\n"); 1719 print_obdevs_ext(shp, sp->smbstr_id, fp); 1720 break; 1721 case SMB_TYPE_PROCESSOR_INFO: 1722 oprintf(fp, "\n"); 1723 print_processor_info(shp, sp->smbstr_id, fp); 1724 break; 1725 case SUN_OEM_EXT_PROCESSOR: 1726 oprintf(fp, "\n"); 1727 print_extprocessor(shp, sp->smbstr_id, fp); 1728 break; 1729 case SUN_OEM_EXT_PORT: 1730 oprintf(fp, "\n"); 1731 print_extport(shp, sp->smbstr_id, fp); 1732 break; 1733 case SUN_OEM_PCIEXRC: 1734 oprintf(fp, "\n"); 1735 print_pciexrc(shp, sp->smbstr_id, fp); 1736 break; 1737 case SUN_OEM_EXT_MEMARRAY: 1738 oprintf(fp, "\n"); 1739 print_extmemarray(shp, sp->smbstr_id, fp); 1740 break; 1741 case SUN_OEM_EXT_MEMDEVICE: 1742 oprintf(fp, "\n"); 1743 print_extmemdevice(shp, sp->smbstr_id, fp); 1744 break; 1745 default: 1746 hex++; 1747 } 1748 1749 if (hex) 1750 print_bytes(sp->smbstr_data, sp->smbstr_size, fp); 1751 else 1752 oprintf(fp, "\n"); 1753 1754 return (0); 1755 } 1756 1757 static uint16_t 1758 getu16(const char *name, const char *s) 1759 { 1760 u_longlong_t val; 1761 char *p; 1762 1763 errno = 0; 1764 val = strtoull(s, &p, 0); 1765 1766 if (errno != 0 || p == s || *p != '\0' || val > UINT16_MAX) { 1767 (void) fprintf(stderr, "%s: invalid %s argument -- %s\n", 1768 g_pname, name, s); 1769 exit(SMBIOS_USAGE); 1770 } 1771 1772 return ((uint16_t)val); 1773 } 1774 1775 static uint16_t 1776 getstype(const char *name, const char *s) 1777 { 1778 const char *ts; 1779 uint16_t t; 1780 1781 for (t = 0; t < SMB_TYPE_OEM_LO; t++) { 1782 if ((ts = smbios_type_name(t)) != NULL && strcmp(s, ts) == 0) 1783 return (t); 1784 } 1785 1786 (void) fprintf(stderr, "%s: invalid %s argument -- %s\n", 1787 g_pname, name, s); 1788 1789 exit(SMBIOS_USAGE); 1790 /*NOTREACHED*/ 1791 } 1792 1793 static int 1794 usage(FILE *fp) 1795 { 1796 (void) fprintf(fp, "Usage: %s " 1797 "[-BeOsx] [-i id] [-t type] [-w file] [file]\n\n", g_pname); 1798 1799 (void) fprintf(fp, 1800 "\t-B disable header validation for broken BIOSes\n" 1801 "\t-e display SMBIOS entry point information\n" 1802 "\t-i display only the specified structure\n" 1803 "\t-O display obsolete structure types\n" 1804 "\t-s display only a summary of structure identifiers and types\n" 1805 "\t-t display only the specified structure type\n" 1806 "\t-w write the raw data to the specified file\n" 1807 "\t-x display raw data for structures\n"); 1808 1809 return (SMBIOS_USAGE); 1810 } 1811 1812 int 1813 main(int argc, char *argv[]) 1814 { 1815 const char *ifile = NULL; 1816 const char *ofile = NULL; 1817 int oflags = 0; 1818 1819 smbios_hdl_t *shp; 1820 smbios_struct_t s; 1821 int err, fd, c; 1822 char *p; 1823 1824 if ((p = strrchr(argv[0], '/')) == NULL) 1825 g_pname = argv[0]; 1826 else 1827 g_pname = p + 1; 1828 1829 while (optind < argc) { 1830 while ((c = getopt(argc, argv, "Bei:Ost:w:xZ")) != EOF) { 1831 switch (c) { 1832 case 'B': 1833 oflags |= SMB_O_NOCKSUM | SMB_O_NOVERS; 1834 break; 1835 case 'e': 1836 opt_e++; 1837 break; 1838 case 'i': 1839 opt_i = getu16("struct ID", optarg); 1840 break; 1841 case 'O': 1842 opt_O++; 1843 break; 1844 case 's': 1845 opt_s++; 1846 break; 1847 case 't': 1848 if (isdigit(optarg[0])) 1849 opt_t = getu16("struct type", optarg); 1850 else 1851 opt_t = getstype("struct type", optarg); 1852 break; 1853 case 'w': 1854 ofile = optarg; 1855 break; 1856 case 'x': 1857 opt_x++; 1858 break; 1859 case 'Z': 1860 oflags |= SMB_O_ZIDS; /* undocumented */ 1861 break; 1862 default: 1863 return (usage(stderr)); 1864 } 1865 } 1866 1867 if (optind < argc) { 1868 if (ifile != NULL) { 1869 (void) fprintf(stderr, "%s: illegal " 1870 "argument -- %s\n", g_pname, argv[optind]); 1871 return (SMBIOS_USAGE); 1872 } 1873 ifile = argv[optind++]; 1874 } 1875 } 1876 1877 if ((shp = smbios_open(ifile, SMB_VERSION, oflags, &err)) == NULL) { 1878 (void) fprintf(stderr, "%s: failed to load SMBIOS: %s\n", 1879 g_pname, smbios_errmsg(err)); 1880 return (SMBIOS_ERROR); 1881 } 1882 1883 if (opt_i == -1 && opt_t == -1 && opt_e == 0 && 1884 smbios_truncated(shp)) 1885 (void) fprintf(stderr, "%s: SMBIOS table is truncated\n", 1886 g_pname); 1887 1888 if (ofile != NULL) { 1889 if ((fd = open(ofile, O_WRONLY|O_CREAT|O_TRUNC, 0666)) == -1) { 1890 (void) fprintf(stderr, "%s: failed to open %s: %s\n", 1891 g_pname, ofile, strerror(errno)); 1892 err = SMBIOS_ERROR; 1893 } else if (smbios_write(shp, fd) != 0) { 1894 (void) fprintf(stderr, "%s: failed to write %s: %s\n", 1895 g_pname, ofile, smbios_errmsg(smbios_errno(shp))); 1896 err = SMBIOS_ERROR; 1897 } 1898 smbios_close(shp); 1899 return (err); 1900 } 1901 1902 if (opt_e) { 1903 print_smbios(shp, stdout); 1904 smbios_close(shp); 1905 return (SMBIOS_SUCCESS); 1906 } 1907 1908 if (opt_O && (opt_i != -1 || opt_t != -1)) 1909 opt_O++; /* -i or -t imply displaying obsolete records */ 1910 1911 if (opt_i != -1) 1912 err = smbios_lookup_id(shp, opt_i, &s); 1913 else 1914 err = smbios_iter(shp, print_struct, stdout); 1915 1916 if (err != 0) { 1917 (void) fprintf(stderr, "%s: failed to access SMBIOS: %s\n", 1918 g_pname, smbios_errmsg(smbios_errno(shp))); 1919 smbios_close(shp); 1920 return (SMBIOS_ERROR); 1921 } 1922 1923 if (opt_i != -1) 1924 (void) print_struct(shp, &s, stdout); 1925 1926 smbios_close(shp); 1927 return (SMBIOS_SUCCESS); 1928 } 1929