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