Thomas Gleixner | 4317cf9 | 2019-05-31 01:09:38 -0700 | [diff] [blame] | 1 | /* SPDX-License-Identifier: GPL-2.0-only */ |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 2 | /* |
Shile Zhang | 1091670 | 2019-12-04 08:46:31 +0800 | [diff] [blame^] | 3 | * sorttable.h |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 4 | * |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 5 | * Copyright 2011 - 2012 Cavium, Inc. |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 6 | * |
| 7 | * Some of this code was taken out of recordmcount.h written by: |
| 8 | * |
Shile Zhang | 6402e14 | 2019-12-04 08:46:28 +0800 | [diff] [blame] | 9 | * Copyright 2009 John F. Reiser <jreiser@BitWagon.com>. All rights reserved. |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 10 | * Copyright 2010 Steven Rostedt <srostedt@redhat.com>, Red Hat Inc. |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 11 | */ |
| 12 | |
| 13 | #undef extable_ent_size |
| 14 | #undef compare_extable |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 15 | #undef do_sort |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 16 | #undef Elf_Addr |
| 17 | #undef Elf_Ehdr |
| 18 | #undef Elf_Shdr |
| 19 | #undef Elf_Rel |
| 20 | #undef Elf_Rela |
| 21 | #undef Elf_Sym |
| 22 | #undef ELF_R_SYM |
| 23 | #undef Elf_r_sym |
| 24 | #undef ELF_R_INFO |
| 25 | #undef Elf_r_info |
| 26 | #undef ELF_ST_BIND |
| 27 | #undef ELF_ST_TYPE |
| 28 | #undef fn_ELF_R_SYM |
| 29 | #undef fn_ELF_R_INFO |
| 30 | #undef uint_t |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 31 | #undef _r |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 32 | #undef _w |
| 33 | |
Shile Zhang | 1091670 | 2019-12-04 08:46:31 +0800 | [diff] [blame^] | 34 | #ifdef SORTTABLE_64 |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 35 | # define extable_ent_size 16 |
| 36 | # define compare_extable compare_extable_64 |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 37 | # define do_sort do_sort_64 |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 38 | # define Elf_Addr Elf64_Addr |
| 39 | # define Elf_Ehdr Elf64_Ehdr |
| 40 | # define Elf_Shdr Elf64_Shdr |
| 41 | # define Elf_Rel Elf64_Rel |
| 42 | # define Elf_Rela Elf64_Rela |
| 43 | # define Elf_Sym Elf64_Sym |
| 44 | # define ELF_R_SYM ELF64_R_SYM |
| 45 | # define Elf_r_sym Elf64_r_sym |
| 46 | # define ELF_R_INFO ELF64_R_INFO |
| 47 | # define Elf_r_info Elf64_r_info |
| 48 | # define ELF_ST_BIND ELF64_ST_BIND |
| 49 | # define ELF_ST_TYPE ELF64_ST_TYPE |
| 50 | # define fn_ELF_R_SYM fn_ELF64_R_SYM |
| 51 | # define fn_ELF_R_INFO fn_ELF64_R_INFO |
| 52 | # define uint_t uint64_t |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 53 | # define _r r8 |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 54 | # define _w w8 |
| 55 | #else |
| 56 | # define extable_ent_size 8 |
| 57 | # define compare_extable compare_extable_32 |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 58 | # define do_sort do_sort_32 |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 59 | # define Elf_Addr Elf32_Addr |
| 60 | # define Elf_Ehdr Elf32_Ehdr |
| 61 | # define Elf_Shdr Elf32_Shdr |
| 62 | # define Elf_Rel Elf32_Rel |
| 63 | # define Elf_Rela Elf32_Rela |
| 64 | # define Elf_Sym Elf32_Sym |
| 65 | # define ELF_R_SYM ELF32_R_SYM |
| 66 | # define Elf_r_sym Elf32_r_sym |
| 67 | # define ELF_R_INFO ELF32_R_INFO |
| 68 | # define Elf_r_info Elf32_r_info |
| 69 | # define ELF_ST_BIND ELF32_ST_BIND |
| 70 | # define ELF_ST_TYPE ELF32_ST_TYPE |
| 71 | # define fn_ELF_R_SYM fn_ELF32_R_SYM |
| 72 | # define fn_ELF_R_INFO fn_ELF32_R_INFO |
| 73 | # define uint_t uint32_t |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 74 | # define _r r |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 75 | # define _w w |
| 76 | #endif |
| 77 | |
| 78 | static int compare_extable(const void *a, const void *b) |
| 79 | { |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 80 | Elf_Addr av = _r(a); |
| 81 | Elf_Addr bv = _r(b); |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 82 | |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 83 | if (av < bv) |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 84 | return -1; |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 85 | if (av > bv) |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 86 | return 1; |
| 87 | return 0; |
| 88 | } |
| 89 | |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 90 | static int do_sort(Elf_Ehdr *ehdr, |
Shile Zhang | 6402e14 | 2019-12-04 08:46:28 +0800 | [diff] [blame] | 91 | char const *const fname, |
| 92 | table_sort_t custom_sort) |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 93 | { |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 94 | Elf_Shdr *s, *shdr = (Elf_Shdr *)((char *)ehdr + _r(&ehdr->e_shoff)); |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 95 | Elf_Shdr *strtab_sec = NULL; |
| 96 | Elf_Shdr *symtab_sec = NULL; |
| 97 | Elf_Shdr *extab_sec = NULL; |
| 98 | Elf_Sym *sym; |
Jamie Iles | 59c3645 | 2013-11-12 15:06:51 -0800 | [diff] [blame] | 99 | const Elf_Sym *symtab; |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 100 | Elf32_Word *symtab_shndx = NULL; |
| 101 | Elf_Sym *sort_needed_sym = NULL; |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 102 | Elf_Shdr *sort_needed_sec; |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 103 | Elf_Rel *relocs = NULL; |
Tim Gardner | 7cbc0ea | 2014-10-13 15:54:20 -0700 | [diff] [blame] | 104 | int relocs_size = 0; |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 105 | uint32_t *sort_needed_loc; |
| 106 | const char *secstrings; |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 107 | const char *strtab; |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 108 | char *extab_image; |
| 109 | int extab_index = 0; |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 110 | int i; |
| 111 | int idx; |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 112 | unsigned int shnum; |
| 113 | unsigned int shstrndx; |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 114 | |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 115 | shstrndx = r2(&ehdr->e_shstrndx); |
| 116 | if (shstrndx == SHN_XINDEX) |
| 117 | shstrndx = r(&shdr[0].sh_link); |
| 118 | secstrings = (const char *)ehdr + _r(&shdr[shstrndx].sh_offset); |
Jamie Iles | 59c3645 | 2013-11-12 15:06:51 -0800 | [diff] [blame] | 119 | |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 120 | shnum = r2(&ehdr->e_shnum); |
| 121 | if (shnum == SHN_UNDEF) |
| 122 | shnum = _r(&shdr[0].sh_size); |
Jamie Iles | 59c3645 | 2013-11-12 15:06:51 -0800 | [diff] [blame] | 123 | |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 124 | for (i = 0, s = shdr; s < shdr + shnum; i++, s++) { |
| 125 | idx = r(&s->sh_name); |
| 126 | if (!strcmp(secstrings + idx, "__ex_table")) { |
| 127 | extab_sec = s; |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 128 | extab_index = i; |
| 129 | } |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 130 | if (!strcmp(secstrings + idx, ".symtab")) |
| 131 | symtab_sec = s; |
| 132 | if (!strcmp(secstrings + idx, ".strtab")) |
| 133 | strtab_sec = s; |
| 134 | |
| 135 | if ((r(&s->sh_type) == SHT_REL || |
| 136 | r(&s->sh_type) == SHT_RELA) && |
| 137 | r(&s->sh_info) == extab_index) { |
| 138 | relocs = (void *)ehdr + _r(&s->sh_offset); |
| 139 | relocs_size = _r(&s->sh_size); |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 140 | } |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 141 | if (r(&s->sh_type) == SHT_SYMTAB_SHNDX) |
| 142 | symtab_shndx = (Elf32_Word *)((const char *)ehdr + |
| 143 | _r(&s->sh_offset)); |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 144 | } |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 145 | |
Shile Zhang | 6402e14 | 2019-12-04 08:46:28 +0800 | [diff] [blame] | 146 | if (!extab_sec) { |
| 147 | fprintf(stderr, "no __ex_table in file: %s\n", fname); |
Shile Zhang | 3c47b78 | 2019-12-04 08:46:27 +0800 | [diff] [blame] | 148 | return -1; |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 149 | } |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 150 | |
| 151 | if (!symtab_sec) { |
| 152 | fprintf(stderr, "no .symtab in file: %s\n", fname); |
| 153 | return -1; |
| 154 | } |
| 155 | |
| 156 | if (!strtab_sec) { |
| 157 | fprintf(stderr, "no .strtab in file: %s\n", fname); |
| 158 | return -1; |
| 159 | } |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 160 | |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 161 | extab_image = (void *)ehdr + _r(&extab_sec->sh_offset); |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 162 | strtab = (const char *)ehdr + _r(&strtab_sec->sh_offset); |
| 163 | symtab = (const Elf_Sym *)((const char *)ehdr + |
| 164 | _r(&symtab_sec->sh_offset)); |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 165 | |
| 166 | if (custom_sort) { |
| 167 | custom_sort(extab_image, _r(&extab_sec->sh_size)); |
| 168 | } else { |
| 169 | int num_entries = _r(&extab_sec->sh_size) / extable_ent_size; |
| 170 | qsort(extab_image, num_entries, |
| 171 | extable_ent_size, compare_extable); |
| 172 | } |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 173 | |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 174 | /* If there were relocations, we no longer need them. */ |
| 175 | if (relocs) |
| 176 | memset(relocs, 0, relocs_size); |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 177 | |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 178 | /* find the flag main_extable_sort_needed */ |
| 179 | for (sym = (void *)ehdr + _r(&symtab_sec->sh_offset); |
| 180 | sym < sym + _r(&symtab_sec->sh_size) / sizeof(Elf_Sym); |
| 181 | sym++) { |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 182 | if (ELF_ST_TYPE(sym->st_info) != STT_OBJECT) |
| 183 | continue; |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 184 | if (!strcmp(strtab + r(&sym->st_name), |
| 185 | "main_extable_sort_needed")) { |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 186 | sort_needed_sym = sym; |
| 187 | break; |
| 188 | } |
| 189 | } |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 190 | |
Shile Zhang | 6402e14 | 2019-12-04 08:46:28 +0800 | [diff] [blame] | 191 | if (!sort_needed_sym) { |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 192 | fprintf(stderr, |
Shile Zhang | 6402e14 | 2019-12-04 08:46:28 +0800 | [diff] [blame] | 193 | "no main_extable_sort_needed symbol in file: %s\n", |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 194 | fname); |
Shile Zhang | 3c47b78 | 2019-12-04 08:46:27 +0800 | [diff] [blame] | 195 | return -1; |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 196 | } |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 197 | |
Jamie Iles | 59c3645 | 2013-11-12 15:06:51 -0800 | [diff] [blame] | 198 | sort_needed_sec = &shdr[get_secindex(r2(&sym->st_shndx), |
| 199 | sort_needed_sym - symtab, |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 200 | symtab_shndx)]; |
| 201 | sort_needed_loc = (void *)ehdr + |
David Daney | d59a168 | 2012-04-24 11:23:14 -0700 | [diff] [blame] | 202 | _r(&sort_needed_sec->sh_offset) + |
| 203 | _r(&sort_needed_sym->st_value) - |
| 204 | _r(&sort_needed_sec->sh_addr); |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 205 | |
Shile Zhang | 57cafdf | 2019-12-04 08:46:30 +0800 | [diff] [blame] | 206 | /* extable has been sorted, clear the flag */ |
| 207 | w(0, sort_needed_loc); |
| 208 | |
Shile Zhang | 3c47b78 | 2019-12-04 08:46:27 +0800 | [diff] [blame] | 209 | return 0; |
David Daney | a79f248 | 2012-04-19 14:59:55 -0700 | [diff] [blame] | 210 | } |