2 * Copyright (c) 1997 Christopher G. Demetriou. All rights reserved.
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
7 * 1. Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * 2. Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 * 3. All advertising materials mentioning features or use of this software
13 * must display the following acknowledgement:
14 * This product includes software developed by Christopher G. Demetriou
15 * for the NetBSD Project.
16 * 4. The name of the author may not be used to endorse or promote products
17 * derived from this software without specific prior written permission
19 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
20 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
21 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
22 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
23 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
24 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
25 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
26 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
28 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31 #include <sys/cdefs.h>
34 __RCSID("$NetBSD: exec_elf32.c,v 1.4 1997/08/12 06:07:24 mikel Exp $");
37 __FBSDID("$FreeBSD$");
43 #include <sys/types.h>
44 #include <sys/endian.h>
55 #if (defined(NLIST_ELF32) && (ELFSIZE == 32)) || \
56 (defined(NLIST_ELF64) && (ELFSIZE == 64))
58 #define __ELF_WORD_SIZE ELFSIZE
60 #include <sys/elf32.h>
61 #define xewtoh(x) ((data == ELFDATA2MSB) ? be32toh(x) : le32toh(x))
62 #define htoxew(x) ((data == ELFDATA2MSB) ? htobe32(x) : htole32(x))
64 #include <sys/elf64.h>
65 #define xewtoh(x) ((data == ELFDATA2MSB) ? be64toh(x) : le64toh(x))
66 #define htoxew(x) ((data == ELFDATA2MSB) ? htobe64(x) : htole64(x))
68 #include <sys/elf_generic.h>
70 #define CONCAT(x,y) __CONCAT(x,y)
71 #define ELFNAME(x) CONCAT(elf,CONCAT(ELFSIZE,CONCAT(_,x)))
72 #define ELFNAME2(x,y) CONCAT(x,CONCAT(_elf,CONCAT(ELFSIZE,CONCAT(_,y))))
73 #define ELFNAMEEND(x) CONCAT(x,CONCAT(_elf,ELFSIZE))
74 #define ELFDEFNNAME(x) CONCAT(ELF,CONCAT(ELFSIZE,CONCAT(_,x)))
76 #define xe16toh(x) ((data == ELFDATA2MSB) ? be16toh(x) : le16toh(x))
77 #define xe32toh(x) ((data == ELFDATA2MSB) ? be32toh(x) : le32toh(x))
78 #define htoxe32(x) ((data == ELFDATA2MSB) ? htobe32(x) : htole32(x))
81 struct listelem *next;
88 xreadatoff(int fd, void *buf, off_t off, size_t size, const char *fn)
92 if (lseek(fd, off, SEEK_SET) != off) {
96 if ((rv = read(fd, buf, size)) != size) {
97 fprintf(stderr, "%s: read error: %s\n", fn,
98 rv == -1 ? strerror(errno) : "short read");
105 xwriteatoff(int fd, void *buf, off_t off, size_t size, const char *fn)
109 if (lseek(fd, off, SEEK_SET) != off) {
113 if ((rv = write(fd, buf, size)) != size) {
114 fprintf(stderr, "%s: write error: %s\n", fn,
115 rv == -1 ? strerror(errno) : "short write");
122 xmalloc(size_t size, const char *fn, const char *use)
128 fprintf(stderr, "%s: out of memory (allocating for %s)\n",
134 ELFNAMEEND(check)(int fd, const char *fn)
141 * Check the header to maek sure it's an ELF file (of the
144 if (fstat(fd, &sb) == -1)
146 if (sb.st_size < sizeof eh)
148 if (read(fd, &eh, sizeof eh) != sizeof eh)
154 data = eh.e_ident[EI_DATA];
156 switch (xe16toh(eh.e_machine)) {
158 case EM_ALPHA: break;
166 case EM_IA_64: break;
172 #define EM_SPARCV9 43
174 case EM_SPARCV9: break;
178 case EM_X86_64: break;
179 /* ELFDEFNNAME(MACHDEP_ID_CASES) */
189 ELFNAMEEND(hide)(int fd, const char *fn)
192 Elf_Shdr *shdrp = NULL, *symtabshdr, *strtabshdr;
193 Elf_Sym *symtabp = NULL;
194 char *strtabp = NULL;
195 Elf_Size *symfwmap = NULL, *symrvmap = NULL, nsyms, nlocalsyms, ewi;
196 struct listelem *relalist = NULL, *rellist = NULL, *tmpl;
202 if (xreadatoff(fd, &ehdr, 0, sizeof ehdr, fn) != sizeof ehdr)
205 data = ehdr.e_ident[EI_DATA];
207 shdrsize = xe16toh(ehdr.e_shnum) * xe16toh(ehdr.e_shentsize);
208 if ((shdrp = xmalloc(shdrsize, fn, "section header table")) == NULL)
210 if (xreadatoff(fd, shdrp, xewtoh(ehdr.e_shoff), shdrsize, fn) !=
214 symtabshdr = strtabshdr = NULL;
216 for (i = 0; i < xe16toh(ehdr.e_shnum); i++) {
217 switch (xe32toh(shdrp[i].sh_type)) {
219 if (symtabshdr != NULL)
221 symtabshdr = &shdrp[i];
222 strtabshdr = &shdrp[xe32toh(shdrp[i].sh_link)];
225 tmpl = xmalloc(sizeof *tmpl, fn, "rela list element");
229 tmpl->file = xewtoh(shdrp[i].sh_offset);
230 tmpl->size = xewtoh(shdrp[i].sh_size);
231 tmpl->next = relalist;
235 tmpl = xmalloc(sizeof *tmpl, fn, "rel list element");
239 tmpl->file = xewtoh(shdrp[i].sh_offset);
240 tmpl->size = xewtoh(shdrp[i].sh_size);
241 tmpl->next = rellist;
246 if (symtabshdr == NULL)
248 if (strtabshdr == NULL)
251 fprintf(stderr, "%s: weird executable (unsupported)\n", fn);
256 * load up everything we need
260 if ((symtabp = xmalloc(xewtoh(symtabshdr->sh_size), fn, "symbol table"))
263 if (xreadatoff(fd, symtabp, xewtoh(symtabshdr->sh_offset),
264 xewtoh(symtabshdr->sh_size), fn) != xewtoh(symtabshdr->sh_size))
268 if ((strtabp = xmalloc(xewtoh(strtabshdr->sh_size), fn, "string table"))
271 if (xreadatoff(fd, strtabp, xewtoh(strtabshdr->sh_offset),
272 xewtoh(strtabshdr->sh_size), fn) != xewtoh(strtabshdr->sh_size))
275 /* any rela tables */
276 for (tmpl = relalist; tmpl != NULL; tmpl = tmpl->next) {
277 if ((tmpl->mem = xmalloc(tmpl->size, fn, "rela table"))
280 if (xreadatoff(fd, tmpl->mem, tmpl->file,
281 tmpl->size, fn) != tmpl->size)
286 for (tmpl = rellist; tmpl != NULL; tmpl = tmpl->next) {
287 if ((tmpl->mem = xmalloc(tmpl->size, fn, "rel table"))
290 if (xreadatoff(fd, tmpl->mem, tmpl->file,
291 tmpl->size, fn) != tmpl->size)
295 /* Prepare data structures for symbol movement. */
296 nsyms = xewtoh(symtabshdr->sh_size) / xewtoh(symtabshdr->sh_entsize);
297 nlocalsyms = xe32toh(symtabshdr->sh_info);
298 if ((symfwmap = xmalloc(nsyms * sizeof (Elf_Size), fn,
299 "symbol forward mapping table")) == NULL)
301 if ((symrvmap = xmalloc(nsyms * sizeof (Elf_Size), fn,
302 "symbol reverse mapping table")) == NULL)
305 /* init location -> symbol # table */
306 for (ewi = 0; ewi < nsyms; ewi++)
309 /* move symbols, making them local */
310 for (ewi = nlocalsyms; ewi < nsyms; ewi++) {
311 Elf_Sym *sp, symswap;
316 /* if it's on our keep list, don't move it */
317 if (in_keep_list(strtabp + xe32toh(sp->st_name)))
320 /* if it's an undefined symbol, keep it */
321 if (xe16toh(sp->st_shndx) == SHN_UNDEF)
324 /* adjust the symbol so that it's local */
326 ELF_ST_INFO(STB_LOCAL, sp->st_info);
327 /* (STB_LOCAL << 4) | ELF_SYM_TYPE(sp->st_info); *//* XXX */
330 * move the symbol to its new location
333 /* note that symbols in those locations have been swapped */
334 mapswap = symrvmap[ewi];
335 symrvmap[ewi] = symrvmap[nlocalsyms];
336 symrvmap[nlocalsyms] = mapswap;
338 /* and swap the symbols */
340 *sp = symtabp[nlocalsyms];
341 symtabp[nlocalsyms] = symswap;
343 nlocalsyms++; /* note new local sym */
345 symtabshdr->sh_info = htoxe32(nlocalsyms);
347 /* set up symbol # -> location mapping table */
348 for (ewi = 0; ewi < nsyms; ewi++)
349 symfwmap[symrvmap[ewi]] = ewi;
351 /* any rela tables */
352 for (tmpl = relalist; tmpl != NULL; tmpl = tmpl->next) {
353 Elf_Rela *relap = tmpl->mem;
355 for (ewi = 0; ewi < tmpl->size / sizeof(*relap); ewi++) {
356 relap[ewi].r_info = htoxew(ELF_R_INFO(
357 symfwmap[ELF_R_SYM(xewtoh(relap[ewi].r_info))],
358 ELF_R_TYPE(xewtoh(relap[ewi].r_info))
364 for (tmpl = rellist; tmpl != NULL; tmpl = tmpl->next) {
365 Elf_Rel *relp = tmpl->mem;
367 for (ewi = 0; ewi < tmpl->size / sizeof *relp; ewi++) {
368 relp[ewi].r_info = htoxew(ELF_R_INFO(
369 symfwmap[ELF_R_SYM(xewtoh(relp[ewi].r_info))],
370 ELF_R_TYPE(xewtoh(relp[ewi].r_info))
376 * write new tables to the file
378 if (xwriteatoff(fd, shdrp, xewtoh(ehdr.e_shoff), shdrsize, fn) !=
381 if (xwriteatoff(fd, symtabp, xewtoh(symtabshdr->sh_offset),
382 xewtoh(symtabshdr->sh_size), fn) != xewtoh(symtabshdr->sh_size))
384 for (tmpl = relalist; tmpl != NULL; tmpl = tmpl->next) {
385 if (xwriteatoff(fd, tmpl->mem, tmpl->file,
386 tmpl->size, fn) != tmpl->size)
389 for (tmpl = rellist; tmpl != NULL; tmpl = tmpl->next) {
390 if (xwriteatoff(fd, tmpl->mem, tmpl->file,
391 tmpl->size, fn) != tmpl->size)
402 if (symfwmap != NULL)
404 if (symrvmap != NULL)
406 while ((tmpl = relalist) != NULL) {
407 relalist = tmpl->next;
408 if (tmpl->mem != NULL)
412 while ((tmpl = rellist) != NULL) {
413 rellist = tmpl->next;
414 if (tmpl->mem != NULL)
425 #endif /* include this size of ELF */