1*f8fb3368SJohn Marino /*-
2*f8fb3368SJohn Marino * Copyright (c) 2006,2008 Joseph Koshy
3*f8fb3368SJohn Marino * All rights reserved.
4*f8fb3368SJohn Marino *
5*f8fb3368SJohn Marino * Redistribution and use in source and binary forms, with or without
6*f8fb3368SJohn Marino * modification, are permitted provided that the following conditions
7*f8fb3368SJohn Marino * are met:
8*f8fb3368SJohn Marino * 1. Redistributions of source code must retain the above copyright
9*f8fb3368SJohn Marino * notice, this list of conditions and the following disclaimer.
10*f8fb3368SJohn Marino * 2. Redistributions in binary form must reproduce the above copyright
11*f8fb3368SJohn Marino * notice, this list of conditions and the following disclaimer in the
12*f8fb3368SJohn Marino * documentation and/or other materials provided with the distribution.
13*f8fb3368SJohn Marino *
14*f8fb3368SJohn Marino * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
15*f8fb3368SJohn Marino * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16*f8fb3368SJohn Marino * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17*f8fb3368SJohn Marino * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
18*f8fb3368SJohn Marino * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19*f8fb3368SJohn Marino * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20*f8fb3368SJohn Marino * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21*f8fb3368SJohn Marino * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22*f8fb3368SJohn Marino * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23*f8fb3368SJohn Marino * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24*f8fb3368SJohn Marino * SUCH DAMAGE.
25*f8fb3368SJohn Marino */
26*f8fb3368SJohn Marino
27*f8fb3368SJohn Marino #include <assert.h>
28*f8fb3368SJohn Marino #include <libelf.h>
29*f8fb3368SJohn Marino
30*f8fb3368SJohn Marino #include "_libelf.h"
31*f8fb3368SJohn Marino
32*f8fb3368SJohn Marino ELFTC_VCSID("$Id: libelf_extended.c 3174 2015-03-27 17:13:41Z emaste $");
33*f8fb3368SJohn Marino
34*f8fb3368SJohn Marino /*
35*f8fb3368SJohn Marino * Retrieve section #0, allocating a new section if needed.
36*f8fb3368SJohn Marino */
37*f8fb3368SJohn Marino static Elf_Scn *
_libelf_getscn0(Elf * e)38*f8fb3368SJohn Marino _libelf_getscn0(Elf *e)
39*f8fb3368SJohn Marino {
40*f8fb3368SJohn Marino Elf_Scn *s;
41*f8fb3368SJohn Marino
42*f8fb3368SJohn Marino if ((s = STAILQ_FIRST(&e->e_u.e_elf.e_scn)) != NULL)
43*f8fb3368SJohn Marino return (s);
44*f8fb3368SJohn Marino
45*f8fb3368SJohn Marino return (_libelf_allocate_scn(e, (size_t) SHN_UNDEF));
46*f8fb3368SJohn Marino }
47*f8fb3368SJohn Marino
48*f8fb3368SJohn Marino int
_libelf_setshnum(Elf * e,void * eh,int ec,size_t shnum)49*f8fb3368SJohn Marino _libelf_setshnum(Elf *e, void *eh, int ec, size_t shnum)
50*f8fb3368SJohn Marino {
51*f8fb3368SJohn Marino Elf_Scn *scn;
52*f8fb3368SJohn Marino
53*f8fb3368SJohn Marino if (shnum >= SHN_LORESERVE) {
54*f8fb3368SJohn Marino if ((scn = _libelf_getscn0(e)) == NULL)
55*f8fb3368SJohn Marino return (0);
56*f8fb3368SJohn Marino
57*f8fb3368SJohn Marino assert(scn->s_ndx == SHN_UNDEF);
58*f8fb3368SJohn Marino
59*f8fb3368SJohn Marino if (ec == ELFCLASS32)
60*f8fb3368SJohn Marino scn->s_shdr.s_shdr32.sh_size = shnum;
61*f8fb3368SJohn Marino else
62*f8fb3368SJohn Marino scn->s_shdr.s_shdr64.sh_size = shnum;
63*f8fb3368SJohn Marino
64*f8fb3368SJohn Marino (void) elf_flagshdr(scn, ELF_C_SET, ELF_F_DIRTY);
65*f8fb3368SJohn Marino
66*f8fb3368SJohn Marino shnum = 0;
67*f8fb3368SJohn Marino }
68*f8fb3368SJohn Marino
69*f8fb3368SJohn Marino if (ec == ELFCLASS32)
70*f8fb3368SJohn Marino ((Elf32_Ehdr *) eh)->e_shnum = shnum & 0xFFFFU;
71*f8fb3368SJohn Marino else
72*f8fb3368SJohn Marino ((Elf64_Ehdr *) eh)->e_shnum = shnum & 0xFFFFU;
73*f8fb3368SJohn Marino
74*f8fb3368SJohn Marino
75*f8fb3368SJohn Marino return (1);
76*f8fb3368SJohn Marino }
77*f8fb3368SJohn Marino
78*f8fb3368SJohn Marino int
_libelf_setshstrndx(Elf * e,void * eh,int ec,size_t shstrndx)79*f8fb3368SJohn Marino _libelf_setshstrndx(Elf *e, void *eh, int ec, size_t shstrndx)
80*f8fb3368SJohn Marino {
81*f8fb3368SJohn Marino Elf_Scn *scn;
82*f8fb3368SJohn Marino
83*f8fb3368SJohn Marino if (shstrndx >= SHN_LORESERVE) {
84*f8fb3368SJohn Marino if ((scn = _libelf_getscn0(e)) == NULL)
85*f8fb3368SJohn Marino return (0);
86*f8fb3368SJohn Marino
87*f8fb3368SJohn Marino assert(scn->s_ndx == SHN_UNDEF);
88*f8fb3368SJohn Marino
89*f8fb3368SJohn Marino if (ec == ELFCLASS32)
90*f8fb3368SJohn Marino scn->s_shdr.s_shdr32.sh_link = shstrndx;
91*f8fb3368SJohn Marino else
92*f8fb3368SJohn Marino scn->s_shdr.s_shdr64.sh_link = shstrndx;
93*f8fb3368SJohn Marino
94*f8fb3368SJohn Marino (void) elf_flagshdr(scn, ELF_C_SET, ELF_F_DIRTY);
95*f8fb3368SJohn Marino
96*f8fb3368SJohn Marino shstrndx = SHN_XINDEX;
97*f8fb3368SJohn Marino }
98*f8fb3368SJohn Marino
99*f8fb3368SJohn Marino if (ec == ELFCLASS32)
100*f8fb3368SJohn Marino ((Elf32_Ehdr *) eh)->e_shstrndx = shstrndx & 0xFFFFU;
101*f8fb3368SJohn Marino else
102*f8fb3368SJohn Marino ((Elf64_Ehdr *) eh)->e_shstrndx = shstrndx & 0xFFFFU;
103*f8fb3368SJohn Marino
104*f8fb3368SJohn Marino return (1);
105*f8fb3368SJohn Marino }
106*f8fb3368SJohn Marino
107*f8fb3368SJohn Marino int
_libelf_setphnum(Elf * e,void * eh,int ec,size_t phnum)108*f8fb3368SJohn Marino _libelf_setphnum(Elf *e, void *eh, int ec, size_t phnum)
109*f8fb3368SJohn Marino {
110*f8fb3368SJohn Marino Elf_Scn *scn;
111*f8fb3368SJohn Marino
112*f8fb3368SJohn Marino if (phnum >= PN_XNUM) {
113*f8fb3368SJohn Marino if ((scn = _libelf_getscn0(e)) == NULL)
114*f8fb3368SJohn Marino return (0);
115*f8fb3368SJohn Marino
116*f8fb3368SJohn Marino assert(scn->s_ndx == SHN_UNDEF);
117*f8fb3368SJohn Marino
118*f8fb3368SJohn Marino if (ec == ELFCLASS32)
119*f8fb3368SJohn Marino scn->s_shdr.s_shdr32.sh_info = phnum;
120*f8fb3368SJohn Marino else
121*f8fb3368SJohn Marino scn->s_shdr.s_shdr64.sh_info = phnum;
122*f8fb3368SJohn Marino
123*f8fb3368SJohn Marino (void) elf_flagshdr(scn, ELF_C_SET, ELF_F_DIRTY);
124*f8fb3368SJohn Marino
125*f8fb3368SJohn Marino phnum = PN_XNUM;
126*f8fb3368SJohn Marino }
127*f8fb3368SJohn Marino
128*f8fb3368SJohn Marino if (ec == ELFCLASS32)
129*f8fb3368SJohn Marino ((Elf32_Ehdr *) eh)->e_phnum = phnum & 0xFFFFU;
130*f8fb3368SJohn Marino else
131*f8fb3368SJohn Marino ((Elf64_Ehdr *) eh)->e_phnum = phnum & 0xFFFFU;
132*f8fb3368SJohn Marino
133*f8fb3368SJohn Marino return (1);
134*f8fb3368SJohn Marino }
135