|  | /* Copyright 2013-2023 Free Software Foundation, Inc. | 
|  | This program is free software; you can redistribute it and/or modify | 
|  | it under the terms of the GNU General Public License as published by | 
|  | the Free Software Foundation; either version 3 of the License, or | 
|  | (at your option) any later version. | 
|  |  | 
|  | This program is distributed in the hope that it will be useful, | 
|  | but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
|  | GNU General Public License for more details. | 
|  |  | 
|  | You should have received a copy of the GNU General Public License | 
|  | along with this program.  If not, see <http://www.gnu.org/licenses/>. | 
|  | */ | 
|  |  | 
|  | #include <unistd.h> | 
|  | #include <fcntl.h> | 
|  | #include <limits.h> | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  | #include <string.h> | 
|  | #include <sys/mman.h> | 
|  | #include <assert.h> | 
|  |  | 
|  | #include "sym-file-loader.h" | 
|  |  | 
|  | #include <inttypes.h> | 
|  | #include <ansidecl.h> | 
|  | #include <elf/common.h> | 
|  | #include <elf/external.h> | 
|  |  | 
|  | #ifdef TARGET_LP64 | 
|  |  | 
|  | typedef Elf64_External_Phdr Elf_External_Phdr; | 
|  | typedef Elf64_External_Ehdr Elf_External_Ehdr; | 
|  | typedef Elf64_External_Shdr Elf_External_Shdr; | 
|  | typedef Elf64_External_Sym Elf_External_Sym; | 
|  | typedef uint64_t Elf_Addr; | 
|  |  | 
|  | #elif defined TARGET_ILP32 | 
|  |  | 
|  | typedef Elf32_External_Phdr Elf_External_Phdr; | 
|  | typedef Elf32_External_Ehdr Elf_External_Ehdr; | 
|  | typedef Elf32_External_Shdr Elf_External_Shdr; | 
|  | typedef Elf32_External_Sym Elf_External_Sym; | 
|  | typedef uint32_t Elf_Addr; | 
|  |  | 
|  | #endif | 
|  |  | 
|  | #define GET(hdr, field) (\ | 
|  | sizeof ((hdr)->field) == 1 ? (uint64_t) (hdr)->field[0] : \ | 
|  | sizeof ((hdr)->field) == 2 ? (uint64_t) *(uint16_t *) (hdr)->field : \ | 
|  | sizeof ((hdr)->field) == 4 ? (uint64_t) *(uint32_t *) (hdr)->field : \ | 
|  | sizeof ((hdr)->field) == 8 ? *(uint64_t *) (hdr)->field : \ | 
|  | *(uint64_t *) NULL) | 
|  |  | 
|  | #define GETADDR(hdr, field) (\ | 
|  | sizeof ((hdr)->field) == sizeof (Elf_Addr) ? *(Elf_Addr *) (hdr)->field : \ | 
|  | *(Elf_Addr *) NULL) | 
|  |  | 
|  | struct segment | 
|  | { | 
|  | uint8_t *mapped_addr; | 
|  | size_t mapped_size; | 
|  | Elf_External_Phdr *phdr; | 
|  | struct segment *next; | 
|  | }; | 
|  |  | 
|  | struct library | 
|  | { | 
|  | int fd; | 
|  | Elf_External_Ehdr *ehdr; | 
|  | struct segment *segments; | 
|  | }; | 
|  |  | 
|  | static Elf_External_Shdr *find_shdr (Elf_External_Ehdr *ehdr, | 
|  | const char *section); | 
|  | static int translate_offset (uint64_t file_offset, struct segment *seg, | 
|  | void **addr); | 
|  |  | 
|  | #ifdef TARGET_LP64 | 
|  |  | 
|  | uint8_t | 
|  | elf_st_type (uint8_t st_info) | 
|  | { | 
|  | return ELF64_ST_TYPE (st_info); | 
|  | } | 
|  |  | 
|  | #elif defined TARGET_ILP32 | 
|  |  | 
|  | uint8_t | 
|  | elf_st_type (uint8_t st_info) | 
|  | { | 
|  | return ELF32_ST_TYPE (st_info); | 
|  | } | 
|  |  | 
|  | #endif | 
|  |  | 
|  | /* Load a program segment.  */ | 
|  |  | 
|  | static struct segment * | 
|  | load (uint8_t *addr, Elf_External_Phdr *phdr, struct segment *tail_seg) | 
|  | { | 
|  | struct segment *seg = NULL; | 
|  | uint8_t *mapped_addr = NULL; | 
|  | size_t mapped_size = 0; | 
|  | void *from = NULL; | 
|  | void *to = NULL; | 
|  |  | 
|  | /* For the sake of simplicity all operations are permitted.  */ | 
|  | unsigned perm = PROT_READ | PROT_WRITE | PROT_EXEC; | 
|  |  | 
|  | mapped_addr = (uint8_t *) mmap ((void *) GETADDR (phdr, p_vaddr), | 
|  | GET (phdr, p_memsz), perm, | 
|  | MAP_ANONYMOUS | MAP_PRIVATE, -1, 0); | 
|  | assert (mapped_addr != MAP_FAILED); | 
|  |  | 
|  | mapped_size = GET (phdr, p_memsz); | 
|  |  | 
|  | from = (void *) (addr + GET (phdr, p_offset)); | 
|  | to = (void *) mapped_addr; | 
|  |  | 
|  | memcpy (to, from, GET (phdr, p_filesz)); | 
|  |  | 
|  | seg = (struct segment *) malloc (sizeof (struct segment)); | 
|  |  | 
|  | if (seg == 0) | 
|  | return 0; | 
|  |  | 
|  | seg->mapped_addr = mapped_addr; | 
|  | seg->mapped_size = mapped_size; | 
|  | seg->phdr = phdr; | 
|  | seg->next = 0; | 
|  |  | 
|  | if (tail_seg != 0) | 
|  | tail_seg->next = seg; | 
|  |  | 
|  | return seg; | 
|  | } | 
|  |  | 
|  | #ifdef __linux__ | 
|  | # define SELF_LINK "/proc/self/exe" | 
|  | #elif defined NETBSD | 
|  | # define SELF_LINK "/proc/curproc/exe" | 
|  | #elif defined __OpenBSD__ || defined __FreeBSD__ || defined __DragonFly__ | 
|  | # define SELF_LINK "/proc/curproc/file" | 
|  | #elif defined SunOS | 
|  | # define SELF_LINK "/proc/self/path/a.out" | 
|  | #endif | 
|  |  | 
|  | /* Like RPATH=$ORIGIN, return the dirname of the current | 
|  | executable.  */ | 
|  |  | 
|  | static const char * | 
|  | get_origin (void) | 
|  | { | 
|  | static char self_path[PATH_MAX]; | 
|  | static ssize_t self_path_len; | 
|  |  | 
|  | if (self_path_len == 0) | 
|  | { | 
|  | #ifdef SELF_LINK | 
|  | self_path_len = readlink (SELF_LINK, self_path, PATH_MAX - 1); | 
|  | if (self_path_len != -1) | 
|  | { | 
|  | char *dirsep; | 
|  |  | 
|  | self_path[self_path_len] = '\0'; | 
|  | dirsep = strrchr (self_path, '/'); | 
|  | *dirsep = '\0'; | 
|  | } | 
|  | #else | 
|  | self_path_len = -1; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | if (self_path_len == -1) | 
|  | return NULL; | 
|  | else | 
|  | return self_path; | 
|  | } | 
|  |  | 
|  | /* Unload/unmap a segment.  */ | 
|  |  | 
|  | static void | 
|  | unload (struct segment *seg) | 
|  | { | 
|  | munmap (seg->mapped_addr, seg->mapped_size); | 
|  | free (seg); | 
|  | } | 
|  |  | 
|  | void | 
|  | unload_shlib (struct library *lib) | 
|  | { | 
|  | struct segment *seg, *next_seg; | 
|  |  | 
|  | for (seg = lib->segments; seg != NULL; seg = next_seg) | 
|  | { | 
|  | next_seg = seg->next; | 
|  | unload (seg); | 
|  | } | 
|  |  | 
|  | close (lib->fd); | 
|  | free (lib); | 
|  | } | 
|  |  | 
|  | /* Mini shared library loader.  No reallocation | 
|  | is performed for the sake of simplicity.  */ | 
|  |  | 
|  | struct library * | 
|  | load_shlib (const char *file) | 
|  | { | 
|  | struct library *lib; | 
|  | uint64_t i; | 
|  | int fd = -1; | 
|  | off_t fsize; | 
|  | uint8_t *addr; | 
|  | Elf_External_Ehdr *ehdr; | 
|  | Elf_External_Phdr *phdr; | 
|  | struct segment *head_seg = NULL; | 
|  | struct segment *tail_seg = NULL; | 
|  | const char *origin; | 
|  | char *path; | 
|  |  | 
|  | /* Map the lib in memory for reading. | 
|  |  | 
|  | If the file name is relative, try looking it up relative to the | 
|  | main executable's path.  I.e., emulate RPATH=$ORIGIN.  */ | 
|  | if (file[0] != '/') | 
|  | { | 
|  | origin = get_origin (); | 
|  | if (origin == NULL) | 
|  | { | 
|  | fprintf (stderr, "get_origin not implemented."); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | path = alloca (strlen (origin) + 1 + strlen (file) + 1); | 
|  | sprintf (path, "%s/%s", origin, file); | 
|  | fd = open (path, O_RDONLY); | 
|  | } | 
|  |  | 
|  | if (fd < 0) | 
|  | fd = open (file, O_RDONLY); | 
|  |  | 
|  | if (fd < 0) | 
|  | { | 
|  | perror ("fopen failed."); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | fsize = lseek (fd, 0, SEEK_END); | 
|  |  | 
|  | if (fsize < 0) | 
|  | { | 
|  | perror ("lseek failed."); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | addr = (uint8_t *) mmap (NULL, fsize, PROT_READ, MAP_PRIVATE, fd, 0); | 
|  | if (addr == MAP_FAILED) | 
|  | { | 
|  | perror ("mmap failed."); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | /* Check if the lib is an ELF file.  */ | 
|  | ehdr = (Elf_External_Ehdr *) addr; | 
|  | if (ehdr->e_ident[EI_MAG0] != ELFMAG0 | 
|  | || ehdr->e_ident[EI_MAG1] != ELFMAG1 | 
|  | || ehdr->e_ident[EI_MAG2] != ELFMAG2 | 
|  | || ehdr->e_ident[EI_MAG3] != ELFMAG3) | 
|  | { | 
|  | printf ("Not an ELF file: %x\n", ehdr->e_ident[EI_MAG0]); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | if (ehdr->e_ident[EI_CLASS] == ELFCLASS32) | 
|  | { | 
|  | if (sizeof (void *) != 4) | 
|  | { | 
|  | printf ("Architecture mismatch."); | 
|  | return NULL; | 
|  | } | 
|  | } | 
|  | else if (ehdr->e_ident[EI_CLASS] == ELFCLASS64) | 
|  | { | 
|  | if (sizeof (void *) != 8) | 
|  | { | 
|  | printf ("Architecture mismatch."); | 
|  | return NULL; | 
|  | } | 
|  | } | 
|  |  | 
|  | lib = malloc (sizeof (struct library)); | 
|  | if (lib == NULL) | 
|  | { | 
|  | printf ("malloc failed."); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | lib->fd = fd; | 
|  |  | 
|  | /* Load the program segments.  For the sake of simplicity | 
|  | assume that no reallocation is needed.  */ | 
|  | phdr = (Elf_External_Phdr *) (addr + GET (ehdr, e_phoff)); | 
|  | for (i = 0; i < GET (ehdr, e_phnum); i++, phdr++) | 
|  | { | 
|  | if (GET (phdr, p_type) == PT_LOAD) | 
|  | { | 
|  | struct segment *next_seg = load (addr, phdr, tail_seg); | 
|  | if (next_seg == 0) | 
|  | continue; | 
|  | tail_seg = next_seg; | 
|  | if (head_seg == 0) | 
|  | head_seg = next_seg; | 
|  | } | 
|  | } | 
|  | lib->ehdr = ehdr; | 
|  | lib->segments = head_seg; | 
|  | return lib; | 
|  | } | 
|  |  | 
|  | int | 
|  | get_text_addr (struct library *lib, void **text_addr) | 
|  | { | 
|  | Elf_External_Shdr *text; | 
|  |  | 
|  | /* Get the text section.  */ | 
|  | text = find_shdr (lib->ehdr, ".text"); | 
|  | if (text == NULL) | 
|  | return -1; | 
|  |  | 
|  | if (translate_offset (GET (text, sh_offset), lib->segments, text_addr) | 
|  | != 0) | 
|  | return -1; | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Return the section-header table.  */ | 
|  |  | 
|  | Elf_External_Shdr * | 
|  | find_shdrtab (Elf_External_Ehdr *ehdr) | 
|  | { | 
|  | return (Elf_External_Shdr *) (((uint8_t *) ehdr) + GET (ehdr, e_shoff)); | 
|  | } | 
|  |  | 
|  | /* Return the string table of the section headers.  */ | 
|  |  | 
|  | const char * | 
|  | find_shstrtab (Elf_External_Ehdr *ehdr, uint64_t *size) | 
|  | { | 
|  | const Elf_External_Shdr *shdr; | 
|  | const Elf_External_Shdr *shstr; | 
|  |  | 
|  | if (GET (ehdr, e_shnum) <= GET (ehdr, e_shstrndx)) | 
|  | { | 
|  | printf ("The index of the string table is corrupt."); | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | shdr = find_shdrtab (ehdr); | 
|  |  | 
|  | shstr = &shdr[GET (ehdr, e_shstrndx)]; | 
|  | *size = GET (shstr, sh_size); | 
|  | return ((const char *) ehdr) + GET (shstr, sh_offset); | 
|  | } | 
|  |  | 
|  | /* Return the string table named SECTION.  */ | 
|  |  | 
|  | const char * | 
|  | find_strtab (Elf_External_Ehdr *ehdr, | 
|  | const char *section, uint64_t *strtab_size) | 
|  | { | 
|  | uint64_t shstrtab_size = 0; | 
|  | const char *shstrtab; | 
|  | uint64_t i; | 
|  | const Elf_External_Shdr *shdr = find_shdrtab (ehdr); | 
|  |  | 
|  | /* Get the string table of the section headers.  */ | 
|  | shstrtab = find_shstrtab (ehdr, &shstrtab_size); | 
|  | if (shstrtab == NULL) | 
|  | return NULL; | 
|  |  | 
|  | for (i = 0; i < GET (ehdr, e_shnum); i++) | 
|  | { | 
|  | uint64_t name = GET (shdr + i, sh_name); | 
|  | if (GET (shdr + i, sh_type) == SHT_STRTAB && name <= shstrtab_size | 
|  | && strcmp ((const char *) &shstrtab[name], section) == 0) | 
|  | { | 
|  | *strtab_size = GET (shdr + i, sh_size); | 
|  | return ((const char *) ehdr) + GET (shdr + i, sh_offset); | 
|  | } | 
|  |  | 
|  | } | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | /* Return the section header named SECTION.  */ | 
|  |  | 
|  | static Elf_External_Shdr * | 
|  | find_shdr (Elf_External_Ehdr *ehdr, const char *section) | 
|  | { | 
|  | uint64_t shstrtab_size = 0; | 
|  | const char *shstrtab; | 
|  | uint64_t i; | 
|  |  | 
|  | /* Get the string table of the section headers.  */ | 
|  | shstrtab = find_shstrtab (ehdr, &shstrtab_size); | 
|  | if (shstrtab == NULL) | 
|  | return NULL; | 
|  |  | 
|  | Elf_External_Shdr *shdr = find_shdrtab (ehdr); | 
|  | for (i = 0; i < GET (ehdr, e_shnum); i++) | 
|  | { | 
|  | uint64_t name = GET (shdr + i, sh_name); | 
|  | if (name <= shstrtab_size) | 
|  | { | 
|  | if (strcmp ((const char *) &shstrtab[name], section) == 0) | 
|  | return &shdr[i]; | 
|  | } | 
|  |  | 
|  | } | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | /* Return the symbol table.  */ | 
|  |  | 
|  | static Elf_External_Sym * | 
|  | find_symtab (Elf_External_Ehdr *ehdr, uint64_t *symtab_size) | 
|  | { | 
|  | uint64_t i; | 
|  | const Elf_External_Shdr *shdr = find_shdrtab (ehdr); | 
|  |  | 
|  | for (i = 0; i < GET (ehdr, e_shnum); i++) | 
|  | { | 
|  | if (GET (shdr + i, sh_type) == SHT_SYMTAB) | 
|  | { | 
|  | *symtab_size = GET (shdr + i, sh_size) / sizeof (Elf_External_Sym); | 
|  | return (Elf_External_Sym *) (((const char *) ehdr) + | 
|  | GET (shdr + i, sh_offset)); | 
|  | } | 
|  | } | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | /* Translate a file offset to an address in a loaded segment.   */ | 
|  |  | 
|  | static int | 
|  | translate_offset (uint64_t file_offset, struct segment *seg, void **addr) | 
|  | { | 
|  | while (seg) | 
|  | { | 
|  | uint64_t p_from, p_to; | 
|  |  | 
|  | Elf_External_Phdr *phdr = seg->phdr; | 
|  |  | 
|  | if (phdr == NULL) | 
|  | { | 
|  | seg = seg->next; | 
|  | continue; | 
|  | } | 
|  |  | 
|  | p_from = GET (phdr, p_offset); | 
|  | p_to = p_from + GET (phdr, p_filesz); | 
|  |  | 
|  | if (p_from <= file_offset && file_offset < p_to) | 
|  | { | 
|  | *addr = (void *) (seg->mapped_addr + (file_offset - p_from)); | 
|  | return 0; | 
|  | } | 
|  | seg = seg->next; | 
|  | } | 
|  |  | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | /* Lookup the address of FUNC.  */ | 
|  |  | 
|  | int | 
|  | lookup_function (struct library *lib, const char *func, void **addr) | 
|  | { | 
|  | const char *strtab; | 
|  | uint64_t strtab_size = 0; | 
|  | Elf_External_Sym *symtab; | 
|  | uint64_t symtab_size = 0; | 
|  | uint64_t i; | 
|  | Elf_External_Ehdr *ehdr = lib->ehdr; | 
|  | struct segment *seg = lib->segments; | 
|  |  | 
|  | /* Get the string table for the symbols.  */ | 
|  | strtab = find_strtab (ehdr, ".strtab", &strtab_size); | 
|  | if (strtab == NULL) | 
|  | { | 
|  | printf (".strtab not found."); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | /* Get the symbol table.  */ | 
|  | symtab = find_symtab (ehdr, &symtab_size); | 
|  | if (symtab == NULL) | 
|  | { | 
|  | printf ("symbol table not found."); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | for (i = 0; i < symtab_size; i++) | 
|  | { | 
|  | Elf_External_Sym *sym = &symtab[i]; | 
|  |  | 
|  | if (elf_st_type (GET (sym, st_info)) != STT_FUNC) | 
|  | continue; | 
|  |  | 
|  | if (GET (sym, st_name) < strtab_size) | 
|  | { | 
|  | const char *name = &strtab[GET (sym, st_name)]; | 
|  | if (strcmp (name, func) == 0) | 
|  | { | 
|  |  | 
|  | uint64_t offset = GET (sym, st_value); | 
|  | return translate_offset (offset, seg, addr); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | return -1; | 
|  | } |