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 2008 Sun Microsystems, Inc. All rights reserved. 24 * Use is subject to license terms. 25 * Copyright 2017-2018 Mark Johnston <markj@FreeBSD.org> 26 */ 27 28 #include <sys/param.h> 29 #include <sys/mman.h> 30 #include <sys/wait.h> 31 32 #include <assert.h> 33 #include <elf.h> 34 #include <sys/types.h> 35 #include <fcntl.h> 36 #include <gelf.h> 37 #include <limits.h> 38 #include <stddef.h> 39 #include <stdio.h> 40 #include <stdlib.h> 41 #include <strings.h> 42 #include <errno.h> 43 #include <unistd.h> 44 45 #include <libelf.h> 46 47 #include <dt_impl.h> 48 #include <dt_provider.h> 49 #include <dt_program.h> 50 #include <dt_string.h> 51 52 #define ESHDR_NULL 0 53 #define ESHDR_SHSTRTAB 1 54 #define ESHDR_DOF 2 55 #define ESHDR_STRTAB 3 56 #define ESHDR_SYMTAB 4 57 #define ESHDR_REL 5 58 #define ESHDR_NUM 6 59 60 #define PWRITE_SCN(index, data) \ 61 (lseek64(fd, (off64_t)elf_file.shdr[(index)].sh_offset, SEEK_SET) != \ 62 (off64_t)elf_file.shdr[(index)].sh_offset || \ 63 dt_write(dtp, fd, (data), elf_file.shdr[(index)].sh_size) != \ 64 elf_file.shdr[(index)].sh_size) 65 66 static const char DTRACE_SHSTRTAB32[] = "\0" 67 ".shstrtab\0" /* 1 */ 68 ".SUNW_dof\0" /* 11 */ 69 ".strtab\0" /* 21 */ 70 ".symtab\0" /* 29 */ 71 ".rel.SUNW_dof"; /* 37 */ 72 73 static const char DTRACE_SHSTRTAB64[] = "\0" 74 ".shstrtab\0" /* 1 */ 75 ".SUNW_dof\0" /* 11 */ 76 ".strtab\0" /* 21 */ 77 ".symtab\0" /* 29 */ 78 ".rela.SUNW_dof"; /* 37 */ 79 80 static const char DOFSTR[] = "__SUNW_dof"; 81 static const char DOFLAZYSTR[] = "___SUNW_dof"; 82 83 typedef struct dt_link_pair { 84 struct dt_link_pair *dlp_next; /* next pair in linked list */ 85 void *dlp_str; /* buffer for string table */ 86 void *dlp_sym; /* buffer for symbol table */ 87 } dt_link_pair_t; 88 89 typedef struct dof_elf32 { 90 uint32_t de_nrel; /* relocation count */ 91 Elf32_Rel *de_rel; /* array of relocations for x86 */ 92 uint32_t de_nsym; /* symbol count */ 93 Elf32_Sym *de_sym; /* array of symbols */ 94 uint32_t de_strlen; /* size of of string table */ 95 char *de_strtab; /* string table */ 96 uint32_t de_global; /* index of the first global symbol */ 97 } dof_elf32_t; 98 99 static int 100 prepare_elf32(dtrace_hdl_t *dtp, const dof_hdr_t *dof, dof_elf32_t *dep) 101 { 102 dof_sec_t *dofs, *s; 103 dof_relohdr_t *dofrh; 104 dof_relodesc_t *dofr; 105 char *strtab; 106 int i, j, nrel; 107 size_t strtabsz = 1; 108 uint32_t count = 0; 109 size_t base; 110 Elf32_Sym *sym; 111 Elf32_Rel *rel; 112 113 /*LINTED*/ 114 dofs = (dof_sec_t *)((char *)dof + dof->dofh_secoff); 115 116 /* 117 * First compute the size of the string table and the number of 118 * relocations present in the DOF. 119 */ 120 for (i = 0; i < dof->dofh_secnum; i++) { 121 if (dofs[i].dofs_type != DOF_SECT_URELHDR) 122 continue; 123 124 /*LINTED*/ 125 dofrh = (dof_relohdr_t *)((char *)dof + dofs[i].dofs_offset); 126 127 s = &dofs[dofrh->dofr_strtab]; 128 strtab = (char *)dof + s->dofs_offset; 129 assert(strtab[0] == '\0'); 130 strtabsz += s->dofs_size - 1; 131 132 s = &dofs[dofrh->dofr_relsec]; 133 /*LINTED*/ 134 dofr = (dof_relodesc_t *)((char *)dof + s->dofs_offset); 135 count += s->dofs_size / s->dofs_entsize; 136 } 137 138 dep->de_strlen = strtabsz; 139 dep->de_nrel = count; 140 dep->de_nsym = count + 1; /* the first symbol is always null */ 141 142 if (dtp->dt_lazyload) { 143 dep->de_strlen += sizeof (DOFLAZYSTR); 144 dep->de_nsym++; 145 } else { 146 dep->de_strlen += sizeof (DOFSTR); 147 dep->de_nsym++; 148 } 149 150 if ((dep->de_rel = calloc(dep->de_nrel, 151 sizeof (dep->de_rel[0]))) == NULL) { 152 return (dt_set_errno(dtp, EDT_NOMEM)); 153 } 154 155 if ((dep->de_sym = calloc(dep->de_nsym, sizeof (Elf32_Sym))) == NULL) { 156 free(dep->de_rel); 157 return (dt_set_errno(dtp, EDT_NOMEM)); 158 } 159 160 if ((dep->de_strtab = calloc(dep->de_strlen, 1)) == NULL) { 161 free(dep->de_rel); 162 free(dep->de_sym); 163 return (dt_set_errno(dtp, EDT_NOMEM)); 164 } 165 166 count = 0; 167 strtabsz = 1; 168 dep->de_strtab[0] = '\0'; 169 rel = dep->de_rel; 170 sym = dep->de_sym; 171 dep->de_global = 1; 172 173 /* 174 * The first symbol table entry must be zeroed and is always ignored. 175 */ 176 bzero(sym, sizeof (Elf32_Sym)); 177 sym++; 178 179 /* 180 * Take a second pass through the DOF sections filling in the 181 * memory we allocated. 182 */ 183 for (i = 0; i < dof->dofh_secnum; i++) { 184 if (dofs[i].dofs_type != DOF_SECT_URELHDR) 185 continue; 186 187 /*LINTED*/ 188 dofrh = (dof_relohdr_t *)((char *)dof + dofs[i].dofs_offset); 189 190 s = &dofs[dofrh->dofr_strtab]; 191 strtab = (char *)dof + s->dofs_offset; 192 bcopy(strtab + 1, dep->de_strtab + strtabsz, s->dofs_size); 193 base = strtabsz; 194 strtabsz += s->dofs_size - 1; 195 196 s = &dofs[dofrh->dofr_relsec]; 197 /*LINTED*/ 198 dofr = (dof_relodesc_t *)((char *)dof + s->dofs_offset); 199 nrel = s->dofs_size / s->dofs_entsize; 200 201 s = &dofs[dofrh->dofr_tgtsec]; 202 203 for (j = 0; j < nrel; j++) { 204 #if defined(__aarch64__) 205 rel->r_offset = s->dofs_offset + 206 dofr[j].dofr_offset; 207 rel->r_info = ELF32_R_INFO(count + dep->de_global, 208 R_ARM_REL32); 209 #elif defined(__arm__) 210 /* XXX */ 211 printf("%s:%s(%d): arm not implemented\n", 212 __FUNCTION__, __FILE__, __LINE__); 213 #elif defined(__i386) || defined(__amd64) 214 rel->r_offset = s->dofs_offset + 215 dofr[j].dofr_offset; 216 rel->r_info = ELF32_R_INFO(count + dep->de_global, 217 R_386_PC32); 218 #elif defined(__powerpc__) 219 /* 220 * Add 4 bytes to hit the low half of this 64-bit 221 * big-endian address. 222 */ 223 rel->r_offset = s->dofs_offset + 224 dofr[j].dofr_offset + 4; 225 rel->r_info = ELF32_R_INFO(count + dep->de_global, 226 R_PPC_REL32); 227 #elif defined(__riscv) 228 rel->r_offset = s->dofs_offset + dofr[j].dofr_offset; 229 rel->r_info = ELF32_R_INFO(count + dep->de_global, 230 R_RISCV_32_PCREL); 231 #else 232 #error unknown ISA 233 #endif 234 235 sym->st_name = base + dofr[j].dofr_name - 1; 236 sym->st_value = 0; 237 sym->st_size = 0; 238 sym->st_info = ELF32_ST_INFO(STB_GLOBAL, STT_FUNC); 239 sym->st_other = ELF32_ST_VISIBILITY(STV_HIDDEN); 240 sym->st_shndx = SHN_UNDEF; 241 242 rel++; 243 sym++; 244 count++; 245 } 246 } 247 248 /* 249 * Add a symbol for the DOF itself. We use a different symbol for 250 * lazily and actively loaded DOF to make them easy to distinguish. 251 */ 252 sym->st_name = strtabsz; 253 sym->st_value = 0; 254 sym->st_size = dof->dofh_filesz; 255 sym->st_info = ELF32_ST_INFO(STB_GLOBAL, STT_OBJECT); 256 sym->st_other = ELF32_ST_VISIBILITY(STV_HIDDEN); 257 sym->st_shndx = ESHDR_DOF; 258 sym++; 259 260 if (dtp->dt_lazyload) { 261 bcopy(DOFLAZYSTR, dep->de_strtab + strtabsz, 262 sizeof (DOFLAZYSTR)); 263 strtabsz += sizeof (DOFLAZYSTR); 264 } else { 265 bcopy(DOFSTR, dep->de_strtab + strtabsz, sizeof (DOFSTR)); 266 strtabsz += sizeof (DOFSTR); 267 } 268 269 assert(count == dep->de_nrel); 270 assert(strtabsz == dep->de_strlen); 271 272 return (0); 273 } 274 275 276 typedef struct dof_elf64 { 277 uint32_t de_nrel; 278 Elf64_Rela *de_rel; 279 uint32_t de_nsym; 280 Elf64_Sym *de_sym; 281 282 uint32_t de_strlen; 283 char *de_strtab; 284 285 uint32_t de_global; 286 } dof_elf64_t; 287 288 static int 289 prepare_elf64(dtrace_hdl_t *dtp, const dof_hdr_t *dof, dof_elf64_t *dep) 290 { 291 dof_sec_t *dofs, *s; 292 dof_relohdr_t *dofrh; 293 dof_relodesc_t *dofr; 294 char *strtab; 295 int i, j, nrel; 296 size_t strtabsz = 1; 297 uint64_t count = 0; 298 size_t base; 299 Elf64_Sym *sym; 300 Elf64_Rela *rel; 301 302 /*LINTED*/ 303 dofs = (dof_sec_t *)((char *)dof + dof->dofh_secoff); 304 305 /* 306 * First compute the size of the string table and the number of 307 * relocations present in the DOF. 308 */ 309 for (i = 0; i < dof->dofh_secnum; i++) { 310 if (dofs[i].dofs_type != DOF_SECT_URELHDR) 311 continue; 312 313 /*LINTED*/ 314 dofrh = (dof_relohdr_t *)((char *)dof + dofs[i].dofs_offset); 315 316 s = &dofs[dofrh->dofr_strtab]; 317 strtab = (char *)dof + s->dofs_offset; 318 assert(strtab[0] == '\0'); 319 strtabsz += s->dofs_size - 1; 320 321 s = &dofs[dofrh->dofr_relsec]; 322 /*LINTED*/ 323 dofr = (dof_relodesc_t *)((char *)dof + s->dofs_offset); 324 count += s->dofs_size / s->dofs_entsize; 325 } 326 327 dep->de_strlen = strtabsz; 328 dep->de_nrel = count; 329 dep->de_nsym = count + 1; /* the first symbol is always null */ 330 331 if (dtp->dt_lazyload) { 332 dep->de_strlen += sizeof (DOFLAZYSTR); 333 dep->de_nsym++; 334 } else { 335 dep->de_strlen += sizeof (DOFSTR); 336 dep->de_nsym++; 337 } 338 339 if ((dep->de_rel = calloc(dep->de_nrel, 340 sizeof (dep->de_rel[0]))) == NULL) { 341 return (dt_set_errno(dtp, EDT_NOMEM)); 342 } 343 344 if ((dep->de_sym = calloc(dep->de_nsym, sizeof (Elf64_Sym))) == NULL) { 345 free(dep->de_rel); 346 return (dt_set_errno(dtp, EDT_NOMEM)); 347 } 348 349 if ((dep->de_strtab = calloc(dep->de_strlen, 1)) == NULL) { 350 free(dep->de_rel); 351 free(dep->de_sym); 352 return (dt_set_errno(dtp, EDT_NOMEM)); 353 } 354 355 count = 0; 356 strtabsz = 1; 357 dep->de_strtab[0] = '\0'; 358 rel = dep->de_rel; 359 sym = dep->de_sym; 360 dep->de_global = 1; 361 362 /* 363 * The first symbol table entry must be zeroed and is always ignored. 364 */ 365 bzero(sym, sizeof (Elf64_Sym)); 366 sym++; 367 368 /* 369 * Take a second pass through the DOF sections filling in the 370 * memory we allocated. 371 */ 372 for (i = 0; i < dof->dofh_secnum; i++) { 373 if (dofs[i].dofs_type != DOF_SECT_URELHDR) 374 continue; 375 376 /*LINTED*/ 377 dofrh = (dof_relohdr_t *)((char *)dof + dofs[i].dofs_offset); 378 379 s = &dofs[dofrh->dofr_strtab]; 380 strtab = (char *)dof + s->dofs_offset; 381 bcopy(strtab + 1, dep->de_strtab + strtabsz, s->dofs_size); 382 base = strtabsz; 383 strtabsz += s->dofs_size - 1; 384 385 s = &dofs[dofrh->dofr_relsec]; 386 /*LINTED*/ 387 dofr = (dof_relodesc_t *)((char *)dof + s->dofs_offset); 388 nrel = s->dofs_size / s->dofs_entsize; 389 390 s = &dofs[dofrh->dofr_tgtsec]; 391 392 for (j = 0; j < nrel; j++) { 393 #if defined(__aarch64__) 394 rel->r_offset = s->dofs_offset + 395 dofr[j].dofr_offset; 396 rel->r_info = ELF64_R_INFO(count + dep->de_global, 397 R_AARCH64_PREL64); 398 #elif defined(__arm__) 399 /* XXX */ 400 #elif defined(__powerpc__) 401 rel->r_offset = s->dofs_offset + 402 dofr[j].dofr_offset; 403 rel->r_info = ELF64_R_INFO(count + dep->de_global, 404 R_PPC64_REL64); 405 #elif defined(__riscv) 406 rel->r_offset = s->dofs_offset + dofr[j].dofr_offset; 407 rel->r_info = ELF64_R_INFO(count + dep->de_global, 408 R_RISCV_32_PCREL); 409 #elif defined(__i386) || defined(__amd64) 410 rel->r_offset = s->dofs_offset + 411 dofr[j].dofr_offset; 412 rel->r_info = ELF64_R_INFO(count + dep->de_global, 413 R_X86_64_PC64); 414 #else 415 #error unknown ISA 416 #endif 417 418 sym->st_name = base + dofr[j].dofr_name - 1; 419 sym->st_value = 0; 420 sym->st_size = 0; 421 sym->st_info = GELF_ST_INFO(STB_GLOBAL, STT_FUNC); 422 sym->st_other = ELF64_ST_VISIBILITY(STV_HIDDEN); 423 sym->st_shndx = SHN_UNDEF; 424 425 rel++; 426 sym++; 427 count++; 428 } 429 } 430 431 /* 432 * Add a symbol for the DOF itself. We use a different symbol for 433 * lazily and actively loaded DOF to make them easy to distinguish. 434 */ 435 sym->st_name = strtabsz; 436 sym->st_value = 0; 437 sym->st_size = dof->dofh_filesz; 438 sym->st_info = GELF_ST_INFO(STB_GLOBAL, STT_OBJECT); 439 sym->st_other = ELF64_ST_VISIBILITY(STV_HIDDEN); 440 sym->st_shndx = ESHDR_DOF; 441 sym++; 442 443 if (dtp->dt_lazyload) { 444 bcopy(DOFLAZYSTR, dep->de_strtab + strtabsz, 445 sizeof (DOFLAZYSTR)); 446 strtabsz += sizeof (DOFLAZYSTR); 447 } else { 448 bcopy(DOFSTR, dep->de_strtab + strtabsz, sizeof (DOFSTR)); 449 strtabsz += sizeof (DOFSTR); 450 } 451 452 assert(count == dep->de_nrel); 453 assert(strtabsz == dep->de_strlen); 454 455 return (0); 456 } 457 458 /* 459 * Write out an ELF32 file prologue consisting of a header, section headers, 460 * and a section header string table. The DOF data will follow this prologue 461 * and complete the contents of the given ELF file. 462 */ 463 static int 464 dump_elf32(dtrace_hdl_t *dtp, const dof_hdr_t *dof, int fd) 465 { 466 struct { 467 Elf32_Ehdr ehdr; 468 Elf32_Shdr shdr[ESHDR_NUM]; 469 } elf_file; 470 471 Elf32_Shdr *shp; 472 Elf32_Off off; 473 dof_elf32_t de; 474 int ret = 0; 475 uint_t nshdr; 476 477 if (prepare_elf32(dtp, dof, &de) != 0) 478 return (-1); /* errno is set for us */ 479 480 /* 481 * If there are no relocations, we only need enough sections for 482 * the shstrtab and the DOF. 483 */ 484 nshdr = de.de_nrel == 0 ? ESHDR_SYMTAB + 1 : ESHDR_NUM; 485 486 bzero(&elf_file, sizeof (elf_file)); 487 488 elf_file.ehdr.e_ident[EI_MAG0] = ELFMAG0; 489 elf_file.ehdr.e_ident[EI_MAG1] = ELFMAG1; 490 elf_file.ehdr.e_ident[EI_MAG2] = ELFMAG2; 491 elf_file.ehdr.e_ident[EI_MAG3] = ELFMAG3; 492 elf_file.ehdr.e_ident[EI_VERSION] = EV_CURRENT; 493 elf_file.ehdr.e_ident[EI_CLASS] = ELFCLASS32; 494 #if BYTE_ORDER == _BIG_ENDIAN 495 elf_file.ehdr.e_ident[EI_DATA] = ELFDATA2MSB; 496 #else 497 elf_file.ehdr.e_ident[EI_DATA] = ELFDATA2LSB; 498 #endif 499 elf_file.ehdr.e_ident[EI_OSABI] = ELFOSABI_FREEBSD; 500 elf_file.ehdr.e_type = ET_REL; 501 #if defined(__arm__) 502 elf_file.ehdr.e_machine = EM_ARM; 503 #elif defined(__powerpc__) 504 elf_file.ehdr.e_machine = EM_PPC; 505 #elif defined(__i386) || defined(__amd64) 506 elf_file.ehdr.e_machine = EM_386; 507 #elif defined(__aarch64__) 508 elf_file.ehdr.e_machine = EM_AARCH64; 509 #elif defined(__riscv) 510 elf_file.ehdr.e_machine = EM_RISCV; 511 512 /* Set the ELF flags according to our current ABI */ 513 #if defined(__riscv_compressed) 514 elf_file.ehdr.e_flags |= EF_RISCV_RVC; 515 #endif 516 #if defined(__riscv_float_abi_soft) 517 elf_file.ehdr.e_flags |= EF_RISCV_FLOAT_ABI_SOFT; 518 #endif 519 #if defined(__riscv_float_abi_single) 520 elf_file.ehdr.e_flags |= EF_RISCV_FLOAT_ABI_SINGLE; 521 #endif 522 #if defined(__riscv_float_abi_double) 523 elf_file.ehdr.e_flags |= EF_RISCV_FLOAT_ABI_DOUBLE; 524 #endif 525 #endif 526 elf_file.ehdr.e_version = EV_CURRENT; 527 elf_file.ehdr.e_shoff = sizeof (Elf32_Ehdr); 528 elf_file.ehdr.e_ehsize = sizeof (Elf32_Ehdr); 529 elf_file.ehdr.e_phentsize = sizeof (Elf32_Phdr); 530 elf_file.ehdr.e_shentsize = sizeof (Elf32_Shdr); 531 elf_file.ehdr.e_shnum = nshdr; 532 elf_file.ehdr.e_shstrndx = ESHDR_SHSTRTAB; 533 off = sizeof (elf_file) + nshdr * sizeof (Elf32_Shdr); 534 535 shp = &elf_file.shdr[ESHDR_SHSTRTAB]; 536 shp->sh_name = 1; /* DTRACE_SHSTRTAB32[1] = ".shstrtab" */ 537 shp->sh_type = SHT_STRTAB; 538 shp->sh_offset = off; 539 shp->sh_size = sizeof (DTRACE_SHSTRTAB32); 540 shp->sh_addralign = sizeof (char); 541 off = roundup2(shp->sh_offset + shp->sh_size, 8); 542 543 shp = &elf_file.shdr[ESHDR_DOF]; 544 shp->sh_name = 11; /* DTRACE_SHSTRTAB32[11] = ".SUNW_dof" */ 545 shp->sh_flags = SHF_ALLOC; 546 shp->sh_type = SHT_SUNW_dof; 547 shp->sh_offset = off; 548 shp->sh_size = dof->dofh_filesz; 549 shp->sh_addralign = 8; 550 off = shp->sh_offset + shp->sh_size; 551 552 shp = &elf_file.shdr[ESHDR_STRTAB]; 553 shp->sh_name = 21; /* DTRACE_SHSTRTAB32[21] = ".strtab" */ 554 shp->sh_flags = SHF_ALLOC; 555 shp->sh_type = SHT_STRTAB; 556 shp->sh_offset = off; 557 shp->sh_size = de.de_strlen; 558 shp->sh_addralign = sizeof (char); 559 off = roundup2(shp->sh_offset + shp->sh_size, 4); 560 561 shp = &elf_file.shdr[ESHDR_SYMTAB]; 562 shp->sh_name = 29; /* DTRACE_SHSTRTAB32[29] = ".symtab" */ 563 shp->sh_flags = SHF_ALLOC; 564 shp->sh_type = SHT_SYMTAB; 565 shp->sh_entsize = sizeof (Elf32_Sym); 566 shp->sh_link = ESHDR_STRTAB; 567 shp->sh_offset = off; 568 shp->sh_info = de.de_global; 569 shp->sh_size = de.de_nsym * sizeof (Elf32_Sym); 570 shp->sh_addralign = 4; 571 off = roundup2(shp->sh_offset + shp->sh_size, 4); 572 573 if (de.de_nrel == 0) { 574 if (dt_write(dtp, fd, &elf_file, 575 sizeof (elf_file)) != sizeof (elf_file) || 576 PWRITE_SCN(ESHDR_SHSTRTAB, DTRACE_SHSTRTAB32) || 577 PWRITE_SCN(ESHDR_STRTAB, de.de_strtab) || 578 PWRITE_SCN(ESHDR_SYMTAB, de.de_sym) || 579 PWRITE_SCN(ESHDR_DOF, dof)) { 580 ret = dt_set_errno(dtp, errno); 581 } 582 } else { 583 shp = &elf_file.shdr[ESHDR_REL]; 584 shp->sh_name = 37; /* DTRACE_SHSTRTAB32[37] = ".rel.SUNW_dof" */ 585 shp->sh_flags = SHF_ALLOC; 586 shp->sh_type = SHT_REL; 587 shp->sh_entsize = sizeof (de.de_rel[0]); 588 shp->sh_link = ESHDR_SYMTAB; 589 shp->sh_info = ESHDR_DOF; 590 shp->sh_offset = off; 591 shp->sh_size = de.de_nrel * sizeof (de.de_rel[0]); 592 shp->sh_addralign = 4; 593 594 if (dt_write(dtp, fd, &elf_file, 595 sizeof (elf_file)) != sizeof (elf_file) || 596 PWRITE_SCN(ESHDR_SHSTRTAB, DTRACE_SHSTRTAB32) || 597 PWRITE_SCN(ESHDR_STRTAB, de.de_strtab) || 598 PWRITE_SCN(ESHDR_SYMTAB, de.de_sym) || 599 PWRITE_SCN(ESHDR_REL, de.de_rel) || 600 PWRITE_SCN(ESHDR_DOF, dof)) { 601 ret = dt_set_errno(dtp, errno); 602 } 603 } 604 605 free(de.de_strtab); 606 free(de.de_sym); 607 free(de.de_rel); 608 609 return (ret); 610 } 611 612 /* 613 * Write out an ELF64 file prologue consisting of a header, section headers, 614 * and a section header string table. The DOF data will follow this prologue 615 * and complete the contents of the given ELF file. 616 */ 617 static int 618 dump_elf64(dtrace_hdl_t *dtp, const dof_hdr_t *dof, int fd) 619 { 620 struct { 621 Elf64_Ehdr ehdr; 622 Elf64_Shdr shdr[ESHDR_NUM]; 623 } elf_file; 624 625 Elf64_Shdr *shp; 626 Elf64_Off off; 627 dof_elf64_t de; 628 int ret = 0; 629 uint_t nshdr; 630 631 if (prepare_elf64(dtp, dof, &de) != 0) 632 return (-1); /* errno is set for us */ 633 634 /* 635 * If there are no relocations, we only need enough sections for 636 * the shstrtab and the DOF. 637 */ 638 nshdr = de.de_nrel == 0 ? ESHDR_SYMTAB + 1 : ESHDR_NUM; 639 640 bzero(&elf_file, sizeof (elf_file)); 641 642 elf_file.ehdr.e_ident[EI_MAG0] = ELFMAG0; 643 elf_file.ehdr.e_ident[EI_MAG1] = ELFMAG1; 644 elf_file.ehdr.e_ident[EI_MAG2] = ELFMAG2; 645 elf_file.ehdr.e_ident[EI_MAG3] = ELFMAG3; 646 elf_file.ehdr.e_ident[EI_VERSION] = EV_CURRENT; 647 elf_file.ehdr.e_ident[EI_CLASS] = ELFCLASS64; 648 #if BYTE_ORDER == _BIG_ENDIAN 649 elf_file.ehdr.e_ident[EI_DATA] = ELFDATA2MSB; 650 #else 651 elf_file.ehdr.e_ident[EI_DATA] = ELFDATA2LSB; 652 #endif 653 elf_file.ehdr.e_ident[EI_OSABI] = ELFOSABI_FREEBSD; 654 elf_file.ehdr.e_type = ET_REL; 655 #if defined(__arm__) 656 elf_file.ehdr.e_machine = EM_ARM; 657 #elif defined(__powerpc64__) 658 #if defined(_CALL_ELF) && _CALL_ELF == 2 659 elf_file.ehdr.e_flags = 2; 660 #endif 661 elf_file.ehdr.e_machine = EM_PPC64; 662 #elif defined(__i386) || defined(__amd64) 663 elf_file.ehdr.e_machine = EM_AMD64; 664 #elif defined(__aarch64__) 665 elf_file.ehdr.e_machine = EM_AARCH64; 666 #elif defined(__riscv) 667 elf_file.ehdr.e_machine = EM_RISCV; 668 669 /* Set the ELF flags according to our current ABI */ 670 #if defined(__riscv_compressed) 671 elf_file.ehdr.e_flags |= EF_RISCV_RVC; 672 #endif 673 #if defined(__riscv_float_abi_soft) 674 elf_file.ehdr.e_flags |= EF_RISCV_FLOAT_ABI_SOFT; 675 #endif 676 #if defined(__riscv_float_abi_single) 677 elf_file.ehdr.e_flags |= EF_RISCV_FLOAT_ABI_SINGLE; 678 #endif 679 #if defined(__riscv_float_abi_double) 680 elf_file.ehdr.e_flags |= EF_RISCV_FLOAT_ABI_DOUBLE; 681 #endif 682 #endif 683 elf_file.ehdr.e_version = EV_CURRENT; 684 elf_file.ehdr.e_shoff = sizeof (Elf64_Ehdr); 685 elf_file.ehdr.e_ehsize = sizeof (Elf64_Ehdr); 686 elf_file.ehdr.e_phentsize = sizeof (Elf64_Phdr); 687 elf_file.ehdr.e_shentsize = sizeof (Elf64_Shdr); 688 elf_file.ehdr.e_shnum = nshdr; 689 elf_file.ehdr.e_shstrndx = ESHDR_SHSTRTAB; 690 off = sizeof (elf_file) + nshdr * sizeof (Elf64_Shdr); 691 692 shp = &elf_file.shdr[ESHDR_SHSTRTAB]; 693 shp->sh_name = 1; /* DTRACE_SHSTRTAB64[1] = ".shstrtab" */ 694 shp->sh_type = SHT_STRTAB; 695 shp->sh_offset = off; 696 shp->sh_size = sizeof (DTRACE_SHSTRTAB64); 697 shp->sh_addralign = sizeof (char); 698 off = roundup2(shp->sh_offset + shp->sh_size, 8); 699 700 shp = &elf_file.shdr[ESHDR_DOF]; 701 shp->sh_name = 11; /* DTRACE_SHSTRTAB64[11] = ".SUNW_dof" */ 702 shp->sh_flags = SHF_ALLOC; 703 shp->sh_type = SHT_SUNW_dof; 704 shp->sh_offset = off; 705 shp->sh_size = dof->dofh_filesz; 706 shp->sh_addralign = 8; 707 off = shp->sh_offset + shp->sh_size; 708 709 shp = &elf_file.shdr[ESHDR_STRTAB]; 710 shp->sh_name = 21; /* DTRACE_SHSTRTAB64[21] = ".strtab" */ 711 shp->sh_flags = SHF_ALLOC; 712 shp->sh_type = SHT_STRTAB; 713 shp->sh_offset = off; 714 shp->sh_size = de.de_strlen; 715 shp->sh_addralign = sizeof (char); 716 off = roundup2(shp->sh_offset + shp->sh_size, 8); 717 718 shp = &elf_file.shdr[ESHDR_SYMTAB]; 719 shp->sh_name = 29; /* DTRACE_SHSTRTAB64[29] = ".symtab" */ 720 shp->sh_flags = SHF_ALLOC; 721 shp->sh_type = SHT_SYMTAB; 722 shp->sh_entsize = sizeof (Elf64_Sym); 723 shp->sh_link = ESHDR_STRTAB; 724 shp->sh_offset = off; 725 shp->sh_info = de.de_global; 726 shp->sh_size = de.de_nsym * sizeof (Elf64_Sym); 727 shp->sh_addralign = 8; 728 off = roundup2(shp->sh_offset + shp->sh_size, 8); 729 730 if (de.de_nrel == 0) { 731 if (dt_write(dtp, fd, &elf_file, 732 sizeof (elf_file)) != sizeof (elf_file) || 733 PWRITE_SCN(ESHDR_SHSTRTAB, DTRACE_SHSTRTAB64) || 734 PWRITE_SCN(ESHDR_STRTAB, de.de_strtab) || 735 PWRITE_SCN(ESHDR_SYMTAB, de.de_sym) || 736 PWRITE_SCN(ESHDR_DOF, dof)) { 737 ret = dt_set_errno(dtp, errno); 738 } 739 } else { 740 shp = &elf_file.shdr[ESHDR_REL]; 741 shp->sh_name = 37; /* DTRACE_SHSTRTAB64[37] = ".rel.SUNW_dof" */ 742 shp->sh_flags = SHF_ALLOC; 743 shp->sh_type = SHT_RELA; 744 shp->sh_entsize = sizeof (de.de_rel[0]); 745 shp->sh_link = ESHDR_SYMTAB; 746 shp->sh_info = ESHDR_DOF; 747 shp->sh_offset = off; 748 shp->sh_size = de.de_nrel * sizeof (de.de_rel[0]); 749 shp->sh_addralign = 8; 750 751 if (dt_write(dtp, fd, &elf_file, 752 sizeof (elf_file)) != sizeof (elf_file) || 753 PWRITE_SCN(ESHDR_SHSTRTAB, DTRACE_SHSTRTAB64) || 754 PWRITE_SCN(ESHDR_STRTAB, de.de_strtab) || 755 PWRITE_SCN(ESHDR_SYMTAB, de.de_sym) || 756 PWRITE_SCN(ESHDR_REL, de.de_rel) || 757 PWRITE_SCN(ESHDR_DOF, dof)) { 758 ret = dt_set_errno(dtp, errno); 759 } 760 } 761 762 free(de.de_strtab); 763 free(de.de_sym); 764 free(de.de_rel); 765 766 return (ret); 767 } 768 769 static int 770 dt_symtab_lookup(Elf_Data *data_sym, int start, int end, uintptr_t addr, 771 uint_t shn, GElf_Sym *sym, int uses_funcdesc, Elf *elf) 772 { 773 Elf64_Addr symval; 774 Elf_Scn *opd_scn; 775 Elf_Data *opd_desc; 776 int i; 777 778 for (i = start; i < end && gelf_getsym(data_sym, i, sym) != NULL; i++) { 779 if (GELF_ST_TYPE(sym->st_info) == STT_FUNC) { 780 symval = sym->st_value; 781 if (uses_funcdesc) { 782 opd_scn = elf_getscn(elf, sym->st_shndx); 783 opd_desc = elf_rawdata(opd_scn, NULL); 784 symval = 785 *(uint64_t*)((char *)opd_desc->d_buf + symval); 786 } 787 if ((uses_funcdesc || shn == sym->st_shndx) && 788 symval <= addr && addr < symval + sym->st_size) 789 return (0); 790 } 791 } 792 793 return (-1); 794 } 795 796 #if defined(__aarch64__) 797 #define DT_OP_NOP 0xd503201f 798 #define DT_OP_RET 0xd65f03c0 799 #define DT_OP_CALL26 0x94000000 800 #define DT_OP_JUMP26 0x14000000 801 #define DT_REL_NONE R_AARCH64_NONE 802 803 static int 804 dt_modtext(dtrace_hdl_t *dtp, char *p, int isenabled, GElf_Rela *rela, 805 uint32_t *off) 806 { 807 uint32_t *ip; 808 809 /* 810 * Ensure that the offset is aligned on an instruction boundary. 811 */ 812 if ((rela->r_offset & (sizeof (uint32_t) - 1)) != 0) 813 return (-1); 814 815 /* 816 * We only know about some specific relocation types. 817 * We also recognize relocation type NONE, since that gets used for 818 * relocations of USDT probes, and we might be re-processing a file. 819 */ 820 if (GELF_R_TYPE(rela->r_info) != R_AARCH64_CALL26 && 821 GELF_R_TYPE(rela->r_info) != R_AARCH64_JUMP26 && 822 GELF_R_TYPE(rela->r_info) != R_AARCH64_NONE) 823 return (-1); 824 825 ip = (uint32_t *)(p + rela->r_offset); 826 827 /* 828 * We may have already processed this object file in an earlier linker 829 * invocation. Check to see if the present instruction sequence matches 830 * the one we would install below. 831 */ 832 if (ip[0] == DT_OP_NOP || ip[0] == DT_OP_RET) 833 return (0); 834 835 /* 836 * We only expect call instructions with a displacement of 0, or a jump 837 * instruction acting as a tail call. 838 */ 839 if (ip[0] != DT_OP_CALL26 && ip[0] != DT_OP_JUMP26) { 840 dt_dprintf("found %x instead of a call or jmp instruction at " 841 "%llx\n", ip[0], (u_longlong_t)rela->r_offset); 842 return (-1); 843 } 844 845 /* 846 * On arm64, we do not have to differentiate between regular probes and 847 * is-enabled probes. Both cases are encoded as a regular branch for 848 * non-tail call locations, and a jump for tail call locations. Calls 849 * are to be converted into a no-op whereas jumps should become a 850 * return. 851 */ 852 if (ip[0] == DT_OP_CALL26) 853 ip[0] = DT_OP_NOP; 854 else 855 ip[0] = DT_OP_RET; 856 857 return (0); 858 } 859 #elif defined(__arm__) 860 #define DT_REL_NONE R_ARM_NONE 861 862 static int 863 dt_modtext(dtrace_hdl_t *dtp, char *p, int isenabled, GElf_Rela *rela, 864 uint32_t *off) 865 { 866 printf("%s:%s(%d): arm not implemented\n", __FUNCTION__, __FILE__, 867 __LINE__); 868 return (-1); 869 } 870 #elif defined(__powerpc__) 871 /* The sentinel is 'xor r3,r3,r3'. */ 872 #define DT_OP_XOR_R3 0x7c631a78 873 874 #define DT_OP_NOP 0x60000000 875 #define DT_OP_BLR 0x4e800020 876 877 /* This captures all forms of branching to address. */ 878 #define DT_IS_BRANCH(inst) ((inst & 0xfc000000) == 0x48000000) 879 #define DT_IS_BL(inst) (DT_IS_BRANCH(inst) && (inst & 0x01)) 880 881 #define DT_REL_NONE R_PPC_NONE 882 883 static int 884 dt_modtext(dtrace_hdl_t *dtp, char *p, int isenabled, GElf_Rela *rela, 885 uint32_t *off) 886 { 887 uint32_t *ip; 888 889 if ((rela->r_offset & (sizeof (uint32_t) - 1)) != 0) 890 return (-1); 891 892 /*LINTED*/ 893 ip = (uint32_t *)(p + rela->r_offset); 894 895 /* 896 * We only know about some specific relocation types. 897 */ 898 if (GELF_R_TYPE(rela->r_info) != R_PPC_REL24 && 899 GELF_R_TYPE(rela->r_info) != R_PPC_PLTREL24 && 900 GELF_R_TYPE(rela->r_info) != R_PPC_NONE) 901 return (-1); 902 903 /* 904 * We may have already processed this object file in an earlier linker 905 * invocation. Check to see if the present instruction sequence matches 906 * the one we would install below. 907 */ 908 if (isenabled) { 909 if (ip[0] == DT_OP_XOR_R3) { 910 (*off) += sizeof (ip[0]); 911 return (0); 912 } 913 } else { 914 if (ip[0] == DT_OP_NOP) { 915 (*off) += sizeof (ip[0]); 916 return (0); 917 } 918 } 919 920 /* 921 * We only expect branch to address instructions. 922 */ 923 if (!DT_IS_BRANCH(ip[0])) { 924 dt_dprintf("found %x instead of a branch instruction at %llx\n", 925 ip[0], (u_longlong_t)rela->r_offset); 926 return (-1); 927 } 928 929 if (isenabled) { 930 /* 931 * It would necessarily indicate incorrect usage if an is- 932 * enabled probe were tail-called so flag that as an error. 933 * It's also potentially (very) tricky to handle gracefully, 934 * but could be done if this were a desired use scenario. 935 */ 936 if (!DT_IS_BL(ip[0])) { 937 dt_dprintf("tail call to is-enabled probe at %llx\n", 938 (u_longlong_t)rela->r_offset); 939 return (-1); 940 } 941 942 ip[0] = DT_OP_XOR_R3; 943 (*off) += sizeof (ip[0]); 944 } else { 945 if (DT_IS_BL(ip[0])) 946 ip[0] = DT_OP_NOP; 947 else 948 ip[0] = DT_OP_BLR; 949 } 950 951 return (0); 952 } 953 #elif defined(__riscv) 954 #define DT_OP_NOP 0x00000013 /* addi x0, x0, 0 */ 955 #define DT_OP_RET 0x00008067 /* jalr x0, x1, 0 */ 956 #define DT_OP_IS_AUIPC(op) (((op) & 0x7f) == 0x17) 957 #define DT_OP_IS_JALR(op) (((op) & 0x707f) == 0x67) 958 #define DT_OP_JALR_CALL 0x000080e7 /* jalr x1, x1, 0 */ 959 #define DT_OP_JALR_TAIL 0x00030067 /* jalr x0, x6, 0 */ 960 #define DT_REL_NONE R_RISCV_NONE 961 962 static int 963 dt_modtext(dtrace_hdl_t *dtp, char *p, int isenabled, GElf_Rela *rela, 964 uint32_t *off) 965 { 966 uint32_t *ip; 967 968 /* 969 * XXX: this implementation is untested, but should serve as a decent 970 * starting point. 971 */ 972 973 /* 974 * Ensure that the offset is aligned on a compressed-instruction 975 * boundary. 976 */ 977 if ((rela->r_offset & (sizeof (uint16_t) - 1)) != 0) 978 return (-1); 979 980 /* 981 * We only know about some specific relocation types. 982 * We also recognize relocation type NONE, since that gets used for 983 * relocations of USDT probes, and we might be re-processing a file. 984 */ 985 if (GELF_R_TYPE(rela->r_info) != R_RISCV_CALL && 986 GELF_R_TYPE(rela->r_info) != R_RISCV_CALL_PLT && 987 GELF_R_TYPE(rela->r_info) != R_RISCV_NONE) 988 return (-1); 989 990 ip = (uint32_t *)(p + rela->r_offset); 991 992 /* 993 * We may have already processed this object file in an earlier linker 994 * invocation. Check to see if the present instruction sequence matches 995 * the one we would install below. 996 */ 997 if (ip[0] == DT_OP_NOP && (ip[1] == DT_OP_NOP || ip[1] == DT_OP_RET)) 998 return (0); 999 1000 /* 1001 * We expect a auipc+jalr pair, either from a call or a tail. 1002 * - call: auipc x1 0; jalr x1, x1, 0 1003 * - tail: auipc x6 0; jalr x0, x6, 0 1004 */ 1005 if (!DT_OP_IS_AUIPC(ip[0]) || !DT_OP_IS_JALR(ip[1])) 1006 return (-1); 1007 1008 /* 1009 * On riscv, we do not have to differentiate between regular probes and 1010 * is-enabled probes. Calls are to be converted into a no-op whereas 1011 * tail calls should become a return. 1012 */ 1013 if (ip[1] == DT_OP_JALR_CALL) { 1014 ip[0] = DT_OP_NOP; 1015 ip[1] = DT_OP_NOP; 1016 } else { 1017 ip[0] = DT_OP_NOP; 1018 ip[1] = DT_OP_RET; 1019 } 1020 1021 return (0); 1022 } 1023 1024 #elif defined(__i386) || defined(__amd64) 1025 1026 #define DT_OP_NOP 0x90 1027 #define DT_OP_RET 0xc3 1028 #define DT_OP_CALL 0xe8 1029 #define DT_OP_JMP32 0xe9 1030 #define DT_OP_REX_RAX 0x48 1031 #define DT_OP_XOR_EAX_0 0x33 1032 #define DT_OP_XOR_EAX_1 0xc0 1033 1034 #define DT_REL_NONE R_386_NONE 1035 1036 static int 1037 dt_modtext(dtrace_hdl_t *dtp, char *p, int isenabled, GElf_Rela *rela, 1038 uint32_t *off) 1039 { 1040 uint8_t *ip = (uint8_t *)(p + rela->r_offset - 1); 1041 uint8_t ret; 1042 1043 /* 1044 * On x86, the first byte of the instruction is the call opcode and 1045 * the next four bytes are the 32-bit address; the relocation is for 1046 * the address operand. We back up the offset to the first byte of 1047 * the instruction. For is-enabled probes, we later advance the offset 1048 * so that it hits the first nop in the instruction sequence. 1049 */ 1050 (*off) -= 1; 1051 1052 /* 1053 * We only know about some specific relocation types. Luckily 1054 * these types have the same values on both 32-bit and 64-bit 1055 * x86 architectures. 1056 */ 1057 if (GELF_R_TYPE(rela->r_info) != R_386_PC32 && 1058 GELF_R_TYPE(rela->r_info) != R_386_PLT32 && 1059 GELF_R_TYPE(rela->r_info) != R_386_NONE) 1060 return (-1); 1061 1062 /* 1063 * We may have already processed this object file in an earlier linker 1064 * invocation. Check to see if the present instruction sequence matches 1065 * the one we would install. For is-enabled probes, we advance the 1066 * offset to the first nop instruction in the sequence to match the 1067 * text modification code below. 1068 */ 1069 if (!isenabled) { 1070 if ((ip[0] == DT_OP_NOP || ip[0] == DT_OP_RET) && 1071 ip[1] == DT_OP_NOP && ip[2] == DT_OP_NOP && 1072 ip[3] == DT_OP_NOP && ip[4] == DT_OP_NOP) 1073 return (0); 1074 } else if (dtp->dt_oflags & DTRACE_O_LP64) { 1075 if (ip[0] == DT_OP_REX_RAX && 1076 ip[1] == DT_OP_XOR_EAX_0 && ip[2] == DT_OP_XOR_EAX_1 && 1077 (ip[3] == DT_OP_NOP || ip[3] == DT_OP_RET) && 1078 ip[4] == DT_OP_NOP) { 1079 (*off) += 3; 1080 return (0); 1081 } 1082 } else { 1083 if (ip[0] == DT_OP_XOR_EAX_0 && ip[1] == DT_OP_XOR_EAX_1 && 1084 (ip[2] == DT_OP_NOP || ip[2] == DT_OP_RET) && 1085 ip[3] == DT_OP_NOP && ip[4] == DT_OP_NOP) { 1086 (*off) += 2; 1087 return (0); 1088 } 1089 } 1090 1091 /* 1092 * We expect either a call instrution with a 32-bit displacement or a 1093 * jmp instruction with a 32-bit displacement acting as a tail-call. 1094 */ 1095 if (ip[0] != DT_OP_CALL && ip[0] != DT_OP_JMP32) { 1096 dt_dprintf("found %x instead of a call or jmp instruction at " 1097 "%llx\n", ip[0], (u_longlong_t)rela->r_offset); 1098 return (-1); 1099 } 1100 1101 ret = (ip[0] == DT_OP_JMP32) ? DT_OP_RET : DT_OP_NOP; 1102 1103 /* 1104 * Establish the instruction sequence -- all nops for probes, and an 1105 * instruction to clear the return value register (%eax/%rax) followed 1106 * by nops for is-enabled probes. For is-enabled probes, we advance 1107 * the offset to the first nop. This isn't stricly necessary but makes 1108 * for more readable disassembly when the probe is enabled. 1109 */ 1110 if (!isenabled) { 1111 ip[0] = ret; 1112 ip[1] = DT_OP_NOP; 1113 ip[2] = DT_OP_NOP; 1114 ip[3] = DT_OP_NOP; 1115 ip[4] = DT_OP_NOP; 1116 } else if (dtp->dt_oflags & DTRACE_O_LP64) { 1117 ip[0] = DT_OP_REX_RAX; 1118 ip[1] = DT_OP_XOR_EAX_0; 1119 ip[2] = DT_OP_XOR_EAX_1; 1120 ip[3] = ret; 1121 ip[4] = DT_OP_NOP; 1122 (*off) += 3; 1123 } else { 1124 ip[0] = DT_OP_XOR_EAX_0; 1125 ip[1] = DT_OP_XOR_EAX_1; 1126 ip[2] = ret; 1127 ip[3] = DT_OP_NOP; 1128 ip[4] = DT_OP_NOP; 1129 (*off) += 2; 1130 } 1131 1132 return (0); 1133 } 1134 1135 #else 1136 #error unknown ISA 1137 #endif 1138 1139 /*PRINTFLIKE5*/ 1140 static int 1141 dt_link_error(dtrace_hdl_t *dtp, Elf *elf, int fd, dt_link_pair_t *bufs, 1142 const char *format, ...) 1143 { 1144 va_list ap; 1145 dt_link_pair_t *pair; 1146 1147 va_start(ap, format); 1148 dt_set_errmsg(dtp, NULL, NULL, NULL, 0, format, ap); 1149 va_end(ap); 1150 1151 if (elf != NULL) 1152 (void) elf_end(elf); 1153 1154 if (fd >= 0) 1155 (void) close(fd); 1156 1157 while ((pair = bufs) != NULL) { 1158 bufs = pair->dlp_next; 1159 dt_free(dtp, pair->dlp_str); 1160 dt_free(dtp, pair->dlp_sym); 1161 dt_free(dtp, pair); 1162 } 1163 1164 return (dt_set_errno(dtp, EDT_COMPILER)); 1165 } 1166 1167 /* 1168 * Provide a unique identifier used when adding global symbols to an object. 1169 * This is the FNV-1a hash of an absolute path for the file. 1170 */ 1171 static unsigned int 1172 hash_obj(const char *obj, int fd) 1173 { 1174 char path[PATH_MAX]; 1175 unsigned int h; 1176 1177 if (realpath(obj, path) == NULL) 1178 return (-1); 1179 1180 for (h = 2166136261u, obj = &path[0]; *obj != '\0'; obj++) 1181 h = (h ^ *obj) * 16777619; 1182 h &= 0x7fffffff; 1183 return (h); 1184 } 1185 1186 static int 1187 process_obj(dtrace_hdl_t *dtp, const char *obj, int *eprobesp) 1188 { 1189 static const char dt_prefix[] = "__dtrace"; 1190 static const char dt_enabled[] = "enabled"; 1191 static const char dt_symprefix[] = "$dtrace"; 1192 static const char dt_symfmt[] = "%s%u.%s"; 1193 static const char dt_weaksymfmt[] = "%s.%s"; 1194 char probename[DTRACE_NAMELEN]; 1195 int fd, i, ndx, eprobe, uses_funcdesc = 0, mod = 0; 1196 Elf *elf = NULL; 1197 GElf_Ehdr ehdr; 1198 Elf_Scn *scn_rel, *scn_sym, *scn_str, *scn_tgt; 1199 Elf_Data *data_rel, *data_sym, *data_str, *data_tgt; 1200 GElf_Shdr shdr_rel, shdr_sym, shdr_str, shdr_tgt; 1201 GElf_Sym rsym, fsym, dsym; 1202 GElf_Rela rela; 1203 char *s, *p, *r; 1204 char pname[DTRACE_PROVNAMELEN]; 1205 dt_provider_t *pvp; 1206 dt_probe_t *prp; 1207 uint32_t off, eclass, emachine1, emachine2; 1208 size_t symsize, osym, nsym, isym, istr, len; 1209 unsigned int objkey; 1210 dt_link_pair_t *pair, *bufs = NULL; 1211 dt_strtab_t *strtab; 1212 void *tmp; 1213 1214 if ((fd = open64(obj, O_RDWR)) == -1) { 1215 return (dt_link_error(dtp, elf, fd, bufs, 1216 "failed to open %s: %s", obj, strerror(errno))); 1217 } 1218 1219 if ((elf = elf_begin(fd, ELF_C_RDWR, NULL)) == NULL) { 1220 return (dt_link_error(dtp, elf, fd, bufs, 1221 "failed to process %s: %s", obj, elf_errmsg(elf_errno()))); 1222 } 1223 1224 switch (elf_kind(elf)) { 1225 case ELF_K_ELF: 1226 break; 1227 case ELF_K_AR: 1228 return (dt_link_error(dtp, elf, fd, bufs, "archives are not " 1229 "permitted; use the contents of the archive instead: %s", 1230 obj)); 1231 default: 1232 return (dt_link_error(dtp, elf, fd, bufs, 1233 "invalid file type: %s", obj)); 1234 } 1235 1236 if (gelf_getehdr(elf, &ehdr) == NULL) { 1237 return (dt_link_error(dtp, elf, fd, bufs, "corrupt file: %s", 1238 obj)); 1239 } 1240 1241 if (dtp->dt_oflags & DTRACE_O_LP64) { 1242 eclass = ELFCLASS64; 1243 #if defined(__powerpc__) 1244 emachine1 = emachine2 = EM_PPC64; 1245 #if !defined(_CALL_ELF) || _CALL_ELF == 1 1246 uses_funcdesc = 1; 1247 #endif 1248 #elif defined(__i386) || defined(__amd64) 1249 emachine1 = emachine2 = EM_AMD64; 1250 #elif defined(__aarch64__) 1251 emachine1 = emachine2 = EM_AARCH64; 1252 #elif defined(__riscv) 1253 emachine1 = emachine2 = EM_RISCV; 1254 #endif 1255 symsize = sizeof (Elf64_Sym); 1256 } else { 1257 eclass = ELFCLASS32; 1258 #if defined(__arm__) 1259 emachine1 = emachine2 = EM_ARM; 1260 #elif defined(__powerpc__) 1261 emachine1 = emachine2 = EM_PPC; 1262 #elif defined(__i386) || defined(__amd64) 1263 emachine1 = emachine2 = EM_386; 1264 #endif 1265 symsize = sizeof (Elf32_Sym); 1266 } 1267 1268 if (ehdr.e_ident[EI_CLASS] != eclass) { 1269 return (dt_link_error(dtp, elf, fd, bufs, 1270 "incorrect ELF class for object file: %s", obj)); 1271 } 1272 1273 if (ehdr.e_machine != emachine1 && ehdr.e_machine != emachine2) { 1274 return (dt_link_error(dtp, elf, fd, bufs, 1275 "incorrect ELF machine type for object file: %s", obj)); 1276 } 1277 1278 /* 1279 * We use this token as a relatively unique handle for this file on the 1280 * system in order to disambiguate potential conflicts between files of 1281 * the same name which contain identially named local symbols. 1282 */ 1283 if ((objkey = hash_obj(obj, fd)) == (unsigned int)-1) 1284 return (dt_link_error(dtp, elf, fd, bufs, 1285 "failed to generate unique key for object file: %s", obj)); 1286 1287 scn_rel = NULL; 1288 while ((scn_rel = elf_nextscn(elf, scn_rel)) != NULL) { 1289 if (gelf_getshdr(scn_rel, &shdr_rel) == NULL) 1290 goto err; 1291 1292 /* 1293 * Skip any non-relocation sections. 1294 */ 1295 if (shdr_rel.sh_type != SHT_RELA && shdr_rel.sh_type != SHT_REL) 1296 continue; 1297 1298 if ((data_rel = elf_getdata(scn_rel, NULL)) == NULL) 1299 goto err; 1300 1301 /* 1302 * Grab the section, section header and section data for the 1303 * symbol table that this relocation section references. 1304 */ 1305 if ((scn_sym = elf_getscn(elf, shdr_rel.sh_link)) == NULL || 1306 gelf_getshdr(scn_sym, &shdr_sym) == NULL || 1307 (data_sym = elf_getdata(scn_sym, NULL)) == NULL) 1308 goto err; 1309 1310 /* 1311 * Ditto for that symbol table's string table. 1312 */ 1313 if ((scn_str = elf_getscn(elf, shdr_sym.sh_link)) == NULL || 1314 gelf_getshdr(scn_str, &shdr_str) == NULL || 1315 (data_str = elf_getdata(scn_str, NULL)) == NULL) 1316 goto err; 1317 1318 /* 1319 * Grab the section, section header and section data for the 1320 * target section for the relocations. For the relocations 1321 * we're looking for -- this will typically be the text of the 1322 * object file. 1323 */ 1324 if ((scn_tgt = elf_getscn(elf, shdr_rel.sh_info)) == NULL || 1325 gelf_getshdr(scn_tgt, &shdr_tgt) == NULL || 1326 (data_tgt = elf_getdata(scn_tgt, NULL)) == NULL) 1327 goto err; 1328 1329 /* 1330 * We're looking for relocations to symbols matching this form: 1331 * 1332 * __dtrace[enabled]_<prov>___<probe> 1333 * 1334 * For the generated object, we need to record the location 1335 * identified by the relocation, and create a new relocation 1336 * in the generated object that will be resolved at link time 1337 * to the location of the function in which the probe is 1338 * embedded. In the target object, we change the matched symbol 1339 * so that it will be ignored at link time, and we modify the 1340 * target (text) section to replace the call instruction with 1341 * one or more nops. 1342 * 1343 * To avoid runtime overhead, the relocations added to the 1344 * generated object should be resolved at static link time. We 1345 * therefore create aliases for the functions that contain 1346 * probes. An alias is global (so that the relocation from the 1347 * generated object can be resolved), and hidden (so that its 1348 * address is known at static link time). Such aliases have this 1349 * form: 1350 * 1351 * $dtrace<key>.<function> 1352 * 1353 * We take a first pass through all the relocations to 1354 * populate our string table and count the number of extra 1355 * symbols we'll require. 1356 * 1357 * We also handle the case where the object has already been 1358 * processed, to support incremental rebuilds. Relocations 1359 * of interest are converted to type NONE, but all information 1360 * needed to reconstruct the output DOF is retained. 1361 */ 1362 strtab = dt_strtab_create(1); 1363 nsym = 0; 1364 isym = data_sym->d_size / symsize; 1365 istr = data_str->d_size; 1366 1367 for (i = 0; i < shdr_rel.sh_size / shdr_rel.sh_entsize; i++) { 1368 if (shdr_rel.sh_type == SHT_RELA) { 1369 if (gelf_getrela(data_rel, i, &rela) == NULL) 1370 continue; 1371 } else { 1372 GElf_Rel rel; 1373 if (gelf_getrel(data_rel, i, &rel) == NULL) 1374 continue; 1375 rela.r_offset = rel.r_offset; 1376 rela.r_info = rel.r_info; 1377 rela.r_addend = 0; 1378 } 1379 1380 if (gelf_getsym(data_sym, GELF_R_SYM(rela.r_info), 1381 &rsym) == NULL) { 1382 dt_strtab_destroy(strtab); 1383 goto err; 1384 } 1385 1386 s = (char *)data_str->d_buf + rsym.st_name; 1387 1388 if (strncmp(s, dt_prefix, sizeof (dt_prefix) - 1) != 0) 1389 continue; 1390 1391 if (dt_symtab_lookup(data_sym, 0, isym, rela.r_offset, 1392 shdr_rel.sh_info, &fsym, uses_funcdesc, 1393 elf) != 0) { 1394 dt_strtab_destroy(strtab); 1395 goto err; 1396 } 1397 1398 if (fsym.st_name > data_str->d_size) { 1399 dt_strtab_destroy(strtab); 1400 goto err; 1401 } 1402 1403 s = (char *)data_str->d_buf + fsym.st_name; 1404 1405 /* 1406 * If this symbol isn't of type function, we've really 1407 * driven off the rails or the object file is corrupt. 1408 */ 1409 if (GELF_ST_TYPE(fsym.st_info) != STT_FUNC) { 1410 dt_strtab_destroy(strtab); 1411 return (dt_link_error(dtp, elf, fd, bufs, 1412 "expected %s to be of type function", s)); 1413 } 1414 1415 /* 1416 * Aliases of weak symbols don't get a uniquifier. 1417 */ 1418 if (GELF_ST_BIND(fsym.st_info) == STB_WEAK) { 1419 len = snprintf(NULL, 0, dt_weaksymfmt, 1420 dt_symprefix, s) + 1; 1421 } else { 1422 len = snprintf(NULL, 0, dt_symfmt, dt_symprefix, 1423 objkey, s) + 1; 1424 } 1425 if ((p = dt_alloc(dtp, len)) == NULL) { 1426 dt_strtab_destroy(strtab); 1427 goto err; 1428 } 1429 if (GELF_ST_BIND(fsym.st_info) == STB_WEAK) { 1430 (void) snprintf(p, len, dt_weaksymfmt, 1431 dt_symprefix, s); 1432 } else { 1433 (void) snprintf(p, len, dt_symfmt, dt_symprefix, 1434 objkey, s); 1435 } 1436 1437 if (dt_strtab_index(strtab, p) == -1) { 1438 /* 1439 * Do not add new symbols if this object file 1440 * has already been processed. 1441 */ 1442 if (GELF_R_TYPE(rela.r_info) != DT_REL_NONE) 1443 nsym++; 1444 (void) dt_strtab_insert(strtab, p); 1445 } 1446 1447 dt_free(dtp, p); 1448 } 1449 1450 /* 1451 * If any new probes were found, allocate the additional space 1452 * for the symbol table and string table, copying the old data 1453 * into the new buffers, and marking the buffers as dirty. We 1454 * inject those newly allocated buffers into the libelf data 1455 * structures, but are still responsible for freeing them once 1456 * we're done with the elf handle. 1457 */ 1458 osym = isym; 1459 if (nsym > 0) { 1460 /* 1461 * The first byte of the string table is reserved for 1462 * the \0 entry. 1463 */ 1464 len = dt_strtab_size(strtab) - 1; 1465 1466 assert(len > 0); 1467 assert(dt_strtab_index(strtab, "") == 0); 1468 1469 dt_strtab_destroy(strtab); 1470 1471 if ((pair = dt_alloc(dtp, sizeof (*pair))) == NULL) 1472 goto err; 1473 1474 if ((pair->dlp_str = dt_alloc(dtp, data_str->d_size + 1475 len)) == NULL) { 1476 dt_free(dtp, pair); 1477 goto err; 1478 } 1479 1480 if ((pair->dlp_sym = dt_alloc(dtp, data_sym->d_size + 1481 nsym * symsize)) == NULL) { 1482 dt_free(dtp, pair->dlp_str); 1483 dt_free(dtp, pair); 1484 goto err; 1485 } 1486 1487 pair->dlp_next = bufs; 1488 bufs = pair; 1489 1490 bcopy(data_str->d_buf, pair->dlp_str, data_str->d_size); 1491 tmp = data_str->d_buf; 1492 data_str->d_buf = pair->dlp_str; 1493 pair->dlp_str = tmp; 1494 data_str->d_size += len; 1495 (void) elf_flagdata(data_str, ELF_C_SET, ELF_F_DIRTY); 1496 1497 shdr_str.sh_size += len; 1498 (void) gelf_update_shdr(scn_str, &shdr_str); 1499 1500 bcopy(data_sym->d_buf, pair->dlp_sym, data_sym->d_size); 1501 tmp = data_sym->d_buf; 1502 data_sym->d_buf = pair->dlp_sym; 1503 pair->dlp_sym = tmp; 1504 data_sym->d_size += nsym * symsize; 1505 (void) elf_flagdata(data_sym, ELF_C_SET, ELF_F_DIRTY); 1506 1507 shdr_sym.sh_size += nsym * symsize; 1508 (void) gelf_update_shdr(scn_sym, &shdr_sym); 1509 1510 nsym += isym; 1511 } else if (dt_strtab_empty(strtab)) { 1512 dt_strtab_destroy(strtab); 1513 continue; 1514 } 1515 1516 /* 1517 * Now that the tables have been allocated, perform the 1518 * modifications described above. 1519 */ 1520 for (i = 0; i < shdr_rel.sh_size / shdr_rel.sh_entsize; i++) { 1521 if (shdr_rel.sh_type == SHT_RELA) { 1522 if (gelf_getrela(data_rel, i, &rela) == NULL) 1523 continue; 1524 } else { 1525 GElf_Rel rel; 1526 if (gelf_getrel(data_rel, i, &rel) == NULL) 1527 continue; 1528 rela.r_offset = rel.r_offset; 1529 rela.r_info = rel.r_info; 1530 rela.r_addend = 0; 1531 } 1532 1533 ndx = GELF_R_SYM(rela.r_info); 1534 1535 if (gelf_getsym(data_sym, ndx, &rsym) == NULL || 1536 rsym.st_name > data_str->d_size) 1537 goto err; 1538 1539 s = (char *)data_str->d_buf + rsym.st_name; 1540 1541 if (strncmp(s, dt_prefix, sizeof (dt_prefix) - 1) != 0) 1542 continue; 1543 1544 s += sizeof (dt_prefix) - 1; 1545 1546 /* 1547 * Check to see if this is an 'is-enabled' check as 1548 * opposed to a normal probe. 1549 */ 1550 if (strncmp(s, dt_enabled, 1551 sizeof (dt_enabled) - 1) == 0) { 1552 s += sizeof (dt_enabled) - 1; 1553 eprobe = 1; 1554 *eprobesp = 1; 1555 dt_dprintf("is-enabled probe\n"); 1556 } else { 1557 eprobe = 0; 1558 dt_dprintf("normal probe\n"); 1559 } 1560 1561 if (*s++ != '_') 1562 goto err; 1563 1564 if ((p = strstr(s, "___")) == NULL || 1565 p - s >= sizeof (pname)) 1566 goto err; 1567 1568 bcopy(s, pname, p - s); 1569 pname[p - s] = '\0'; 1570 1571 if (dt_symtab_lookup(data_sym, osym, isym, 1572 rela.r_offset, shdr_rel.sh_info, &fsym, 1573 uses_funcdesc, elf) == 0) { 1574 if (fsym.st_name > data_str->d_size) 1575 goto err; 1576 1577 r = s = (char *) data_str->d_buf + fsym.st_name; 1578 assert(strstr(s, dt_symprefix) == s); 1579 s = strchr(s, '.') + 1; 1580 } else if (dt_symtab_lookup(data_sym, 0, osym, 1581 rela.r_offset, shdr_rel.sh_info, &fsym, 1582 uses_funcdesc, elf) == 0) { 1583 u_int bind; 1584 1585 bind = GELF_ST_BIND(fsym.st_info) == STB_WEAK ? 1586 STB_WEAK : STB_GLOBAL; 1587 s = (char *) data_str->d_buf + fsym.st_name; 1588 if (GELF_R_TYPE(rela.r_info) != DT_REL_NONE) { 1589 /* 1590 * Emit an alias for the symbol. It 1591 * needs to be non-preemptible so that 1592 * .SUNW_dof relocations may be resolved 1593 * at static link time. Aliases of weak 1594 * symbols are given a non-unique name 1595 * so that they may be merged by the 1596 * linker. 1597 */ 1598 dsym = fsym; 1599 dsym.st_name = istr; 1600 dsym.st_info = GELF_ST_INFO(bind, 1601 STT_FUNC); 1602 dsym.st_other = 1603 GELF_ST_VISIBILITY(STV_HIDDEN); 1604 (void) gelf_update_sym(data_sym, isym, 1605 &dsym); 1606 isym++; 1607 assert(isym <= nsym); 1608 1609 r = (char *) data_str->d_buf + istr; 1610 if (bind == STB_WEAK) { 1611 istr += sprintf(r, 1612 dt_weaksymfmt, dt_symprefix, 1613 s); 1614 } else { 1615 istr += sprintf(r, dt_symfmt, 1616 dt_symprefix, objkey, s); 1617 } 1618 istr++; 1619 } else { 1620 if (bind == STB_WEAK) { 1621 (void) asprintf(&r, 1622 dt_weaksymfmt, dt_symprefix, 1623 s); 1624 } else { 1625 (void) asprintf(&r, dt_symfmt, 1626 dt_symprefix, objkey, s); 1627 } 1628 } 1629 } else { 1630 goto err; 1631 } 1632 1633 if ((pvp = dt_provider_lookup(dtp, pname)) == NULL) { 1634 return (dt_link_error(dtp, elf, fd, bufs, 1635 "no such provider %s", pname)); 1636 } 1637 1638 if (strlcpy(probename, p + 3, sizeof (probename)) >= 1639 sizeof (probename)) 1640 return (dt_link_error(dtp, elf, fd, bufs, 1641 "invalid probe name %s", probename)); 1642 (void) strhyphenate(probename); 1643 if ((prp = dt_probe_lookup(pvp, probename)) == NULL) 1644 return (dt_link_error(dtp, elf, fd, bufs, 1645 "no such probe %s", probename)); 1646 1647 assert(fsym.st_value <= rela.r_offset); 1648 1649 off = rela.r_offset - fsym.st_value; 1650 if (dt_modtext(dtp, data_tgt->d_buf, eprobe, 1651 &rela, &off) != 0) 1652 goto err; 1653 1654 if (dt_probe_define(pvp, prp, s, r, off, eprobe) != 0) { 1655 return (dt_link_error(dtp, elf, fd, bufs, 1656 "failed to allocate space for probe")); 1657 } 1658 1659 /* 1660 * We are done with this relocation, but it must be 1661 * preserved in order to support incremental rebuilds. 1662 */ 1663 if (shdr_rel.sh_type == SHT_RELA) { 1664 rela.r_info = GELF_R_INFO( 1665 GELF_R_SYM(rela.r_info), DT_REL_NONE); 1666 (void) gelf_update_rela(data_rel, i, &rela); 1667 } else { 1668 GElf_Rel rel; 1669 rel.r_offset = rela.r_offset; 1670 rel.r_info = GELF_R_INFO( 1671 GELF_R_SYM(rela.r_info), DT_REL_NONE); 1672 (void) gelf_update_rel(data_rel, i, &rel); 1673 } 1674 1675 mod = 1; 1676 (void) elf_flagdata(data_tgt, ELF_C_SET, ELF_F_DIRTY); 1677 1678 /* 1679 * This symbol may already have been marked to 1680 * be ignored by another relocation referencing 1681 * the same symbol or if this object file has 1682 * already been processed by an earlier link 1683 * invocation. 1684 */ 1685 if (rsym.st_shndx != SHN_ABS) { 1686 rsym.st_info = GELF_ST_INFO(STB_WEAK, STT_FUNC); 1687 rsym.st_shndx = SHN_ABS; 1688 (void) gelf_update_sym(data_sym, ndx, &rsym); 1689 } 1690 } 1691 } 1692 1693 if (mod && elf_update(elf, ELF_C_WRITE) == -1) 1694 goto err; 1695 1696 (void) elf_end(elf); 1697 (void) close(fd); 1698 1699 while ((pair = bufs) != NULL) { 1700 bufs = pair->dlp_next; 1701 dt_free(dtp, pair->dlp_str); 1702 dt_free(dtp, pair->dlp_sym); 1703 dt_free(dtp, pair); 1704 } 1705 1706 return (0); 1707 1708 err: 1709 return (dt_link_error(dtp, elf, fd, bufs, 1710 "an error was encountered while processing %s", obj)); 1711 } 1712 1713 int 1714 dtrace_program_link(dtrace_hdl_t *dtp, dtrace_prog_t *pgp, uint_t dflags, 1715 const char *file, int objc, char *const objv[]) 1716 { 1717 char tfile[PATH_MAX]; 1718 char drti[PATH_MAX]; 1719 dof_hdr_t *dof; 1720 int fd, status, i, cur; 1721 char *cmd, tmp; 1722 size_t len; 1723 int eprobes = 0, ret = 0; 1724 1725 /* 1726 * A NULL program indicates a special use in which we just link 1727 * together a bunch of object files specified in objv and then 1728 * unlink(2) those object files. 1729 */ 1730 if (pgp == NULL) { 1731 const char *fmt = "%s -o %s -r"; 1732 1733 len = snprintf(&tmp, 1, fmt, dtp->dt_ld_path, file) + 1; 1734 1735 for (i = 0; i < objc; i++) 1736 len += strlen(objv[i]) + 1; 1737 1738 cmd = alloca(len); 1739 1740 cur = snprintf(cmd, len, fmt, dtp->dt_ld_path, file); 1741 1742 for (i = 0; i < objc; i++) 1743 cur += snprintf(cmd + cur, len - cur, " %s", objv[i]); 1744 1745 if ((status = system(cmd)) == -1) { 1746 return (dt_link_error(dtp, NULL, -1, NULL, 1747 "failed to run %s: %s", dtp->dt_ld_path, 1748 strerror(errno))); 1749 } 1750 1751 if (WIFSIGNALED(status)) { 1752 return (dt_link_error(dtp, NULL, -1, NULL, 1753 "failed to link %s: %s failed due to signal %d", 1754 file, dtp->dt_ld_path, WTERMSIG(status))); 1755 } 1756 1757 if (WEXITSTATUS(status) != 0) { 1758 return (dt_link_error(dtp, NULL, -1, NULL, 1759 "failed to link %s: %s exited with status %d\n", 1760 file, dtp->dt_ld_path, WEXITSTATUS(status))); 1761 } 1762 1763 for (i = 0; i < objc; i++) { 1764 if (strcmp(objv[i], file) != 0) 1765 (void) unlink(objv[i]); 1766 } 1767 1768 return (0); 1769 } 1770 1771 for (i = 0; i < objc; i++) { 1772 if (process_obj(dtp, objv[i], &eprobes) != 0) 1773 return (-1); /* errno is set for us */ 1774 } 1775 1776 /* 1777 * If there are is-enabled probes then we need to force use of DOF 1778 * version 2. 1779 */ 1780 if (eprobes && pgp->dp_dofversion < DOF_VERSION_2) 1781 pgp->dp_dofversion = DOF_VERSION_2; 1782 1783 if ((dof = dtrace_dof_create(dtp, pgp, dflags)) == NULL) 1784 return (-1); /* errno is set for us */ 1785 1786 snprintf(tfile, sizeof(tfile), "%s.XXXXXX", file); 1787 if ((fd = mkostemp(tfile, O_CLOEXEC)) == -1) 1788 return (dt_link_error(dtp, NULL, -1, NULL, 1789 "failed to create temporary file %s: %s", 1790 tfile, strerror(errno))); 1791 1792 /* 1793 * If -xlinktype=DOF has been selected, just write out the DOF. 1794 * Otherwise proceed to the default of generating and linking ELF. 1795 */ 1796 switch (dtp->dt_linktype) { 1797 case DT_LTYP_DOF: 1798 if (dt_write(dtp, fd, dof, dof->dofh_filesz) < dof->dofh_filesz) 1799 ret = errno; 1800 1801 if (close(fd) != 0 && ret == 0) 1802 ret = errno; 1803 1804 if (ret != 0) { 1805 return (dt_link_error(dtp, NULL, -1, NULL, 1806 "failed to write %s: %s", file, strerror(ret))); 1807 } 1808 1809 return (0); 1810 1811 case DT_LTYP_ELF: 1812 break; /* fall through to the rest of dtrace_program_link() */ 1813 1814 default: 1815 return (dt_link_error(dtp, NULL, -1, NULL, 1816 "invalid link type %u\n", dtp->dt_linktype)); 1817 } 1818 1819 1820 if (dtp->dt_oflags & DTRACE_O_LP64) 1821 status = dump_elf64(dtp, dof, fd); 1822 else 1823 status = dump_elf32(dtp, dof, fd); 1824 1825 if (status != 0) 1826 return (dt_link_error(dtp, NULL, -1, NULL, 1827 "failed to write %s: %s", tfile, 1828 strerror(dtrace_errno(dtp)))); 1829 1830 if (!dtp->dt_lazyload) { 1831 const char *fmt = "%s -o %s -r %s %s"; 1832 dt_dirpath_t *dp = dt_list_next(&dtp->dt_lib_path); 1833 1834 (void) snprintf(drti, sizeof (drti), "%s/drti.o", dp->dir_path); 1835 1836 len = snprintf(&tmp, 1, fmt, dtp->dt_ld_path, file, tfile, 1837 drti) + 1; 1838 1839 cmd = alloca(len); 1840 1841 (void) snprintf(cmd, len, fmt, dtp->dt_ld_path, file, tfile, 1842 drti); 1843 if ((status = system(cmd)) == -1) { 1844 ret = dt_link_error(dtp, NULL, fd, NULL, 1845 "failed to run %s: %s", dtp->dt_ld_path, 1846 strerror(errno)); 1847 goto done; 1848 } 1849 1850 if (WIFSIGNALED(status)) { 1851 ret = dt_link_error(dtp, NULL, fd, NULL, 1852 "failed to link %s: %s failed due to signal %d", 1853 file, dtp->dt_ld_path, WTERMSIG(status)); 1854 goto done; 1855 } 1856 1857 if (WEXITSTATUS(status) != 0) { 1858 ret = dt_link_error(dtp, NULL, fd, NULL, 1859 "failed to link %s: %s exited with status %d\n", 1860 file, dtp->dt_ld_path, WEXITSTATUS(status)); 1861 goto done; 1862 } 1863 (void) close(fd); /* release temporary file */ 1864 1865 /* 1866 * Now that we've linked drti.o, reduce the global __SUNW_dof 1867 * symbol to a local symbol. This is needed to so that multiple 1868 * generated object files (for different providers, for 1869 * instance) can be linked together. This is accomplished using 1870 * the -Blocal flag with Sun's linker, but GNU ld doesn't appear 1871 * to have an equivalent option. 1872 */ 1873 asprintf(&cmd, "%s --localize-hidden %s", dtp->dt_objcopy_path, 1874 file); 1875 if ((status = system(cmd)) == -1) { 1876 ret = dt_link_error(dtp, NULL, -1, NULL, 1877 "failed to run %s: %s", dtp->dt_objcopy_path, 1878 strerror(errno)); 1879 free(cmd); 1880 goto done; 1881 } 1882 free(cmd); 1883 1884 if (WIFSIGNALED(status)) { 1885 ret = dt_link_error(dtp, NULL, -1, NULL, 1886 "failed to link %s: %s failed due to signal %d", 1887 file, dtp->dt_objcopy_path, WTERMSIG(status)); 1888 goto done; 1889 } 1890 1891 if (WEXITSTATUS(status) != 0) { 1892 ret = dt_link_error(dtp, NULL, -1, NULL, 1893 "failed to link %s: %s exited with status %d\n", 1894 file, dtp->dt_objcopy_path, WEXITSTATUS(status)); 1895 goto done; 1896 } 1897 } else { 1898 if (rename(tfile, file) != 0) { 1899 ret = dt_link_error(dtp, NULL, fd, NULL, 1900 "failed to rename %s to %s: %s", tfile, file, 1901 strerror(errno)); 1902 goto done; 1903 } 1904 (void) close(fd); 1905 } 1906 1907 done: 1908 dtrace_dof_destroy(dtp, dof); 1909 1910 if (!dtp->dt_lazyload) 1911 (void) unlink(tfile); 1912 return (ret); 1913 } 1914