aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPedro Alves <palves@redhat.com>2014-04-15 14:02:34 +0100
committerPedro Alves <palves@redhat.com>2014-04-15 14:23:37 +0100
commit35e5d2f0f81d97f9bd41586b1979345072b7989d (patch)
tree7855d882067bbb98544809b10df256ebc2ccc1d3
parenteb4c17106b7303565b8d0ce8b572fa64945d1e7d (diff)
downloadgdb-35e5d2f0f81d97f9bd41586b1979345072b7989d.zip
gdb-35e5d2f0f81d97f9bd41586b1979345072b7989d.tar.gz
gdb-35e5d2f0f81d97f9bd41586b1979345072b7989d.tar.bz2
gdb.base/sym-file.exp, hide guts of the custom loader.
This test uses a simple custom elf loader, implemented in gdb.base/sym-file-loader.h|c. This loader doesn't have a dlclose-like function today, but I'll need one. But, I found that the guts of the loader are exposed too much to the client, making the interface more complicated than necessary. It's simpler if the loader just exports a few dlopen/dlsym -style functions. That's what this patch does. Tested on x86_86 Fedora 17, native and gdbserver. gdb/testsuite/ 2014-04-15 Pedro Alves <palves@redhat.com> * gdb.base/sym-file-loader.h: Move inclusion of <inttypes.h>, <ansidecl.h>, <elf/common.h> and <elf/external.h> to sym-file-loader.c. (Elf_External_Phdr, Elf_External_Ehdr, Elf_External_Shdr) (Elf_External_Sym, Elf_Addr, GET, GETADDR, struct segment): Move to sym-file-loader.c. (struct library): Forward declare. (load_shlib, lookup_function): Change prototypes. (find_shstrtab, find_strtab, find_shdr, find_symtab) (translate_offset): Remove declarations. (get_text_addr): New declaration. * gdb.base/sym-file-loader.c: Move inclusion of <inttypes.h>, <ansidecl.h>, <elf/common.h> and <elf/external.h> here from sym-file-loader.h. (Elf_External_Phdr, Elf_External_Ehdr, Elf_External_Shdr) (Elf_External_Sym, Elf_Addr, GET, GETADDR, struct segment): Move here from sym-file-loader.h. (struct library): New structure. (load_shlib, lookup_function): Change prototypes and adjust to work with a struct library. (find_shstrtab, find_strtab, find_shdr, find_symtab) (translate_offset): Make static. (get_text_addr): New function. * gdb.base/sym-file-main.c (main): Adjust to new loader interface.
-rw-r--r--gdb/testsuite/ChangeLog27
-rw-r--r--gdb/testsuite/gdb.base/sym-file-loader.c114
-rw-r--r--gdb/testsuite/gdb.base/sym-file-loader.h77
-rw-r--r--gdb/testsuite/gdb.base/sym-file-main.c28
4 files changed, 141 insertions, 105 deletions
diff --git a/gdb/testsuite/ChangeLog b/gdb/testsuite/ChangeLog
index ca002e9..bbea89d 100644
--- a/gdb/testsuite/ChangeLog
+++ b/gdb/testsuite/ChangeLog
@@ -1,5 +1,32 @@
2014-04-15 Pedro Alves <palves@redhat.com>
+ * gdb.base/sym-file-loader.h: Move inclusion of <inttypes.h>,
+ <ansidecl.h>, <elf/common.h> and <elf/external.h> to
+ sym-file-loader.c.
+ (Elf_External_Phdr, Elf_External_Ehdr, Elf_External_Shdr)
+ (Elf_External_Sym, Elf_Addr, GET, GETADDR, struct segment): Move
+ to sym-file-loader.c.
+ (struct library): Forward declare.
+ (load_shlib, lookup_function): Change prototypes.
+ (find_shstrtab, find_strtab, find_shdr, find_symtab)
+ (translate_offset): Remove declarations.
+ (get_text_addr): New declaration.
+ * gdb.base/sym-file-loader.c: Move inclusion of <inttypes.h>,
+ <ansidecl.h>, <elf/common.h> and <elf/external.h> here from
+ sym-file-loader.h.
+ (Elf_External_Phdr, Elf_External_Ehdr, Elf_External_Shdr)
+ (Elf_External_Sym, Elf_Addr, GET, GETADDR, struct segment): Move
+ here from sym-file-loader.h.
+ (struct library): New structure.
+ (load_shlib, lookup_function): Change prototypes and adjust to
+ work with a struct library.
+ (find_shstrtab, find_strtab, find_shdr, find_symtab)
+ (translate_offset): Make static.
+ (get_text_addr): New function.
+ * gdb.base/sym-file-main.c (main): Adjust to new loader interface.
+
+2014-04-15 Pedro Alves <palves@redhat.com>
+
* gdb.base/sym-file-loader.c: Fix typo. SELF_LINK, not SELK_LINK.
2014-04-15 Pedro Alves <palves@redhat.com>
diff --git a/gdb/testsuite/gdb.base/sym-file-loader.c b/gdb/testsuite/gdb.base/sym-file-loader.c
index c30ffe1..d10065e 100644
--- a/gdb/testsuite/gdb.base/sym-file-loader.c
+++ b/gdb/testsuite/gdb.base/sym-file-loader.c
@@ -23,6 +23,59 @@
#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;
+ 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
@@ -123,10 +176,10 @@ get_origin (void)
/* 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)
+struct library *
+load_shlib (const char *file)
{
+ struct library *lib;
uint64_t i;
int fd = -1;
off_t fsize;
@@ -148,7 +201,7 @@ load_shlib (const char *file, Elf_External_Ehdr **ehdr_out,
if (origin == NULL)
{
fprintf (stderr, "get_origin not implemented.");
- return -1;
+ return NULL;
}
path = alloca (strlen (origin) + 1 + strlen (file) + 1);
@@ -162,7 +215,7 @@ load_shlib (const char *file, Elf_External_Ehdr **ehdr_out,
if (fd < 0)
{
perror ("fopen failed.");
- return -1;
+ return NULL;
}
fsize = lseek (fd, 0, SEEK_END);
@@ -170,14 +223,14 @@ load_shlib (const char *file, Elf_External_Ehdr **ehdr_out,
if (fsize < 0)
{
perror ("lseek failed.");
- return -1;
+ return NULL;
}
addr = (uint8_t *) mmap (NULL, fsize, PROT_READ, MAP_PRIVATE, fd, 0);
if (addr == (uint8_t *) -1)
{
perror ("mmap failed.");
- return -1;
+ return NULL;
}
/* Check if the lib is an ELF file. */
@@ -188,7 +241,7 @@ load_shlib (const char *file, Elf_External_Ehdr **ehdr_out,
|| ehdr->e_ident[EI_MAG3] != ELFMAG3)
{
printf ("Not an ELF file: %x\n", ehdr->e_ident[EI_MAG0]);
- return -1;
+ return NULL;
}
if (ehdr->e_ident[EI_CLASS] == ELFCLASS32)
@@ -196,7 +249,7 @@ load_shlib (const char *file, Elf_External_Ehdr **ehdr_out,
if (sizeof (void *) != 4)
{
printf ("Architecture mismatch.");
- return -1;
+ return NULL;
}
}
else if (ehdr->e_ident[EI_CLASS] == ELFCLASS64)
@@ -204,10 +257,19 @@ load_shlib (const char *file, Elf_External_Ehdr **ehdr_out,
if (sizeof (void *) != 8)
{
printf ("Architecture mismatch.");
- return -1;
+ 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));
@@ -223,8 +285,25 @@ load_shlib (const char *file, Elf_External_Ehdr **ehdr_out,
head_seg = next_seg;
}
}
- *ehdr_out = ehdr;
- *seg_out = head_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;
}
@@ -289,7 +368,7 @@ find_strtab (Elf_External_Ehdr *ehdr,
/* Return the section header named SECTION. */
-Elf_External_Shdr *
+static Elf_External_Shdr *
find_shdr (Elf_External_Ehdr *ehdr, const char *section)
{
uint64_t shstrtab_size = 0;
@@ -317,7 +396,7 @@ find_shdr (Elf_External_Ehdr *ehdr, const char *section)
/* Return the symbol table. */
-Elf_External_Sym *
+static Elf_External_Sym *
find_symtab (Elf_External_Ehdr *ehdr, uint64_t *symtab_size)
{
uint64_t i;
@@ -337,7 +416,7 @@ find_symtab (Elf_External_Ehdr *ehdr, uint64_t *symtab_size)
/* Translate a file offset to an address in a loaded segment. */
-int
+static int
translate_offset (uint64_t file_offset, struct segment *seg, void **addr)
{
while (seg)
@@ -369,14 +448,15 @@ translate_offset (uint64_t file_offset, struct segment *seg, void **addr)
/* Lookup the address of FUNC. */
int
-lookup_function (const char *func,
- Elf_External_Ehdr *ehdr, struct segment *seg, void **addr)
+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);
diff --git a/gdb/testsuite/gdb.base/sym-file-loader.h b/gdb/testsuite/gdb.base/sym-file-loader.h
index de9e8be..c6b1af3 100644
--- a/gdb/testsuite/gdb.base/sym-file-loader.h
+++ b/gdb/testsuite/gdb.base/sym-file-loader.h
@@ -16,84 +16,21 @@
#ifndef __SYM_FILE_LOADER__
#define __SYM_FILE_LOADER__
-#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;
- Elf_External_Phdr *phdr;
- struct segment *next;
-};
+struct library;
/* 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);
-
-/* Return the section-header table. */
+/* Load a library. */
-Elf_External_Shdr *find_shdrtab (Elf_External_Ehdr *ehdr);
+struct library *load_shlib (const char *file);
-/* Return the string table of the section headers. */
-
-const char *find_shstrtab (Elf_External_Ehdr *ehdr, uint64_t *size);
-
-/* Return the string table named SECTION. */
-
-const char *find_strtab (Elf_External_Ehdr *ehdr,
- const char *section, uint64_t *strtab_size);
-
-/* Return the section header named SECTION. */
-
-Elf_External_Shdr *find_shdr (Elf_External_Ehdr *ehdr, const char *section);
-
-/* Return the symbol table. */
-
-Elf_External_Sym *find_symtab (Elf_External_Ehdr *ehdr,
- uint64_t *symtab_size);
-
-/* Translate a file offset to an address in a loaded segment. */
+/* Lookup the address of FUNC. */
-int translate_offset (uint64_t file_offset, struct segment *seg, void **addr);
+int lookup_function (struct library *lib, const char *func, void **addr);
-/* Lookup the address of FUNC. */
+/* Return the library's loaded text address. */
-int
-lookup_function (const char *func, Elf_External_Ehdr* ehdr,
- struct segment *seg, void **addr);
+int get_text_addr (struct library *lib, void **text_addr);
#endif
diff --git a/gdb/testsuite/gdb.base/sym-file-main.c b/gdb/testsuite/gdb.base/sym-file-main.c
index 2746566..b48a1c2 100644
--- a/gdb/testsuite/gdb.base/sym-file-main.c
+++ b/gdb/testsuite/gdb.base/sym-file-main.c
@@ -38,37 +38,29 @@ int
main (int argc, const char *argv[])
{
const char *file = SHLIB_NAME;
- Elf_External_Ehdr *ehdr = NULL;
- struct segment *head_seg = NULL;
- Elf_External_Shdr *text;
+ struct library *lib;
char *text_addr = NULL;
int (*pbar) () = NULL;
int (*pfoo) (int) = NULL;
- if (load_shlib (file, &ehdr, &head_seg) != 0)
- return -1;
+ lib = load_shlib (file);
+ if (lib == NULL)
+ return 1;
- /* Get the text section. */
- text = find_shdr (ehdr, ".text");
- if (text == NULL)
- return -1;
-
- /* Notify GDB to add the symbol file. */
- if (translate_offset (GET (text, sh_offset), head_seg, (void **) &text_addr)
- != 0)
- return -1;
+ if (get_text_addr (lib, (void **) &text_addr) != 0)
+ return 1;
gdb_add_symbol_file (text_addr, file);
/* Call bar from SHLIB_NAME. */
- if (lookup_function ("bar", ehdr, head_seg, (void *) &pbar) != 0)
- return -1;
+ if (lookup_function (lib, "bar", (void *) &pbar) != 0)
+ return 1;
(*pbar) ();
/* Call foo from SHLIB_NAME. */
- if (lookup_function ("foo", ehdr, head_seg, (void *) &pfoo) != 0)
- return -1;
+ if (lookup_function (lib, "foo", (void *) &pfoo) != 0)
+ return 1;
(*pfoo) (2);