/* * Copyright 2008-2009, Ingo Weinhold, ingo_weinhold@gmx.de. * Copyright 2003-2012, Axel Dörfler, axeld@pinc-software.de. * Distributed under the terms of the MIT License. * * Copyright 2002, Manuel J. Petit. All rights reserved. * Copyright 2001, Travis Geiselbrecht. All rights reserved. * Distributed under the terms of the NewOS License. */ #include "elf_load_image.h" #include #include #include #include "add_ons.h" #include "elf_haiku_version.h" #include "elf_symbol_lookup.h" #include "elf_tls.h" #include "elf_versioning.h" #include "images.h" #include "runtime_loader_private.h" static const char* sSearchPathSubDir = NULL; static const char* get_program_path() { return gProgramImage != NULL ? gProgramImage->path : NULL; } static int32 count_regions(const char* imagePath, char const* buff, int phnum, int phentsize) { elf_phdr* pheaders; int32 count = 0; int i; for (i = 0; i < phnum; i++) { pheaders = (elf_phdr*)(buff + i * phentsize); switch (pheaders->p_type) { case PT_NULL: // NOP header break; case PT_LOAD: count += 1; if (pheaders->p_memsz != pheaders->p_filesz) { addr_t A = TO_PAGE_SIZE(pheaders->p_vaddr + pheaders->p_memsz); addr_t B = TO_PAGE_SIZE(pheaders->p_vaddr + pheaders->p_filesz); if (A != B) count += 1; } break; case PT_DYNAMIC: // will be handled at some other place break; case PT_INTERP: // should check here for appropriate interpreter break; case PT_NOTE: // unsupported break; case PT_SHLIB: // undefined semantics break; case PT_PHDR: // we don't use it break; case PT_EH_FRAME: case PT_RELRO: // not implemented yet, but can be ignored break; case PT_STACK: // we don't use it break; case PT_TLS: // will be handled at some other place break; case PT_ARM_UNWIND: // will be handled in libgcc_s.so.1 break; default: FATAL("%s: Unhandled pheader type in count 0x%" B_PRIx32 "\n", imagePath, pheaders->p_type); break; } } return count; } static status_t parse_program_headers(image_t* image, char* buff, int phnum, int phentsize) { elf_phdr* pheader; int regcount; int i; image->dso_tls_id = unsigned(-1); regcount = 0; for (i = 0; i < phnum; i++) { pheader = (elf_phdr*)(buff + i * phentsize); switch (pheader->p_type) { case PT_NULL: /* NOP header */ break; case PT_LOAD: if (pheader->p_memsz == pheader->p_filesz) { /* * everything in one area */ image->regions[regcount].start = pheader->p_vaddr; image->regions[regcount].size = pheader->p_memsz; image->regions[regcount].vmstart = PAGE_BASE(pheader->p_vaddr); image->regions[regcount].vmsize = TO_PAGE_SIZE(pheader->p_memsz + PAGE_OFFSET(pheader->p_vaddr)); image->regions[regcount].fdstart = pheader->p_offset; image->regions[regcount].fdsize = pheader->p_filesz; image->regions[regcount].delta = 0; image->regions[regcount].flags = 0; if (pheader->p_flags & PF_WRITE) { // this is a writable segment image->regions[regcount].flags |= RFLAG_RW; } } else { /* * may require splitting */ addr_t A = TO_PAGE_SIZE(pheader->p_vaddr + pheader->p_memsz); addr_t B = TO_PAGE_SIZE(pheader->p_vaddr + pheader->p_filesz); image->regions[regcount].start = pheader->p_vaddr; image->regions[regcount].size = pheader->p_filesz; image->regions[regcount].vmstart = PAGE_BASE(pheader->p_vaddr); image->regions[regcount].vmsize = TO_PAGE_SIZE(pheader->p_filesz + PAGE_OFFSET(pheader->p_vaddr)); image->regions[regcount].fdstart = pheader->p_offset; image->regions[regcount].fdsize = pheader->p_filesz; image->regions[regcount].delta = 0; image->regions[regcount].flags = 0; if (pheader->p_flags & PF_WRITE) { // this is a writable segment image->regions[regcount].flags |= RFLAG_RW; } if (A != B) { /* * yeah, it requires splitting */ regcount += 1; image->regions[regcount].start = pheader->p_vaddr; image->regions[regcount].size = pheader->p_memsz - pheader->p_filesz; image->regions[regcount].vmstart = image->regions[regcount-1].vmstart + image->regions[regcount-1].vmsize; image->regions[regcount].vmsize = TO_PAGE_SIZE(pheader->p_memsz + PAGE_OFFSET(pheader->p_vaddr)) - image->regions[regcount-1].vmsize; image->regions[regcount].fdstart = 0; image->regions[regcount].fdsize = 0; image->regions[regcount].delta = 0; image->regions[regcount].flags = RFLAG_ANON; if (pheader->p_flags & PF_WRITE) { // this is a writable segment image->regions[regcount].flags |= RFLAG_RW; } } } regcount += 1; break; case PT_DYNAMIC: image->dynamic_ptr = pheader->p_vaddr; break; case PT_INTERP: // should check here for appropiate interpreter break; case PT_NOTE: // unsupported break; case PT_SHLIB: // undefined semantics break; case PT_PHDR: // we don't use it break; case PT_EH_FRAME: case PT_RELRO: // not implemented yet, but can be ignored break; case PT_STACK: // we don't use it break; case PT_TLS: image->dso_tls_id = TLSBlockTemplates::Get().Register( TLSBlockTemplate((void*)pheader->p_vaddr, pheader->p_filesz, pheader->p_memsz)); break; case PT_ARM_UNWIND: // will be handled in libgcc_s.so.1 break; default: FATAL("%s: Unhandled pheader type in parse 0x%" B_PRIx32 "\n", image->path, pheader->p_type); return B_BAD_DATA; } } return B_OK; } static bool assert_dynamic_loadable(image_t* image) { uint32 i; if (!image->dynamic_ptr) return true; for (i = 0; i < image->num_regions; i++) { if (image->dynamic_ptr >= image->regions[i].start && image->dynamic_ptr < image->regions[i].start + image->regions[i].size) { return true; } } return false; } static bool parse_dynamic_segment(image_t* image) { elf_dyn* d; int i; int sonameOffset = -1; image->symhash = 0; image->syms = 0; image->strtab = 0; d = (elf_dyn*)image->dynamic_ptr; if (!d) return true; for (i = 0; d[i].d_tag != DT_NULL; i++) { switch (d[i].d_tag) { case DT_NEEDED: image->num_needed += 1; break; case DT_HASH: image->symhash = (uint32*)(d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_STRTAB: image->strtab = (char*)(d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_SYMTAB: image->syms = (elf_sym*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_REL: image->rel = (elf_rel*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_RELSZ: image->rel_len = d[i].d_un.d_val; break; case DT_RELA: image->rela = (elf_rela*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_RELASZ: image->rela_len = d[i].d_un.d_val; break; case DT_JMPREL: // procedure linkage table relocations image->pltrel = (elf_rel*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_PLTRELSZ: image->pltrel_len = d[i].d_un.d_val; break; case DT_INIT: image->init_routine = (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_FINI: image->term_routine = (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_SONAME: sonameOffset = d[i].d_un.d_val; break; case DT_VERSYM: image->symbol_versions = (elf_versym*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_VERDEF: image->version_definitions = (elf_verdef*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_VERDEFNUM: image->num_version_definitions = d[i].d_un.d_val; break; case DT_VERNEED: image->needed_versions = (elf_verneed*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_VERNEEDNUM: image->num_needed_versions = d[i].d_un.d_val; break; case DT_SYMBOLIC: image->flags |= RFLAG_SYMBOLIC; break; case DT_FLAGS: { uint32 flags = d[i].d_un.d_val; if ((flags & DF_SYMBOLIC) != 0) image->flags |= RFLAG_SYMBOLIC; if ((flags & DF_STATIC_TLS) != 0) { FATAL("Static TLS model is not supported.\n"); return false; } break; } case DT_INIT_ARRAY: // array of pointers to initialization functions image->init_array = (addr_t*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_INIT_ARRAYSZ: // size in bytes of the array of initialization functions image->init_array_len = d[i].d_un.d_val; break; case DT_PREINIT_ARRAY: // array of pointers to pre-initialization functions image->preinit_array = (addr_t*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_PREINIT_ARRAYSZ: // size in bytes of the array of pre-initialization functions image->preinit_array_len = d[i].d_un.d_val; break; case DT_FINI_ARRAY: // array of pointers to termination functions image->term_array = (addr_t*) (d[i].d_un.d_ptr + image->regions[0].delta); break; case DT_FINI_ARRAYSZ: // size in bytes of the array of termination functions image->term_array_len = d[i].d_un.d_val; break; default: continue; // TODO: Implement: // DT_RELENT: The size of a DT_REL entry. // DT_RELAENT: The size of a DT_RELA entry. // DT_SYMENT: The size of a symbol table entry. // DT_PLTREL: The type of the PLT relocation entries (DT_JMPREL). // DT_BIND_NOW/DF_BIND_NOW: No lazy binding allowed. // DT_RUNPATH: Library search path (supersedes DT_RPATH). // DT_TEXTREL/DF_TEXTREL: Indicates whether text relocations are // required (for optimization purposes only). } } // lets make sure we found all the required sections if (!image->symhash || !image->syms || !image->strtab) return false; if (sonameOffset >= 0) strlcpy(image->name, STRING(image, sonameOffset), sizeof(image->name)); return true; } // #pragma mark - status_t parse_elf_header(elf_ehdr* eheader, int32* _pheaderSize, int32* _sheaderSize) { if (memcmp(eheader->e_ident, ELFMAG, 4) != 0) return B_NOT_AN_EXECUTABLE; if (eheader->e_ident[4] != ELF_CLASS) return B_NOT_AN_EXECUTABLE; if (eheader->e_phoff == 0) return B_NOT_AN_EXECUTABLE; if (eheader->e_phentsize < sizeof(elf_phdr)) return B_NOT_AN_EXECUTABLE; *_pheaderSize = eheader->e_phentsize * eheader->e_phnum; *_sheaderSize = eheader->e_shentsize * eheader->e_shnum; if (*_pheaderSize <= 0) return B_NOT_AN_EXECUTABLE; return B_OK; } #if defined(_COMPAT_MODE) #if defined(__x86_64__) status_t parse_elf32_header(Elf32_Ehdr* eheader, int32* _pheaderSize, int32* _sheaderSize) { if (memcmp(eheader->e_ident, ELFMAG, 4) != 0) return B_NOT_AN_EXECUTABLE; if (eheader->e_ident[4] != ELFCLASS32) return B_NOT_AN_EXECUTABLE; if (eheader->e_phoff == 0) return B_NOT_AN_EXECUTABLE; if (eheader->e_phentsize < sizeof(Elf32_Phdr)) return B_NOT_AN_EXECUTABLE; *_pheaderSize = eheader->e_phentsize * eheader->e_phnum; *_sheaderSize = eheader->e_shentsize * eheader->e_shnum; if (*_pheaderSize <= 0) return B_NOT_AN_EXECUTABLE; return B_OK; } #else status_t parse_elf64_header(Elf64_Ehdr* eheader, int32* _pheaderSize, int32* _sheaderSize) { if (memcmp(eheader->e_ident, ELFMAG, 4) != 0) return B_NOT_AN_EXECUTABLE; if (eheader->e_ident[4] != ELFCLASS64) return B_NOT_AN_EXECUTABLE; if (eheader->e_phoff == 0) return B_NOT_AN_EXECUTABLE; if (eheader->e_phentsize < sizeof(Elf64_Phdr)) return B_NOT_AN_EXECUTABLE; *_pheaderSize = eheader->e_phentsize * eheader->e_phnum; *_sheaderSize = eheader->e_shentsize * eheader->e_shnum; if (*_pheaderSize <= 0) return B_NOT_AN_EXECUTABLE; return B_OK; } #endif // __x86_64__ #endif // _COMPAT_MODE status_t load_image(char const* name, image_type type, const char* rpath, const char* requestingObjectPath, image_t** _image) { int32 pheaderSize, sheaderSize; char path[PATH_MAX]; ssize_t length; char pheaderBuffer[4096]; int32 numRegions; image_t* found; image_t* image; status_t status; int fd; elf_ehdr eheader; // Have we already loaded that image? Don't check for add-ons -- we always // reload them. if (type != B_ADD_ON_IMAGE) { found = find_loaded_image_by_name(name, APP_OR_LIBRARY_TYPE); if (found == NULL && type != B_APP_IMAGE && gProgramImage != NULL) { // Special case for add-ons that link against the application // executable, with the executable not having a soname set. if (const char* lastSlash = strrchr(name, '/')) { if (strcmp(gProgramImage->name, lastSlash + 1) == 0) found = gProgramImage; } } if (found) { atomic_add(&found->ref_count, 1); *_image = found; KTRACE("rld: load_container(\"%s\", type: %d, rpath: \"%s\") " "already loaded", name, type, rpath); return B_OK; } } KTRACE("rld: load_container(\"%s\", type: %d, rpath: \"%s\")", name, type, rpath); strlcpy(path, name, sizeof(path)); // find and open the file fd = open_executable(path, type, rpath, get_program_path(), requestingObjectPath, sSearchPathSubDir); if (fd < 0) { FATAL("Cannot open file %s (needed by %s): %s\n", name, requestingObjectPath, strerror(fd)); KTRACE("rld: load_container(\"%s\"): failed to open file", name); return fd; } // normalize the image path status = _kern_normalize_path(path, true, path); if (status != B_OK) goto err1; // Test again if this image has been registered already - this time, // we can check the full path, not just its name as noted. // You could end up loading an image twice with symbolic links, else. if (type != B_ADD_ON_IMAGE) { found = find_loaded_image_by_name(path, APP_OR_LIBRARY_TYPE); if (found) { atomic_add(&found->ref_count, 1); *_image = found; _kern_close(fd); KTRACE("rld: load_container(\"%s\"): already loaded after all", name); return B_OK; } } length = _kern_read(fd, 0, &eheader, sizeof(eheader)); if (length != sizeof(eheader)) { status = B_NOT_AN_EXECUTABLE; FATAL("%s: Troubles reading ELF header\n", path); goto err1; } status = parse_elf_header(&eheader, &pheaderSize, &sheaderSize); if (status < B_OK) { FATAL("%s: Incorrect ELF header\n", path); goto err1; } // ToDo: what to do about this restriction?? if (pheaderSize > (int)sizeof(pheaderBuffer)) { FATAL("%s: Cannot handle program headers bigger than %lu\n", path, sizeof(pheaderBuffer)); status = B_UNSUPPORTED; goto err1; } length = _kern_read(fd, eheader.e_phoff, pheaderBuffer, pheaderSize); if (length != pheaderSize) { FATAL("%s: Could not read program headers: %s\n", path, strerror(length)); status = B_BAD_DATA; goto err1; } numRegions = count_regions(path, pheaderBuffer, eheader.e_phnum, eheader.e_phentsize); if (numRegions <= 0) { FATAL("%s: Troubles parsing Program headers, numRegions = %" B_PRId32 "\n", path, numRegions); status = B_BAD_DATA; goto err1; } image = create_image(name, path, numRegions); if (image == NULL) { FATAL("%s: Failed to allocate image_t object\n", path); status = B_NO_MEMORY; goto err1; } status = parse_program_headers(image, pheaderBuffer, eheader.e_phnum, eheader.e_phentsize); if (status < B_OK) goto err2; if (!assert_dynamic_loadable(image)) { FATAL("%s: Dynamic segment must be loadable (implementation " "restriction)\n", image->path); status = B_UNSUPPORTED; goto err2; } status = map_image(fd, path, image, eheader.e_type == ET_EXEC); if (status < B_OK) { FATAL("%s: Could not map image: %s\n", image->path, strerror(status)); status = B_ERROR; goto err2; } if (!parse_dynamic_segment(image)) { FATAL("%s: Troubles handling dynamic section\n", image->path); status = B_BAD_DATA; goto err3; } if (eheader.e_entry != 0) image->entry_point = eheader.e_entry + image->regions[0].delta; analyze_image_haiku_version_and_abi(fd, image, eheader, sheaderSize, pheaderBuffer, sizeof(pheaderBuffer)); // If sSearchPathSubDir is unset (meaning, this is the first image we're // loading) we init the search path subdir if the compiler version doesn't // match ours. if (sSearchPathSubDir == NULL) { #if __GNUC__ == 2 || (defined(_COMPAT_MODE) && !defined(__x86_64__)) if ((image->abi & B_HAIKU_ABI_MAJOR) == B_HAIKU_ABI_GCC_4) sSearchPathSubDir = "x86"; #endif #if __GNUC__ >= 4 || (defined(_COMPAT_MODE) && !defined(__x86_64__)) if ((image->abi & B_HAIKU_ABI_MAJOR) == B_HAIKU_ABI_GCC_2) sSearchPathSubDir = "x86_gcc2"; #endif } set_abi_api_version(image->abi, image->api_version); // init gcc version dependent image flags // symbol resolution strategy if (image->abi == B_HAIKU_ABI_GCC_2_ANCIENT) image->find_undefined_symbol = find_undefined_symbol_beos; // init version infos status = init_image_version_infos(image); image->type = type; register_image(image, fd, path); image_event(image, IMAGE_EVENT_LOADED); _kern_close(fd); enqueue_loaded_image(image); *_image = image; KTRACE("rld: load_container(\"%s\"): done: id: %" B_PRId32 " (ABI: %#" B_PRIx32 ")", name, image->id, image->abi); return B_OK; err3: unmap_image(image); err2: delete_image_struct(image); err1: _kern_close(fd); KTRACE("rld: load_container(\"%s\"): failed: %s", name, strerror(status)); return status; }