xref: /freebsd/contrib/elftoolchain/elfcopy/sections.c (revision d0ba1baed3f6e4936a0c1b89c25f6c59168ef6de)
1 /*-
2  * Copyright (c) 2007-2011,2014 Kai Wang
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions
7  * are met:
8  * 1. Redistributions of source code must retain the above copyright
9  *    notice, this list of conditions and the following disclaimer.
10  * 2. Redistributions in binary form must reproduce the above copyright
11  *    notice, this list of conditions and the following disclaimer in the
12  *    documentation and/or other materials provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24  * SUCH DAMAGE.
25  */
26 
27 #include <sys/param.h>
28 #include <sys/stat.h>
29 #include <err.h>
30 #include <libgen.h>
31 #include <stdio.h>
32 #include <stdlib.h>
33 #include <string.h>
34 
35 #include "elfcopy.h"
36 
37 ELFTC_VCSID("$Id: sections.c 3443 2016-04-15 18:57:54Z kaiwang27 $");
38 
39 static void	add_gnu_debuglink(struct elfcopy *ecp);
40 static uint32_t calc_crc32(const char *p, size_t len, uint32_t crc);
41 static void	check_section_rename(struct elfcopy *ecp, struct section *s);
42 static void	filter_reloc(struct elfcopy *ecp, struct section *s);
43 static int	get_section_flags(struct elfcopy *ecp, const char *name);
44 static void	insert_sections(struct elfcopy *ecp);
45 static void	insert_to_strtab(struct section *t, const char *s);
46 static int	is_append_section(struct elfcopy *ecp, const char *name);
47 static int	is_compress_section(struct elfcopy *ecp, const char *name);
48 static int	is_debug_section(const char *name);
49 static int	is_dwo_section(const char *name);
50 static int	is_modify_section(struct elfcopy *ecp, const char *name);
51 static int	is_print_section(struct elfcopy *ecp, const char *name);
52 static int	lookup_string(struct section *t, const char *s);
53 static void	modify_section(struct elfcopy *ecp, struct section *s);
54 static void	pad_section(struct elfcopy *ecp, struct section *s);
55 static void	print_data(const char *d, size_t sz);
56 static void	print_section(struct section *s);
57 static void	*read_section(struct section *s, size_t *size);
58 static void	update_reloc(struct elfcopy *ecp, struct section *s);
59 static void	update_section_group(struct elfcopy *ecp, struct section *s);
60 
61 int
62 is_remove_section(struct elfcopy *ecp, const char *name)
63 {
64 
65 	/* Always keep section name table */
66 	if (strcmp(name, ".shstrtab") == 0)
67 		return 0;
68 	if (strcmp(name, ".symtab") == 0 ||
69 	    strcmp(name, ".strtab") == 0) {
70 		if (ecp->strip == STRIP_ALL && lookup_symop_list(
71 		    ecp, NULL, SYMOP_KEEP) == NULL)
72 			return (1);
73 		else
74 			return (0);
75 	}
76 
77 	if (ecp->strip == STRIP_DWO && is_dwo_section(name))
78 		return (1);
79 	if (ecp->strip == STRIP_NONDWO && !is_dwo_section(name))
80 		return (1);
81 
82 	if (is_debug_section(name)) {
83 		if (ecp->strip == STRIP_ALL ||
84 		    ecp->strip == STRIP_DEBUG ||
85 		    ecp->strip == STRIP_UNNEEDED ||
86 		    (ecp->flags & DISCARD_LOCAL))
87 			return (1);
88 		if (ecp->strip == STRIP_NONDEBUG)
89 			return (0);
90 	}
91 
92 	if ((ecp->flags & SEC_REMOVE) || (ecp->flags & SEC_COPY)) {
93 		struct sec_action *sac;
94 
95 		sac = lookup_sec_act(ecp, name, 0);
96 		if ((ecp->flags & SEC_REMOVE) && sac != NULL && sac->remove)
97 			return (1);
98 		if ((ecp->flags & SEC_COPY) && (sac == NULL || !sac->copy))
99 			return (1);
100 	}
101 
102 	return (0);
103 }
104 
105 /*
106  * Relocation section needs to be removed if the section it applies to
107  * will be removed.
108  */
109 int
110 is_remove_reloc_sec(struct elfcopy *ecp, uint32_t sh_info)
111 {
112 	const char	*name;
113 	GElf_Shdr	 ish;
114 	Elf_Scn		*is;
115 	size_t		 indx;
116 	int		 elferr;
117 
118 	if (elf_getshstrndx(ecp->ein, &indx) == 0)
119 		errx(EXIT_FAILURE, "elf_getshstrndx failed: %s",
120 		    elf_errmsg(-1));
121 
122 	is = NULL;
123 	while ((is = elf_nextscn(ecp->ein, is)) != NULL) {
124 		if (sh_info == elf_ndxscn(is)) {
125 			if (gelf_getshdr(is, &ish) == NULL)
126 				errx(EXIT_FAILURE, "gelf_getshdr failed: %s",
127 				    elf_errmsg(-1));
128 			if ((name = elf_strptr(ecp->ein, indx, ish.sh_name)) ==
129 			    NULL)
130 				errx(EXIT_FAILURE, "elf_strptr failed: %s",
131 				    elf_errmsg(-1));
132 			if (is_remove_section(ecp, name))
133 				return (1);
134 			else
135 				return (0);
136 		}
137 	}
138 	elferr = elf_errno();
139 	if (elferr != 0)
140 		errx(EXIT_FAILURE, "elf_nextscn failed: %s",
141 		    elf_errmsg(elferr));
142 
143 	/* Remove reloc section if we can't find the target section. */
144 	return (1);
145 }
146 
147 static int
148 is_append_section(struct elfcopy *ecp, const char *name)
149 {
150 	struct sec_action *sac;
151 
152 	sac = lookup_sec_act(ecp, name, 0);
153 	if (sac != NULL && sac->append != 0 && sac->string != NULL)
154 		return (1);
155 
156 	return (0);
157 }
158 
159 static int
160 is_compress_section(struct elfcopy *ecp, const char *name)
161 {
162 	struct sec_action *sac;
163 
164 	sac = lookup_sec_act(ecp, name, 0);
165 	if (sac != NULL && sac->compress != 0)
166 		return (1);
167 
168 	return (0);
169 }
170 
171 static void
172 check_section_rename(struct elfcopy *ecp, struct section *s)
173 {
174 	struct sec_action *sac;
175 	char *prefix;
176 	size_t namelen;
177 
178 	if (s->pseudo)
179 		return;
180 
181 	sac = lookup_sec_act(ecp, s->name, 0);
182 	if (sac != NULL && sac->rename)
183 		s->name = sac->newname;
184 
185 	if (!strcmp(s->name, ".symtab") ||
186 	    !strcmp(s->name, ".strtab") ||
187 	    !strcmp(s->name, ".shstrtab"))
188 		return;
189 
190 	prefix = NULL;
191 	if (s->loadable && ecp->prefix_alloc != NULL)
192 		prefix = ecp->prefix_alloc;
193 	else if (ecp->prefix_sec != NULL)
194 		prefix = ecp->prefix_sec;
195 
196 	if (prefix != NULL) {
197 		namelen = strlen(s->name) + strlen(prefix) + 1;
198 		if ((s->newname = malloc(namelen)) == NULL)
199 			err(EXIT_FAILURE, "malloc failed");
200 		snprintf(s->newname, namelen, "%s%s", prefix, s->name);
201 		s->name = s->newname;
202 	}
203 }
204 
205 static int
206 get_section_flags(struct elfcopy *ecp, const char *name)
207 {
208 	struct sec_action *sac;
209 
210 	sac = lookup_sec_act(ecp, name, 0);
211 	if (sac != NULL && sac->flags)
212 		return sac->flags;
213 
214 	return (0);
215 }
216 
217 /*
218  * Determine whether the section are debugging section.
219  * According to libbfd, debugging sections are recognized
220  * only by name.
221  */
222 static int
223 is_debug_section(const char *name)
224 {
225 	const char *dbg_sec[] = {
226 		".apple_",
227 		".debug",
228 		".gnu.linkonce.wi.",
229 		".line",
230 		".stab",
231 		NULL
232 	};
233 	const char **p;
234 
235 	for(p = dbg_sec; *p; p++) {
236 		if (strncmp(name, *p, strlen(*p)) == 0)
237 			return (1);
238 	}
239 
240 	return (0);
241 }
242 
243 static int
244 is_dwo_section(const char *name)
245 {
246 	size_t len;
247 
248 	if ((len = strlen(name)) > 4 && strcmp(name + len - 4, ".dwo") == 0)
249 		return (1);
250 	return (0);
251 }
252 
253 static int
254 is_print_section(struct elfcopy *ecp, const char *name)
255 {
256 	struct sec_action *sac;
257 
258 	sac = lookup_sec_act(ecp, name, 0);
259 	if (sac != NULL && sac->print != 0)
260 		return (1);
261 
262 	return (0);
263 }
264 
265 static int
266 is_modify_section(struct elfcopy *ecp, const char *name)
267 {
268 
269 	if (is_append_section(ecp, name) ||
270 	    is_compress_section(ecp, name))
271 		return (1);
272 
273 	return (0);
274 }
275 
276 struct sec_action*
277 lookup_sec_act(struct elfcopy *ecp, const char *name, int add)
278 {
279 	struct sec_action *sac;
280 
281 	if (name == NULL)
282 		return NULL;
283 
284 	STAILQ_FOREACH(sac, &ecp->v_sac, sac_list) {
285 		if (strcmp(name, sac->name) == 0)
286 			return sac;
287 	}
288 
289 	if (add == 0)
290 		return NULL;
291 
292 	if ((sac = malloc(sizeof(*sac))) == NULL)
293 		errx(EXIT_FAILURE, "not enough memory");
294 	memset(sac, 0, sizeof(*sac));
295 	sac->name = name;
296 	STAILQ_INSERT_TAIL(&ecp->v_sac, sac, sac_list);
297 
298 	return (sac);
299 }
300 
301 void
302 free_sec_act(struct elfcopy *ecp)
303 {
304 	struct sec_action *sac, *sac_temp;
305 
306 	STAILQ_FOREACH_SAFE(sac, &ecp->v_sac, sac_list, sac_temp) {
307 		STAILQ_REMOVE(&ecp->v_sac, sac, sec_action, sac_list);
308 		free(sac);
309 	}
310 }
311 
312 void
313 insert_to_sec_list(struct elfcopy *ecp, struct section *sec, int tail)
314 {
315 	struct section *s;
316 
317 	if (!tail) {
318 		TAILQ_FOREACH(s, &ecp->v_sec, sec_list) {
319 			if (sec->off < s->off) {
320 				TAILQ_INSERT_BEFORE(s, sec, sec_list);
321 				goto inc_nos;
322 			}
323 		}
324 	}
325 
326 	TAILQ_INSERT_TAIL(&ecp->v_sec, sec, sec_list);
327 
328 inc_nos:
329 	if (sec->pseudo == 0)
330 		ecp->nos++;
331 }
332 
333 /*
334  * First step of section creation: create scn and internal section
335  * structure, discard sections to be removed.
336  */
337 void
338 create_scn(struct elfcopy *ecp)
339 {
340 	struct section	*s;
341 	const char	*name;
342 	Elf_Scn		*is;
343 	GElf_Shdr	 ish;
344 	size_t		 indx;
345 	uint64_t	 oldndx, newndx;
346 	int		 elferr, sec_flags, reorder;
347 
348 	/*
349 	 * Insert a pseudo section that contains the ELF header
350 	 * and program header. Used as reference for section offset
351 	 * or load address adjustment.
352 	 */
353 	if ((s = calloc(1, sizeof(*s))) == NULL)
354 		err(EXIT_FAILURE, "calloc failed");
355 	s->off = 0;
356 	s->sz = gelf_fsize(ecp->eout, ELF_T_EHDR, 1, EV_CURRENT) +
357 	    gelf_fsize(ecp->eout, ELF_T_PHDR, ecp->ophnum, EV_CURRENT);
358 	s->align = 1;
359 	s->pseudo = 1;
360 	s->loadable = add_to_inseg_list(ecp, s);
361 	insert_to_sec_list(ecp, s, 0);
362 
363 	/* Create internal .shstrtab section. */
364 	init_shstrtab(ecp);
365 
366 	if (elf_getshstrndx(ecp->ein, &indx) == 0)
367 		errx(EXIT_FAILURE, "elf_getshstrndx failed: %s",
368 		    elf_errmsg(-1));
369 
370 	reorder = 0;
371 	is = NULL;
372 	while ((is = elf_nextscn(ecp->ein, is)) != NULL) {
373 		if (gelf_getshdr(is, &ish) == NULL)
374 			errx(EXIT_FAILURE, "gelf_getshdr failed: %s",
375 			    elf_errmsg(-1));
376 		if ((name = elf_strptr(ecp->ein, indx, ish.sh_name)) == NULL)
377 			errx(EXIT_FAILURE, "elf_strptr failed: %s",
378 			    elf_errmsg(-1));
379 
380 		/* Skip sections to be removed. */
381 		if (is_remove_section(ecp, name))
382 			continue;
383 
384 		/*
385 		 * Relocation section need to be remove if the section
386 		 * it applies will be removed.
387 		 */
388 		if (ish.sh_type == SHT_REL || ish.sh_type == SHT_RELA)
389 			if (ish.sh_info != 0 &&
390 			    is_remove_reloc_sec(ecp, ish.sh_info))
391 				continue;
392 
393 		/*
394 		 * Section groups should be removed if symbol table will
395 		 * be removed. (section group's signature stored in symbol
396 		 * table)
397 		 */
398 		if (ish.sh_type == SHT_GROUP && ecp->strip == STRIP_ALL)
399 			continue;
400 
401 		/* Get section flags set by user. */
402 		sec_flags = get_section_flags(ecp, name);
403 
404 		/* Create internal section object. */
405 		if (strcmp(name, ".shstrtab") != 0) {
406 			if ((s = calloc(1, sizeof(*s))) == NULL)
407 				err(EXIT_FAILURE, "calloc failed");
408 			s->name		= name;
409 			s->is		= is;
410 			s->off		= ish.sh_offset;
411 			s->sz		= ish.sh_size;
412 			s->align	= ish.sh_addralign;
413 			s->type		= ish.sh_type;
414 			s->flags	= ish.sh_flags;
415 			s->vma		= ish.sh_addr;
416 
417 			/*
418 			 * Search program headers to determine whether section
419 			 * is loadable, but if user explicitly set section flags
420 			 * while neither "load" nor "alloc" is set, we make the
421 			 * section unloadable.
422 			 *
423 			 * Sections in relocatable object is loadable if
424 			 * section flag SHF_ALLOC is set.
425 			 */
426 			if (sec_flags &&
427 			    (sec_flags & (SF_LOAD | SF_ALLOC)) == 0)
428 				s->loadable = 0;
429 			else {
430 				s->loadable = add_to_inseg_list(ecp, s);
431 				if ((ecp->flags & RELOCATABLE) &&
432 				    (ish.sh_flags & SHF_ALLOC))
433 					s->loadable = 1;
434 			}
435 		} else {
436 			/* Assuming .shstrtab is "unloadable". */
437 			s		= ecp->shstrtab;
438 			s->off		= ish.sh_offset;
439 		}
440 
441 		oldndx = newndx = SHN_UNDEF;
442 		if (strcmp(name, ".symtab") != 0 &&
443 		    strcmp(name, ".strtab") != 0) {
444 			if (!strcmp(name, ".shstrtab")) {
445 				/*
446 				 * Add sections specified by --add-section and
447 				 * gnu debuglink. we want these sections have
448 				 * smaller index than .shstrtab section.
449 				 */
450 				if (ecp->debuglink != NULL)
451 					add_gnu_debuglink(ecp);
452 				if (ecp->flags & SEC_ADD)
453 					insert_sections(ecp);
454 			}
455  			if ((s->os = elf_newscn(ecp->eout)) == NULL)
456 				errx(EXIT_FAILURE, "elf_newscn failed: %s",
457 				    elf_errmsg(-1));
458 			if ((newndx = elf_ndxscn(s->os)) == SHN_UNDEF)
459 				errx(EXIT_FAILURE, "elf_ndxscn failed: %s",
460 				    elf_errmsg(-1));
461 		}
462 		if ((oldndx = elf_ndxscn(is)) == SHN_UNDEF)
463 			errx(EXIT_FAILURE, "elf_ndxscn failed: %s",
464 			    elf_errmsg(-1));
465 		if (oldndx != SHN_UNDEF && newndx != SHN_UNDEF)
466 			ecp->secndx[oldndx] = newndx;
467 
468 		/*
469 		 * If strip action is STRIP_NONDEBUG(only keep debug),
470 		 * change sections type of loadable sections and section
471 		 * groups to SHT_NOBITS, and the content of those sections
472 		 * will be discarded. However, SHT_NOTE sections should
473 		 * be kept.
474 		 */
475 		if (ecp->strip == STRIP_NONDEBUG) {
476 			if (((ish.sh_flags & SHF_ALLOC) ||
477 			    (ish.sh_flags & SHF_GROUP)) &&
478 			    ish.sh_type != SHT_NOTE)
479 				s->type = SHT_NOBITS;
480 		}
481 
482 		check_section_rename(ecp, s);
483 
484 		/* create section header based on input object. */
485 		if (strcmp(name, ".symtab") != 0 &&
486 		    strcmp(name, ".strtab") != 0 &&
487 		    strcmp(name, ".shstrtab") != 0) {
488 			copy_shdr(ecp, s, NULL, 0, sec_flags);
489 			/*
490 			 * elfcopy puts .symtab, .strtab and .shstrtab
491 			 * sections in the end of the output object.
492 			 * If the input objects have more sections
493 			 * after any of these 3 sections, the section
494 			 * table will be reordered. section symbols
495 			 * should be regenerated for relocations.
496 			 */
497 			if (reorder)
498 				ecp->flags &= ~SYMTAB_INTACT;
499 		} else
500 			reorder = 1;
501 
502 		if (strcmp(name, ".symtab") == 0) {
503 			ecp->flags |= SYMTAB_EXIST;
504 			ecp->symtab = s;
505 		}
506 		if (strcmp(name, ".strtab") == 0)
507 			ecp->strtab = s;
508 
509 		insert_to_sec_list(ecp, s, 0);
510 	}
511 	elferr = elf_errno();
512 	if (elferr != 0)
513 		errx(EXIT_FAILURE, "elf_nextscn failed: %s",
514 		    elf_errmsg(elferr));
515 }
516 
517 struct section *
518 insert_shtab(struct elfcopy *ecp, int tail)
519 {
520 	struct section	*s, *shtab;
521 	GElf_Ehdr	 ieh;
522 	int		 nsecs;
523 
524 	/*
525 	 * Treat section header table as a "pseudo" section, insert it
526 	 * into section list, so later it will get sorted and resynced
527 	 * just as normal sections.
528 	 */
529 	if ((shtab = calloc(1, sizeof(*shtab))) == NULL)
530 		errx(EXIT_FAILURE, "calloc failed");
531 	if (!tail) {
532 		/*
533 		 * "shoff" of input object is used as a hint for section
534 		 * resync later.
535 		 */
536 		if (gelf_getehdr(ecp->ein, &ieh) == NULL)
537 			errx(EXIT_FAILURE, "gelf_getehdr() failed: %s",
538 			    elf_errmsg(-1));
539 		shtab->off = ieh.e_shoff;
540 	} else
541 		shtab->off = 0;
542 	/* Calculate number of sections in the output object. */
543 	nsecs = 0;
544 	TAILQ_FOREACH(s, &ecp->v_sec, sec_list) {
545 		if (!s->pseudo)
546 			nsecs++;
547 	}
548 	/* Remember there is always a null section, so we +1 here. */
549 	shtab->sz = gelf_fsize(ecp->eout, ELF_T_SHDR, nsecs + 1, EV_CURRENT);
550 	if (shtab->sz == 0)
551 		errx(EXIT_FAILURE, "gelf_fsize() failed: %s", elf_errmsg(-1));
552 	shtab->align = (ecp->oec == ELFCLASS32 ? 4 : 8);
553 	shtab->loadable = 0;
554 	shtab->pseudo = 1;
555 	insert_to_sec_list(ecp, shtab, tail);
556 
557 	return (shtab);
558 }
559 
560 void
561 copy_content(struct elfcopy *ecp)
562 {
563 	struct section *s;
564 
565 	TAILQ_FOREACH(s, &ecp->v_sec, sec_list) {
566 		/* Skip pseudo section. */
567 		if (s->pseudo)
568 			continue;
569 
570 		/* Skip special sections. */
571 		if (strcmp(s->name, ".symtab") == 0 ||
572 		    strcmp(s->name, ".strtab") == 0 ||
573 		    strcmp(s->name, ".shstrtab") == 0)
574 			continue;
575 
576 		/*
577 		 * If strip action is STRIP_ALL, relocation info need
578 		 * to be stripped. Skip filtering otherwisw.
579 		 */
580 		if (ecp->strip == STRIP_ALL &&
581 		    (s->type == SHT_REL || s->type == SHT_RELA))
582 			filter_reloc(ecp, s);
583 
584 		/*
585 		 * The section indices in the SHT_GROUP section needs
586 		 * to be updated since we might have stripped some
587 		 * sections and changed section numbering.
588 		 */
589 		if (s->type == SHT_GROUP)
590 			update_section_group(ecp, s);
591 
592 		if (is_modify_section(ecp, s->name))
593 			modify_section(ecp, s);
594 
595 		copy_data(s);
596 
597 		/*
598 		 * If symbol table is modified, relocation info might
599 		 * need update, as symbol index may have changed.
600 		 */
601 		if ((ecp->flags & SYMTAB_INTACT) == 0 &&
602 		    (ecp->flags & SYMTAB_EXIST) &&
603 		    (s->type == SHT_REL || s->type == SHT_RELA))
604 			update_reloc(ecp, s);
605 
606 		if (is_print_section(ecp, s->name))
607 			print_section(s);
608 	}
609 }
610 
611 
612 /*
613  * Update section group section. The section indices in the SHT_GROUP
614  * section need update after section numbering changed.
615  */
616 static void
617 update_section_group(struct elfcopy *ecp, struct section *s)
618 {
619 	GElf_Shdr	 ish;
620 	Elf_Data	*id;
621 	uint32_t	*ws, *wd;
622 	uint64_t	 n;
623 	size_t		 ishnum;
624 	int		 i, j;
625 
626 	if (!elf_getshnum(ecp->ein, &ishnum))
627 		errx(EXIT_FAILURE, "elf_getshnum failed: %s",
628 		    elf_errmsg(-1));
629 
630 	if (gelf_getshdr(s->is, &ish) == NULL)
631 		errx(EXIT_FAILURE, "gelf_getehdr() failed: %s",
632 		    elf_errmsg(-1));
633 
634 	if ((id = elf_getdata(s->is, NULL)) == NULL)
635 		errx(EXIT_FAILURE, "elf_getdata() failed: %s",
636 		    elf_errmsg(-1));
637 
638 	if (ish.sh_size == 0)
639 		return;
640 
641 	if (ish.sh_entsize == 0)
642 		ish.sh_entsize = 4;
643 
644 	ws = id->d_buf;
645 
646 	/* We only support COMDAT section. */
647 #ifndef GRP_COMDAT
648 #define	GRP_COMDAT 0x1
649 #endif
650 	if ((*ws & GRP_COMDAT) == 0)
651 		return;
652 
653 	if ((s->buf = malloc(ish.sh_size)) == NULL)
654 		err(EXIT_FAILURE, "malloc failed");
655 
656 	s->sz = ish.sh_size;
657 
658 	wd = s->buf;
659 
660 	/* Copy the flag word as-is. */
661 	*wd = *ws;
662 
663 	/* Update the section indices. */
664 	n = ish.sh_size / ish.sh_entsize;
665 	for(i = 1, j = 1; (uint64_t)i < n; i++) {
666 		if (ws[i] != SHN_UNDEF && ws[i] < ishnum &&
667 		    ecp->secndx[ws[i]] != 0)
668 			wd[j++] = ecp->secndx[ws[i]];
669 		else
670 			s->sz -= 4;
671 	}
672 
673 	s->nocopy = 1;
674 }
675 
676 /*
677  * Filter relocation entries, only keep those entries whose
678  * symbol is in the keep list.
679  */
680 static void
681 filter_reloc(struct elfcopy *ecp, struct section *s)
682 {
683 	const char	*name;
684 	GElf_Shdr	 ish;
685 	GElf_Rel	 rel;
686 	GElf_Rela	 rela;
687 	Elf32_Rel	*rel32;
688 	Elf64_Rel	*rel64;
689 	Elf32_Rela	*rela32;
690 	Elf64_Rela	*rela64;
691 	Elf_Data	*id;
692 	uint64_t	 cap, n, nrels;
693 	int		 elferr, i;
694 
695 	if (gelf_getshdr(s->is, &ish) == NULL)
696 		errx(EXIT_FAILURE, "gelf_getehdr() failed: %s",
697 		    elf_errmsg(-1));
698 
699 	/* We don't want to touch relocation info for dynamic symbols. */
700 	if ((ecp->flags & SYMTAB_EXIST) == 0) {
701 		if (ish.sh_link == 0 || ecp->secndx[ish.sh_link] == 0) {
702 			/*
703 			 * This reloc section applies to the symbol table
704 			 * that was stripped, so discard whole section.
705 			 */
706 			s->nocopy = 1;
707 			s->sz = 0;
708 		}
709 		return;
710 	} else {
711 		/* Symbol table exist, check if index equals. */
712 		if (ish.sh_link != elf_ndxscn(ecp->symtab->is))
713 			return;
714 	}
715 
716 #define	COPYREL(REL, SZ) do {					\
717 	if (nrels == 0) {					\
718 		if ((REL##SZ = malloc(cap *			\
719 		    sizeof(Elf##SZ##_Rel))) == NULL)		\
720 			err(EXIT_FAILURE, "malloc failed");	\
721 	}							\
722 	if (nrels >= cap) {					\
723 		cap *= 2;					\
724 		if ((REL##SZ = realloc(REL##SZ, cap *		\
725 		    sizeof(Elf##SZ##_Rel))) == NULL)		\
726 			err(EXIT_FAILURE, "realloc failed");	\
727 	}							\
728 	REL##SZ[nrels].r_offset = REL.r_offset;			\
729 	REL##SZ[nrels].r_info	= REL.r_info;			\
730 	if (s->type == SHT_RELA)				\
731 		rela##SZ[nrels].r_addend = rela.r_addend;	\
732 	nrels++;						\
733 } while (0)
734 
735 	nrels = 0;
736 	cap = 4;		/* keep list is usually small. */
737 	rel32 = NULL;
738 	rel64 = NULL;
739 	rela32 = NULL;
740 	rela64 = NULL;
741 	if ((id = elf_getdata(s->is, NULL)) == NULL)
742 		errx(EXIT_FAILURE, "elf_getdata() failed: %s",
743 		    elf_errmsg(-1));
744 	n = ish.sh_size / ish.sh_entsize;
745 	for(i = 0; (uint64_t)i < n; i++) {
746 		if (s->type == SHT_REL) {
747 			if (gelf_getrel(id, i, &rel) != &rel)
748 				errx(EXIT_FAILURE, "gelf_getrel failed: %s",
749 				    elf_errmsg(-1));
750 		} else {
751 			if (gelf_getrela(id, i, &rela) != &rela)
752 				errx(EXIT_FAILURE, "gelf_getrel failed: %s",
753 				    elf_errmsg(-1));
754 		}
755 		name = elf_strptr(ecp->ein, elf_ndxscn(ecp->strtab->is),
756 		    GELF_R_SYM(rel.r_info));
757 		if (name == NULL)
758 			errx(EXIT_FAILURE, "elf_strptr failed: %s",
759 			    elf_errmsg(-1));
760 		if (lookup_symop_list(ecp, name, SYMOP_KEEP) != NULL) {
761 			if (ecp->oec == ELFCLASS32) {
762 				if (s->type == SHT_REL)
763 					COPYREL(rel, 32);
764 				else
765 					COPYREL(rela, 32);
766 			} else {
767 				if (s->type == SHT_REL)
768 					COPYREL(rel, 64);
769 				else
770 					COPYREL(rela, 64);
771 			}
772 		}
773 	}
774 	elferr = elf_errno();
775 	if (elferr != 0)
776 		errx(EXIT_FAILURE, "elf_getdata() failed: %s",
777 		    elf_errmsg(elferr));
778 
779 	if (ecp->oec == ELFCLASS32) {
780 		if (s->type == SHT_REL)
781 			s->buf = rel32;
782 		else
783 			s->buf = rela32;
784 	} else {
785 		if (s->type == SHT_REL)
786 			s->buf = rel64;
787 		else
788 			s->buf = rela64;
789 	}
790 	s->sz = gelf_fsize(ecp->eout, (s->type == SHT_REL ? ELF_T_REL :
791 	    ELF_T_RELA), nrels, EV_CURRENT);
792 	s->nocopy = 1;
793 }
794 
795 static void
796 update_reloc(struct elfcopy *ecp, struct section *s)
797 {
798 	GElf_Shdr	 osh;
799 	GElf_Rel	 rel;
800 	GElf_Rela	 rela;
801 	Elf_Data	*od;
802 	uint64_t	 n;
803 	int		 i;
804 
805 #define UPDATEREL(REL) do {						\
806 	if (gelf_get##REL(od, i, &REL) != &REL)				\
807 		errx(EXIT_FAILURE, "gelf_get##REL failed: %s",		\
808 		    elf_errmsg(-1));					\
809 	REL.r_info = GELF_R_INFO(ecp->symndx[GELF_R_SYM(REL.r_info)],	\
810 	    GELF_R_TYPE(REL.r_info));					\
811 	if (!gelf_update_##REL(od, i, &REL))				\
812 		errx(EXIT_FAILURE, "gelf_update_##REL failed: %s",	\
813 		    elf_errmsg(-1));					\
814 } while(0)
815 
816 	if (s->sz == 0)
817 		return;
818 	if (gelf_getshdr(s->os, &osh) == NULL)
819 		errx(EXIT_FAILURE, "gelf_getehdr() failed: %s",
820 		    elf_errmsg(-1));
821 	/* Only process .symtab reloc info. */
822 	if (osh.sh_link != elf_ndxscn(ecp->symtab->is))
823 		return;
824 	if ((od = elf_getdata(s->os, NULL)) == NULL)
825 		errx(EXIT_FAILURE, "elf_getdata() failed: %s",
826 		    elf_errmsg(-1));
827 	n = osh.sh_size / osh.sh_entsize;
828 	for(i = 0; (uint64_t)i < n; i++) {
829 		if (s->type == SHT_REL)
830 			UPDATEREL(rel);
831 		else
832 			UPDATEREL(rela);
833 	}
834 }
835 
836 static void
837 pad_section(struct elfcopy *ecp, struct section *s)
838 {
839 	GElf_Shdr	 osh;
840 	Elf_Data	*od;
841 
842 	if (s == NULL || s->pad_sz == 0)
843 		return;
844 
845 	if ((s->pad = malloc(s->pad_sz)) == NULL)
846 		err(EXIT_FAILURE, "malloc failed");
847 	memset(s->pad, ecp->fill, s->pad_sz);
848 
849 	/* Create a new Elf_Data to contain the padding bytes. */
850 	if ((od = elf_newdata(s->os)) == NULL)
851 		errx(EXIT_FAILURE, "elf_newdata() failed: %s",
852 		    elf_errmsg(-1));
853 	od->d_align = 1;
854 	od->d_off = s->sz;
855 	od->d_buf = s->pad;
856 	od->d_type = ELF_T_BYTE;
857 	od->d_size = s->pad_sz;
858 	od->d_version = EV_CURRENT;
859 
860 	/* Update section header. */
861 	if (gelf_getshdr(s->os, &osh) == NULL)
862 		errx(EXIT_FAILURE, "gelf_getshdr() failed: %s",
863 		    elf_errmsg(-1));
864 	osh.sh_size = s->sz + s->pad_sz;
865 	if (!gelf_update_shdr(s->os, &osh))
866 		errx(EXIT_FAILURE, "elf_update_shdr failed: %s",
867 		    elf_errmsg(-1));
868 }
869 
870 void
871 resync_sections(struct elfcopy *ecp)
872 {
873 	struct section	*s, *ps;
874 	GElf_Shdr	 osh;
875 	uint64_t	 off;
876 	int		 first;
877 
878 	ps = NULL;
879 	first = 1;
880 	off = 0;
881 	TAILQ_FOREACH(s, &ecp->v_sec, sec_list) {
882 		if (first) {
883 			off = s->off;
884 			first = 0;
885 		}
886 
887 		/*
888 		 * Ignore TLS sections with load address 0 and without
889 		 * content. We don't need to adjust their file offset or
890 		 * VMA, only the size matters.
891 		 */
892 		if (s->seg_tls != NULL && s->type == SHT_NOBITS &&
893 		    s->off == 0)
894 			continue;
895 
896 		/* Align section offset. */
897 		if (s->align == 0)
898 			s->align = 1;
899 		if (off <= s->off) {
900 			if (!s->loadable || (ecp->flags & RELOCATABLE))
901 				s->off = roundup(off, s->align);
902 		} else {
903 			if (s->loadable && (ecp->flags & RELOCATABLE) == 0)
904 				warnx("moving loadable section %s, "
905 				    "is this intentional?", s->name);
906 			s->off = roundup(off, s->align);
907 		}
908 
909 		/* Calculate next section offset. */
910 		off = s->off;
911 		if (s->pseudo || (s->type != SHT_NOBITS && s->type != SHT_NULL))
912 			off += s->sz;
913 
914 		if (s->pseudo) {
915 			ps = NULL;
916 			continue;
917 		}
918 
919 		/* Count padding bytes added through --pad-to. */
920 		if (s->pad_sz > 0)
921 			off += s->pad_sz;
922 
923 		/* Update section header accordingly. */
924 		if (gelf_getshdr(s->os, &osh) == NULL)
925 			errx(EXIT_FAILURE, "gelf_getshdr() failed: %s",
926 			    elf_errmsg(-1));
927 		osh.sh_addr = s->vma;
928 		osh.sh_offset = s->off;
929 		osh.sh_size = s->sz;
930 		if (!gelf_update_shdr(s->os, &osh))
931 			errx(EXIT_FAILURE, "elf_update_shdr failed: %s",
932 			    elf_errmsg(-1));
933 
934 		/* Add padding for previous section, if need. */
935 		if (ps != NULL) {
936 			if (ps->pad_sz > 0) {
937 				/* Apply padding added by --pad-to. */
938 				pad_section(ecp, ps);
939 			} else if ((ecp->flags & GAP_FILL) &&
940 			    (ps->off + ps->sz < s->off)) {
941 				/*
942 				 * Fill the gap between sections by padding
943 				 * the section with lower address.
944 				 */
945 				ps->pad_sz = s->off - (ps->off + ps->sz);
946 				pad_section(ecp, ps);
947 			}
948 		}
949 
950 		ps = s;
951 	}
952 
953 	/* Pad the last section, if need. */
954 	if (ps != NULL && ps->pad_sz > 0)
955 		pad_section(ecp, ps);
956 }
957 
958 static void
959 modify_section(struct elfcopy *ecp, struct section *s)
960 {
961 	struct sec_action	*sac;
962 	size_t			 srcsz, dstsz, p, len;
963 	char			*b, *c, *d, *src, *end;
964 	int			 dupe;
965 
966 	src = read_section(s, &srcsz);
967 	if (src == NULL || srcsz == 0) {
968 		/* For empty section, we proceed if we need to append. */
969 		if (!is_append_section(ecp, s->name))
970 			return;
971 	}
972 
973 	/* Allocate buffer needed for new section data. */
974 	dstsz = srcsz;
975 	if (is_append_section(ecp, s->name)) {
976 		sac = lookup_sec_act(ecp, s->name, 0);
977 		dstsz += strlen(sac->string) + 1;
978 	}
979 	if ((b = malloc(dstsz)) == NULL)
980 		err(EXIT_FAILURE, "malloc failed");
981 	s->buf = b;
982 
983 	/* Compress section. */
984 	p = 0;
985 	if (is_compress_section(ecp, s->name)) {
986 		end = src + srcsz;
987 		for(c = src; c < end;) {
988 			len = 0;
989 			while(c + len < end && c[len] != '\0')
990 				len++;
991 			if (c + len == end) {
992 				/* XXX should we warn here? */
993 				strncpy(&b[p], c, len);
994 				p += len;
995 				break;
996 			}
997 			dupe = 0;
998 			for (d = b; d < b + p; ) {
999 				if (strcmp(d, c) == 0) {
1000 					dupe = 1;
1001 					break;
1002 				}
1003 				d += strlen(d) + 1;
1004 			}
1005 			if (!dupe) {
1006 				strncpy(&b[p], c, len);
1007 				b[p + len] = '\0';
1008 				p += len + 1;
1009 			}
1010 			c += len + 1;
1011 		}
1012 	} else {
1013 		memcpy(b, src, srcsz);
1014 		p += srcsz;
1015 	}
1016 
1017 	/* Append section. */
1018 	if (is_append_section(ecp, s->name)) {
1019 		sac = lookup_sec_act(ecp, s->name, 0);
1020 		len = strlen(sac->string);
1021 		strncpy(&b[p], sac->string, len);
1022 		b[p + len] = '\0';
1023 		p += len + 1;
1024 	}
1025 
1026 	s->sz = p;
1027 	s->nocopy = 1;
1028 }
1029 
1030 static void
1031 print_data(const char *d, size_t sz)
1032 {
1033 	const char *c;
1034 
1035 	for (c = d; c < d + sz; c++) {
1036 		if (*c == '\0')
1037 			putchar('\n');
1038 		else
1039 			putchar(*c);
1040 	}
1041 }
1042 
1043 static void
1044 print_section(struct section *s)
1045 {
1046 	Elf_Data	*id;
1047 	int		 elferr;
1048 
1049 	if (s->buf != NULL && s->sz > 0) {
1050 		print_data(s->buf, s->sz);
1051 	} else {
1052 		id = NULL;
1053 		while ((id = elf_getdata(s->is, id)) != NULL ||
1054 		    (id = elf_rawdata(s->is, id)) != NULL) {
1055 			(void) elf_errno();
1056 			print_data(id->d_buf, id->d_size);
1057 		}
1058 		elferr = elf_errno();
1059 		if (elferr != 0)
1060 			errx(EXIT_FAILURE, "elf_getdata() failed: %s",
1061 			    elf_errmsg(elferr));
1062 	}
1063 	putchar('\n');
1064 }
1065 
1066 static void *
1067 read_section(struct section *s, size_t *size)
1068 {
1069 	Elf_Data	*id;
1070 	char		*b;
1071 	size_t		 sz;
1072 	int		 elferr;
1073 
1074 	sz = 0;
1075 	b = NULL;
1076 	id = NULL;
1077 	while ((id = elf_getdata(s->is, id)) != NULL ||
1078 	    (id = elf_rawdata(s->is, id)) != NULL) {
1079 		(void) elf_errno();
1080 		if (b == NULL)
1081 			b = malloc(id->d_size);
1082 		else
1083 			b = malloc(sz + id->d_size);
1084 		if (b == NULL)
1085 			err(EXIT_FAILURE, "malloc or realloc failed");
1086 
1087 		memcpy(&b[sz], id->d_buf, id->d_size);
1088 		sz += id->d_size;
1089 	}
1090 	elferr = elf_errno();
1091 	if (elferr != 0)
1092 		errx(EXIT_FAILURE, "elf_getdata() failed: %s",
1093 		    elf_errmsg(elferr));
1094 
1095 	*size = sz;
1096 
1097 	return (b);
1098 }
1099 
1100 void
1101 copy_shdr(struct elfcopy *ecp, struct section *s, const char *name, int copy,
1102     int sec_flags)
1103 {
1104 	GElf_Shdr ish, osh;
1105 
1106 	if (gelf_getshdr(s->is, &ish) == NULL)
1107 		errx(EXIT_FAILURE, "gelf_getshdr() failed: %s",
1108 		    elf_errmsg(-1));
1109 	if (gelf_getshdr(s->os, &osh) == NULL)
1110 		errx(EXIT_FAILURE, "gelf_getshdr() failed: %s",
1111 		    elf_errmsg(-1));
1112 
1113 	if (copy)
1114 		(void) memcpy(&osh, &ish, sizeof(ish));
1115 	else {
1116 		osh.sh_type		= s->type;
1117 		osh.sh_addr		= s->vma;
1118 		osh.sh_offset		= s->off;
1119 		osh.sh_size		= s->sz;
1120 		osh.sh_link		= ish.sh_link;
1121 		osh.sh_info		= ish.sh_info;
1122 		osh.sh_addralign	= s->align;
1123 		osh.sh_entsize		= ish.sh_entsize;
1124 
1125 		if (sec_flags) {
1126 			osh.sh_flags = 0;
1127 			if (sec_flags & SF_ALLOC)
1128 				osh.sh_flags |= SHF_ALLOC;
1129 			if ((sec_flags & SF_READONLY) == 0)
1130 				osh.sh_flags |= SHF_WRITE;
1131 			if (sec_flags & SF_CODE)
1132 				osh.sh_flags |= SHF_EXECINSTR;
1133 			if ((sec_flags & SF_CONTENTS) &&
1134 			    s->type == SHT_NOBITS && s->sz > 0) {
1135 				/*
1136 				 * Convert SHT_NOBITS section to section with
1137 				 * (zero'ed) content on file.
1138 				 */
1139 				osh.sh_type = s->type = SHT_PROGBITS;
1140 				if ((s->buf = calloc(1, s->sz)) == NULL)
1141 					err(EXIT_FAILURE, "malloc failed");
1142 				s->nocopy = 1;
1143 			}
1144 		} else {
1145 			osh.sh_flags = ish.sh_flags;
1146 			/*
1147 			 * Newer binutils as(1) emits the section flag
1148 			 * SHF_INFO_LINK for relocation sections. elfcopy
1149 			 * emits this flag in the output section if it's
1150 			 * missing in the input section, to remain compatible
1151 			 * with binutils.
1152 			 */
1153 			if (ish.sh_type == SHT_REL || ish.sh_type == SHT_RELA)
1154 				osh.sh_flags |= SHF_INFO_LINK;
1155 		}
1156 	}
1157 
1158 	if (name == NULL)
1159 		add_to_shstrtab(ecp, s->name);
1160 	else
1161 		add_to_shstrtab(ecp, name);
1162 
1163 	if (!gelf_update_shdr(s->os, &osh))
1164 		errx(EXIT_FAILURE, "elf_update_shdr failed: %s",
1165 		    elf_errmsg(-1));
1166 }
1167 
1168 void
1169 copy_data(struct section *s)
1170 {
1171 	Elf_Data	*id, *od;
1172 	int		 elferr;
1173 
1174 	if (s->nocopy && s->buf == NULL)
1175 		return;
1176 
1177 	if ((id = elf_getdata(s->is, NULL)) == NULL) {
1178 		(void) elf_errno();
1179 		if ((id = elf_rawdata(s->is, NULL)) == NULL) {
1180 			elferr = elf_errno();
1181 			if (elferr != 0)
1182 				errx(EXIT_FAILURE, "failed to read section:"
1183 				    " %s", s->name);
1184 			return;
1185 		}
1186 	}
1187 
1188 	if ((od = elf_newdata(s->os)) == NULL)
1189 		errx(EXIT_FAILURE, "elf_newdata() failed: %s",
1190 		    elf_errmsg(-1));
1191 
1192 	if (s->nocopy) {
1193 		/* Use s->buf as content if s->nocopy is set. */
1194 		od->d_align	= id->d_align;
1195 		od->d_off	= 0;
1196 		od->d_buf	= s->buf;
1197 		od->d_type	= id->d_type;
1198 		od->d_size	= s->sz;
1199 		od->d_version	= id->d_version;
1200 	} else {
1201 		od->d_align	= id->d_align;
1202 		od->d_off	= id->d_off;
1203 		od->d_buf	= id->d_buf;
1204 		od->d_type	= id->d_type;
1205 		od->d_size	= id->d_size;
1206 		od->d_version	= id->d_version;
1207 	}
1208 
1209 	/*
1210 	 * Alignment Fixup. libelf does not allow the alignment for
1211 	 * Elf_Data descriptor to be set to 0. In this case we workaround
1212 	 * it by setting the alignment to 1.
1213 	 *
1214 	 * According to the ELF ABI, alignment 0 and 1 has the same
1215 	 * meaning: the section has no alignment constraints.
1216 	 */
1217 	if (od->d_align == 0)
1218 		od->d_align = 1;
1219 }
1220 
1221 struct section *
1222 create_external_section(struct elfcopy *ecp, const char *name, char *newname,
1223     void *buf, uint64_t size, uint64_t off, uint64_t stype, Elf_Type dtype,
1224     uint64_t flags, uint64_t align, uint64_t vma, int loadable)
1225 {
1226 	struct section	*s;
1227 	Elf_Scn		*os;
1228 	Elf_Data	*od;
1229 	GElf_Shdr	 osh;
1230 
1231 	if ((os = elf_newscn(ecp->eout)) == NULL)
1232 		errx(EXIT_FAILURE, "elf_newscn() failed: %s",
1233 		    elf_errmsg(-1));
1234 	if ((s = calloc(1, sizeof(*s))) == NULL)
1235 		err(EXIT_FAILURE, "calloc failed");
1236 	s->name = name;
1237 	s->newname = newname;	/* needs to be free()'ed */
1238 	s->off = off;
1239 	s->sz = size;
1240 	s->vma = vma;
1241 	s->align = align;
1242 	s->loadable = loadable;
1243 	s->is = NULL;
1244 	s->os = os;
1245 	s->type = stype;
1246 	s->nocopy = 1;
1247 	insert_to_sec_list(ecp, s, 1);
1248 
1249 	if (gelf_getshdr(os, &osh) == NULL)
1250 		errx(EXIT_FAILURE, "gelf_getshdr() failed: %s",
1251 		    elf_errmsg(-1));
1252 	osh.sh_flags = flags;
1253 	osh.sh_type = s->type;
1254 	osh.sh_addr = s->vma;
1255 	osh.sh_addralign = s->align;
1256 	if (!gelf_update_shdr(os, &osh))
1257 		errx(EXIT_FAILURE, "gelf_update_shdr() failed: %s",
1258 		    elf_errmsg(-1));
1259 	add_to_shstrtab(ecp, name);
1260 
1261 	if (buf != NULL && size != 0) {
1262 		if ((od = elf_newdata(os)) == NULL)
1263 			errx(EXIT_FAILURE, "elf_newdata() failed: %s",
1264 			    elf_errmsg(-1));
1265 		od->d_align = align;
1266 		od->d_off = 0;
1267 		od->d_buf = buf;
1268 		od->d_size = size;
1269 		od->d_type = dtype;
1270 		od->d_version = EV_CURRENT;
1271 	}
1272 
1273 	/*
1274 	 * Clear SYMTAB_INTACT, as we probably need to update/add new
1275 	 * STT_SECTION symbols into the symbol table.
1276 	 */
1277 	ecp->flags &= ~SYMTAB_INTACT;
1278 
1279 	return (s);
1280 }
1281 
1282 /*
1283  * Insert sections specified by --add-section to the end of section list.
1284  */
1285 static void
1286 insert_sections(struct elfcopy *ecp)
1287 {
1288 	struct sec_add	*sa;
1289 	struct section	*s;
1290 	size_t		 off;
1291 	uint64_t	 stype;
1292 
1293 	/* Put these sections in the end of current list. */
1294 	off = 0;
1295 	TAILQ_FOREACH(s, &ecp->v_sec, sec_list) {
1296 		if (s->type != SHT_NOBITS && s->type != SHT_NULL)
1297 			off = s->off + s->sz;
1298 		else
1299 			off = s->off;
1300 	}
1301 
1302 	STAILQ_FOREACH(sa, &ecp->v_sadd, sadd_list) {
1303 
1304 		/* TODO: Add section header vma/lma, flag changes here */
1305 
1306 		/*
1307 		 * The default section type for user added section is
1308 		 * SHT_PROGBITS. If the section name match certain patterns,
1309 		 * elfcopy will try to set a more appropriate section type.
1310 		 * However, data type is always set to ELF_T_BYTE and no
1311 		 * translation is performed by libelf.
1312 		 */
1313 		stype = SHT_PROGBITS;
1314 		if (strcmp(sa->name, ".note") == 0 ||
1315 		    strncmp(sa->name, ".note.", strlen(".note.")) == 0)
1316 			stype = SHT_NOTE;
1317 
1318 		(void) create_external_section(ecp, sa->name, NULL, sa->content,
1319 		    sa->size, off, stype, ELF_T_BYTE, 0, 1, 0, 0);
1320 	}
1321 }
1322 
1323 void
1324 add_to_shstrtab(struct elfcopy *ecp, const char *name)
1325 {
1326 	struct section *s;
1327 
1328 	s = ecp->shstrtab;
1329 	insert_to_strtab(s, name);
1330 }
1331 
1332 void
1333 update_shdr(struct elfcopy *ecp, int update_link)
1334 {
1335 	struct section	*s;
1336 	GElf_Shdr	 osh;
1337 	int		 elferr;
1338 
1339 	TAILQ_FOREACH(s, &ecp->v_sec, sec_list) {
1340 		if (s->pseudo)
1341 			continue;
1342 
1343 		if (gelf_getshdr(s->os, &osh) == NULL)
1344 			errx(EXIT_FAILURE, "gelf_getshdr failed: %s",
1345 			    elf_errmsg(-1));
1346 
1347 		/* Find section name in string table and set sh_name. */
1348 		osh.sh_name = lookup_string(ecp->shstrtab, s->name);
1349 
1350 		/*
1351 		 * sh_link needs to be updated, since the index of the
1352 		 * linked section might have changed.
1353 		 */
1354 		if (update_link && osh.sh_link != 0)
1355 			osh.sh_link = ecp->secndx[osh.sh_link];
1356 
1357 		/*
1358 		 * sh_info of relocation section links to the section to which
1359 		 * its relocation info applies. So it may need update as well.
1360 		 */
1361 		if ((s->type == SHT_REL || s->type == SHT_RELA) &&
1362 		    osh.sh_info != 0)
1363 			osh.sh_info = ecp->secndx[osh.sh_info];
1364 
1365 		/*
1366 		 * sh_info of SHT_GROUP section needs to point to the correct
1367 		 * string in the symbol table.
1368 		 */
1369 		if (s->type == SHT_GROUP && (ecp->flags & SYMTAB_EXIST) &&
1370 		    (ecp->flags & SYMTAB_INTACT) == 0)
1371 			osh.sh_info = ecp->symndx[osh.sh_info];
1372 
1373 		if (!gelf_update_shdr(s->os, &osh))
1374 			errx(EXIT_FAILURE, "gelf_update_shdr() failed: %s",
1375 			    elf_errmsg(-1));
1376 	}
1377 	elferr = elf_errno();
1378 	if (elferr != 0)
1379 		errx(EXIT_FAILURE, "elf_nextscn failed: %s",
1380 		    elf_errmsg(elferr));
1381 }
1382 
1383 void
1384 init_shstrtab(struct elfcopy *ecp)
1385 {
1386 	struct section *s;
1387 
1388 	if ((ecp->shstrtab = calloc(1, sizeof(*ecp->shstrtab))) == NULL)
1389 		err(EXIT_FAILURE, "calloc failed");
1390 	s = ecp->shstrtab;
1391 	s->name = ".shstrtab";
1392 	s->is = NULL;
1393 	s->sz = 0;
1394 	s->align = 1;
1395 	s->loadable = 0;
1396 	s->type = SHT_STRTAB;
1397 	s->vma = 0;
1398 
1399 	insert_to_strtab(s, "");
1400 	insert_to_strtab(s, ".symtab");
1401 	insert_to_strtab(s, ".strtab");
1402 	insert_to_strtab(s, ".shstrtab");
1403 }
1404 
1405 void
1406 set_shstrtab(struct elfcopy *ecp)
1407 {
1408 	struct section	*s;
1409 	Elf_Data	*data;
1410 	GElf_Shdr	 sh;
1411 
1412 	s = ecp->shstrtab;
1413 
1414 	if (s->os == NULL) {
1415 		/* Input object does not contain .shstrtab section */
1416 		if ((s->os = elf_newscn(ecp->eout)) == NULL)
1417 			errx(EXIT_FAILURE, "elf_newscn failed: %s",
1418 			    elf_errmsg(-1));
1419 		insert_to_sec_list(ecp, s, 1);
1420 	}
1421 
1422 	if (gelf_getshdr(s->os, &sh) == NULL)
1423 		errx(EXIT_FAILURE, "gelf_getshdr() failed: %s",
1424 		    elf_errmsg(-1));
1425 	sh.sh_addr	= 0;
1426 	sh.sh_addralign	= 1;
1427 	sh.sh_offset	= s->off;
1428 	sh.sh_type	= SHT_STRTAB;
1429 	sh.sh_flags	= 0;
1430 	sh.sh_entsize	= 0;
1431 	sh.sh_info	= 0;
1432 	sh.sh_link	= 0;
1433 
1434 	if ((data = elf_newdata(s->os)) == NULL)
1435 		errx(EXIT_FAILURE, "elf_newdata() failed: %s",
1436 		    elf_errmsg(-1));
1437 
1438 	/*
1439 	 * If we don't have a symbol table, skip those a few bytes
1440 	 * which are reserved for this in the beginning of shstrtab.
1441 	 */
1442 	if (!(ecp->flags & SYMTAB_EXIST)) {
1443 		s->sz -= sizeof(".symtab\0.strtab");
1444 		memmove(s->buf, (char *)s->buf + sizeof(".symtab\0.strtab"),
1445 		    s->sz);
1446 	}
1447 
1448 	sh.sh_size	= s->sz;
1449 	if (!gelf_update_shdr(s->os, &sh))
1450 		errx(EXIT_FAILURE, "gelf_update_shdr() failed: %s",
1451 		    elf_errmsg(-1));
1452 
1453 	data->d_align	= 1;
1454 	data->d_buf	= s->buf;
1455 	data->d_size	= s->sz;
1456 	data->d_off	= 0;
1457 	data->d_type	= ELF_T_BYTE;
1458 	data->d_version	= EV_CURRENT;
1459 
1460 	if (!elf_setshstrndx(ecp->eout, elf_ndxscn(s->os)))
1461 		errx(EXIT_FAILURE, "elf_setshstrndx() failed: %s",
1462 		     elf_errmsg(-1));
1463 }
1464 
1465 void
1466 add_section(struct elfcopy *ecp, const char *arg)
1467 {
1468 	struct sec_add	*sa;
1469 	struct stat	 sb;
1470 	const char	*s, *fn;
1471 	FILE		*fp;
1472 	int		 len;
1473 
1474 	if ((s = strchr(arg, '=')) == NULL)
1475 		errx(EXIT_FAILURE,
1476 		    "illegal format for --add-section option");
1477 	if ((sa = malloc(sizeof(*sa))) == NULL)
1478 		err(EXIT_FAILURE, "malloc failed");
1479 
1480 	len = s - arg;
1481 	if ((sa->name = malloc(len + 1)) == NULL)
1482 		err(EXIT_FAILURE, "malloc failed");
1483 	strncpy(sa->name, arg, len);
1484 	sa->name[len] = '\0';
1485 
1486 	fn = s + 1;
1487 	if (stat(fn, &sb) == -1)
1488 		err(EXIT_FAILURE, "stat failed");
1489 	sa->size = sb.st_size;
1490 	if (sa->size > 0) {
1491 		if ((sa->content = malloc(sa->size)) == NULL)
1492 			err(EXIT_FAILURE, "malloc failed");
1493 		if ((fp = fopen(fn, "r")) == NULL)
1494 			err(EXIT_FAILURE, "can not open %s", fn);
1495 		if (fread(sa->content, 1, sa->size, fp) == 0 ||
1496 		    ferror(fp))
1497 			err(EXIT_FAILURE, "fread failed");
1498 		fclose(fp);
1499 	} else
1500 		sa->content = NULL;
1501 
1502 	STAILQ_INSERT_TAIL(&ecp->v_sadd, sa, sadd_list);
1503 	ecp->flags |= SEC_ADD;
1504 }
1505 
1506 void
1507 free_sec_add(struct elfcopy *ecp)
1508 {
1509 	struct sec_add *sa, *sa_temp;
1510 
1511 	STAILQ_FOREACH_SAFE(sa, &ecp->v_sadd, sadd_list, sa_temp) {
1512 		STAILQ_REMOVE(&ecp->v_sadd, sa, sec_add, sadd_list);
1513 		free(sa->name);
1514 		free(sa->content);
1515 		free(sa);
1516 	}
1517 }
1518 
1519 static void
1520 add_gnu_debuglink(struct elfcopy *ecp)
1521 {
1522 	struct sec_add	*sa;
1523 	struct stat	 sb;
1524 	FILE		*fp;
1525 	char		*fnbase, *buf;
1526 	int		 crc_off;
1527 	int		 crc;
1528 
1529 	if (ecp->debuglink == NULL)
1530 		return;
1531 
1532 	/* Read debug file content. */
1533 	if ((sa = malloc(sizeof(*sa))) == NULL)
1534 		err(EXIT_FAILURE, "malloc failed");
1535 	if ((sa->name = strdup(".gnu_debuglink")) == NULL)
1536 		err(EXIT_FAILURE, "strdup failed");
1537 	if (stat(ecp->debuglink, &sb) == -1)
1538 		err(EXIT_FAILURE, "stat failed");
1539 	if (sb.st_size == 0)
1540 		errx(EXIT_FAILURE, "empty debug link target %s",
1541 		    ecp->debuglink);
1542 	if ((buf = malloc(sb.st_size)) == NULL)
1543 		err(EXIT_FAILURE, "malloc failed");
1544 	if ((fp = fopen(ecp->debuglink, "r")) == NULL)
1545 		err(EXIT_FAILURE, "can not open %s", ecp->debuglink);
1546 	if (fread(buf, 1, sb.st_size, fp) == 0 ||
1547 	    ferror(fp))
1548 		err(EXIT_FAILURE, "fread failed");
1549 	fclose(fp);
1550 
1551 	/* Calculate crc checksum.  */
1552 	crc = calc_crc32(buf, sb.st_size, 0xFFFFFFFF);
1553 	free(buf);
1554 
1555 	/* Calculate section size and the offset to store crc checksum. */
1556 	if ((fnbase = basename(ecp->debuglink)) == NULL)
1557 		err(EXIT_FAILURE, "basename failed");
1558 	crc_off = roundup(strlen(fnbase) + 1, 4);
1559 	sa->size = crc_off + 4;
1560 
1561 	/* Section content. */
1562 	if ((sa->content = calloc(1, sa->size)) == NULL)
1563 		err(EXIT_FAILURE, "malloc failed");
1564 	strncpy(sa->content, fnbase, strlen(fnbase));
1565 	if (ecp->oed == ELFDATA2LSB) {
1566 		sa->content[crc_off] = crc & 0xFF;
1567 		sa->content[crc_off + 1] = (crc >> 8) & 0xFF;
1568 		sa->content[crc_off + 2] = (crc >> 16) & 0xFF;
1569 		sa->content[crc_off + 3] = crc >> 24;
1570 	} else {
1571 		sa->content[crc_off] = crc >> 24;
1572 		sa->content[crc_off + 1] = (crc >> 16) & 0xFF;
1573 		sa->content[crc_off + 2] = (crc >> 8) & 0xFF;
1574 		sa->content[crc_off + 3] = crc & 0xFF;
1575 	}
1576 
1577 	STAILQ_INSERT_TAIL(&ecp->v_sadd, sa, sadd_list);
1578 	ecp->flags |= SEC_ADD;
1579 }
1580 
1581 static void
1582 insert_to_strtab(struct section *t, const char *s)
1583 {
1584 	const char	*r;
1585 	char		*b, *c;
1586 	size_t		 len, slen;
1587 	int		 append;
1588 
1589 	if (t->sz == 0) {
1590 		t->cap = 512;
1591 		if ((t->buf = malloc(t->cap)) == NULL)
1592 			err(EXIT_FAILURE, "malloc failed");
1593 	}
1594 
1595 	slen = strlen(s);
1596 	append = 0;
1597 	b = t->buf;
1598 	for (c = b; c < b + t->sz;) {
1599 		len = strlen(c);
1600 		if (!append && len >= slen) {
1601 			r = c + (len - slen);
1602 			if (strcmp(r, s) == 0)
1603 				return;
1604 		} else if (len < slen && len != 0) {
1605 			r = s + (slen - len);
1606 			if (strcmp(c, r) == 0) {
1607 				t->sz -= len + 1;
1608 				memmove(c, c + len + 1, t->sz - (c - b));
1609 				append = 1;
1610 				continue;
1611 			}
1612 		}
1613 		c += len + 1;
1614 	}
1615 
1616 	while (t->sz + slen + 1 >= t->cap) {
1617 		t->cap *= 2;
1618 		if ((t->buf = realloc(t->buf, t->cap)) == NULL)
1619 			err(EXIT_FAILURE, "realloc failed");
1620 	}
1621 	b = t->buf;
1622 	strncpy(&b[t->sz], s, slen);
1623 	b[t->sz + slen] = '\0';
1624 	t->sz += slen + 1;
1625 }
1626 
1627 static int
1628 lookup_string(struct section *t, const char *s)
1629 {
1630 	const char	*b, *c, *r;
1631 	size_t		 len, slen;
1632 
1633 	slen = strlen(s);
1634 	b = t->buf;
1635 	for (c = b; c < b + t->sz;) {
1636 		len = strlen(c);
1637 		if (len >= slen) {
1638 			r = c + (len - slen);
1639 			if (strcmp(r, s) == 0)
1640 				return (r - b);
1641 		}
1642 		c += len + 1;
1643 	}
1644 
1645 	return (-1);
1646 }
1647 
1648 static uint32_t crctable[256] =
1649 {
1650 	0x00000000L, 0x77073096L, 0xEE0E612CL, 0x990951BAL,
1651 	0x076DC419L, 0x706AF48FL, 0xE963A535L, 0x9E6495A3L,
1652 	0x0EDB8832L, 0x79DCB8A4L, 0xE0D5E91EL, 0x97D2D988L,
1653 	0x09B64C2BL, 0x7EB17CBDL, 0xE7B82D07L, 0x90BF1D91L,
1654 	0x1DB71064L, 0x6AB020F2L, 0xF3B97148L, 0x84BE41DEL,
1655 	0x1ADAD47DL, 0x6DDDE4EBL, 0xF4D4B551L, 0x83D385C7L,
1656 	0x136C9856L, 0x646BA8C0L, 0xFD62F97AL, 0x8A65C9ECL,
1657 	0x14015C4FL, 0x63066CD9L, 0xFA0F3D63L, 0x8D080DF5L,
1658 	0x3B6E20C8L, 0x4C69105EL, 0xD56041E4L, 0xA2677172L,
1659 	0x3C03E4D1L, 0x4B04D447L, 0xD20D85FDL, 0xA50AB56BL,
1660 	0x35B5A8FAL, 0x42B2986CL, 0xDBBBC9D6L, 0xACBCF940L,
1661 	0x32D86CE3L, 0x45DF5C75L, 0xDCD60DCFL, 0xABD13D59L,
1662 	0x26D930ACL, 0x51DE003AL, 0xC8D75180L, 0xBFD06116L,
1663 	0x21B4F4B5L, 0x56B3C423L, 0xCFBA9599L, 0xB8BDA50FL,
1664 	0x2802B89EL, 0x5F058808L, 0xC60CD9B2L, 0xB10BE924L,
1665 	0x2F6F7C87L, 0x58684C11L, 0xC1611DABL, 0xB6662D3DL,
1666 	0x76DC4190L, 0x01DB7106L, 0x98D220BCL, 0xEFD5102AL,
1667 	0x71B18589L, 0x06B6B51FL, 0x9FBFE4A5L, 0xE8B8D433L,
1668 	0x7807C9A2L, 0x0F00F934L, 0x9609A88EL, 0xE10E9818L,
1669 	0x7F6A0DBBL, 0x086D3D2DL, 0x91646C97L, 0xE6635C01L,
1670 	0x6B6B51F4L, 0x1C6C6162L, 0x856530D8L, 0xF262004EL,
1671 	0x6C0695EDL, 0x1B01A57BL, 0x8208F4C1L, 0xF50FC457L,
1672 	0x65B0D9C6L, 0x12B7E950L, 0x8BBEB8EAL, 0xFCB9887CL,
1673 	0x62DD1DDFL, 0x15DA2D49L, 0x8CD37CF3L, 0xFBD44C65L,
1674 	0x4DB26158L, 0x3AB551CEL, 0xA3BC0074L, 0xD4BB30E2L,
1675 	0x4ADFA541L, 0x3DD895D7L, 0xA4D1C46DL, 0xD3D6F4FBL,
1676 	0x4369E96AL, 0x346ED9FCL, 0xAD678846L, 0xDA60B8D0L,
1677 	0x44042D73L, 0x33031DE5L, 0xAA0A4C5FL, 0xDD0D7CC9L,
1678 	0x5005713CL, 0x270241AAL, 0xBE0B1010L, 0xC90C2086L,
1679 	0x5768B525L, 0x206F85B3L, 0xB966D409L, 0xCE61E49FL,
1680 	0x5EDEF90EL, 0x29D9C998L, 0xB0D09822L, 0xC7D7A8B4L,
1681 	0x59B33D17L, 0x2EB40D81L, 0xB7BD5C3BL, 0xC0BA6CADL,
1682 	0xEDB88320L, 0x9ABFB3B6L, 0x03B6E20CL, 0x74B1D29AL,
1683 	0xEAD54739L, 0x9DD277AFL, 0x04DB2615L, 0x73DC1683L,
1684 	0xE3630B12L, 0x94643B84L, 0x0D6D6A3EL, 0x7A6A5AA8L,
1685 	0xE40ECF0BL, 0x9309FF9DL, 0x0A00AE27L, 0x7D079EB1L,
1686 	0xF00F9344L, 0x8708A3D2L, 0x1E01F268L, 0x6906C2FEL,
1687 	0xF762575DL, 0x806567CBL, 0x196C3671L, 0x6E6B06E7L,
1688 	0xFED41B76L, 0x89D32BE0L, 0x10DA7A5AL, 0x67DD4ACCL,
1689 	0xF9B9DF6FL, 0x8EBEEFF9L, 0x17B7BE43L, 0x60B08ED5L,
1690 	0xD6D6A3E8L, 0xA1D1937EL, 0x38D8C2C4L, 0x4FDFF252L,
1691 	0xD1BB67F1L, 0xA6BC5767L, 0x3FB506DDL, 0x48B2364BL,
1692 	0xD80D2BDAL, 0xAF0A1B4CL, 0x36034AF6L, 0x41047A60L,
1693 	0xDF60EFC3L, 0xA867DF55L, 0x316E8EEFL, 0x4669BE79L,
1694 	0xCB61B38CL, 0xBC66831AL, 0x256FD2A0L, 0x5268E236L,
1695 	0xCC0C7795L, 0xBB0B4703L, 0x220216B9L, 0x5505262FL,
1696 	0xC5BA3BBEL, 0xB2BD0B28L, 0x2BB45A92L, 0x5CB36A04L,
1697 	0xC2D7FFA7L, 0xB5D0CF31L, 0x2CD99E8BL, 0x5BDEAE1DL,
1698 	0x9B64C2B0L, 0xEC63F226L, 0x756AA39CL, 0x026D930AL,
1699 	0x9C0906A9L, 0xEB0E363FL, 0x72076785L, 0x05005713L,
1700 	0x95BF4A82L, 0xE2B87A14L, 0x7BB12BAEL, 0x0CB61B38L,
1701 	0x92D28E9BL, 0xE5D5BE0DL, 0x7CDCEFB7L, 0x0BDBDF21L,
1702 	0x86D3D2D4L, 0xF1D4E242L, 0x68DDB3F8L, 0x1FDA836EL,
1703 	0x81BE16CDL, 0xF6B9265BL, 0x6FB077E1L, 0x18B74777L,
1704 	0x88085AE6L, 0xFF0F6A70L, 0x66063BCAL, 0x11010B5CL,
1705 	0x8F659EFFL, 0xF862AE69L, 0x616BFFD3L, 0x166CCF45L,
1706 	0xA00AE278L, 0xD70DD2EEL, 0x4E048354L, 0x3903B3C2L,
1707 	0xA7672661L, 0xD06016F7L, 0x4969474DL, 0x3E6E77DBL,
1708 	0xAED16A4AL, 0xD9D65ADCL, 0x40DF0B66L, 0x37D83BF0L,
1709 	0xA9BCAE53L, 0xDEBB9EC5L, 0x47B2CF7FL, 0x30B5FFE9L,
1710 	0xBDBDF21CL, 0xCABAC28AL, 0x53B39330L, 0x24B4A3A6L,
1711 	0xBAD03605L, 0xCDD70693L, 0x54DE5729L, 0x23D967BFL,
1712 	0xB3667A2EL, 0xC4614AB8L, 0x5D681B02L, 0x2A6F2B94L,
1713 	0xB40BBE37L, 0xC30C8EA1L, 0x5A05DF1BL, 0x2D02EF8DL
1714 };
1715 
1716 static uint32_t
1717 calc_crc32(const char *p, size_t len, uint32_t crc)
1718 {
1719 	uint32_t i;
1720 
1721 	for (i = 0; i < len; i++) {
1722 		crc = crctable[(crc ^ *p++) & 0xFFL] ^ (crc >> 8);
1723 	}
1724 
1725 	return (crc ^ 0xFFFFFFFF);
1726 }
1727