| Index: gdb/testsuite/gdb.base/sym-file-loader.c
|
| diff --git a/gdb/testsuite/gdb.base/sym-file-loader.c b/gdb/testsuite/gdb.base/sym-file-loader.c
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..c55076fef0a3683477f4c6dd6fc0f792d62c5473
|
| --- /dev/null
|
| +++ b/gdb/testsuite/gdb.base/sym-file-loader.c
|
| @@ -0,0 +1,353 @@
|
| +/* Copyright 2013 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 <stdio.h>
|
| +#include <stdlib.h>
|
| +#include <string.h>
|
| +#include <sys/mman.h>
|
| +
|
| +#include "sym-file-loader.h"
|
| +
|
| +#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;
|
| + 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);
|
| +
|
| + 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->phdr = phdr;
|
| + seg->next = 0;
|
| +
|
| + if (tail_seg != 0)
|
| + tail_seg->next = seg;
|
| +
|
| + return seg;
|
| +}
|
| +
|
| +/* Mini shared library loader. No reallocation
|
| + is performed for the sake of simplicity. */
|
| +
|
| +int
|
| +load_shlib (const char *file, Elf_External_Ehdr **ehdr_out,
|
| + struct segment **seg_out)
|
| +{
|
| + uint64_t i;
|
| + int fd;
|
| + off_t fsize;
|
| + uint8_t *addr;
|
| + Elf_External_Ehdr *ehdr;
|
| + Elf_External_Phdr *phdr;
|
| + struct segment *head_seg = NULL;
|
| + struct segment *tail_seg = NULL;
|
| +
|
| + /* Map the lib in memory for reading. */
|
| + fd = open (file, O_RDONLY);
|
| + if (fd < 0)
|
| + {
|
| + perror ("fopen failed.");
|
| + return -1;
|
| + }
|
| +
|
| + fsize = lseek (fd, 0, SEEK_END);
|
| +
|
| + if (fsize < 0)
|
| + {
|
| + perror ("lseek failed.");
|
| + return -1;
|
| + }
|
| +
|
| + addr = (uint8_t *) mmap (NULL, fsize, PROT_READ, MAP_PRIVATE, fd, 0);
|
| + if (addr == (uint8_t *) -1)
|
| + {
|
| + perror ("mmap failed.");
|
| + return -1;
|
| + }
|
| +
|
| + /* 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 -1;
|
| + }
|
| +
|
| + if (ehdr->e_ident[EI_CLASS] == ELFCLASS32)
|
| + {
|
| + if (sizeof (void *) != 4)
|
| + {
|
| + printf ("Architecture mismatch.");
|
| + return -1;
|
| + }
|
| + }
|
| + else if (ehdr->e_ident[EI_CLASS] == ELFCLASS64)
|
| + {
|
| + if (sizeof (void *) != 8)
|
| + {
|
| + printf ("Architecture mismatch.");
|
| + return -1;
|
| + }
|
| + }
|
| +
|
| + /* 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;
|
| + }
|
| + }
|
| + *ehdr_out = ehdr;
|
| + *seg_out = head_seg;
|
| + 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. */
|
| +
|
| +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. */
|
| +
|
| +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. */
|
| +
|
| +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 (const char *func,
|
| + Elf_External_Ehdr *ehdr, struct segment *seg, void **addr)
|
| +{
|
| + const char *strtab;
|
| + uint64_t strtab_size = 0;
|
| + Elf_External_Sym *symtab;
|
| + uint64_t symtab_size = 0;
|
| + uint64_t i;
|
| +
|
| + /* 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;
|
| +}
|
|
|