Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 1 | /* |
Domenico Andreoli | e714d2e | 2019-01-15 18:28:24 +0100 | [diff] [blame] | 2 | SPDX-License-Identifier: GPL-2.0-only |
| 3 | |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 4 | Copyright (C) 2009 Red Hat Inc. |
| 5 | Copyright (C) 2009 Arnaldo Carvalho de Melo <acme@redhat.com> |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 6 | */ |
| 7 | |
| 8 | #include <malloc.h> |
| 9 | #include <stdio.h> |
| 10 | #include <string.h> |
| 11 | |
| 12 | #include "dutil.h" |
| 13 | #include "elf_symtab.h" |
| 14 | |
| 15 | #define HASHSYMS__BITS 8 |
| 16 | #define HASHSYMS__SIZE (1UL << HASHSYMS__BITS) |
| 17 | |
Arnaldo Carvalho de Melo | 2440419 | 2021-08-05 16:05:45 -0300 | [diff] [blame] | 18 | struct elf_symtab *elf_symtab__new(const char *name, Elf *elf) |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 19 | { |
Jiri Olsa | 1bb4989 | 2021-01-24 23:15:19 +0100 | [diff] [blame] | 20 | size_t symtab_index; |
| 21 | |
Arnaldo Carvalho de Melo | ba47890 | 2009-03-25 18:02:48 -0300 | [diff] [blame] | 22 | if (name == NULL) |
| 23 | name = ".symtab"; |
| 24 | |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 25 | GElf_Shdr shdr; |
Arnaldo Carvalho de Melo | 74c2078 | 2021-08-05 10:21:15 -0300 | [diff] [blame] | 26 | Elf_Scn *sec = elf_section_by_name(elf, &shdr, name, &symtab_index); |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 27 | |
| 28 | if (sec == NULL) |
| 29 | return NULL; |
| 30 | |
| 31 | if (gelf_getshdr(sec, &shdr) == NULL) |
| 32 | return NULL; |
| 33 | |
Jiri Olsa | 1bb4989 | 2021-01-24 23:15:19 +0100 | [diff] [blame] | 34 | struct elf_symtab *symtab = zalloc(sizeof(*symtab)); |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 35 | if (symtab == NULL) |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 36 | return NULL; |
| 37 | |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 38 | symtab->name = strdup(name); |
| 39 | if (symtab->name == NULL) |
Arnaldo Carvalho de Melo | ba47890 | 2009-03-25 18:02:48 -0300 | [diff] [blame] | 40 | goto out_delete; |
| 41 | |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 42 | symtab->syms = elf_getdata(sec, NULL); |
| 43 | if (symtab->syms == NULL) |
Arnaldo Carvalho de Melo | ba47890 | 2009-03-25 18:02:48 -0300 | [diff] [blame] | 44 | goto out_free_name; |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 45 | |
Jiri Olsa | 1bb4989 | 2021-01-24 23:15:19 +0100 | [diff] [blame] | 46 | /* |
| 47 | * This returns extended section index table's |
| 48 | * section index, if it exists. |
| 49 | */ |
| 50 | int symtab_xindex = elf_scnshndx(sec); |
| 51 | |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 52 | sec = elf_getscn(elf, shdr.sh_link); |
| 53 | if (sec == NULL) |
Arnaldo Carvalho de Melo | ba47890 | 2009-03-25 18:02:48 -0300 | [diff] [blame] | 54 | goto out_free_name; |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 55 | |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 56 | symtab->symstrs = elf_getdata(sec, NULL); |
| 57 | if (symtab->symstrs == NULL) |
Arnaldo Carvalho de Melo | ba47890 | 2009-03-25 18:02:48 -0300 | [diff] [blame] | 58 | goto out_free_name; |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 59 | |
Jiri Olsa | 1bb4989 | 2021-01-24 23:15:19 +0100 | [diff] [blame] | 60 | /* |
| 61 | * The .symtab section has optional extended section index |
| 62 | * table, load its data so it can be used to resolve symbol's |
| 63 | * section index. |
| 64 | **/ |
| 65 | if (symtab_xindex > 0) { |
| 66 | GElf_Shdr shdr_xindex; |
| 67 | Elf_Scn *sec_xindex; |
| 68 | |
| 69 | sec_xindex = elf_getscn(elf, symtab_xindex); |
| 70 | if (sec_xindex == NULL) |
| 71 | goto out_free_name; |
| 72 | |
| 73 | if (gelf_getshdr(sec_xindex, &shdr_xindex) == NULL) |
| 74 | goto out_free_name; |
| 75 | |
| 76 | /* Extra check to verify it's correct type */ |
| 77 | if (shdr_xindex.sh_type != SHT_SYMTAB_SHNDX) |
| 78 | goto out_free_name; |
| 79 | |
| 80 | /* Extra check to verify it belongs to the .symtab */ |
| 81 | if (symtab_index != shdr_xindex.sh_link) |
| 82 | goto out_free_name; |
| 83 | |
| 84 | symtab->syms_sec_idx_table = elf_getdata(elf_getscn(elf, symtab_xindex), NULL); |
| 85 | if (symtab->syms_sec_idx_table == NULL) |
| 86 | goto out_free_name; |
| 87 | } |
| 88 | |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 89 | symtab->nr_syms = shdr.sh_size / shdr.sh_entsize; |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 90 | |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 91 | return symtab; |
Arnaldo Carvalho de Melo | ba47890 | 2009-03-25 18:02:48 -0300 | [diff] [blame] | 92 | out_free_name: |
Arnaldo Carvalho de Melo | 1105b7d | 2021-05-27 10:35:14 -0300 | [diff] [blame] | 93 | zfree(&symtab->name); |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 94 | out_delete: |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 95 | free(symtab); |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 96 | return NULL; |
| 97 | } |
| 98 | |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 99 | void elf_symtab__delete(struct elf_symtab *symtab) |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 100 | { |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 101 | if (symtab == NULL) |
Arnaldo Carvalho de Melo | 33ed25d | 2009-04-03 09:41:09 -0300 | [diff] [blame] | 102 | return; |
Arnaldo Carvalho de Melo | 1105b7d | 2021-05-27 10:35:14 -0300 | [diff] [blame] | 103 | zfree(&symtab->name); |
Arnaldo Carvalho de Melo | a54515f | 2012-08-17 18:47:15 -0300 | [diff] [blame] | 104 | free(symtab); |
Arnaldo Carvalho de Melo | 0954d75 | 2009-03-24 16:58:44 -0300 | [diff] [blame] | 105 | } |