1 /* simple-object-elf.c -- routines to manipulate ELF object files.
2 Copyright 2010 Free Software Foundation, Inc.
3 Written by Ian Lance Taylor, Google.
4
5 This program is free software; you can redistribute it and/or modify it
6 under the terms of the GNU General Public License as published by the
7 Free Software Foundation; either version 2, or (at your option) any
8 later version.
9
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
14
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, 51 Franklin Street - Fifth Floor,
18 Boston, MA 02110-1301, USA. */
19
20 #include "config.h"
21 #include "libiberty.h"
22 #include "simple-object.h"
23
24 #include <errno.h>
25 #include <stddef.h>
26
27 #ifdef HAVE_STDLIB_H
28 #include <stdlib.h>
29 #endif
30
31 #ifdef HAVE_STDINT_H
32 #include <stdint.h>
33 #endif
34
35 #ifdef HAVE_STRING_H
36 #include <string.h>
37 #endif
38
39 #ifdef HAVE_INTTYPES_H
40 #include <inttypes.h>
41 #endif
42
43 #include "simple-object-common.h"
44
45 /* ELF structures and constants. */
46
47 /* 32-bit ELF file header. */
48
49 typedef struct {
50 unsigned char e_ident[16]; /* ELF "magic number" */
51 unsigned char e_type[2]; /* Identifies object file type */
52 unsigned char e_machine[2]; /* Specifies required architecture */
53 unsigned char e_version[4]; /* Identifies object file version */
54 unsigned char e_entry[4]; /* Entry point virtual address */
55 unsigned char e_phoff[4]; /* Program header table file offset */
56 unsigned char e_shoff[4]; /* Section header table file offset */
57 unsigned char e_flags[4]; /* Processor-specific flags */
58 unsigned char e_ehsize[2]; /* ELF header size in bytes */
59 unsigned char e_phentsize[2]; /* Program header table entry size */
60 unsigned char e_phnum[2]; /* Program header table entry count */
61 unsigned char e_shentsize[2]; /* Section header table entry size */
62 unsigned char e_shnum[2]; /* Section header table entry count */
63 unsigned char e_shstrndx[2]; /* Section header string table index */
64 } Elf32_External_Ehdr;
65
66 /* 64-bit ELF file header. */
67
68 typedef struct {
69 unsigned char e_ident[16]; /* ELF "magic number" */
70 unsigned char e_type[2]; /* Identifies object file type */
71 unsigned char e_machine[2]; /* Specifies required architecture */
72 unsigned char e_version[4]; /* Identifies object file version */
73 unsigned char e_entry[8]; /* Entry point virtual address */
74 unsigned char e_phoff[8]; /* Program header table file offset */
75 unsigned char e_shoff[8]; /* Section header table file offset */
76 unsigned char e_flags[4]; /* Processor-specific flags */
77 unsigned char e_ehsize[2]; /* ELF header size in bytes */
78 unsigned char e_phentsize[2]; /* Program header table entry size */
79 unsigned char e_phnum[2]; /* Program header table entry count */
80 unsigned char e_shentsize[2]; /* Section header table entry size */
81 unsigned char e_shnum[2]; /* Section header table entry count */
82 unsigned char e_shstrndx[2]; /* Section header string table index */
83 } Elf64_External_Ehdr;
84
85 /* Indexes and values in e_ident field of Ehdr. */
86
87 #define EI_MAG0 0 /* File identification byte 0 index */
88 #define ELFMAG0 0x7F /* Magic number byte 0 */
89
90 #define EI_MAG1 1 /* File identification byte 1 index */
91 #define ELFMAG1 'E' /* Magic number byte 1 */
92
93 #define EI_MAG2 2 /* File identification byte 2 index */
94 #define ELFMAG2 'L' /* Magic number byte 2 */
95
96 #define EI_MAG3 3 /* File identification byte 3 index */
97 #define ELFMAG3 'F' /* Magic number byte 3 */
98
99 #define EI_CLASS 4 /* File class */
100 #define ELFCLASSNONE 0 /* Invalid class */
101 #define ELFCLASS32 1 /* 32-bit objects */
102 #define ELFCLASS64 2 /* 64-bit objects */
103
104 #define EI_DATA 5 /* Data encoding */
105 #define ELFDATANONE 0 /* Invalid data encoding */
106 #define ELFDATA2LSB 1 /* 2's complement, little endian */
107 #define ELFDATA2MSB 2 /* 2's complement, big endian */
108
109 #define EI_VERSION 6 /* File version */
110 #define EV_CURRENT 1 /* Current version */
111
112 #define EI_OSABI 7 /* Operating System/ABI indication */
113
114 /* Values for e_type field of Ehdr. */
115
116 #define ET_REL 1 /* Relocatable file */
117
118 /* Values for e_machine field of Ehdr. */
119
120 #define EM_SPARC 2 /* SUN SPARC */
121 #define EM_SPARC32PLUS 18 /* Sun's "v8plus" */
122
123 /* Special section index values. */
124
125 #define SHN_LORESERVE 0xFF00 /* Begin range of reserved indices */
126 #define SHN_XINDEX 0xFFFF /* Section index is held elsewhere */
127
128 /* 32-bit ELF program header. */
129
130 typedef struct {
131 unsigned char p_type[4]; /* Identifies program segment type */
132 unsigned char p_offset[4]; /* Segment file offset */
133 unsigned char p_vaddr[4]; /* Segment virtual address */
134 unsigned char p_paddr[4]; /* Segment physical address */
135 unsigned char p_filesz[4]; /* Segment size in file */
136 unsigned char p_memsz[4]; /* Segment size in memory */
137 unsigned char p_flags[4]; /* Segment flags */
138 unsigned char p_align[4]; /* Segment alignment, file & memory */
139 } Elf32_External_Phdr;
140
141 /* 64-bit ELF program header. */
142
143 typedef struct {
144 unsigned char p_type[4]; /* Identifies program segment type */
145 unsigned char p_flags[4]; /* Segment flags */
146 unsigned char p_offset[8]; /* Segment file offset */
147 unsigned char p_vaddr[8]; /* Segment virtual address */
148 unsigned char p_paddr[8]; /* Segment physical address */
149 unsigned char p_filesz[8]; /* Segment size in file */
150 unsigned char p_memsz[8]; /* Segment size in memory */
151 unsigned char p_align[8]; /* Segment alignment, file & memory */
152 } Elf64_External_Phdr;
153
154 /* 32-bit ELF section header */
155
156 typedef struct {
157 unsigned char sh_name[4]; /* Section name, index in string tbl */
158 unsigned char sh_type[4]; /* Type of section */
159 unsigned char sh_flags[4]; /* Miscellaneous section attributes */
160 unsigned char sh_addr[4]; /* Section virtual addr at execution */
161 unsigned char sh_offset[4]; /* Section file offset */
162 unsigned char sh_size[4]; /* Size of section in bytes */
163 unsigned char sh_link[4]; /* Index of another section */
164 unsigned char sh_info[4]; /* Additional section information */
165 unsigned char sh_addralign[4]; /* Section alignment */
166 unsigned char sh_entsize[4]; /* Entry size if section holds table */
167 } Elf32_External_Shdr;
168
169 /* 64-bit ELF section header. */
170
171 typedef struct {
172 unsigned char sh_name[4]; /* Section name, index in string tbl */
173 unsigned char sh_type[4]; /* Type of section */
174 unsigned char sh_flags[8]; /* Miscellaneous section attributes */
175 unsigned char sh_addr[8]; /* Section virtual addr at execution */
176 unsigned char sh_offset[8]; /* Section file offset */
177 unsigned char sh_size[8]; /* Size of section in bytes */
178 unsigned char sh_link[4]; /* Index of another section */
179 unsigned char sh_info[4]; /* Additional section information */
180 unsigned char sh_addralign[8]; /* Section alignment */
181 unsigned char sh_entsize[8]; /* Entry size if section holds table */
182 } Elf64_External_Shdr;
183
184 /* Values for sh_type field. */
185
186 #define SHT_PROGBITS 1 /* Program data */
187 #define SHT_STRTAB 3 /* A string table */
188
189 /* Functions to fetch and store different ELF types, depending on the
190 endianness and size. */
191
192 struct elf_type_functions
193 {
194 unsigned short (*fetch_Elf_Half) (const unsigned char *);
195 unsigned int (*fetch_Elf_Word) (const unsigned char *);
196 ulong_type (*fetch_Elf_Addr) (const unsigned char *);
197 void (*set_Elf_Half) (unsigned char *, unsigned short);
198 void (*set_Elf_Word) (unsigned char *, unsigned int);
199 void (*set_Elf_Addr) (unsigned char *, ulong_type);
200 };
201
202 static const struct elf_type_functions elf_big_32_functions =
203 {
204 simple_object_fetch_big_16,
205 simple_object_fetch_big_32,
206 simple_object_fetch_big_32_ulong,
207 simple_object_set_big_16,
208 simple_object_set_big_32,
209 simple_object_set_big_32_ulong
210 };
211
212 static const struct elf_type_functions elf_little_32_functions =
213 {
214 simple_object_fetch_little_16,
215 simple_object_fetch_little_32,
216 simple_object_fetch_little_32_ulong,
217 simple_object_set_little_16,
218 simple_object_set_little_32,
219 simple_object_set_little_32_ulong
220 };
221
222 #ifdef UNSIGNED_64BIT_TYPE
223
224 static const struct elf_type_functions elf_big_64_functions =
225 {
226 simple_object_fetch_big_16,
227 simple_object_fetch_big_32,
228 simple_object_fetch_big_64,
229 simple_object_set_big_16,
230 simple_object_set_big_32,
231 simple_object_set_big_64
232 };
233
234 static const struct elf_type_functions elf_little_64_functions =
235 {
236 simple_object_fetch_little_16,
237 simple_object_fetch_little_32,
238 simple_object_fetch_little_64,
239 simple_object_set_little_16,
240 simple_object_set_little_32,
241 simple_object_set_little_64
242 };
243
244 #endif
245
246 /* Hideous macro to fetch the value of a field from an external ELF
247 struct of some sort. TYPEFUNCS is the set of type functions.
248 BUFFER points to the external data. STRUCTTYPE is the appropriate
249 struct type. FIELD is a field within the struct. TYPE is the type
250 of the field in the struct: Elf_Half, Elf_Word, or Elf_Addr. */
251
252 #define ELF_FETCH_STRUCT_FIELD(TYPEFUNCS, STRUCTTYPE, FIELD, BUFFER, TYPE) \
253 ((TYPEFUNCS)->fetch_ ## TYPE ((BUFFER) + offsetof (STRUCTTYPE, FIELD)))
254
255 /* Even more hideous macro to fetch the value of FIELD from BUFFER.
256 SIZE is 32 or 64. STRUCTTYPE is the name of the struct from
257 elf/external.h: Ehdr, Shdr, etc. FIELD is the name of a field in
258 the struct. TYPE is the type of the field in the struct: Elf_Half,
259 Elf_Word, or Elf_Addr. */
260
261 #define ELF_FETCH_SIZED_FIELD(TYPEFUNCS, SIZE, STRUCTTYPE, BUFFER, \
262 FIELD, TYPE) \
263 ELF_FETCH_STRUCT_FIELD (TYPEFUNCS, \
264 Elf ## SIZE ## _External_ ## STRUCTTYPE, \
265 FIELD, BUFFER, TYPE)
266
267 /* Like ELF_FETCH_SIZED_FIELD but taking an ELFCLASS value. */
268
269 #define ELF_FETCH_FIELD(TYPEFUNCS, CLASS, STRUCTTYPE, BUFFER, \
270 FIELD, TYPE) \
271 ((CLASS) == ELFCLASS32 \
272 ? ELF_FETCH_SIZED_FIELD (TYPEFUNCS, 32, STRUCTTYPE, BUFFER, FIELD, \
273 TYPE) \
274 : ELF_FETCH_SIZED_FIELD (TYPEFUNCS, 64, STRUCTTYPE, BUFFER, FIELD, \
275 TYPE))
276
277 /* Hideous macro to set the value of a field in an external ELF
278 structure to VAL. TYPEFUNCS is the set of type functions. BUFFER
279 points to the external data. STRUCTTYPE is the appropriate
280 structure type. FIELD is a field within the struct. TYPE is the
281 type of the field in the struct: Elf_Half, Elf_Word, or
282 Elf_Addr. */
283
284 #define ELF_SET_STRUCT_FIELD(TYPEFUNCS, STRUCTTYPE, FIELD, BUFFER, TYPE, VAL) \
285 (TYPEFUNCS)->set_ ## TYPE ((BUFFER) + offsetof (STRUCTTYPE, FIELD), (VAL))
286
287 /* Even more hideous macro to set the value of FIELD in BUFFER to VAL.
288 SIZE is 32 or 64. STRUCTTYPE is the name of the struct from
289 elf/external.h: Ehdr, Shdr, etc. FIELD is the name of a field in
290 the struct. TYPE is the type of the field in the struct: Elf_Half,
291 Elf_Word, or Elf_Addr. */
292
293 #define ELF_SET_SIZED_FIELD(TYPEFUNCS, SIZE, STRUCTTYPE, BUFFER, FIELD, \
294 TYPE, VAL) \
295 ELF_SET_STRUCT_FIELD (TYPEFUNCS, \
296 Elf ## SIZE ## _External_ ## STRUCTTYPE, \
297 FIELD, BUFFER, TYPE, VAL)
298
299 /* Like ELF_SET_SIZED_FIELD but taking an ELFCLASS value. */
300
301 #define ELF_SET_FIELD(TYPEFUNCS, CLASS, STRUCTTYPE, BUFFER, FIELD, \
302 TYPE, VAL) \
303 ((CLASS) == ELFCLASS32 \
304 ? ELF_SET_SIZED_FIELD (TYPEFUNCS, 32, STRUCTTYPE, BUFFER, FIELD, \
305 TYPE, VAL) \
306 : ELF_SET_SIZED_FIELD (TYPEFUNCS, 64, STRUCTTYPE, BUFFER, FIELD, \
307 TYPE, VAL))
308
309 /* Private data for an simple_object_read. */
310
311 struct simple_object_elf_read
312 {
313 /* Type functions. */
314 const struct elf_type_functions* type_functions;
315 /* Elf data. */
316 unsigned char ei_data;
317 /* Elf class. */
318 unsigned char ei_class;
319 /* ELF OS ABI. */
320 unsigned char ei_osabi;
321 /* Elf machine number. */
322 unsigned short machine;
323 /* Processor specific flags. */
324 unsigned int flags;
325 /* File offset of section headers. */
326 ulong_type shoff;
327 /* Number of sections. */
328 unsigned int shnum;
329 /* Index of string table section header. */
330 unsigned int shstrndx;
331 };
332
333 /* Private data for an simple_object_attributes. */
334
335 struct simple_object_elf_attributes
336 {
337 /* Type functions. */
338 const struct elf_type_functions* type_functions;
339 /* Elf data. */
340 unsigned char ei_data;
341 /* Elf class. */
342 unsigned char ei_class;
343 /* ELF OS ABI. */
344 unsigned char ei_osabi;
345 /* Elf machine number. */
346 unsigned short machine;
347 /* Processor specific flags. */
348 unsigned int flags;
349 };
350
351 /* See if we have an ELF file. */
352
353 static void *
simple_object_elf_match(unsigned char header[SIMPLE_OBJECT_MATCH_HEADER_LEN],int descriptor,off_t offset,const char * segment_name ATTRIBUTE_UNUSED,const char ** errmsg,int * err)354 simple_object_elf_match (unsigned char header[SIMPLE_OBJECT_MATCH_HEADER_LEN],
355 int descriptor, off_t offset,
356 const char *segment_name ATTRIBUTE_UNUSED,
357 const char **errmsg, int *err)
358 {
359 unsigned char ei_data;
360 unsigned char ei_class;
361 const struct elf_type_functions *type_functions;
362 unsigned char ehdr[sizeof (Elf64_External_Ehdr)];
363 struct simple_object_elf_read *eor;
364
365 if (header[EI_MAG0] != ELFMAG0
366 || header[EI_MAG1] != ELFMAG1
367 || header[EI_MAG2] != ELFMAG2
368 || header[EI_MAG3] != ELFMAG3
369 || header[EI_VERSION] != EV_CURRENT)
370 {
371 *errmsg = NULL;
372 *err = 0;
373 return NULL;
374 }
375
376 ei_data = header[EI_DATA];
377 if (ei_data != ELFDATA2LSB && ei_data != ELFDATA2MSB)
378 {
379 *errmsg = "unknown ELF endianness";
380 *err = 0;
381 return NULL;
382 }
383
384 ei_class = header[EI_CLASS];
385 switch (ei_class)
386 {
387 case ELFCLASS32:
388 type_functions = (ei_data == ELFDATA2LSB
389 ? &elf_little_32_functions
390 : &elf_big_32_functions);
391 break;
392
393 case ELFCLASS64:
394 #ifndef UNSIGNED_64BIT_TYPE
395 *errmsg = "64-bit ELF objects not supported";
396 *err = 0;
397 return NULL;
398 #else
399 type_functions = (ei_data == ELFDATA2LSB
400 ? &elf_little_64_functions
401 : &elf_big_64_functions);
402 break;
403 #endif
404
405 default:
406 *errmsg = "unrecognized ELF size";
407 *err = 0;
408 return NULL;
409 }
410
411 if (!simple_object_internal_read (descriptor, offset, ehdr, sizeof ehdr,
412 errmsg, err))
413 return NULL;
414
415 eor = XNEW (struct simple_object_elf_read);
416 eor->type_functions = type_functions;
417 eor->ei_data = ei_data;
418 eor->ei_class = ei_class;
419 eor->ei_osabi = header[EI_OSABI];
420 eor->machine = ELF_FETCH_FIELD (type_functions, ei_class, Ehdr, ehdr,
421 e_machine, Elf_Half);
422 eor->flags = ELF_FETCH_FIELD (type_functions, ei_class, Ehdr, ehdr,
423 e_flags, Elf_Word);
424 eor->shoff = ELF_FETCH_FIELD (type_functions, ei_class, Ehdr, ehdr,
425 e_shoff, Elf_Addr);
426 eor->shnum = ELF_FETCH_FIELD (type_functions, ei_class, Ehdr, ehdr,
427 e_shnum, Elf_Half);
428 eor->shstrndx = ELF_FETCH_FIELD (type_functions, ei_class, Ehdr, ehdr,
429 e_shstrndx, Elf_Half);
430
431 if ((eor->shnum == 0 || eor->shstrndx == SHN_XINDEX)
432 && eor->shoff != 0)
433 {
434 unsigned char shdr[sizeof (Elf64_External_Shdr)];
435
436 /* Object file has more than 0xffff sections. */
437
438 if (!simple_object_internal_read (descriptor, offset + eor->shoff, shdr,
439 (ei_class == ELFCLASS32
440 ? sizeof (Elf32_External_Shdr)
441 : sizeof (Elf64_External_Shdr)),
442 errmsg, err))
443 {
444 XDELETE (eor);
445 return NULL;
446 }
447
448 if (eor->shnum == 0)
449 eor->shnum = ELF_FETCH_FIELD (type_functions, ei_class, Shdr,
450 shdr, sh_size, Elf_Addr);
451
452 if (eor->shstrndx == SHN_XINDEX)
453 {
454 eor->shstrndx = ELF_FETCH_FIELD (type_functions, ei_class, Shdr,
455 shdr, sh_link, Elf_Word);
456
457 /* Versions of the GNU binutils between 2.12 and 2.18 did
458 not handle objects with more than SHN_LORESERVE sections
459 correctly. All large section indexes were offset by
460 0x100. There is more information at
461 http://sourceware.org/bugzilla/show_bug.cgi?id-5900 .
462 Fortunately these object files are easy to detect, as the
463 GNU binutils always put the section header string table
464 near the end of the list of sections. Thus if the
465 section header string table index is larger than the
466 number of sections, then we know we have to subtract
467 0x100 to get the real section index. */
468 if (eor->shstrndx >= eor->shnum
469 && eor->shstrndx >= SHN_LORESERVE + 0x100)
470 eor->shstrndx -= 0x100;
471 }
472 }
473
474 if (eor->shstrndx >= eor->shnum)
475 {
476 *errmsg = "invalid ELF shstrndx >= shnum";
477 *err = 0;
478 XDELETE (eor);
479 return NULL;
480 }
481
482 return (void *) eor;
483 }
484
485 /* Find all sections in an ELF file. */
486
487 static const char *
simple_object_elf_find_sections(simple_object_read * sobj,int (* pfn)(void *,const char *,off_t offset,off_t length),void * data,int * err)488 simple_object_elf_find_sections (simple_object_read *sobj,
489 int (*pfn) (void *, const char *,
490 off_t offset, off_t length),
491 void *data,
492 int *err)
493 {
494 struct simple_object_elf_read *eor =
495 (struct simple_object_elf_read *) sobj->data;
496 const struct elf_type_functions *type_functions = eor->type_functions;
497 unsigned char ei_class = eor->ei_class;
498 size_t shdr_size;
499 unsigned int shnum;
500 unsigned char *shdrs;
501 const char *errmsg;
502 unsigned char *shstrhdr;
503 size_t name_size;
504 off_t shstroff;
505 unsigned char *names;
506 unsigned int i;
507
508 shdr_size = (ei_class == ELFCLASS32
509 ? sizeof (Elf32_External_Shdr)
510 : sizeof (Elf64_External_Shdr));
511
512 /* Read the section headers. We skip section 0, which is not a
513 useful section. */
514
515 shnum = eor->shnum;
516 shdrs = XNEWVEC (unsigned char, shdr_size * (shnum - 1));
517
518 if (!simple_object_internal_read (sobj->descriptor,
519 sobj->offset + eor->shoff + shdr_size,
520 shdrs,
521 shdr_size * (shnum - 1),
522 &errmsg, err))
523 {
524 XDELETEVEC (shdrs);
525 return errmsg;
526 }
527
528 /* Read the section names. */
529
530 shstrhdr = shdrs + (eor->shstrndx - 1) * shdr_size;
531 name_size = ELF_FETCH_FIELD (type_functions, ei_class, Shdr,
532 shstrhdr, sh_size, Elf_Addr);
533 shstroff = ELF_FETCH_FIELD (type_functions, ei_class, Shdr,
534 shstrhdr, sh_offset, Elf_Addr);
535 names = XNEWVEC (unsigned char, name_size);
536 if (!simple_object_internal_read (sobj->descriptor,
537 sobj->offset + shstroff,
538 names, name_size, &errmsg, err))
539 {
540 XDELETEVEC (names);
541 XDELETEVEC (shdrs);
542 return errmsg;
543 }
544
545 for (i = 1; i < shnum; ++i)
546 {
547 unsigned char *shdr;
548 unsigned int sh_name;
549 const char *name;
550 off_t offset;
551 off_t length;
552
553 shdr = shdrs + (i - 1) * shdr_size;
554 sh_name = ELF_FETCH_FIELD (type_functions, ei_class, Shdr,
555 shdr, sh_name, Elf_Word);
556 if (sh_name >= name_size)
557 {
558 *err = 0;
559 XDELETEVEC (names);
560 XDELETEVEC (shdrs);
561 return "ELF section name out of range";
562 }
563
564 name = (const char *) names + sh_name;
565 offset = ELF_FETCH_FIELD (type_functions, ei_class, Shdr,
566 shdr, sh_offset, Elf_Addr);
567 length = ELF_FETCH_FIELD (type_functions, ei_class, Shdr,
568 shdr, sh_size, Elf_Addr);
569
570 if (!(*pfn) (data, name, offset, length))
571 break;
572 }
573
574 XDELETEVEC (names);
575 XDELETEVEC (shdrs);
576
577 return NULL;
578 }
579
580 /* Fetch the attributes for an simple_object_read. */
581
582 static void *
simple_object_elf_fetch_attributes(simple_object_read * sobj,const char ** errmsg ATTRIBUTE_UNUSED,int * err ATTRIBUTE_UNUSED)583 simple_object_elf_fetch_attributes (simple_object_read *sobj,
584 const char **errmsg ATTRIBUTE_UNUSED,
585 int *err ATTRIBUTE_UNUSED)
586 {
587 struct simple_object_elf_read *eor =
588 (struct simple_object_elf_read *) sobj->data;
589 struct simple_object_elf_attributes *ret;
590
591 ret = XNEW (struct simple_object_elf_attributes);
592 ret->type_functions = eor->type_functions;
593 ret->ei_data = eor->ei_data;
594 ret->ei_class = eor->ei_class;
595 ret->ei_osabi = eor->ei_osabi;
596 ret->machine = eor->machine;
597 ret->flags = eor->flags;
598 return ret;
599 }
600
601 /* Release the privata data for an simple_object_read. */
602
603 static void
simple_object_elf_release_read(void * data)604 simple_object_elf_release_read (void *data)
605 {
606 XDELETE (data);
607 }
608
609 /* Compare two attributes structures. */
610
611 static const char *
simple_object_elf_attributes_merge(void * todata,void * fromdata,int * err)612 simple_object_elf_attributes_merge (void *todata, void *fromdata, int *err)
613 {
614 struct simple_object_elf_attributes *to =
615 (struct simple_object_elf_attributes *) todata;
616 struct simple_object_elf_attributes *from =
617 (struct simple_object_elf_attributes *) fromdata;
618
619 if (to->ei_data != from->ei_data || to->ei_class != from->ei_class)
620 {
621 *err = 0;
622 return "ELF object format mismatch";
623 }
624
625 if (to->machine != from->machine)
626 {
627 int ok;
628
629 /* EM_SPARC and EM_SPARC32PLUS are compatible and force an
630 output of EM_SPARC32PLUS. */
631 ok = 0;
632 switch (to->machine)
633 {
634 case EM_SPARC:
635 if (from->machine == EM_SPARC32PLUS)
636 {
637 to->machine = from->machine;
638 ok = 1;
639 }
640 break;
641
642 case EM_SPARC32PLUS:
643 if (from->machine == EM_SPARC)
644 ok = 1;
645 break;
646
647 default:
648 break;
649 }
650
651 if (!ok)
652 {
653 *err = 0;
654 return "ELF machine number mismatch";
655 }
656 }
657
658 return NULL;
659 }
660
661 /* Release the private data for an attributes structure. */
662
663 static void
simple_object_elf_release_attributes(void * data)664 simple_object_elf_release_attributes (void *data)
665 {
666 XDELETE (data);
667 }
668
669 /* Prepare to write out a file. */
670
671 static void *
simple_object_elf_start_write(void * attributes_data,const char ** errmsg ATTRIBUTE_UNUSED,int * err ATTRIBUTE_UNUSED)672 simple_object_elf_start_write (void *attributes_data,
673 const char **errmsg ATTRIBUTE_UNUSED,
674 int *err ATTRIBUTE_UNUSED)
675 {
676 struct simple_object_elf_attributes *attrs =
677 (struct simple_object_elf_attributes *) attributes_data;
678 struct simple_object_elf_attributes *ret;
679
680 /* We're just going to record the attributes, but we need to make a
681 copy because the user may delete them. */
682 ret = XNEW (struct simple_object_elf_attributes);
683 *ret = *attrs;
684 return ret;
685 }
686
687 /* Write out an ELF ehdr. */
688
689 static int
simple_object_elf_write_ehdr(simple_object_write * sobj,int descriptor,const char ** errmsg,int * err)690 simple_object_elf_write_ehdr (simple_object_write *sobj, int descriptor,
691 const char **errmsg, int *err)
692 {
693 struct simple_object_elf_attributes *attrs =
694 (struct simple_object_elf_attributes *) sobj->data;
695 const struct elf_type_functions* fns;
696 unsigned char cl;
697 size_t ehdr_size;
698 unsigned char buf[sizeof (Elf64_External_Ehdr)];
699 simple_object_write_section *section;
700 unsigned int shnum;
701 unsigned int shstrndx;
702
703 fns = attrs->type_functions;
704 cl = attrs->ei_class;
705
706 shnum = 0;
707 for (section = sobj->sections; section != NULL; section = section->next)
708 ++shnum;
709 if (shnum > 0)
710 {
711 /* Add a section header for the dummy section and one for
712 .shstrtab. */
713 shnum += 2;
714 }
715
716 ehdr_size = (cl == ELFCLASS32
717 ? sizeof (Elf32_External_Ehdr)
718 : sizeof (Elf64_External_Ehdr));
719 memset (buf, 0, sizeof (Elf64_External_Ehdr));
720
721 buf[EI_MAG0] = ELFMAG0;
722 buf[EI_MAG1] = ELFMAG1;
723 buf[EI_MAG2] = ELFMAG2;
724 buf[EI_MAG3] = ELFMAG3;
725 buf[EI_CLASS] = cl;
726 buf[EI_DATA] = attrs->ei_data;
727 buf[EI_VERSION] = EV_CURRENT;
728 buf[EI_OSABI] = attrs->ei_osabi;
729
730 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_type, Elf_Half, ET_REL);
731 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_machine, Elf_Half, attrs->machine);
732 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_version, Elf_Word, EV_CURRENT);
733 /* e_entry left as zero. */
734 /* e_phoff left as zero. */
735 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_shoff, Elf_Addr, ehdr_size);
736 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_flags, Elf_Word, attrs->flags);
737 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_ehsize, Elf_Half, ehdr_size);
738 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_phentsize, Elf_Half,
739 (cl == ELFCLASS32
740 ? sizeof (Elf32_External_Phdr)
741 : sizeof (Elf64_External_Phdr)));
742 /* e_phnum left as zero. */
743 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_shentsize, Elf_Half,
744 (cl == ELFCLASS32
745 ? sizeof (Elf32_External_Shdr)
746 : sizeof (Elf64_External_Shdr)));
747 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_shnum, Elf_Half,
748 shnum >= SHN_LORESERVE ? 0 : shnum);
749 if (shnum == 0)
750 shstrndx = 0;
751 else
752 {
753 shstrndx = shnum - 1;
754 if (shstrndx >= SHN_LORESERVE)
755 shstrndx = SHN_XINDEX;
756 }
757 ELF_SET_FIELD (fns, cl, Ehdr, buf, e_shstrndx, Elf_Half, shstrndx);
758
759 return simple_object_internal_write (descriptor, 0, buf, ehdr_size,
760 errmsg, err);
761 }
762
763 /* Write out an ELF shdr. */
764
765 static int
simple_object_elf_write_shdr(simple_object_write * sobj,int descriptor,off_t offset,unsigned int sh_name,unsigned int sh_type,unsigned int sh_flags,unsigned int sh_offset,unsigned int sh_size,unsigned int sh_link,unsigned int sh_addralign,const char ** errmsg,int * err)766 simple_object_elf_write_shdr (simple_object_write *sobj, int descriptor,
767 off_t offset, unsigned int sh_name,
768 unsigned int sh_type, unsigned int sh_flags,
769 unsigned int sh_offset, unsigned int sh_size,
770 unsigned int sh_link, unsigned int sh_addralign,
771 const char **errmsg, int *err)
772 {
773 struct simple_object_elf_attributes *attrs =
774 (struct simple_object_elf_attributes *) sobj->data;
775 const struct elf_type_functions* fns;
776 unsigned char cl;
777 size_t shdr_size;
778 unsigned char buf[sizeof (Elf64_External_Shdr)];
779
780 fns = attrs->type_functions;
781 cl = attrs->ei_class;
782
783 shdr_size = (cl == ELFCLASS32
784 ? sizeof (Elf32_External_Shdr)
785 : sizeof (Elf64_External_Shdr));
786 memset (buf, 0, sizeof (Elf64_External_Shdr));
787
788 ELF_SET_FIELD (fns, cl, Shdr, buf, sh_name, Elf_Word, sh_name);
789 ELF_SET_FIELD (fns, cl, Shdr, buf, sh_type, Elf_Word, sh_type);
790 ELF_SET_FIELD (fns, cl, Shdr, buf, sh_flags, Elf_Addr, sh_flags);
791 ELF_SET_FIELD (fns, cl, Shdr, buf, sh_offset, Elf_Addr, sh_offset);
792 ELF_SET_FIELD (fns, cl, Shdr, buf, sh_size, Elf_Addr, sh_size);
793 ELF_SET_FIELD (fns, cl, Shdr, buf, sh_link, Elf_Word, sh_link);
794 /* sh_info left as zero. */
795 ELF_SET_FIELD (fns, cl, Shdr, buf, sh_addralign, Elf_Addr, sh_addralign);
796 /* sh_entsize left as zero. */
797
798 return simple_object_internal_write (descriptor, offset, buf, shdr_size,
799 errmsg, err);
800 }
801
802 /* Write out a complete ELF file.
803 Ehdr
804 initial dummy Shdr
805 user-created Shdrs
806 .shstrtab Shdr
807 user-created section data
808 .shstrtab data */
809
810 static const char *
simple_object_elf_write_to_file(simple_object_write * sobj,int descriptor,int * err)811 simple_object_elf_write_to_file (simple_object_write *sobj, int descriptor,
812 int *err)
813 {
814 struct simple_object_elf_attributes *attrs =
815 (struct simple_object_elf_attributes *) sobj->data;
816 unsigned char cl;
817 size_t ehdr_size;
818 size_t shdr_size;
819 const char *errmsg;
820 simple_object_write_section *section;
821 unsigned int shnum;
822 size_t shdr_offset;
823 size_t sh_offset;
824 unsigned int first_sh_size;
825 unsigned int first_sh_link;
826 size_t sh_name;
827 unsigned char zero;
828
829 if (!simple_object_elf_write_ehdr (sobj, descriptor, &errmsg, err))
830 return errmsg;
831
832 cl = attrs->ei_class;
833 if (cl == ELFCLASS32)
834 {
835 ehdr_size = sizeof (Elf32_External_Ehdr);
836 shdr_size = sizeof (Elf32_External_Shdr);
837 }
838 else
839 {
840 ehdr_size = sizeof (Elf64_External_Ehdr);
841 shdr_size = sizeof (Elf64_External_Shdr);
842 }
843
844 shnum = 0;
845 for (section = sobj->sections; section != NULL; section = section->next)
846 ++shnum;
847 if (shnum == 0)
848 return NULL;
849
850 /* Add initial dummy Shdr and .shstrtab. */
851 shnum += 2;
852
853 shdr_offset = ehdr_size;
854 sh_offset = shdr_offset + shnum * shdr_size;
855
856 if (shnum < SHN_LORESERVE)
857 first_sh_size = 0;
858 else
859 first_sh_size = shnum;
860 if (shnum - 1 < SHN_LORESERVE)
861 first_sh_link = 0;
862 else
863 first_sh_link = shnum - 1;
864 if (!simple_object_elf_write_shdr (sobj, descriptor, shdr_offset,
865 0, 0, 0, 0, first_sh_size, first_sh_link,
866 0, &errmsg, err))
867 return errmsg;
868
869 shdr_offset += shdr_size;
870
871 sh_name = 1;
872 for (section = sobj->sections; section != NULL; section = section->next)
873 {
874 size_t mask;
875 size_t new_sh_offset;
876 size_t sh_size;
877 struct simple_object_write_section_buffer *buffer;
878
879 mask = (1U << section->align) - 1;
880 new_sh_offset = sh_offset + mask;
881 new_sh_offset &= ~ mask;
882 while (new_sh_offset > sh_offset)
883 {
884 unsigned char zeroes[16];
885 size_t write;
886
887 memset (zeroes, 0, sizeof zeroes);
888 write = new_sh_offset - sh_offset;
889 if (write > sizeof zeroes)
890 write = sizeof zeroes;
891 if (!simple_object_internal_write (descriptor, sh_offset, zeroes,
892 write, &errmsg, err))
893 return errmsg;
894 sh_offset += write;
895 }
896
897 sh_size = 0;
898 for (buffer = section->buffers; buffer != NULL; buffer = buffer->next)
899 {
900 if (!simple_object_internal_write (descriptor, sh_offset + sh_size,
901 ((const unsigned char *)
902 buffer->buffer),
903 buffer->size, &errmsg, err))
904 return errmsg;
905 sh_size += buffer->size;
906 }
907
908 if (!simple_object_elf_write_shdr (sobj, descriptor, shdr_offset,
909 sh_name, SHT_PROGBITS, 0, sh_offset,
910 sh_size, 0, 1U << section->align,
911 &errmsg, err))
912 return errmsg;
913
914 shdr_offset += shdr_size;
915 sh_name += strlen (section->name) + 1;
916 sh_offset += sh_size;
917 }
918
919 if (!simple_object_elf_write_shdr (sobj, descriptor, shdr_offset,
920 sh_name, SHT_STRTAB, 0, sh_offset,
921 sh_name + strlen (".shstrtab") + 1, 0,
922 1, &errmsg, err))
923 return errmsg;
924
925 /* .shstrtab has a leading zero byte. */
926 zero = 0;
927 if (!simple_object_internal_write (descriptor, sh_offset, &zero, 1,
928 &errmsg, err))
929 return errmsg;
930 ++sh_offset;
931
932 for (section = sobj->sections; section != NULL; section = section->next)
933 {
934 size_t len;
935
936 len = strlen (section->name) + 1;
937 if (!simple_object_internal_write (descriptor, sh_offset,
938 (const unsigned char *) section->name,
939 len, &errmsg, err))
940 return errmsg;
941 sh_offset += len;
942 }
943
944 if (!simple_object_internal_write (descriptor, sh_offset,
945 (const unsigned char *) ".shstrtab",
946 strlen (".shstrtab") + 1, &errmsg, err))
947 return errmsg;
948
949 return NULL;
950 }
951
952 /* Release the private data for an simple_object_write structure. */
953
954 static void
simple_object_elf_release_write(void * data)955 simple_object_elf_release_write (void *data)
956 {
957 XDELETE (data);
958 }
959
960 /* The ELF functions. */
961
962 const struct simple_object_functions simple_object_elf_functions =
963 {
964 simple_object_elf_match,
965 simple_object_elf_find_sections,
966 simple_object_elf_fetch_attributes,
967 simple_object_elf_release_read,
968 simple_object_elf_attributes_merge,
969 simple_object_elf_release_attributes,
970 simple_object_elf_start_write,
971 simple_object_elf_write_to_file,
972 simple_object_elf_release_write
973 };
974