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 2007 Sun Microsystems, Inc. All rights reserved. 24 * Use is subject to license terms. 25 */ 26 #pragma ident "%Z%%M% %I% %E% SMI" 27 28 /* 29 * Dump an elf file. 30 */ 31 #include <machdep.h> 32 #include <sys/elf_386.h> 33 #include <sys/elf_amd64.h> 34 #include <sys/elf_SPARC.h> 35 #include <dwarf.h> 36 #include <unistd.h> 37 #include <errno.h> 38 #include <strings.h> 39 #include <debug.h> 40 #include <conv.h> 41 #include <msg.h> 42 #include <_elfdump.h> 43 44 45 46 /* 47 * VERSYM_STATE is used to maintain information about the VERSYM section 48 * in the object being analyzed. It is filled in by versions(), and used 49 * by init_symtbl_state() when displaying symbol information. 50 * 51 * Note that the value of the gnu field is a hueristic guess, 52 * based on the section names. 53 */ 54 typedef struct { 55 Cache *cache; /* Pointer to cache entry for VERSYM */ 56 Versym *data; /* Pointer to versym array */ 57 int num_verdef; /* # of versions defined in object */ 58 int gnu; /* True if we think obj produced by GNU tools */ 59 } VERSYM_STATE; 60 61 /* 62 * SYMTBL_STATE is used to maintain information about a single symbol 63 * table section, for use by the routines that display symbol information. 64 */ 65 typedef struct { 66 const char *file; /* Name of file */ 67 Ehdr *ehdr; /* ELF header for file */ 68 Cache *cache; /* Cache of all section headers */ 69 Word shnum; /* # of sections in cache */ 70 Cache *seccache; /* Cache of symbol table section hdr */ 71 Word secndx; /* Index of symbol table section hdr */ 72 const char *secname; /* Name of section */ 73 uint_t flags; /* Command line option flags */ 74 struct { /* Extended section index data */ 75 int checked; /* TRUE if already checked for shxndx */ 76 Word *data; /* NULL, or extended section index */ 77 /* used for symbol table entries */ 78 uint_t n; /* # items in shxndx.data */ 79 } shxndx; 80 VERSYM_STATE *versym; /* NULL, or associated VERSYM section */ 81 Sym *sym; /* Array of symbols */ 82 Word symn; /* # of symbols */ 83 } SYMTBL_STATE; 84 85 86 87 /* 88 * Focal point for verifying symbol names. 89 */ 90 static const char * 91 string(Cache *refsec, Word ndx, Cache *strsec, const char *file, Word name) 92 { 93 /* 94 * If an error in this routine is due to a property of the string 95 * section, as opposed to a bad offset into the section (a property of 96 * the referencing section), then we will detect the same error on 97 * every call involving those sections. We use these static variables 98 * to retain the information needed to only issue each such error once. 99 */ 100 static Cache *last_refsec; /* Last referencing section seen */ 101 static int strsec_err; /* True if error issued */ 102 103 const char *strs; 104 Word strn; 105 106 if (strsec->c_data == NULL) 107 return (NULL); 108 109 strs = (char *)strsec->c_data->d_buf; 110 strn = strsec->c_data->d_size; 111 112 /* 113 * We only print a diagnostic regarding a bad string table once per 114 * input section being processed. If the refsec has changed, reset 115 * our retained error state. 116 */ 117 if (last_refsec != refsec) { 118 last_refsec = refsec; 119 strsec_err = 0; 120 } 121 122 /* Verify that strsec really is a string table */ 123 if (strsec->c_shdr->sh_type != SHT_STRTAB) { 124 if (!strsec_err) { 125 (void) fprintf(stderr, MSG_INTL(MSG_ERR_NOTSTRTAB), 126 file, strsec->c_ndx, refsec->c_ndx); 127 strsec_err = 1; 128 } 129 return (MSG_INTL(MSG_STR_UNKNOWN)); 130 } 131 132 /* 133 * Is the string table offset within range of the available strings? 134 */ 135 if (name >= strn) { 136 /* 137 * Do we have a empty string table? 138 */ 139 if (strs == 0) { 140 if (!strsec_err) { 141 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 142 file, strsec->c_name); 143 strsec_err = 1; 144 } 145 } else { 146 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSTOFF), 147 file, refsec->c_name, EC_WORD(ndx), strsec->c_name, 148 EC_WORD(name), EC_WORD(strn - 1)); 149 } 150 151 /* 152 * Return the empty string so that the calling function can 153 * continue it's output diagnostics. 154 */ 155 return (MSG_INTL(MSG_STR_UNKNOWN)); 156 } 157 return (strs + name); 158 } 159 160 /* 161 * Relocations can reference section symbols and standard symbols. If the 162 * former, establish the section name. 163 */ 164 static const char * 165 relsymname(Cache *cache, Cache *csec, Cache *strsec, Word symndx, Word symnum, 166 Word relndx, Sym *syms, char *secstr, size_t secsz, const char *file, 167 uint_t flags) 168 { 169 Sym *sym; 170 171 if (symndx >= symnum) { 172 (void) fprintf(stderr, MSG_INTL(MSG_ERR_RELBADSYMNDX), 173 file, EC_WORD(symndx), EC_WORD(relndx)); 174 return (MSG_INTL(MSG_STR_UNKNOWN)); 175 } 176 177 sym = (Sym *)(syms + symndx); 178 179 /* 180 * If the symbol represents a section offset construct an appropriate 181 * string. 182 */ 183 if ((ELF_ST_TYPE(sym->st_info) == STT_SECTION) && (sym->st_name == 0)) { 184 if (flags & FLG_LONGNAME) 185 (void) snprintf(secstr, secsz, 186 MSG_INTL(MSG_STR_L_SECTION), 187 cache[sym->st_shndx].c_name); 188 else 189 (void) snprintf(secstr, secsz, 190 MSG_INTL(MSG_STR_SECTION), 191 cache[sym->st_shndx].c_name); 192 return ((const char *)secstr); 193 } 194 195 return (string(csec, symndx, strsec, file, sym->st_name)); 196 } 197 198 /* 199 * Focal point for establishing a string table section. Data such as the 200 * dynamic information simply points to a string table. Data such as 201 * relocations, reference a symbol table, which in turn is associated with a 202 * string table. 203 */ 204 static int 205 stringtbl(Cache *cache, int symtab, Word ndx, Word shnum, const char *file, 206 Word *symnum, Cache **symsec, Cache **strsec) 207 { 208 Shdr *shdr = cache[ndx].c_shdr; 209 210 if (symtab) { 211 /* 212 * Validate the symbol table section. 213 */ 214 if ((shdr->sh_link == 0) || (shdr->sh_link >= shnum)) { 215 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHLINK), 216 file, cache[ndx].c_name, EC_WORD(shdr->sh_link)); 217 return (0); 218 } 219 if ((shdr->sh_entsize == 0) || (shdr->sh_size == 0)) { 220 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 221 file, cache[ndx].c_name); 222 return (0); 223 } 224 225 /* 226 * Obtain, and verify the symbol table data. 227 */ 228 if ((cache[ndx].c_data == NULL) || 229 (cache[ndx].c_data->d_buf == NULL)) { 230 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 231 file, cache[ndx].c_name); 232 return (0); 233 } 234 235 /* 236 * Establish the string table index. 237 */ 238 ndx = shdr->sh_link; 239 shdr = cache[ndx].c_shdr; 240 241 /* 242 * Return symbol table information. 243 */ 244 if (symnum) 245 *symnum = (shdr->sh_size / shdr->sh_entsize); 246 if (symsec) 247 *symsec = &cache[ndx]; 248 } 249 250 /* 251 * Validate the associated string table section. 252 */ 253 if ((shdr->sh_link == 0) || (shdr->sh_link >= shnum)) { 254 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHLINK), 255 file, cache[ndx].c_name, EC_WORD(shdr->sh_link)); 256 return (0); 257 } 258 259 if (strsec) 260 *strsec = &cache[shdr->sh_link]; 261 262 return (1); 263 } 264 265 /* 266 * Lookup a symbol and set Sym accordingly. 267 */ 268 static int 269 symlookup(const char *name, Cache *cache, Word shnum, Sym **sym, 270 Cache *symtab, const char *file) 271 { 272 Shdr *shdr; 273 Word symn, cnt; 274 Sym *syms; 275 276 if (symtab == 0) 277 return (0); 278 279 shdr = symtab->c_shdr; 280 281 /* 282 * Determine the symbol data and number. 283 */ 284 if ((shdr->sh_entsize == 0) || (shdr->sh_size == 0)) { 285 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 286 file, symtab->c_name); 287 return (0); 288 } 289 if (symtab->c_data == NULL) 290 return (0); 291 292 /* LINTED */ 293 symn = (Word)(shdr->sh_size / shdr->sh_entsize); 294 syms = (Sym *)symtab->c_data->d_buf; 295 296 /* 297 * Get the associated string table section. 298 */ 299 if ((shdr->sh_link == 0) || (shdr->sh_link >= shnum)) { 300 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHLINK), 301 file, symtab->c_name, EC_WORD(shdr->sh_link)); 302 return (0); 303 } 304 305 /* 306 * Loop through the symbol table to find a match. 307 */ 308 for (cnt = 0; cnt < symn; syms++, cnt++) { 309 const char *symname; 310 311 symname = string(symtab, cnt, &cache[shdr->sh_link], file, 312 syms->st_name); 313 314 if (symname && (strcmp(name, symname) == 0)) { 315 *sym = syms; 316 return (1); 317 } 318 } 319 return (0); 320 } 321 322 /* 323 * Print section headers. 324 */ 325 static void 326 sections(const char *file, Cache *cache, Word shnum, Ehdr *ehdr, 327 const char *name) 328 { 329 size_t seccnt; 330 331 for (seccnt = 1; seccnt < shnum; seccnt++) { 332 Cache *_cache = &cache[seccnt]; 333 Shdr *shdr = _cache->c_shdr; 334 const char *secname = _cache->c_name; 335 336 /* 337 * Although numerous section header entries can be zero, it's 338 * usually a sign of trouble if the type is zero. 339 */ 340 if (shdr->sh_type == 0) { 341 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHTYPE), 342 file, secname, EC_WORD(shdr->sh_type)); 343 } 344 345 if (name && strcmp(name, secname)) 346 continue; 347 348 /* 349 * Identify any sections that are suspicious. A .got section 350 * shouldn't exist in a relocatable object. 351 */ 352 if (ehdr->e_type == ET_REL) { 353 if (strncmp(secname, MSG_ORIG(MSG_ELF_GOT), 354 MSG_ELF_GOT_SIZE) == 0) { 355 (void) fprintf(stderr, 356 MSG_INTL(MSG_GOT_UNEXPECTED), file, 357 secname); 358 } 359 } 360 361 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 362 dbg_print(0, MSG_INTL(MSG_ELF_SHDR), EC_WORD(seccnt), secname); 363 Elf_shdr(0, ehdr->e_machine, shdr); 364 } 365 } 366 367 /* 368 * A couple of instances of unwind data are printed as tables of 8 data items 369 * expressed as 0x?? integers. 370 */ 371 #define UNWINDTBLSZ 10 + (8 * 5) + 1 372 373 static void 374 unwindtbl(uint64_t *ndx, uint_t len, uchar_t *data, uint64_t doff, 375 const char *msg, const char *pre, size_t plen) 376 { 377 char buffer[UNWINDTBLSZ]; 378 uint_t boff = plen, cnt = 0; 379 380 dbg_print(0, msg); 381 (void) strncpy(buffer, pre, UNWINDTBLSZ); 382 383 while (*ndx < (len + 4)) { 384 if (cnt == 8) { 385 dbg_print(0, buffer); 386 boff = plen; 387 cnt = 0; 388 } 389 (void) snprintf(&buffer[boff], UNWINDTBLSZ - boff, 390 MSG_ORIG(MSG_UNW_TBLENTRY), data[doff + (*ndx)++]); 391 boff += 5; 392 cnt++; 393 } 394 if (cnt) 395 dbg_print(0, buffer); 396 } 397 398 /* 399 * Obtain a specified Phdr entry. 400 */ 401 static Phdr * 402 getphdr(Word phnum, Word type, const char *file, Elf *elf) 403 { 404 Word cnt; 405 Phdr *phdr; 406 407 if ((phdr = elf_getphdr(elf)) == NULL) { 408 failure(file, MSG_ORIG(MSG_ELF_GETPHDR)); 409 return (0); 410 } 411 412 for (cnt = 0; cnt < phnum; phdr++, cnt++) { 413 if (phdr->p_type == type) 414 return (phdr); 415 } 416 return (0); 417 } 418 419 static void 420 unwind(Cache *cache, Word shnum, Word phnum, Ehdr *ehdr, const char *name, 421 const char *file, Elf *elf) 422 { 423 Word cnt; 424 Phdr *uphdr = 0; 425 426 /* 427 * For the moment - UNWIND is only relevant for a AMD64 object. 428 */ 429 if (ehdr->e_machine != EM_AMD64) 430 return; 431 432 if (phnum) 433 uphdr = getphdr(phnum, PT_SUNW_UNWIND, file, elf); 434 435 for (cnt = 1; cnt < shnum; cnt++) { 436 Cache *_cache = &cache[cnt]; 437 Shdr *shdr = _cache->c_shdr; 438 uchar_t *data; 439 size_t datasize; 440 uint64_t off, ndx, frame_ptr, fde_cnt, tabndx; 441 uint_t vers, frame_ptr_enc, fde_cnt_enc, table_enc; 442 443 /* 444 * AMD64 - this is a strmcp() just to find the gcc produced 445 * sections. Soon gcc should be setting the section type - and 446 * we'll not need this strcmp(). 447 */ 448 if ((shdr->sh_type != SHT_AMD64_UNWIND) && 449 (strncmp(_cache->c_name, MSG_ORIG(MSG_SCN_FRM), 450 MSG_SCN_FRM_SIZE) != 0) && 451 (strncmp(_cache->c_name, MSG_ORIG(MSG_SCN_FRMHDR), 452 MSG_SCN_FRMHDR_SIZE) != 0)) 453 continue; 454 if (name && strcmp(name, _cache->c_name)) 455 continue; 456 457 if (_cache->c_data == NULL) 458 continue; 459 460 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 461 dbg_print(0, MSG_INTL(MSG_ELF_SCN_UNWIND), _cache->c_name); 462 463 data = (uchar_t *)(_cache->c_data->d_buf); 464 datasize = _cache->c_data->d_size; 465 off = 0; 466 467 /* 468 * Is this a .eh_frame_hdr 469 */ 470 if ((uphdr && (shdr->sh_addr == uphdr->p_vaddr)) || 471 (strncmp(_cache->c_name, MSG_ORIG(MSG_SCN_FRMHDR), 472 MSG_SCN_FRMHDR_SIZE) == 0)) { 473 474 dbg_print(0, MSG_ORIG(MSG_UNW_FRMHDR)); 475 ndx = 0; 476 477 vers = data[ndx++]; 478 frame_ptr_enc = data[ndx++]; 479 fde_cnt_enc = data[ndx++]; 480 table_enc = data[ndx++]; 481 482 dbg_print(0, MSG_ORIG(MSG_UNW_FRMVERS), vers); 483 484 frame_ptr = dwarf_ehe_extract(data, &ndx, frame_ptr_enc, 485 ehdr->e_ident, shdr->sh_addr + ndx); 486 487 dbg_print(0, MSG_ORIG(MSG_UNW_FRPTRENC), 488 conv_dwarf_ehe(frame_ptr_enc), EC_XWORD(frame_ptr)); 489 490 fde_cnt = dwarf_ehe_extract(data, &ndx, fde_cnt_enc, 491 ehdr->e_ident, shdr->sh_addr + ndx); 492 493 dbg_print(0, MSG_ORIG(MSG_UNW_FDCNENC), 494 conv_dwarf_ehe(fde_cnt_enc), EC_XWORD(fde_cnt)); 495 dbg_print(0, MSG_ORIG(MSG_UNW_TABENC), 496 conv_dwarf_ehe(table_enc)); 497 dbg_print(0, MSG_ORIG(MSG_UNW_BINSRTAB1)); 498 dbg_print(0, MSG_ORIG(MSG_UNW_BINSRTAB2)); 499 500 for (tabndx = 0; tabndx < fde_cnt; tabndx++) { 501 dbg_print(0, MSG_ORIG(MSG_UNW_BINSRTABENT), 502 EC_XWORD(dwarf_ehe_extract(data, &ndx, 503 table_enc, ehdr->e_ident, shdr->sh_addr)), 504 EC_XWORD(dwarf_ehe_extract(data, &ndx, 505 table_enc, ehdr->e_ident, shdr->sh_addr))); 506 } 507 continue; 508 } 509 510 /* 511 * Walk the Eh_frame's 512 */ 513 while (off < datasize) { 514 uint_t cieid, cielength, cieversion, 515 cieretaddr; 516 int cieRflag, cieLflag, ciePflag, cieZflag; 517 uint_t cieaugndx, length, id; 518 uint64_t ciecalign, ciedalign; 519 char *cieaugstr; 520 521 ndx = 0; 522 /* 523 * extract length in lsb format 524 */ 525 length = LSB32EXTRACT(data + off + ndx); 526 ndx += 4; 527 528 /* 529 * extract CIE id in lsb format 530 */ 531 id = LSB32EXTRACT(data + off + ndx); 532 ndx += 4; 533 534 /* 535 * A CIE record has a id of '0', otherwise this is a 536 * FDE entry and the 'id' is the CIE pointer. 537 */ 538 if (id == 0) { 539 uint64_t persVal; 540 541 cielength = length; 542 cieid = id; 543 cieLflag = ciePflag = cieRflag = cieZflag = 0; 544 545 dbg_print(0, MSG_ORIG(MSG_UNW_CIE), 546 EC_XWORD(shdr->sh_addr + off)); 547 dbg_print(0, MSG_ORIG(MSG_UNW_CIELNGTH), 548 cielength, cieid); 549 550 cieversion = data[off + ndx]; 551 ndx += 1; 552 cieaugstr = (char *)(&data[off + ndx]); 553 ndx += strlen(cieaugstr) + 1; 554 555 dbg_print(0, MSG_ORIG(MSG_UNW_CIEVERS), 556 cieversion, cieaugstr); 557 558 ciecalign = uleb_extract(&data[off], &ndx); 559 ciedalign = sleb_extract(&data[off], &ndx); 560 cieretaddr = data[off + ndx]; 561 ndx += 1; 562 563 dbg_print(0, MSG_ORIG(MSG_UNW_CIECALGN), 564 EC_XWORD(ciecalign), EC_XWORD(ciedalign), 565 cieretaddr); 566 567 if (cieaugstr[0]) 568 dbg_print(0, MSG_ORIG(MSG_UNW_CIEAUXVAL)); 569 570 for (cieaugndx = 0; cieaugstr[cieaugndx]; 571 cieaugndx++) { 572 uint_t val; 573 574 switch (cieaugstr[cieaugndx]) { 575 case 'z': 576 val = uleb_extract(&data[off], 577 &ndx); 578 dbg_print(0, 579 MSG_ORIG(MSG_UNW_CIEAUXSIZE), 580 val); 581 cieZflag = 1; 582 break; 583 case 'P': 584 ciePflag = data[off + ndx]; 585 ndx += 1; 586 587 persVal = dwarf_ehe_extract( 588 &data[off], 589 &ndx, ciePflag, ehdr->e_ident, 590 shdr->sh_addr + off + ndx); 591 dbg_print(0, 592 MSG_ORIG(MSG_UNW_CIEAUXPERS), 593 ciePflag, 594 conv_dwarf_ehe(ciePflag), 595 EC_XWORD(persVal)); 596 break; 597 case 'R': 598 val = data[off + ndx]; 599 ndx += 1; 600 dbg_print(0, 601 MSG_ORIG(MSG_UNW_CIEAUXCENC), 602 val, conv_dwarf_ehe(val)); 603 cieRflag = val; 604 break; 605 case 'L': 606 val = data[off + ndx]; 607 ndx += 1; 608 dbg_print(0, 609 MSG_ORIG(MSG_UNW_CIEAUXLSDA), 610 val, conv_dwarf_ehe(val)); 611 cieLflag = val; 612 break; 613 default: 614 dbg_print(0, 615 MSG_ORIG(MSG_UNW_CIEAUXUNEC), 616 cieaugstr[cieaugndx]); 617 break; 618 } 619 } 620 if ((cielength + 4) > ndx) 621 unwindtbl(&ndx, cielength, data, off, 622 MSG_ORIG(MSG_UNW_CIECFI), 623 MSG_ORIG(MSG_UNW_CIEPRE), 624 MSG_UNW_CIEPRE_SIZE); 625 off += cielength + 4; 626 627 } else { 628 uint_t fdelength = length; 629 int fdecieptr = id; 630 uint64_t fdeinitloc, fdeaddrrange; 631 632 dbg_print(0, MSG_ORIG(MSG_UNW_FDE), 633 EC_XWORD(shdr->sh_addr + off)); 634 dbg_print(0, MSG_ORIG(MSG_UNW_FDELNGTH), 635 fdelength, fdecieptr); 636 637 fdeinitloc = dwarf_ehe_extract(&data[off], 638 &ndx, cieRflag, ehdr->e_ident, 639 shdr->sh_addr + off + ndx); 640 fdeaddrrange = dwarf_ehe_extract(&data[off], 641 &ndx, (cieRflag & ~DW_EH_PE_pcrel), 642 ehdr->e_ident, 643 shdr->sh_addr + off + ndx); 644 645 dbg_print(0, MSG_ORIG(MSG_UNW_FDEINITLOC), 646 EC_XWORD(fdeinitloc), 647 EC_XWORD(fdeaddrrange)); 648 649 if (cieaugstr[0]) 650 dbg_print(0, 651 MSG_ORIG(MSG_UNW_FDEAUXVAL)); 652 if (cieZflag) { 653 uint64_t val; 654 val = uleb_extract(&data[off], &ndx); 655 dbg_print(0, 656 MSG_ORIG(MSG_UNW_FDEAUXSIZE), 657 EC_XWORD(val)); 658 if (val & cieLflag) { 659 fdeinitloc = dwarf_ehe_extract( 660 &data[off], &ndx, cieLflag, 661 ehdr->e_ident, 662 shdr->sh_addr + off + ndx); 663 dbg_print(0, 664 MSG_ORIG(MSG_UNW_FDEAUXLSDA), 665 EC_XWORD(val)); 666 } 667 } 668 if ((fdelength + 4) > ndx) 669 unwindtbl(&ndx, fdelength, data, off, 670 MSG_ORIG(MSG_UNW_FDECFI), 671 MSG_ORIG(MSG_UNW_FDEPRE), 672 MSG_UNW_FDEPRE_SIZE); 673 off += fdelength + 4; 674 } 675 } 676 } 677 } 678 679 /* 680 * Print the hardware/software capabilities. For executables and shared objects 681 * this should be accompanied with a program header. 682 */ 683 static void 684 cap(const char *file, Cache *cache, Word shnum, Word phnum, Ehdr *ehdr, 685 Elf *elf) 686 { 687 Word cnt; 688 Shdr *cshdr = 0; 689 Cache *ccache; 690 Off cphdr_off = 0; 691 Xword cphdr_sz; 692 693 /* 694 * Determine if a hardware/software capabilities header exists. 695 */ 696 if (phnum) { 697 Phdr *phdr; 698 699 if ((phdr = elf_getphdr(elf)) == NULL) { 700 failure(file, MSG_ORIG(MSG_ELF_GETPHDR)); 701 return; 702 } 703 704 for (cnt = 0; cnt < phnum; phdr++, cnt++) { 705 if (phdr->p_type == PT_SUNWCAP) { 706 cphdr_off = phdr->p_offset; 707 cphdr_sz = phdr->p_filesz; 708 break; 709 } 710 } 711 } 712 713 /* 714 * Determine if a hardware/software capabilities section exists. 715 */ 716 for (cnt = 1; cnt < shnum; cnt++) { 717 Cache *_cache = &cache[cnt]; 718 Shdr *shdr = _cache->c_shdr; 719 720 if (shdr->sh_type != SHT_SUNW_cap) 721 continue; 722 723 if (cphdr_off && ((cphdr_off < shdr->sh_offset) || 724 (cphdr_off + cphdr_sz) > (shdr->sh_offset + shdr->sh_size))) 725 continue; 726 727 if (_cache->c_data == NULL) 728 continue; 729 730 ccache = _cache; 731 cshdr = shdr; 732 break; 733 } 734 735 if ((cshdr == 0) && (cphdr_off == 0)) 736 return; 737 738 if ((cshdr->sh_entsize == 0) || (cshdr->sh_size == 0)) { 739 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 740 file, ccache->c_name); 741 return; 742 } 743 744 /* 745 * Print the hardware/software capabilities section. 746 */ 747 if (cshdr) { 748 Word ndx, capn; 749 Cap *cap = (Cap *)ccache->c_data->d_buf; 750 751 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 752 dbg_print(0, MSG_INTL(MSG_ELF_SCN_CAP), ccache->c_name); 753 754 Elf_cap_title(0); 755 756 capn = (Word)(cshdr->sh_size / cshdr->sh_entsize); 757 758 for (ndx = 0; ndx < capn; cap++, ndx++) { 759 if (cap->c_tag != CA_SUNW_NULL) 760 Elf_cap_entry(0, cap, ndx, ehdr->e_machine); 761 } 762 } else 763 (void) fprintf(stderr, MSG_INTL(MSG_WARN_INVCAP1), file); 764 765 /* 766 * If this object is an executable or shared object, then the 767 * hardware/software capabilities section should have an accompanying 768 * program header. 769 */ 770 if (cshdr && ((ehdr->e_type == ET_EXEC) || (ehdr->e_type == ET_DYN))) { 771 if (cphdr_off == 0) 772 (void) fprintf(stderr, MSG_INTL(MSG_WARN_INVCAP2), 773 file, ccache->c_name); 774 else if ((cphdr_off != cshdr->sh_offset) || 775 (cphdr_sz != cshdr->sh_size)) 776 (void) fprintf(stderr, MSG_INTL(MSG_WARN_INVCAP3), 777 file, ccache->c_name); 778 } 779 } 780 781 /* 782 * Print the interpretor. 783 */ 784 static void 785 interp(const char *file, Cache *cache, Word shnum, Word phnum, Elf *elf) 786 { 787 Word cnt; 788 Shdr *ishdr = 0; 789 Cache *icache; 790 Off iphdr_off = 0; 791 Xword iphdr_fsz; 792 793 /* 794 * Determine if an interp header exists. 795 */ 796 if (phnum) { 797 Phdr *phdr; 798 799 if ((phdr = getphdr(phnum, PT_INTERP, file, elf)) != 0) { 800 iphdr_off = phdr->p_offset; 801 iphdr_fsz = phdr->p_filesz; 802 } 803 } 804 805 if (iphdr_off == 0) 806 return; 807 808 /* 809 * Determine if an interp section exists. 810 */ 811 for (cnt = 1; cnt < shnum; cnt++) { 812 Cache *_cache = &cache[cnt]; 813 Shdr *shdr = _cache->c_shdr; 814 815 /* 816 * Scan sections to find a section which contains the PT_INTERP 817 * string. The target section can't be in a NOBITS section. 818 */ 819 if ((shdr->sh_type == SHT_NOBITS) || 820 (iphdr_off < shdr->sh_offset) || 821 (iphdr_off + iphdr_fsz) > (shdr->sh_offset + shdr->sh_size)) 822 continue; 823 824 icache = _cache; 825 ishdr = shdr; 826 break; 827 } 828 829 /* 830 * Print the interpreter string based on the offset defined in the 831 * program header, as this is the offset used by the kernel. 832 */ 833 if (ishdr && icache->c_data) { 834 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 835 dbg_print(0, MSG_INTL(MSG_ELF_SCN_INTERP), icache->c_name); 836 dbg_print(0, MSG_ORIG(MSG_FMT_INDENT), 837 (char *)icache->c_data->d_buf + 838 (iphdr_off - ishdr->sh_offset)); 839 } else 840 (void) fprintf(stderr, MSG_INTL(MSG_WARN_INVINTERP1), file); 841 842 /* 843 * If there are any inconsistences between the program header and 844 * section information, flag them. 845 */ 846 if (ishdr && ((iphdr_off != ishdr->sh_offset) || 847 (iphdr_fsz != ishdr->sh_size))) { 848 (void) fprintf(stderr, MSG_INTL(MSG_WARN_INVINTERP2), file, 849 icache->c_name); 850 } 851 } 852 853 /* 854 * Print the syminfo section. 855 */ 856 static void 857 syminfo(Cache *cache, Word shnum, const char *file) 858 { 859 Shdr *infoshdr; 860 Syminfo *info; 861 Sym *syms; 862 Dyn *dyns; 863 Word infonum, cnt, ndx, symnum; 864 Cache *infocache = 0, *symsec, *strsec; 865 866 for (cnt = 1; cnt < shnum; cnt++) { 867 if (cache[cnt].c_shdr->sh_type == SHT_SUNW_syminfo) { 868 infocache = &cache[cnt]; 869 break; 870 } 871 } 872 if (infocache == 0) 873 return; 874 875 infoshdr = infocache->c_shdr; 876 if ((infoshdr->sh_entsize == 0) || (infoshdr->sh_size == 0)) { 877 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 878 file, infocache->c_name); 879 return; 880 } 881 if (infocache->c_data == NULL) 882 return; 883 884 infonum = (Word)(infoshdr->sh_size / infoshdr->sh_entsize); 885 info = (Syminfo *)infocache->c_data->d_buf; 886 887 /* 888 * Get the data buffer of the associated dynamic section. 889 */ 890 if ((infoshdr->sh_info == 0) || (infoshdr->sh_info >= shnum)) { 891 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHINFO), 892 file, infocache->c_name, EC_WORD(infoshdr->sh_info)); 893 return; 894 } 895 if (cache[infoshdr->sh_info].c_data == NULL) 896 return; 897 898 dyns = cache[infoshdr->sh_info].c_data->d_buf; 899 if (dyns == 0) { 900 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 901 file, cache[infoshdr->sh_info].c_name); 902 return; 903 } 904 905 /* 906 * Get the data buffer for the associated symbol table and string table. 907 */ 908 if (stringtbl(cache, 1, cnt, shnum, file, 909 &symnum, &symsec, &strsec) == 0) 910 return; 911 912 syms = symsec->c_data->d_buf; 913 914 /* 915 * Loop through the syminfo entries. 916 */ 917 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 918 dbg_print(0, MSG_INTL(MSG_ELF_SCN_SYMINFO), infocache->c_name); 919 Elf_syminfo_title(0); 920 921 for (ndx = 1, info++; ndx < infonum; ndx++, info++) { 922 Sym *sym; 923 const char *needed = 0, *name; 924 925 if ((info->si_flags == 0) && (info->si_boundto == 0)) 926 continue; 927 928 sym = &syms[ndx]; 929 name = string(infocache, ndx, strsec, file, sym->st_name); 930 931 if (info->si_boundto < SYMINFO_BT_LOWRESERVE) { 932 Dyn *dyn = &dyns[info->si_boundto]; 933 934 needed = string(infocache, info->si_boundto, 935 strsec, file, dyn->d_un.d_val); 936 } 937 Elf_syminfo_entry(0, ndx, info, name, needed); 938 } 939 } 940 941 /* 942 * Print version definition section entries. 943 */ 944 static void 945 version_def(Verdef *vdf, Word shnum, Cache *vcache, Cache *scache, 946 const char *file) 947 { 948 Word cnt; 949 char index[MAXNDXSIZE]; 950 951 Elf_ver_def_title(0); 952 953 for (cnt = 1; cnt <= shnum; cnt++, 954 vdf = (Verdef *)((uintptr_t)vdf + vdf->vd_next)) { 955 const char *name, *dep; 956 Half vcnt = vdf->vd_cnt - 1; 957 Half ndx = vdf->vd_ndx; 958 Verdaux *vdap = (Verdaux *)((uintptr_t)vdf + 959 vdf->vd_aux); 960 961 /* 962 * Obtain the name and first dependency (if any). 963 */ 964 name = string(vcache, cnt, scache, file, vdap->vda_name); 965 vdap = (Verdaux *)((uintptr_t)vdap + vdap->vda_next); 966 if (vcnt) 967 dep = string(vcache, cnt, scache, file, vdap->vda_name); 968 else 969 dep = MSG_ORIG(MSG_STR_EMPTY); 970 971 (void) snprintf(index, MAXNDXSIZE, MSG_ORIG(MSG_FMT_INDEX), 972 EC_XWORD(ndx)); 973 Elf_ver_line_1(0, index, name, dep, 974 conv_ver_flags(vdf->vd_flags)); 975 976 /* 977 * Print any additional dependencies. 978 */ 979 if (vcnt) { 980 vdap = (Verdaux *)((uintptr_t)vdap + vdap->vda_next); 981 for (vcnt--; vcnt; vcnt--, 982 vdap = (Verdaux *)((uintptr_t)vdap + 983 vdap->vda_next)) { 984 dep = string(vcache, cnt, scache, file, 985 vdap->vda_name); 986 Elf_ver_line_2(0, MSG_ORIG(MSG_STR_EMPTY), dep); 987 } 988 } 989 } 990 } 991 992 /* 993 * Print a version needed section entries. 994 */ 995 static void 996 version_need(Verneed *vnd, Word shnum, Cache *vcache, Cache *scache, 997 const char *file) 998 { 999 Word cnt; 1000 1001 Elf_ver_need_title(0); 1002 1003 for (cnt = 1; cnt <= shnum; cnt++, 1004 vnd = (Verneed *)((uintptr_t)vnd + vnd->vn_next)) { 1005 const char *name, *dep; 1006 Half vcnt = vnd->vn_cnt; 1007 Vernaux *vnap = (Vernaux *)((uintptr_t)vnd + 1008 vnd->vn_aux); 1009 1010 /* 1011 * Obtain the name of the needed file and the version name 1012 * within it that we're dependent on. Note that the count 1013 * should be at least one, otherwise this is a pretty bogus 1014 * entry. 1015 */ 1016 name = string(vcache, cnt, scache, file, vnd->vn_file); 1017 if (vcnt) 1018 dep = string(vcache, cnt, scache, file, vnap->vna_name); 1019 else 1020 dep = MSG_INTL(MSG_STR_NULL); 1021 1022 Elf_ver_line_1(0, MSG_ORIG(MSG_STR_EMPTY), name, dep, 1023 conv_ver_flags(vnap->vna_flags)); 1024 1025 /* 1026 * Print any additional version dependencies. 1027 */ 1028 if (vcnt) { 1029 vnap = (Vernaux *)((uintptr_t)vnap + vnap->vna_next); 1030 for (vcnt--; vcnt; vcnt--, 1031 vnap = (Vernaux *)((uintptr_t)vnap + 1032 vnap->vna_next)) { 1033 dep = string(vcache, cnt, scache, file, 1034 vnap->vna_name); 1035 Elf_ver_line_3(0, MSG_ORIG(MSG_STR_EMPTY), dep, 1036 conv_ver_flags(vnap->vna_flags)); 1037 } 1038 } 1039 } 1040 } 1041 1042 /* 1043 * Display version section information if the flags require it. 1044 * Return version information needed by other output. 1045 * 1046 * entry: 1047 * cache - Cache of all section headers 1048 * shnum - # of sections in cache 1049 * file - Name of file 1050 * flags - Command line option flags 1051 * versym - VERSYM_STATE block to be filled in. 1052 */ 1053 static void 1054 versions(Cache *cache, Word shnum, const char *file, uint_t flags, 1055 VERSYM_STATE *versym) 1056 { 1057 GElf_Word cnt; 1058 const char *gnu_prefix; 1059 size_t gnu_prefix_len; 1060 1061 bzero(versym, sizeof (*versym)); 1062 gnu_prefix = MSG_ORIG(MSG_GNU_VERNAMPREFIX); 1063 gnu_prefix_len = strlen(gnu_prefix); 1064 1065 for (cnt = 1; cnt < shnum; cnt++) { 1066 void *ver; 1067 uint_t num; 1068 Cache *_cache = &cache[cnt]; 1069 Shdr *shdr = _cache->c_shdr; 1070 const char *secname = _cache->c_name; 1071 1072 /* 1073 * If the section names starts with the .gnu.version prefix, 1074 * then this object was almost certainly produced by the 1075 * GNU ld and not the native Solaris ld. 1076 */ 1077 if (strncmp(gnu_prefix, secname, gnu_prefix_len) == 0) 1078 versym->gnu = 1; 1079 1080 /* 1081 * If this is the version symbol table record its data 1082 * address for later symbol processing. 1083 */ 1084 if ((shdr->sh_type == SHT_SUNW_versym) && 1085 (_cache->c_data != NULL)) { 1086 versym->cache = _cache; 1087 versym->data = _cache->c_data->d_buf; 1088 continue; 1089 } 1090 1091 /* 1092 * If this is a version definition section, retain # of 1093 * version definitions for later symbol processing. 1094 */ 1095 if (shdr->sh_type == SHT_SUNW_verdef) 1096 versym->num_verdef = shdr->sh_info; 1097 1098 if ((flags & FLG_VERSIONS) == 0) 1099 continue; 1100 1101 if ((shdr->sh_type != SHT_SUNW_verdef) && 1102 (shdr->sh_type != SHT_SUNW_verneed)) 1103 continue; 1104 1105 /* 1106 * Determine the version section data and number. 1107 */ 1108 if ((_cache->c_data == NULL) || 1109 ((ver = (void *)_cache->c_data->d_buf) == NULL)) { 1110 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 1111 file, secname); 1112 continue; 1113 } 1114 if ((num = shdr->sh_info) == 0) { 1115 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHINFO), 1116 file, secname, EC_WORD(shdr->sh_info)); 1117 continue; 1118 } 1119 1120 /* 1121 * Get the data buffer for the associated string table. 1122 */ 1123 if ((shdr->sh_link == 0) || (shdr->sh_link >= shnum)) { 1124 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHLINK), 1125 file, secname, EC_WORD(shdr->sh_link)); 1126 continue; 1127 } 1128 1129 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 1130 if (shdr->sh_type == SHT_SUNW_verdef) { 1131 dbg_print(0, MSG_INTL(MSG_ELF_SCN_VERDEF), secname); 1132 version_def((Verdef *)ver, num, _cache, 1133 &cache[shdr->sh_link], file); 1134 } else if (shdr->sh_type == SHT_SUNW_verneed) { 1135 dbg_print(0, MSG_INTL(MSG_ELF_SCN_VERNEED), secname); 1136 version_need((Verneed *)ver, num, _cache, 1137 &cache[shdr->sh_link], file); 1138 } 1139 } 1140 } 1141 1142 /* 1143 * Initialize a symbol table state structure 1144 * 1145 * entry: 1146 * state - State structure to be initialized 1147 * cache - Cache of all section headers 1148 * shnum - # of sections in cache 1149 * secndx - Index of symbol table section 1150 * ehdr - ELF header for file 1151 * versym - Information about versym section 1152 * file - Name of file 1153 * flags - Command line option flags 1154 */ 1155 static int 1156 init_symtbl_state(SYMTBL_STATE *state, Cache *cache, Word shnum, Word secndx, 1157 Ehdr *ehdr, VERSYM_STATE *versym, const char *file, uint_t flags) 1158 { 1159 Shdr *shdr; 1160 1161 state->file = file; 1162 state->ehdr = ehdr; 1163 state->cache = cache; 1164 state->shnum = shnum; 1165 state->seccache = &cache[secndx]; 1166 state->secndx = secndx; 1167 state->secname = state->seccache->c_name; 1168 state->flags = flags; 1169 state->shxndx.checked = 0; 1170 state->shxndx.data = NULL; 1171 state->shxndx.n = 0; 1172 1173 shdr = state->seccache->c_shdr; 1174 1175 /* 1176 * Check the symbol data and per-item size. 1177 */ 1178 if ((shdr->sh_entsize == 0) || (shdr->sh_size == 0)) { 1179 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 1180 file, state->secname); 1181 return (0); 1182 } 1183 if (state->seccache->c_data == NULL) 1184 return (0); 1185 1186 /* LINTED */ 1187 state->symn = (Word)(shdr->sh_size / shdr->sh_entsize); 1188 state->sym = (Sym *)state->seccache->c_data->d_buf; 1189 1190 /* 1191 * Check associated string table section. 1192 */ 1193 if ((shdr->sh_link == 0) || (shdr->sh_link >= shnum)) { 1194 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHLINK), 1195 file, state->secname, EC_WORD(shdr->sh_link)); 1196 return (0); 1197 } 1198 1199 /* 1200 * Determine if there is a associated Versym section 1201 * with this Symbol Table. 1202 */ 1203 if (versym->cache && 1204 (versym->cache->c_shdr->sh_link == state->secndx)) 1205 state->versym = versym; 1206 else 1207 state->versym = NULL; 1208 1209 1210 return (1); 1211 } 1212 1213 /* 1214 * Determine the extended section index used for symbol tables entries. 1215 */ 1216 static void 1217 symbols_getxindex(SYMTBL_STATE * state) 1218 { 1219 uint_t symn; 1220 Word symcnt; 1221 1222 state->shxndx.checked = 1; /* Note that we've been called */ 1223 for (symcnt = 1; symcnt < state->shnum; symcnt++) { 1224 Cache *_cache = &state->cache[symcnt]; 1225 Shdr *shdr = _cache->c_shdr; 1226 1227 if ((shdr->sh_type != SHT_SYMTAB_SHNDX) || 1228 (shdr->sh_link != state->secndx)) 1229 continue; 1230 1231 if ((shdr->sh_entsize) && 1232 /* LINTED */ 1233 ((symn = (uint_t)(shdr->sh_size / shdr->sh_entsize)) == 0)) 1234 continue; 1235 1236 if (_cache->c_data == NULL) 1237 continue; 1238 1239 state->shxndx.data = _cache->c_data->d_buf; 1240 state->shxndx.n = symn; 1241 return; 1242 } 1243 } 1244 1245 /* 1246 * Produce a line of output for the given symbol 1247 * 1248 * entry: 1249 * state - Symbol table state 1250 * symndx - Index of symbol within the table 1251 * symndx_disp - Index to display. This may not be the same 1252 * as symndx if the display is relative to the logical 1253 * combination of the SUNW_ldynsym/dynsym tables. 1254 * sym - Symbol to display 1255 */ 1256 static void 1257 output_symbol(SYMTBL_STATE *state, Word symndx, Word disp_symndx, Sym *sym) 1258 { 1259 /* 1260 * Symbol types for which we check that the specified 1261 * address/size land inside the target section. 1262 */ 1263 static const int addr_symtype[STT_NUM] = { 1264 0, /* STT_NOTYPE */ 1265 1, /* STT_OBJECT */ 1266 1, /* STT_FUNC */ 1267 0, /* STT_SECTION */ 1268 0, /* STT_FILE */ 1269 1, /* STT_COMMON */ 1270 0, /* STT_TLS */ 1271 }; 1272 #if STT_NUM != (STT_TLS + 1) 1273 #error "STT_NUM has grown. Update addr_symtype[]" 1274 #endif 1275 1276 char index[MAXNDXSIZE], *sec; 1277 const char *symname; 1278 Versym verndx; 1279 uchar_t type; 1280 Shdr *tshdr; 1281 Word shndx; 1282 1283 /* Ensure symbol index is in range */ 1284 if (symndx >= state->symn) { 1285 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSORTNDX), 1286 state->file, state->secname, EC_WORD(symndx)); 1287 return; 1288 } 1289 1290 /* 1291 * If we are using extended symbol indexes, find the 1292 * corresponding SHN_SYMTAB_SHNDX table. 1293 */ 1294 if ((sym->st_shndx == SHN_XINDEX) && (state->shxndx.checked == 0)) 1295 symbols_getxindex(state); 1296 1297 /* LINTED */ 1298 symname = string(state->seccache, symndx, 1299 &state->cache[state->seccache->c_shdr->sh_link], state->file, 1300 sym->st_name); 1301 1302 tshdr = 0; 1303 sec = NULL; 1304 1305 if ((state->ehdr->e_type == ET_CORE)) 1306 sec = (char *)MSG_INTL(MSG_STR_UNKNOWN); 1307 else if ((sym->st_shndx < SHN_LORESERVE) && 1308 (sym->st_shndx < state->shnum)) { 1309 shndx = sym->st_shndx; 1310 tshdr = state->cache[shndx].c_shdr; 1311 sec = state->cache[shndx].c_name; 1312 } else if (sym->st_shndx == SHN_XINDEX) { 1313 if (state->shxndx.data) { 1314 Word _shxndx; 1315 1316 if (symndx > state->shxndx.n) { 1317 (void) fprintf(stderr, 1318 MSG_INTL(MSG_ERR_BADSYMXINDEX1), 1319 state->file, state->secname, EC_WORD(symndx)); 1320 } else if ((_shxndx = 1321 state->shxndx.data[symndx]) > state->shnum) { 1322 (void) fprintf(stderr, 1323 MSG_INTL(MSG_ERR_BADSYMXINDEX2), 1324 state->file, state->secname, EC_WORD(symndx), 1325 EC_WORD(_shxndx)); 1326 } else { 1327 shndx = _shxndx; 1328 tshdr = state->cache[shndx].c_shdr; 1329 sec = state->cache[shndx].c_name; 1330 } 1331 } else { 1332 (void) fprintf(stderr, 1333 MSG_INTL(MSG_ERR_BADSYMXINDEX3), 1334 state->file, state->secname, EC_WORD(symndx)); 1335 } 1336 } else if ((sym->st_shndx < SHN_LORESERVE) && 1337 (sym->st_shndx >= state->shnum)) { 1338 (void) fprintf(stderr, 1339 MSG_INTL(MSG_ERR_BADSYM5), state->file, 1340 state->secname, demangle(symname, state->flags), 1341 sym->st_shndx); 1342 } 1343 1344 /* 1345 * If versioning is available display the 1346 * version index. If not, then use 0. 1347 */ 1348 if (state->versym) { 1349 verndx = state->versym->data[symndx]; 1350 1351 /* 1352 * Check to see if this is a defined symbol with a 1353 * version index that is outside the valid range for 1354 * the file. If so, then there are two possiblities: 1355 * 1356 * - Files produced by the GNU ld use the top (16th) bit 1357 * as a "hidden symbol" marker. If we have 1358 * detected that this object comes from GNU ld, 1359 * then check to see if this is the case and that 1360 * the resulting masked version is in range. If so, 1361 * issue a warning describing it. 1362 * - If this is not a GNU "hidden bit" issue, then 1363 * issue a generic "out of range" error. 1364 */ 1365 if (VERNDX_INVALID(sym->st_shndx, state->versym->num_verdef, 1366 state->versym->data, verndx)) { 1367 if (state->versym->gnu && (verndx & 0x8000) && 1368 ((verndx & ~0x8000) <= 1369 state->versym->num_verdef)) { 1370 (void) fprintf(stderr, 1371 MSG_INTL(MSG_WARN_GNUVER), state->file, 1372 state->secname, EC_WORD(symndx), 1373 EC_HALF(verndx & ~0x8000)); 1374 } else { /* Generic version range error */ 1375 (void) fprintf(stderr, 1376 MSG_INTL(MSG_ERR_BADVER), state->file, 1377 state->secname, EC_WORD(symndx), 1378 EC_HALF(verndx), state->versym->num_verdef); 1379 } 1380 } 1381 } else { 1382 verndx = 0; 1383 } 1384 1385 /* 1386 * Error checking for TLS. 1387 */ 1388 type = ELF_ST_TYPE(sym->st_info); 1389 if (type == STT_TLS) { 1390 if (tshdr && 1391 (sym->st_shndx != SHN_UNDEF) && 1392 ((tshdr->sh_flags & SHF_TLS) == 0)) { 1393 (void) fprintf(stderr, 1394 MSG_INTL(MSG_ERR_BADSYM3), state->file, 1395 state->secname, demangle(symname, state->flags)); 1396 } 1397 } else if ((type != STT_SECTION) && sym->st_size && 1398 tshdr && (tshdr->sh_flags & SHF_TLS)) { 1399 (void) fprintf(stderr, 1400 MSG_INTL(MSG_ERR_BADSYM4), state->file, 1401 state->secname, demangle(symname, state->flags)); 1402 } 1403 1404 /* 1405 * If a symbol with non-zero size has a type that 1406 * specifies an address, then make sure the location 1407 * it references is actually contained within the 1408 * section. UNDEF symbols don't count in this case, 1409 * so we ignore them. 1410 * 1411 * The meaning of the st_value field in a symbol 1412 * depends on the type of object. For a relocatable 1413 * object, it is the offset within the section. 1414 * For sharable objects, it is the offset relative to 1415 * the base of the object, and for other types, it is 1416 * the virtual address. To get an offset within the 1417 * section for non-ET_REL files, we subtract the 1418 * base address of the section. 1419 */ 1420 if (addr_symtype[type] && (sym->st_size > 0) && 1421 (sym->st_shndx != SHN_UNDEF) && ((sym->st_shndx < SHN_LORESERVE) || 1422 (sym->st_shndx == SHN_XINDEX)) && (tshdr != NULL)) { 1423 Word v = sym->st_value; 1424 if (state->ehdr->e_type != ET_REL) 1425 v -= tshdr->sh_addr; 1426 if (((v + sym->st_size) > tshdr->sh_size)) { 1427 (void) fprintf(stderr, 1428 MSG_INTL(MSG_ERR_BADSYM6), state->file, 1429 state->secname, demangle(symname, state->flags), 1430 EC_WORD(shndx), EC_XWORD(tshdr->sh_size), 1431 EC_XWORD(sym->st_value), EC_XWORD(sym->st_size)); 1432 } 1433 } 1434 1435 (void) snprintf(index, MAXNDXSIZE, 1436 MSG_ORIG(MSG_FMT_INDEX), EC_XWORD(disp_symndx)); 1437 Elf_syms_table_entry(0, ELF_DBG_ELFDUMP, index, 1438 state->ehdr->e_machine, sym, verndx, sec, symname); 1439 } 1440 1441 /* 1442 * Search for and process any symbol tables. 1443 */ 1444 void 1445 symbols(Cache *cache, Word shnum, Ehdr *ehdr, const char *name, 1446 VERSYM_STATE *versym, const char *file, uint_t flags) 1447 { 1448 SYMTBL_STATE state; 1449 Cache *_cache; 1450 Word secndx; 1451 1452 for (secndx = 1; secndx < shnum; secndx++) { 1453 Word symcnt; 1454 Shdr *shdr; 1455 1456 _cache = &cache[secndx]; 1457 shdr = _cache->c_shdr; 1458 1459 if ((shdr->sh_type != SHT_SYMTAB) && 1460 (shdr->sh_type != SHT_DYNSYM) && 1461 (shdr->sh_type != SHT_SUNW_LDYNSYM)) 1462 continue; 1463 if (name && strcmp(name, _cache->c_name)) 1464 continue; 1465 1466 if (!init_symtbl_state(&state, cache, shnum, secndx, ehdr, 1467 versym, file, flags)) 1468 continue; 1469 /* 1470 * Loop through the symbol tables entries. 1471 */ 1472 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 1473 dbg_print(0, MSG_INTL(MSG_ELF_SCN_SYMTAB), state.secname); 1474 Elf_syms_table_title(0, ELF_DBG_ELFDUMP); 1475 1476 for (symcnt = 0; symcnt < state.symn; symcnt++) 1477 output_symbol(&state, symcnt, symcnt, 1478 state.sym + symcnt); 1479 } 1480 } 1481 1482 /* 1483 * Search for and process any SHT_SUNW_symsort or SHT_SUNW_tlssort sections. 1484 * These sections are always associated with the .SUNW_ldynsym./.dynsym pair. 1485 */ 1486 static void 1487 sunw_sort(Cache *cache, Word shnum, Ehdr *ehdr, const char *name, 1488 VERSYM_STATE *versym, const char *file, uint_t flags) 1489 { 1490 SYMTBL_STATE ldynsym_state, dynsym_state; 1491 Cache *sortcache, *symcache; 1492 Shdr *sortshdr, *symshdr; 1493 Word sortsecndx, symsecndx; 1494 Word ldynsym_cnt; 1495 Word *ndx; 1496 Word ndxn; 1497 int output_cnt = 0; 1498 1499 for (sortsecndx = 1; sortsecndx < shnum; sortsecndx++) { 1500 1501 sortcache = &cache[sortsecndx]; 1502 sortshdr = sortcache->c_shdr; 1503 1504 if ((sortshdr->sh_type != SHT_SUNW_symsort) && 1505 (sortshdr->sh_type != SHT_SUNW_tlssort)) 1506 continue; 1507 if (name && strcmp(name, sortcache->c_name)) 1508 continue; 1509 1510 /* 1511 * If the section references a SUNW_ldynsym, then we 1512 * expect to see the associated .dynsym immediately 1513 * following. If it references a .dynsym, there is no 1514 * SUNW_ldynsym. If it is any other type, then we don't 1515 * know what to do with it. 1516 */ 1517 if ((sortshdr->sh_link == 0) || (sortshdr->sh_link >= shnum)) { 1518 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHLINK), 1519 file, sortcache->c_name, 1520 EC_WORD(sortshdr->sh_link)); 1521 continue; 1522 } 1523 symcache = &cache[sortshdr->sh_link]; 1524 symshdr = symcache->c_shdr; 1525 symsecndx = sortshdr->sh_link; 1526 ldynsym_cnt = 0; 1527 switch (symshdr->sh_type) { 1528 case SHT_SUNW_LDYNSYM: 1529 if (!init_symtbl_state(&ldynsym_state, cache, shnum, 1530 symsecndx, ehdr, versym, file, flags)) 1531 continue; 1532 ldynsym_cnt = ldynsym_state.symn; 1533 /* 1534 * We know that the dynsym follows immediately 1535 * after the SUNW_ldynsym, and so, should be at 1536 * (sortshdr->sh_link + 1). However, elfdump is a 1537 * diagnostic tool, so we do the full paranoid 1538 * search instead. 1539 */ 1540 for (symsecndx = 1; symsecndx < shnum; symsecndx++) { 1541 symcache = &cache[symsecndx]; 1542 symshdr = symcache->c_shdr; 1543 if (symshdr->sh_type == SHT_DYNSYM) 1544 break; 1545 } 1546 if (symsecndx >= shnum) { /* Dynsym not found! */ 1547 (void) fprintf(stderr, 1548 MSG_INTL(MSG_ERR_NODYNSYM), 1549 file, sortcache->c_name); 1550 continue; 1551 } 1552 /* Fallthrough to process associated dynsym */ 1553 /*FALLTHROUGH*/ 1554 case SHT_DYNSYM: 1555 if (!init_symtbl_state(&dynsym_state, cache, shnum, 1556 symsecndx, ehdr, versym, file, flags)) 1557 continue; 1558 break; 1559 default: 1560 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADNDXSEC), 1561 file, sortcache->c_name, conv_sec_type( 1562 ehdr->e_machine, symshdr->sh_type, 0)); 1563 continue; 1564 } 1565 1566 /* 1567 * Output header 1568 */ 1569 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 1570 if (ldynsym_cnt > 0) { 1571 dbg_print(0, MSG_INTL(MSG_ELF_SCN_SYMSORT2), 1572 sortcache->c_name, ldynsym_state.secname, 1573 dynsym_state.secname); 1574 /* 1575 * The data for .SUNW_ldynsym and dynsym sections 1576 * is supposed to be adjacent with SUNW_ldynsym coming 1577 * first. Check, and issue a warning if it isn't so. 1578 */ 1579 if ((ldynsym_state.sym + ldynsym_state.symn) 1580 != dynsym_state.sym) 1581 (void) fprintf(stderr, 1582 MSG_INTL(MSG_ERR_LDYNNOTADJ), file, 1583 ldynsym_state.secname, 1584 dynsym_state.secname); 1585 } else { 1586 dbg_print(0, MSG_INTL(MSG_ELF_SCN_SYMSORT1), 1587 sortcache->c_name, dynsym_state.secname); 1588 } 1589 Elf_syms_table_title(0, ELF_DBG_ELFDUMP); 1590 1591 /* If not first one, insert a line of whitespace */ 1592 if (output_cnt++ > 0) 1593 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 1594 1595 /* 1596 * SUNW_dynsymsort and SUNW_dyntlssort are arrays of 1597 * symbol indices. Iterate over the array entries, 1598 * dispaying the referenced symbols. 1599 */ 1600 ndxn = sortshdr->sh_size / sortshdr->sh_entsize; 1601 ndx = (Word *)sortcache->c_data->d_buf; 1602 for (; ndxn-- > 0; ndx++) { 1603 if (*ndx >= ldynsym_cnt) { 1604 Word sec_ndx = *ndx - ldynsym_cnt; 1605 1606 output_symbol(&dynsym_state, sec_ndx, 1607 *ndx, dynsym_state.sym + sec_ndx); 1608 } else { 1609 output_symbol(&ldynsym_state, *ndx, 1610 *ndx, ldynsym_state.sym + *ndx); 1611 } 1612 } 1613 } 1614 } 1615 1616 /* 1617 * Search for and process any relocation sections. 1618 */ 1619 static void 1620 reloc(Cache *cache, Word shnum, Ehdr *ehdr, const char *name, const char *file, 1621 uint_t flags) 1622 { 1623 Word cnt; 1624 1625 for (cnt = 1; cnt < shnum; cnt++) { 1626 Word type, symnum; 1627 Xword relndx, relnum, relsize; 1628 void *rels; 1629 Sym *syms; 1630 Cache *symsec, *strsec; 1631 Cache *_cache = &cache[cnt]; 1632 Shdr *shdr = _cache->c_shdr; 1633 char *relname = _cache->c_name; 1634 1635 if (((type = shdr->sh_type) != SHT_RELA) && 1636 (type != SHT_REL)) 1637 continue; 1638 if (name && strcmp(name, relname)) 1639 continue; 1640 1641 /* 1642 * Decide entry size. 1643 */ 1644 if (((relsize = shdr->sh_entsize) == 0) || 1645 (relsize > shdr->sh_size)) { 1646 if (type == SHT_RELA) 1647 relsize = sizeof (Rela); 1648 else 1649 relsize = sizeof (Rel); 1650 } 1651 1652 /* 1653 * Determine the number of relocations available. 1654 */ 1655 if (shdr->sh_size == 0) { 1656 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 1657 file, relname); 1658 continue; 1659 } 1660 if (_cache->c_data == NULL) 1661 continue; 1662 1663 rels = _cache->c_data->d_buf; 1664 relnum = shdr->sh_size / relsize; 1665 1666 /* 1667 * Get the data buffer for the associated symbol table and 1668 * string table. 1669 */ 1670 if (stringtbl(cache, 1, cnt, shnum, file, 1671 &symnum, &symsec, &strsec) == 0) 1672 continue; 1673 1674 syms = symsec->c_data->d_buf; 1675 1676 /* 1677 * Loop through the relocation entries. 1678 */ 1679 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 1680 dbg_print(0, MSG_INTL(MSG_ELF_SCN_RELOC), _cache->c_name); 1681 Elf_reloc_title(0, ELF_DBG_ELFDUMP, type); 1682 1683 for (relndx = 0; relndx < relnum; relndx++, 1684 rels = (void *)((char *)rels + relsize)) { 1685 char section[BUFSIZ]; 1686 const char *symname; 1687 Word symndx, reltype; 1688 Rela *rela; 1689 Rel *rel; 1690 1691 /* 1692 * Unravel the relocation and determine the symbol with 1693 * which this relocation is associated. 1694 */ 1695 if (type == SHT_RELA) { 1696 rela = (Rela *)rels; 1697 symndx = ELF_R_SYM(rela->r_info); 1698 reltype = ELF_R_TYPE(rela->r_info); 1699 } else { 1700 rel = (Rel *)rels; 1701 symndx = ELF_R_SYM(rel->r_info); 1702 reltype = ELF_R_TYPE(rel->r_info); 1703 } 1704 1705 symname = relsymname(cache, _cache, strsec, symndx, 1706 symnum, relndx, syms, section, BUFSIZ, file, 1707 flags); 1708 1709 /* 1710 * A zero symbol index is only valid for a few 1711 * relocations. 1712 */ 1713 if (symndx == 0) { 1714 Half mach = ehdr->e_machine; 1715 int badrel = 0; 1716 1717 if ((mach == EM_SPARC) || 1718 (mach == EM_SPARC32PLUS) || 1719 (mach == EM_SPARCV9)) { 1720 if ((reltype != R_SPARC_NONE) && 1721 (reltype != R_SPARC_REGISTER) && 1722 (reltype != R_SPARC_RELATIVE)) 1723 badrel++; 1724 } else if (mach == EM_386) { 1725 if ((reltype != R_386_NONE) && 1726 (reltype != R_386_RELATIVE)) 1727 badrel++; 1728 } else if (mach == EM_AMD64) { 1729 if ((reltype != R_AMD64_NONE) && 1730 (reltype != R_AMD64_RELATIVE)) 1731 badrel++; 1732 } 1733 1734 if (badrel) { 1735 (void) fprintf(stderr, 1736 MSG_INTL(MSG_ERR_BADREL1), file, 1737 conv_reloc_type(mach, reltype, 0)); 1738 } 1739 } 1740 1741 Elf_reloc_entry_1(0, ELF_DBG_ELFDUMP, 1742 MSG_ORIG(MSG_STR_EMPTY), ehdr->e_machine, type, 1743 rels, relname, symname, 0); 1744 } 1745 } 1746 } 1747 1748 /* 1749 * Search for and process a .dynamic section. 1750 */ 1751 static void 1752 dynamic(Cache *cache, Word shnum, Ehdr *ehdr, const char *file) 1753 { 1754 Word cnt; 1755 1756 for (cnt = 1; cnt < shnum; cnt++) { 1757 Dyn *dyn; 1758 ulong_t numdyn; 1759 int ndx, end_ndx; 1760 Cache *_cache = &cache[cnt], *strsec; 1761 Shdr *shdr = _cache->c_shdr; 1762 1763 if (shdr->sh_type != SHT_DYNAMIC) 1764 continue; 1765 1766 /* 1767 * Verify the associated string table section. 1768 */ 1769 if (stringtbl(cache, 0, cnt, shnum, file, 0, 0, &strsec) == 0) 1770 continue; 1771 1772 if ((shdr->sh_entsize == 0) || (shdr->sh_size == 0)) { 1773 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 1774 file, _cache->c_name); 1775 continue; 1776 } 1777 if (_cache->c_data == NULL) 1778 continue; 1779 1780 numdyn = shdr->sh_size / shdr->sh_entsize; 1781 dyn = (Dyn *)_cache->c_data->d_buf; 1782 1783 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 1784 dbg_print(0, MSG_INTL(MSG_ELF_SCN_DYNAMIC), _cache->c_name); 1785 1786 Elf_dyn_title(0); 1787 1788 for (ndx = 0; ndx < numdyn; dyn++, ndx++) { 1789 const char *name; 1790 1791 /* 1792 * Print the information numerically, and if possible 1793 * as a string. 1794 */ 1795 switch (dyn->d_tag) { 1796 case DT_NULL: 1797 /* 1798 * Special case: DT_NULLs can come in groups 1799 * that we prefer to reduce to a single line. 1800 */ 1801 end_ndx = ndx; 1802 while ((end_ndx < (numdyn - 1)) && 1803 ((dyn + 1)->d_tag == DT_NULL)) { 1804 dyn++; 1805 end_ndx++; 1806 } 1807 Elf_dyn_null_entry(0, dyn, ndx, end_ndx); 1808 ndx = end_ndx; 1809 continue; 1810 1811 /* 1812 * Print the information numerically, and if possible 1813 * as a string. 1814 */ 1815 case DT_NEEDED: 1816 case DT_SONAME: 1817 case DT_FILTER: 1818 case DT_AUXILIARY: 1819 case DT_CONFIG: 1820 case DT_RPATH: 1821 case DT_RUNPATH: 1822 case DT_USED: 1823 case DT_DEPAUDIT: 1824 case DT_AUDIT: 1825 case DT_SUNW_AUXILIARY: 1826 case DT_SUNW_FILTER: 1827 name = string(_cache, ndx, strsec, 1828 file, dyn->d_un.d_ptr); 1829 break; 1830 1831 case DT_FLAGS: 1832 name = conv_dyn_flag(dyn->d_un.d_val, 0); 1833 break; 1834 case DT_FLAGS_1: 1835 name = conv_dyn_flag1(dyn->d_un.d_val); 1836 break; 1837 case DT_POSFLAG_1: 1838 name = conv_dyn_posflag1(dyn->d_un.d_val, 0); 1839 break; 1840 case DT_FEATURE_1: 1841 name = conv_dyn_feature1(dyn->d_un.d_val, 0); 1842 break; 1843 case DT_DEPRECATED_SPARC_REGISTER: 1844 name = MSG_INTL(MSG_STR_DEPRECATED); 1845 break; 1846 default: 1847 name = MSG_ORIG(MSG_STR_EMPTY); 1848 break; 1849 } 1850 1851 Elf_dyn_entry(0, dyn, ndx, name, ehdr->e_machine); 1852 } 1853 } 1854 } 1855 1856 /* 1857 * Search for and process a MOVE section. 1858 */ 1859 static void 1860 move(Cache *cache, Word shnum, const char *name, const char *file, uint_t flags) 1861 { 1862 Word cnt; 1863 const char *fmt = 0; 1864 1865 for (cnt = 1; cnt < shnum; cnt++) { 1866 Word movenum, symnum, ndx; 1867 Sym *syms; 1868 Cache *_cache = &cache[cnt]; 1869 Shdr *shdr = _cache->c_shdr; 1870 Cache *symsec, *strsec; 1871 Move *move; 1872 1873 if (shdr->sh_type != SHT_SUNW_move) 1874 continue; 1875 if (name && strcmp(name, _cache->c_name)) 1876 continue; 1877 1878 /* 1879 * Determine the move data and number. 1880 */ 1881 if ((shdr->sh_entsize == 0) || (shdr->sh_size == 0)) { 1882 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 1883 file, _cache->c_name); 1884 continue; 1885 } 1886 if (_cache->c_data == NULL) 1887 continue; 1888 1889 move = (Move *)_cache->c_data->d_buf; 1890 movenum = shdr->sh_size / shdr->sh_entsize; 1891 1892 /* 1893 * Get the data buffer for the associated symbol table and 1894 * string table. 1895 */ 1896 if (stringtbl(cache, 1, cnt, shnum, file, 1897 &symnum, &symsec, &strsec) == 0) 1898 return; 1899 1900 syms = (Sym *)symsec->c_data->d_buf; 1901 1902 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 1903 dbg_print(0, MSG_INTL(MSG_ELF_SCN_MOVE), _cache->c_name); 1904 dbg_print(0, MSG_INTL(MSG_MOVE_TITLE)); 1905 1906 if (fmt == 0) 1907 fmt = MSG_INTL(MSG_MOVE_ENTRY); 1908 1909 for (ndx = 0; ndx < movenum; move++, ndx++) { 1910 const char *symname; 1911 char index[MAXNDXSIZE], section[BUFSIZ]; 1912 Word symndx, shndx; 1913 Sym *sym; 1914 1915 /* 1916 * Check for null entries 1917 */ 1918 if ((move->m_info == 0) && (move->m_value == 0) && 1919 (move->m_poffset == 0) && (move->m_repeat == 0) && 1920 (move->m_stride == 0)) { 1921 dbg_print(0, fmt, MSG_ORIG(MSG_STR_EMPTY), 1922 EC_XWORD(move->m_poffset), 0, 0, 0, 1923 EC_LWORD(0), MSG_ORIG(MSG_STR_EMPTY)); 1924 continue; 1925 } 1926 if (((symndx = ELF_M_SYM(move->m_info)) == 0) || 1927 (symndx >= symnum)) { 1928 (void) fprintf(stderr, 1929 MSG_INTL(MSG_ERR_BADMINFO), file, 1930 _cache->c_name, EC_XWORD(move->m_info)); 1931 1932 (void) snprintf(index, MAXNDXSIZE, 1933 MSG_ORIG(MSG_FMT_INDEX), EC_XWORD(symndx)); 1934 dbg_print(0, fmt, index, 1935 EC_XWORD(move->m_poffset), 1936 ELF_M_SIZE(move->m_info), move->m_repeat, 1937 move->m_stride, move->m_value, 1938 MSG_INTL(MSG_STR_UNKNOWN)); 1939 continue; 1940 } 1941 1942 symname = relsymname(cache, _cache, strsec, 1943 symndx, symnum, ndx, syms, section, BUFSIZ, file, 1944 flags); 1945 sym = (Sym *)(syms + symndx); 1946 1947 /* 1948 * Additional sanity check. 1949 */ 1950 shndx = sym->st_shndx; 1951 if (!((shndx == SHN_COMMON) || 1952 (((shndx >= 1) && (shndx <= shnum)) && 1953 (cache[shndx].c_shdr)->sh_type == SHT_NOBITS))) { 1954 (void) fprintf(stderr, 1955 MSG_INTL(MSG_ERR_BADSYM2), file, 1956 _cache->c_name, demangle(symname, flags)); 1957 } 1958 1959 (void) snprintf(index, MAXNDXSIZE, 1960 MSG_ORIG(MSG_FMT_INDEX), EC_XWORD(symndx)); 1961 dbg_print(0, fmt, index, EC_XWORD(move->m_poffset), 1962 ELF_M_SIZE(move->m_info), move->m_repeat, 1963 move->m_stride, move->m_value, 1964 demangle(symname, flags)); 1965 } 1966 } 1967 } 1968 1969 /* 1970 * Traverse a note section analyzing each note information block. 1971 * The data buffers size is used to validate references before they are made, 1972 * and is decremented as each element is processed. 1973 */ 1974 void 1975 note_entry(Cache *cache, Word *data, size_t size, const char *file) 1976 { 1977 size_t bsize = size; 1978 1979 /* 1980 * Print out a single `note' information block. 1981 */ 1982 while (size > 0) { 1983 size_t namesz, descsz, type, pad, noteoff; 1984 1985 noteoff = bsize - size; 1986 /* 1987 * Make sure we can at least reference the 3 initial entries 1988 * (4-byte words) of the note information block. 1989 */ 1990 if (size >= (sizeof (Word) * 3)) 1991 size -= (sizeof (Word) * 3); 1992 else { 1993 (void) fprintf(stderr, MSG_INTL(MSG_NOTE_BADDATASZ), 1994 file, cache->c_name, EC_WORD(noteoff)); 1995 return; 1996 } 1997 1998 /* 1999 * Make sure any specified name string can be referenced. 2000 */ 2001 if ((namesz = *data++) != 0) { 2002 if (size >= namesz) 2003 size -= namesz; 2004 else { 2005 (void) fprintf(stderr, 2006 MSG_INTL(MSG_NOTE_BADNMSZ), file, 2007 cache->c_name, EC_WORD(noteoff), 2008 EC_WORD(namesz)); 2009 return; 2010 } 2011 } 2012 2013 /* 2014 * Make sure any specified descriptor can be referenced. 2015 */ 2016 if ((descsz = *data++) != 0) { 2017 /* 2018 * If namesz isn't a 4-byte multiple, account for any 2019 * padding that must exist before the descriptor. 2020 */ 2021 if ((pad = (namesz & (sizeof (Word) - 1))) != 0) { 2022 pad = sizeof (Word) - pad; 2023 size -= pad; 2024 } 2025 if (size >= descsz) 2026 size -= descsz; 2027 else { 2028 (void) fprintf(stderr, 2029 MSG_INTL(MSG_NOTE_BADDESZ), file, 2030 cache->c_name, EC_WORD(noteoff), 2031 EC_WORD(namesz)); 2032 return; 2033 } 2034 } 2035 2036 type = *data++; 2037 2038 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 2039 dbg_print(0, MSG_ORIG(MSG_NOTE_TYPE), EC_WORD(type)); 2040 2041 dbg_print(0, MSG_ORIG(MSG_NOTE_NAMESZ), EC_WORD(namesz)); 2042 if (namesz) { 2043 char *name = (char *)data; 2044 2045 /* 2046 * Since the name string may have 'null' bytes 2047 * in it (ia32 .string) - we just write the 2048 * whole stream in a single fwrite. 2049 */ 2050 (void) fwrite(name, namesz, 1, stdout); 2051 name = name + ((namesz + (sizeof (Word) - 1)) & 2052 ~(sizeof (Word) - 1)); 2053 /* LINTED */ 2054 data = (Word *)name; 2055 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 2056 } 2057 2058 /* 2059 * If multiple information blocks exist within a .note section 2060 * account for any padding that must exist before the next 2061 * information block. 2062 */ 2063 if ((pad = (descsz & (sizeof (Word) - 1))) != 0) { 2064 pad = sizeof (Word) - pad; 2065 if (size > pad) 2066 size -= pad; 2067 } 2068 2069 dbg_print(0, MSG_ORIG(MSG_NOTE_DESCSZ), EC_WORD(descsz)); 2070 if (descsz) { 2071 int ndx, byte, word; 2072 char string[58], *str = string; 2073 uchar_t *desc = (uchar_t *)data; 2074 2075 /* 2076 * Dump descriptor bytes. 2077 */ 2078 for (ndx = byte = word = 0; descsz; descsz--, desc++) { 2079 int tok = *desc; 2080 2081 (void) snprintf(str, 58, MSG_ORIG(MSG_NOTE_TOK), 2082 tok); 2083 str += 3; 2084 2085 if (++byte == 4) { 2086 *str++ = ' ', *str++ = ' '; 2087 word++; 2088 byte = 0; 2089 } 2090 if (word == 4) { 2091 *str = '\0'; 2092 dbg_print(0, MSG_ORIG(MSG_NOTE_DESC), 2093 ndx, string); 2094 word = 0; 2095 ndx += 16; 2096 str = string; 2097 } 2098 } 2099 if (byte || word) { 2100 *str = '\0'; 2101 dbg_print(0, MSG_ORIG(MSG_NOTE_DESC), 2102 ndx, string); 2103 } 2104 2105 desc += pad; 2106 /* LINTED */ 2107 data = (Word *)desc; 2108 } 2109 } 2110 } 2111 2112 /* 2113 * Search for and process a .note section. 2114 */ 2115 static void 2116 note(Cache *cache, Word shnum, const char *name, const char *file) 2117 { 2118 Word cnt; 2119 2120 /* 2121 * Otherwise look for any .note sections. 2122 */ 2123 for (cnt = 1; cnt < shnum; cnt++) { 2124 Cache *_cache = &cache[cnt]; 2125 Shdr *shdr = _cache->c_shdr; 2126 2127 if (shdr->sh_type != SHT_NOTE) 2128 continue; 2129 if (name && strcmp(name, _cache->c_name)) 2130 continue; 2131 2132 /* 2133 * As these sections are often hand rolled, make sure they're 2134 * properly aligned before proceeding. 2135 */ 2136 if (shdr->sh_offset & (sizeof (Word) - 1)) { 2137 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADALIGN), 2138 file, _cache->c_name); 2139 continue; 2140 } 2141 if (_cache->c_data == NULL) 2142 continue; 2143 2144 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 2145 dbg_print(0, MSG_INTL(MSG_ELF_SCN_NOTE), _cache->c_name); 2146 note_entry(_cache, (Word *)_cache->c_data->d_buf, 2147 /* LINTED */ 2148 (Word)_cache->c_data->d_size, file); 2149 } 2150 } 2151 2152 /* 2153 * Determine an individual hash entry. This may be the initial hash entry, 2154 * or an associated chain entry. 2155 */ 2156 static void 2157 hash_entry(Cache *refsec, Cache *strsec, const char *hsecname, Word hashndx, 2158 Word symndx, Word symn, Sym *syms, const char *file, ulong_t bkts, 2159 uint_t flags, int chain) 2160 { 2161 Sym *sym; 2162 const char *symname, *str; 2163 char _bucket[MAXNDXSIZE], _symndx[MAXNDXSIZE]; 2164 ulong_t nbkt, nhash; 2165 2166 if (symndx > symn) { 2167 (void) fprintf(stderr, MSG_INTL(MSG_ERR_HSBADSYMNDX), file, 2168 EC_WORD(symndx), EC_WORD(hashndx)); 2169 symname = MSG_INTL(MSG_STR_UNKNOWN); 2170 } else { 2171 sym = (Sym *)(syms + symndx); 2172 symname = string(refsec, symndx, strsec, file, sym->st_name); 2173 } 2174 2175 if (chain == 0) { 2176 (void) snprintf(_bucket, MAXNDXSIZE, MSG_ORIG(MSG_FMT_INTEGER), 2177 hashndx); 2178 str = (const char *)_bucket; 2179 } else 2180 str = MSG_ORIG(MSG_STR_EMPTY); 2181 2182 (void) snprintf(_symndx, MAXNDXSIZE, MSG_ORIG(MSG_FMT_INDEX2), 2183 EC_WORD(symndx)); 2184 dbg_print(0, MSG_ORIG(MSG_FMT_HASH_INFO), str, _symndx, 2185 demangle(symname, flags)); 2186 2187 /* 2188 * Determine if this string is in the correct bucket. 2189 */ 2190 nhash = elf_hash(symname); 2191 nbkt = nhash % bkts; 2192 2193 if (nbkt != hashndx) { 2194 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADHASH), file, 2195 hsecname, symname, EC_WORD(hashndx), nbkt); 2196 } 2197 } 2198 2199 #define MAXCOUNT 500 2200 2201 static void 2202 hash(Cache *cache, Word shnum, const char *name, const char *file, uint_t flags) 2203 { 2204 static int count[MAXCOUNT]; 2205 Word cnt; 2206 ulong_t ndx, bkts; 2207 char number[MAXNDXSIZE]; 2208 2209 for (cnt = 1; cnt < shnum; cnt++) { 2210 uint_t *hash, *chain; 2211 Cache *_cache = &cache[cnt]; 2212 Shdr *sshdr, *hshdr = _cache->c_shdr; 2213 char *ssecname, *hsecname = _cache->c_name; 2214 Sym *syms; 2215 Word symn; 2216 2217 if (hshdr->sh_type != SHT_HASH) 2218 continue; 2219 if (name && strcmp(name, hsecname)) 2220 continue; 2221 2222 /* 2223 * Determine the hash table data and size. 2224 */ 2225 if ((hshdr->sh_entsize == 0) || (hshdr->sh_size == 0)) { 2226 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 2227 file, hsecname); 2228 continue; 2229 } 2230 if (_cache->c_data == NULL) 2231 continue; 2232 2233 hash = (uint_t *)_cache->c_data->d_buf; 2234 bkts = *hash; 2235 chain = hash + 2 + bkts; 2236 hash += 2; 2237 2238 /* 2239 * Get the data buffer for the associated symbol table. 2240 */ 2241 if ((hshdr->sh_link == 0) || (hshdr->sh_link >= shnum)) { 2242 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHLINK), 2243 file, hsecname, EC_WORD(hshdr->sh_link)); 2244 continue; 2245 } 2246 2247 _cache = &cache[hshdr->sh_link]; 2248 ssecname = _cache->c_name; 2249 2250 if (_cache->c_data == NULL) 2251 continue; 2252 2253 if ((syms = (Sym *)_cache->c_data->d_buf) == NULL) { 2254 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 2255 file, ssecname); 2256 continue; 2257 } 2258 2259 sshdr = _cache->c_shdr; 2260 /* LINTED */ 2261 symn = (Word)(sshdr->sh_size / sshdr->sh_entsize); 2262 2263 /* 2264 * Get the associated string table section. 2265 */ 2266 if ((sshdr->sh_link == 0) || (sshdr->sh_link >= shnum)) { 2267 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHLINK), 2268 file, ssecname, EC_WORD(sshdr->sh_link)); 2269 continue; 2270 } 2271 2272 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 2273 dbg_print(0, MSG_INTL(MSG_ELF_SCN_HASH), hsecname); 2274 dbg_print(0, MSG_INTL(MSG_ELF_HASH_INFO)); 2275 2276 /* 2277 * Loop through the hash buckets, printing the appropriate 2278 * symbols. 2279 */ 2280 for (ndx = 0; ndx < bkts; ndx++, hash++) { 2281 Word _ndx, _cnt; 2282 2283 if (*hash == 0) { 2284 count[0]++; 2285 continue; 2286 } 2287 2288 hash_entry(_cache, &cache[sshdr->sh_link], hsecname, 2289 ndx, *hash, symn, syms, file, bkts, flags, 0); 2290 2291 /* 2292 * Determine if any other symbols are chained to this 2293 * bucket. 2294 */ 2295 _ndx = chain[*hash]; 2296 _cnt = 1; 2297 while (_ndx) { 2298 hash_entry(_cache, &cache[sshdr->sh_link], 2299 hsecname, ndx, _ndx, symn, syms, file, 2300 bkts, flags, 1); 2301 _ndx = chain[_ndx]; 2302 _cnt++; 2303 } 2304 2305 if (_cnt >= MAXCOUNT) { 2306 (void) fprintf(stderr, 2307 MSG_INTL(MSG_HASH_OVERFLW), file, 2308 _cache->c_name, EC_WORD(ndx), 2309 EC_WORD(_cnt)); 2310 } else 2311 count[_cnt]++; 2312 } 2313 break; 2314 } 2315 2316 /* 2317 * Print out the count information. 2318 */ 2319 bkts = cnt = 0; 2320 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 2321 2322 for (ndx = 0; ndx < MAXCOUNT; ndx++) { 2323 Word _cnt; 2324 2325 if ((_cnt = count[ndx]) == 0) 2326 continue; 2327 2328 (void) snprintf(number, MAXNDXSIZE, 2329 MSG_ORIG(MSG_FMT_INTEGER), _cnt); 2330 dbg_print(0, MSG_INTL(MSG_ELF_HASH_BKTS1), number, 2331 EC_WORD(ndx)); 2332 bkts += _cnt; 2333 cnt += (Word)(ndx * _cnt); 2334 } 2335 if (cnt) { 2336 (void) snprintf(number, MAXNDXSIZE, MSG_ORIG(MSG_FMT_INTEGER), 2337 bkts); 2338 dbg_print(0, MSG_INTL(MSG_ELF_HASH_BKTS2), number, 2339 EC_WORD(cnt)); 2340 } 2341 } 2342 2343 static void 2344 group(Cache *cache, Word shnum, const char *name, const char *file, 2345 uint_t flags) 2346 { 2347 Word scnt; 2348 2349 for (scnt = 1; scnt < shnum; scnt++) { 2350 Cache *_cache = &cache[scnt]; 2351 Shdr *shdr = _cache->c_shdr; 2352 Word *grpdata, gcnt, grpcnt, symnum, unknown; 2353 Cache *symsec, *strsec; 2354 Sym *syms, *sym; 2355 char flgstrbuf[MSG_GRP_COMDAT_SIZE + 10]; 2356 2357 if (shdr->sh_type != SHT_GROUP) 2358 continue; 2359 if (name && strcmp(name, _cache->c_name)) 2360 continue; 2361 if ((_cache->c_data == NULL) || 2362 ((grpdata = (Word *)_cache->c_data->d_buf) == NULL)) 2363 continue; 2364 grpcnt = shdr->sh_size / sizeof (Word); 2365 2366 /* 2367 * Get the data buffer for the associated symbol table and 2368 * string table. 2369 */ 2370 if (stringtbl(cache, 1, scnt, shnum, file, 2371 &symnum, &symsec, &strsec) == 0) 2372 return; 2373 2374 syms = symsec->c_data->d_buf; 2375 2376 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 2377 dbg_print(0, MSG_INTL(MSG_ELF_SCN_GRP), _cache->c_name); 2378 dbg_print(0, MSG_INTL(MSG_GRP_TITLE)); 2379 2380 /* 2381 * The first element of the group defines the group. The 2382 * associated symbol is defined by the sh_link field. 2383 */ 2384 if ((shdr->sh_info == SHN_UNDEF) || (shdr->sh_info > symnum)) { 2385 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSHINFO), 2386 file, _cache->c_name, EC_WORD(shdr->sh_info)); 2387 return; 2388 } 2389 2390 (void) strcpy(flgstrbuf, MSG_ORIG(MSG_STR_OSQBRKT)); 2391 if (grpdata[0] & GRP_COMDAT) { 2392 (void) strcat(flgstrbuf, MSG_ORIG(MSG_GRP_COMDAT)); 2393 } 2394 if ((unknown = (grpdata[0] & ~GRP_COMDAT)) != 0) { 2395 size_t len = strlen(flgstrbuf); 2396 2397 (void) snprintf(&flgstrbuf[len], 2398 (MSG_GRP_COMDAT_SIZE + 10 - len), 2399 MSG_ORIG(MSG_GRP_UNKNOWN), unknown); 2400 } 2401 (void) strcat(flgstrbuf, MSG_ORIG(MSG_STR_CSQBRKT)); 2402 sym = (Sym *)(syms + shdr->sh_info); 2403 2404 dbg_print(0, MSG_INTL(MSG_GRP_SIGNATURE), flgstrbuf, 2405 demangle(string(_cache, 0, strsec, file, sym->st_name), 2406 flags)); 2407 2408 for (gcnt = 1; gcnt < grpcnt; gcnt++) { 2409 char index[MAXNDXSIZE]; 2410 const char *name; 2411 2412 (void) snprintf(index, MAXNDXSIZE, 2413 MSG_ORIG(MSG_FMT_INDEX), EC_XWORD(gcnt)); 2414 2415 if (grpdata[gcnt] >= shnum) 2416 name = MSG_INTL(MSG_GRP_INVALSCN); 2417 else 2418 name = cache[grpdata[gcnt]].c_name; 2419 2420 (void) printf(MSG_ORIG(MSG_GRP_ENTRY), index, name, 2421 EC_XWORD(grpdata[gcnt])); 2422 } 2423 } 2424 } 2425 2426 static void 2427 got(Cache *cache, Word shnum, Ehdr *ehdr, const char *file, uint_t flags) 2428 { 2429 Cache *gotcache = 0, *symtab = 0, *_cache; 2430 Addr gotbgn, gotend; 2431 Shdr *gotshdr; 2432 Word cnt, gotents, gotndx; 2433 size_t gentsize; 2434 Got_info *gottable; 2435 char *gotdata; 2436 Sym *gotsym; 2437 Xword gotsymaddr; 2438 2439 /* 2440 * First, find the got. 2441 */ 2442 for (cnt = 1; cnt < shnum; cnt++) { 2443 _cache = &cache[cnt]; 2444 if (strncmp(_cache->c_name, MSG_ORIG(MSG_ELF_GOT), 2445 MSG_ELF_GOT_SIZE) == 0) { 2446 gotcache = _cache; 2447 break; 2448 } 2449 } 2450 if (gotcache == 0) 2451 return; 2452 2453 /* 2454 * A got section within a relocatable object is suspicious. 2455 */ 2456 if (ehdr->e_type == ET_REL) { 2457 (void) fprintf(stderr, MSG_INTL(MSG_GOT_UNEXPECTED), file, 2458 _cache->c_name); 2459 } 2460 2461 gotshdr = gotcache->c_shdr; 2462 if (gotshdr->sh_size == 0) { 2463 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 2464 file, gotcache->c_name); 2465 return; 2466 } 2467 2468 gotbgn = gotshdr->sh_addr; 2469 gotend = gotbgn + gotshdr->sh_size; 2470 2471 /* 2472 * Some architectures don't properly set the sh_entsize for the GOT 2473 * table. If it's not set, default to a size of a pointer. 2474 */ 2475 if ((gentsize = gotshdr->sh_entsize) == 0) 2476 gentsize = sizeof (Xword); 2477 2478 if (gotcache->c_data == NULL) 2479 return; 2480 2481 /* LINTED */ 2482 gotents = (Word)(gotshdr->sh_size / gentsize); 2483 gotdata = gotcache->c_data->d_buf; 2484 2485 if ((gottable = calloc(gotents, sizeof (Got_info))) == 0) { 2486 int err = errno; 2487 (void) fprintf(stderr, MSG_INTL(MSG_ERR_MALLOC), file, 2488 strerror(err)); 2489 return; 2490 } 2491 2492 /* 2493 * Now we scan through all the sections looking for any relocations 2494 * that may be against the GOT. Since these may not be isolated to a 2495 * .rel[a].got section we check them all. 2496 * While scanning sections save the symbol table entry (a symtab 2497 * overriding a dynsym) so that we can lookup _GLOBAL_OFFSET_TABLE_. 2498 */ 2499 for (cnt = 1; cnt < shnum; cnt++) { 2500 Word type, symnum; 2501 Xword relndx, relnum, relsize; 2502 void *rels; 2503 Sym *syms; 2504 Cache *symsec, *strsec; 2505 Cache *_cache = &cache[cnt]; 2506 Shdr *shdr; 2507 2508 shdr = _cache->c_shdr; 2509 type = shdr->sh_type; 2510 2511 if ((symtab == 0) && (type == SHT_DYNSYM)) { 2512 symtab = _cache; 2513 continue; 2514 } 2515 if (type == SHT_SYMTAB) { 2516 symtab = _cache; 2517 continue; 2518 } 2519 if ((type != SHT_RELA) && (type != SHT_REL)) 2520 continue; 2521 2522 /* 2523 * Decide entry size. 2524 */ 2525 if (((relsize = shdr->sh_entsize) == 0) || 2526 (relsize > shdr->sh_size)) { 2527 if (type == SHT_RELA) 2528 relsize = sizeof (Rela); 2529 else 2530 relsize = sizeof (Rel); 2531 } 2532 2533 /* 2534 * Determine the number of relocations available. 2535 */ 2536 if (shdr->sh_size == 0) { 2537 (void) fprintf(stderr, MSG_INTL(MSG_ERR_BADSZ), 2538 file, _cache->c_name); 2539 continue; 2540 } 2541 if (_cache->c_data == NULL) 2542 continue; 2543 2544 rels = _cache->c_data->d_buf; 2545 relnum = shdr->sh_size / relsize; 2546 2547 /* 2548 * Get the data buffer for the associated symbol table and 2549 * string table. 2550 */ 2551 if (stringtbl(cache, 1, cnt, shnum, file, 2552 &symnum, &symsec, &strsec) == 0) 2553 continue; 2554 2555 syms = symsec->c_data->d_buf; 2556 2557 /* 2558 * Loop through the relocation entries. 2559 */ 2560 for (relndx = 0; relndx < relnum; relndx++, 2561 rels = (void *)((char *)rels + relsize)) { 2562 char section[BUFSIZ]; 2563 Addr offset; 2564 Got_info *gip; 2565 Word symndx, reltype; 2566 Rela *rela; 2567 Rel *rel; 2568 2569 /* 2570 * Unravel the relocation. 2571 */ 2572 if (type == SHT_RELA) { 2573 rela = (Rela *)rels; 2574 symndx = ELF_R_SYM(rela->r_info); 2575 reltype = ELF_R_TYPE(rela->r_info); 2576 offset = rela->r_offset; 2577 } else { 2578 rel = (Rel *)rels; 2579 symndx = ELF_R_SYM(rel->r_info); 2580 reltype = ELF_R_TYPE(rel->r_info); 2581 offset = rel->r_offset; 2582 } 2583 2584 /* 2585 * Only pay attention to relocations against the GOT. 2586 */ 2587 if ((offset < gotbgn) || (offset > gotend)) 2588 continue; 2589 2590 /* LINTED */ 2591 gotndx = (Word)((offset - gotbgn) / 2592 gotshdr->sh_entsize); 2593 gip = &gottable[gotndx]; 2594 2595 if (gip->g_reltype != 0) { 2596 (void) fprintf(stderr, 2597 MSG_INTL(MSG_GOT_MULTIPLE), file, 2598 EC_WORD(gotndx), EC_ADDR(offset)); 2599 continue; 2600 } 2601 2602 if (symndx) 2603 gip->g_symname = relsymname(cache, _cache, 2604 strsec, symndx, symnum, relndx, syms, 2605 section, BUFSIZ, file, flags); 2606 gip->g_reltype = reltype; 2607 gip->g_rel = rels; 2608 } 2609 } 2610 2611 if (symlookup(MSG_ORIG(MSG_GOT_SYM), cache, shnum, &gotsym, symtab, 2612 file)) 2613 gotsymaddr = gotsym->st_value; 2614 else 2615 gotsymaddr = gotbgn; 2616 2617 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 2618 dbg_print(0, MSG_INTL(MSG_ELF_SCN_GOT), gotcache->c_name); 2619 Elf_got_title(0); 2620 2621 for (gotndx = 0; gotndx < gotents; gotndx++) { 2622 Got_info *gip; 2623 Sword gindex; 2624 Addr gaddr; 2625 Xword gotentry; 2626 2627 gip = &gottable[gotndx]; 2628 2629 gaddr = gotbgn + (gotndx * gentsize); 2630 gindex = (Sword)(gaddr - gotsymaddr) / (Sword)gentsize; 2631 2632 if (gentsize == sizeof (Word)) 2633 /* LINTED */ 2634 gotentry = (Xword)(*((Word *)(gotdata) + gotndx)); 2635 else 2636 /* LINTED */ 2637 gotentry = *((Xword *)(gotdata) + gotndx); 2638 2639 Elf_got_entry(0, gindex, gaddr, gotentry, ehdr->e_machine, 2640 gip->g_reltype, gip->g_rel, gip->g_symname); 2641 } 2642 free(gottable); 2643 } 2644 2645 void 2646 checksum(Elf *elf) 2647 { 2648 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 2649 dbg_print(0, MSG_INTL(MSG_STR_CHECKSUM), elf_checksum(elf)); 2650 } 2651 2652 void 2653 regular(const char *file, Elf *elf, uint_t flags, char *Nname, int wfd) 2654 { 2655 Elf_Scn *scn; 2656 Ehdr *ehdr; 2657 Elf_Data *data; 2658 size_t cnt, shstrndx, shnum, phnum; 2659 Shdr *nameshdr, *shdr; 2660 char *names = 0; 2661 Cache *cache, *_cache; 2662 VERSYM_STATE versym; 2663 2664 if ((ehdr = elf_getehdr(elf)) == NULL) { 2665 failure(file, MSG_ORIG(MSG_ELF_GETEHDR)); 2666 return; 2667 } 2668 2669 if (elf_getshnum(elf, &shnum) == 0) { 2670 failure(file, MSG_ORIG(MSG_ELF_GETSHNUM)); 2671 return; 2672 } 2673 2674 if (elf_getshstrndx(elf, &shstrndx) == 0) { 2675 failure(file, MSG_ORIG(MSG_ELF_GETSHSTRNDX)); 2676 return; 2677 } 2678 2679 if (elf_getphnum(elf, &phnum) == 0) { 2680 failure(file, MSG_ORIG(MSG_ELF_GETPHNUM)); 2681 return; 2682 } 2683 2684 if ((scn = elf_getscn(elf, 0)) != NULL) { 2685 if ((shdr = elf_getshdr(scn)) == NULL) { 2686 failure(file, MSG_ORIG(MSG_ELF_GETSHDR)); 2687 (void) fprintf(stderr, MSG_INTL(MSG_ELF_ERR_SCN), 0); 2688 return; 2689 } 2690 } else 2691 shdr = 0; 2692 2693 /* 2694 * Print the elf header. 2695 */ 2696 if (flags & FLG_EHDR) 2697 Elf_ehdr(0, ehdr, shdr); 2698 2699 /* 2700 * Print the program headers. 2701 */ 2702 if ((flags & FLG_PHDR) && (phnum != 0)) { 2703 Phdr *phdr; 2704 2705 if ((phdr = elf_getphdr(elf)) == NULL) { 2706 failure(file, MSG_ORIG(MSG_ELF_GETPHDR)); 2707 return; 2708 } 2709 2710 for (cnt = 0; cnt < phnum; phdr++, cnt++) { 2711 2712 if (Nname && 2713 (strcmp(Nname, conv_phdr_type(ehdr->e_machine, 2714 phdr->p_type, CONV_FMT_ALTFILE)) != 0)) 2715 continue; 2716 2717 dbg_print(0, MSG_ORIG(MSG_STR_EMPTY)); 2718 dbg_print(0, MSG_INTL(MSG_ELF_PHDR), EC_WORD(cnt)); 2719 Elf_phdr(0, ehdr->e_machine, phdr); 2720 } 2721 } 2722 2723 /* 2724 * Return now if there are no section, if there's just one section to 2725 * act as an extension of the ELF header, or if on section information 2726 * was requested. 2727 */ 2728 if ((shnum <= 1) || (flags && (flags & ~(FLG_EHDR | FLG_PHDR)) == 0)) { 2729 if ((ehdr->e_type == ET_CORE) && (flags & FLG_NOTE)) 2730 note(0, shnum, 0, file); 2731 return; 2732 } 2733 2734 /* 2735 * Obtain the .shstrtab data buffer to provide the required section 2736 * name strings. 2737 */ 2738 if ((scn = elf_getscn(elf, shstrndx)) == NULL) { 2739 failure(file, MSG_ORIG(MSG_ELF_GETSCN)); 2740 (void) fprintf(stderr, MSG_INTL(MSG_ELF_ERR_SHDR), 2741 EC_XWORD(shstrndx)); 2742 2743 } else if ((data = elf_getdata(scn, NULL)) == NULL) { 2744 failure(file, MSG_ORIG(MSG_ELF_GETDATA)); 2745 (void) fprintf(stderr, MSG_INTL(MSG_ELF_ERR_DATA), 2746 EC_XWORD(shstrndx)); 2747 2748 } else if ((nameshdr = elf_getshdr(scn)) == NULL) { 2749 failure(file, MSG_ORIG(MSG_ELF_GETSHDR)); 2750 (void) fprintf(stderr, MSG_INTL(MSG_ELF_ERR_SCN), 2751 EC_WORD(elf_ndxscn(scn))); 2752 2753 } else if ((names = data->d_buf) == 0) 2754 (void) fprintf(stderr, MSG_INTL(MSG_ERR_SHSTRNULL), file); 2755 2756 /* 2757 * Allocate a cache to maintain a descriptor for each section. 2758 */ 2759 if ((cache = malloc(shnum * sizeof (Cache))) == 0) { 2760 int err = errno; 2761 (void) fprintf(stderr, MSG_INTL(MSG_ERR_MALLOC), 2762 file, strerror(err)); 2763 return; 2764 } 2765 2766 *cache = cache_init; 2767 _cache = cache; 2768 _cache++; 2769 2770 /* 2771 * Traverse the sections of the file. This gathering of data is 2772 * carried out in two passes. First, the section headers are captured 2773 * and the section header names are evaluated. A verification pass is 2774 * then carried out over the section information. Files have been 2775 * known to exhibit overlapping (and hence erroneous) section header 2776 * information. 2777 * 2778 * Finally, the data for each section is obtained. This processing is 2779 * carried out after section verification because should any section 2780 * header overlap occur, and a file needs translating (ie. xlate'ing 2781 * information from a non-native architecture file), then the process 2782 * of translation can corrupt the section header information. Of 2783 * course, if there is any section overlap, the data related to the 2784 * sections is going to be compromised. However, it is the translation 2785 * of this data that has caused problems with elfdump()'s ability to 2786 * extract the data. 2787 */ 2788 for (cnt = 1, scn = NULL; scn = elf_nextscn(elf, scn); 2789 cnt++, _cache++) { 2790 char scnndxnm[100]; 2791 2792 _cache->c_ndx = cnt; 2793 _cache->c_scn = scn; 2794 2795 if ((_cache->c_shdr = elf_getshdr(scn)) == NULL) { 2796 failure(file, MSG_ORIG(MSG_ELF_GETSHDR)); 2797 (void) fprintf(stderr, MSG_INTL(MSG_ELF_ERR_SCN), 2798 EC_WORD(elf_ndxscn(scn))); 2799 } 2800 2801 /* 2802 * If a shstrtab exists, assign the section name. 2803 */ 2804 if (names && _cache->c_shdr) { 2805 if (_cache->c_shdr->sh_name && 2806 /* LINTED */ 2807 (nameshdr->sh_size > _cache->c_shdr->sh_name)) { 2808 _cache->c_name = 2809 names + _cache->c_shdr->sh_name; 2810 continue; 2811 } 2812 2813 /* 2814 * Generate an error if the section name index is zero 2815 * or exceeds the shstrtab data. Fall through to 2816 * fabricate a section name. 2817 */ 2818 if ((_cache->c_shdr->sh_name == 0) || 2819 /* LINTED */ 2820 (nameshdr->sh_size <= _cache->c_shdr->sh_name)) { 2821 (void) fprintf(stderr, 2822 MSG_INTL(MSG_ERR_BADSHNAME), file, 2823 EC_WORD(cnt), 2824 EC_XWORD(_cache->c_shdr->sh_name)); 2825 } 2826 } 2827 2828 /* 2829 * If there exists no shstrtab data, or a section header has no 2830 * name (an invalid index of 0), then compose a name for the 2831 * section. 2832 */ 2833 (void) snprintf(scnndxnm, sizeof (scnndxnm), 2834 MSG_INTL(MSG_FMT_SCNNDX), cnt); 2835 2836 if ((_cache->c_name = malloc(strlen(scnndxnm) + 1)) == 0) { 2837 int err = errno; 2838 (void) fprintf(stderr, MSG_INTL(MSG_ERR_MALLOC), 2839 file, strerror(err)); 2840 return; 2841 } 2842 (void) strcpy(_cache->c_name, scnndxnm); 2843 } 2844 2845 /* 2846 * Having collected all the sections, validate their address range. 2847 * Cases have existed where the section information has been invalid. 2848 * This can lead to all sorts of other, hard to diagnose errors, as 2849 * each section is processed individually (ie. with elf_getdata()). 2850 * Here, we carry out some address comparisons to catch a family of 2851 * overlapping memory issues we have observed (likely, there are others 2852 * that we have yet to discover). 2853 * 2854 * Note, should any memory overlap occur, obtaining any additional 2855 * data from the file is questionable. However, it might still be 2856 * possible to inspect the ELF header, Programs headers, or individual 2857 * sections, so rather than bailing on an error condition, continue 2858 * processing to see if any data can be salvaged. 2859 */ 2860 for (cnt = 1; cnt < shnum; cnt++) { 2861 Cache *_cache = &cache[cnt]; 2862 Shdr *shdr = _cache->c_shdr; 2863 Off bgn1, bgn = shdr->sh_offset; 2864 Off end1, end = shdr->sh_offset + shdr->sh_size; 2865 int cnt1; 2866 2867 if ((shdr->sh_size == 0) || (shdr->sh_type == SHT_NOBITS)) 2868 continue; 2869 2870 for (cnt1 = 1; cnt1 < shnum; cnt1++) { 2871 Cache *_cache1 = &cache[cnt1]; 2872 Shdr *shdr1 = _cache1->c_shdr; 2873 2874 bgn1 = shdr1->sh_offset; 2875 end1 = shdr1->sh_offset + shdr1->sh_size; 2876 2877 if ((cnt1 == cnt) || (shdr->sh_size == 0) || 2878 (shdr1->sh_type == SHT_NOBITS)) 2879 continue; 2880 2881 if (((bgn1 <= bgn) && (end1 > bgn)) || 2882 ((bgn1 < end) && (end1 >= end))) { 2883 (void) fprintf(stderr, 2884 MSG_INTL(MSG_ERR_SECMEMOVER), file, 2885 EC_WORD(elf_ndxscn(_cache1->c_scn)), 2886 _cache1->c_name, EC_OFF(bgn1), EC_OFF(end1), 2887 EC_WORD(elf_ndxscn(_cache->c_scn)), 2888 _cache->c_name, EC_OFF(bgn), EC_OFF(end)); 2889 } 2890 } 2891 2892 /* 2893 * And finally, make sure this section doesn't overlap the 2894 * section header itself. 2895 */ 2896 bgn1 = ehdr->e_shoff; 2897 end1 = ehdr->e_shoff + (ehdr->e_shentsize * ehdr->e_shnum); 2898 2899 if (((bgn1 <= bgn) && (end1 > bgn)) || 2900 ((bgn1 < end) && (end1 >= end))) { 2901 (void) fprintf(stderr, 2902 MSG_INTL(MSG_ERR_SHDRMEMOVER), file, EC_OFF(bgn1), 2903 EC_OFF(end1), 2904 EC_WORD(elf_ndxscn(_cache->c_scn)), 2905 _cache->c_name, EC_OFF(bgn), EC_OFF(end)); 2906 } 2907 } 2908 2909 /* 2910 * Finally, obtain the data for each section. 2911 */ 2912 for (cnt = 1; cnt < shnum; cnt++) { 2913 Cache *_cache = &cache[cnt]; 2914 Elf_Scn *scn = _cache->c_scn; 2915 2916 if ((_cache->c_data = elf_getdata(scn, NULL)) == NULL) { 2917 failure(file, MSG_ORIG(MSG_ELF_GETDATA)); 2918 (void) fprintf(stderr, MSG_INTL(MSG_ELF_ERR_SCNDATA), 2919 EC_WORD(elf_ndxscn(scn))); 2920 } 2921 2922 /* 2923 * Do we wish to write the section out? 2924 */ 2925 if (wfd && Nname && (strcmp(Nname, _cache->c_name) == 0) && 2926 _cache->c_data) { 2927 (void) write(wfd, _cache->c_data->d_buf, 2928 _cache->c_data->d_size); 2929 } 2930 } 2931 2932 if (flags & FLG_SHDR) 2933 sections(file, cache, shnum, ehdr, Nname); 2934 2935 if (flags & FLG_INTERP) 2936 interp(file, cache, shnum, phnum, elf); 2937 2938 versions(cache, shnum, file, flags, &versym); 2939 2940 if (flags & FLG_SYMBOLS) 2941 symbols(cache, shnum, ehdr, Nname, &versym, file, flags); 2942 2943 if (flags & FLG_SORT) 2944 sunw_sort(cache, shnum, ehdr, Nname, &versym, file, flags); 2945 2946 if (flags & FLG_HASH) 2947 hash(cache, shnum, Nname, file, flags); 2948 2949 if (flags & FLG_GOT) 2950 got(cache, shnum, ehdr, file, flags); 2951 2952 if (flags & FLG_GROUP) 2953 group(cache, shnum, Nname, file, flags); 2954 2955 if (flags & FLG_SYMINFO) 2956 syminfo(cache, shnum, file); 2957 2958 if (flags & FLG_RELOC) 2959 reloc(cache, shnum, ehdr, Nname, file, flags); 2960 2961 if (flags & FLG_DYNAMIC) 2962 dynamic(cache, shnum, ehdr, file); 2963 2964 if (flags & FLG_NOTE) 2965 note(cache, shnum, Nname, file); 2966 2967 if (flags & FLG_MOVE) 2968 move(cache, shnum, Nname, file, flags); 2969 2970 if (flags & FLG_CHECKSUM) 2971 checksum(elf); 2972 2973 if (flags & FLG_CAP) 2974 cap(file, cache, shnum, phnum, ehdr, elf); 2975 2976 if (flags & FLG_UNWIND) 2977 unwind(cache, shnum, phnum, ehdr, Nname, file, elf); 2978 2979 free(cache); 2980 } 2981