1*7304104dSAndroid Build Coastguard Worker /* Get ELF program header table.
2*7304104dSAndroid Build Coastguard Worker Copyright (C) 1998-2010, 2014, 2015 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker This file is part of elfutils.
4*7304104dSAndroid Build Coastguard Worker Written by Ulrich Drepper <[email protected]>, 1998.
5*7304104dSAndroid Build Coastguard Worker
6*7304104dSAndroid Build Coastguard Worker This file is free software; you can redistribute it and/or modify
7*7304104dSAndroid Build Coastguard Worker it under the terms of either
8*7304104dSAndroid Build Coastguard Worker
9*7304104dSAndroid Build Coastguard Worker * the GNU Lesser General Public License as published by the Free
10*7304104dSAndroid Build Coastguard Worker Software Foundation; either version 3 of the License, or (at
11*7304104dSAndroid Build Coastguard Worker your option) any later version
12*7304104dSAndroid Build Coastguard Worker
13*7304104dSAndroid Build Coastguard Worker or
14*7304104dSAndroid Build Coastguard Worker
15*7304104dSAndroid Build Coastguard Worker * the GNU General Public License as published by the Free
16*7304104dSAndroid Build Coastguard Worker Software Foundation; either version 2 of the License, or (at
17*7304104dSAndroid Build Coastguard Worker your option) any later version
18*7304104dSAndroid Build Coastguard Worker
19*7304104dSAndroid Build Coastguard Worker or both in parallel, as here.
20*7304104dSAndroid Build Coastguard Worker
21*7304104dSAndroid Build Coastguard Worker elfutils is distributed in the hope that it will be useful, but
22*7304104dSAndroid Build Coastguard Worker WITHOUT ANY WARRANTY; without even the implied warranty of
23*7304104dSAndroid Build Coastguard Worker MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
24*7304104dSAndroid Build Coastguard Worker General Public License for more details.
25*7304104dSAndroid Build Coastguard Worker
26*7304104dSAndroid Build Coastguard Worker You should have received copies of the GNU General Public License and
27*7304104dSAndroid Build Coastguard Worker the GNU Lesser General Public License along with this program. If
28*7304104dSAndroid Build Coastguard Worker not, see <http://www.gnu.org/licenses/>. */
29*7304104dSAndroid Build Coastguard Worker
30*7304104dSAndroid Build Coastguard Worker #ifdef HAVE_CONFIG_H
31*7304104dSAndroid Build Coastguard Worker # include <config.h>
32*7304104dSAndroid Build Coastguard Worker #endif
33*7304104dSAndroid Build Coastguard Worker
34*7304104dSAndroid Build Coastguard Worker #include <errno.h>
35*7304104dSAndroid Build Coastguard Worker #include <stdbool.h>
36*7304104dSAndroid Build Coastguard Worker #include <stdlib.h>
37*7304104dSAndroid Build Coastguard Worker #include <assert.h>
38*7304104dSAndroid Build Coastguard Worker
39*7304104dSAndroid Build Coastguard Worker #include "libelfP.h"
40*7304104dSAndroid Build Coastguard Worker #include "common.h"
41*7304104dSAndroid Build Coastguard Worker
42*7304104dSAndroid Build Coastguard Worker #ifndef LIBELFBITS
43*7304104dSAndroid Build Coastguard Worker # define LIBELFBITS 32
44*7304104dSAndroid Build Coastguard Worker #endif
45*7304104dSAndroid Build Coastguard Worker
ElfW2(LIBELFBITS,Phdr)46*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Phdr) *
47*7304104dSAndroid Build Coastguard Worker __elfw2(LIBELFBITS,getphdr_wrlock) (Elf *elf)
48*7304104dSAndroid Build Coastguard Worker {
49*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Phdr) *result;
50*7304104dSAndroid Build Coastguard Worker
51*7304104dSAndroid Build Coastguard Worker /* If the program header entry has already been filled in the code
52*7304104dSAndroid Build Coastguard Worker below must already have been run. So the class is set, too. No
53*7304104dSAndroid Build Coastguard Worker need to waste any more time here. */
54*7304104dSAndroid Build Coastguard Worker result = elf->state.ELFW(elf,LIBELFBITS).phdr;
55*7304104dSAndroid Build Coastguard Worker if (likely (result != NULL))
56*7304104dSAndroid Build Coastguard Worker return result;
57*7304104dSAndroid Build Coastguard Worker
58*7304104dSAndroid Build Coastguard Worker if (elf->class == 0)
59*7304104dSAndroid Build Coastguard Worker elf->class = ELFW(ELFCLASS,LIBELFBITS);
60*7304104dSAndroid Build Coastguard Worker else if (elf->class != ELFW(ELFCLASS,LIBELFBITS))
61*7304104dSAndroid Build Coastguard Worker {
62*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_INVALID_CLASS);
63*7304104dSAndroid Build Coastguard Worker result = NULL;
64*7304104dSAndroid Build Coastguard Worker goto out;
65*7304104dSAndroid Build Coastguard Worker }
66*7304104dSAndroid Build Coastguard Worker
67*7304104dSAndroid Build Coastguard Worker if (likely (result == NULL))
68*7304104dSAndroid Build Coastguard Worker {
69*7304104dSAndroid Build Coastguard Worker /* Read the section header table. */
70*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Ehdr) *ehdr = elf->state.ELFW(elf,LIBELFBITS).ehdr;
71*7304104dSAndroid Build Coastguard Worker
72*7304104dSAndroid Build Coastguard Worker /* If no program header exists return NULL. */
73*7304104dSAndroid Build Coastguard Worker size_t phnum;
74*7304104dSAndroid Build Coastguard Worker if (__elf_getphdrnum_rdlock (elf, &phnum) != 0)
75*7304104dSAndroid Build Coastguard Worker goto out;
76*7304104dSAndroid Build Coastguard Worker if (phnum == 0 || ehdr->e_phoff == 0)
77*7304104dSAndroid Build Coastguard Worker {
78*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_NO_PHDR);
79*7304104dSAndroid Build Coastguard Worker goto out;
80*7304104dSAndroid Build Coastguard Worker }
81*7304104dSAndroid Build Coastguard Worker
82*7304104dSAndroid Build Coastguard Worker /* Check this doesn't overflow. */
83*7304104dSAndroid Build Coastguard Worker size_t size = phnum * sizeof (ElfW2(LIBELFBITS,Phdr));
84*7304104dSAndroid Build Coastguard Worker
85*7304104dSAndroid Build Coastguard Worker if (phnum > SIZE_MAX / sizeof (ElfW2(LIBELFBITS,Phdr))
86*7304104dSAndroid Build Coastguard Worker || ehdr->e_phoff > elf->maximum_size
87*7304104dSAndroid Build Coastguard Worker || elf->maximum_size - ehdr->e_phoff < size)
88*7304104dSAndroid Build Coastguard Worker {
89*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_INVALID_DATA);
90*7304104dSAndroid Build Coastguard Worker goto out;
91*7304104dSAndroid Build Coastguard Worker }
92*7304104dSAndroid Build Coastguard Worker
93*7304104dSAndroid Build Coastguard Worker if (elf->map_address != NULL)
94*7304104dSAndroid Build Coastguard Worker {
95*7304104dSAndroid Build Coastguard Worker /* First see whether the information in the ELF header is
96*7304104dSAndroid Build Coastguard Worker valid and it does not ask for too much. */
97*7304104dSAndroid Build Coastguard Worker if (unlikely (ehdr->e_phoff >= elf->maximum_size)
98*7304104dSAndroid Build Coastguard Worker || unlikely (elf->maximum_size - ehdr->e_phoff < size))
99*7304104dSAndroid Build Coastguard Worker {
100*7304104dSAndroid Build Coastguard Worker /* Something is wrong. */
101*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_INVALID_PHDR);
102*7304104dSAndroid Build Coastguard Worker goto out;
103*7304104dSAndroid Build Coastguard Worker }
104*7304104dSAndroid Build Coastguard Worker
105*7304104dSAndroid Build Coastguard Worker /* All the data is already mapped. Use it. */
106*7304104dSAndroid Build Coastguard Worker void *file_phdr = ((char *) elf->map_address
107*7304104dSAndroid Build Coastguard Worker + elf->start_offset + ehdr->e_phoff);
108*7304104dSAndroid Build Coastguard Worker if (ehdr->e_ident[EI_DATA] == MY_ELFDATA
109*7304104dSAndroid Build Coastguard Worker && (ALLOW_UNALIGNED
110*7304104dSAndroid Build Coastguard Worker || ((uintptr_t) file_phdr
111*7304104dSAndroid Build Coastguard Worker & (__alignof__ (ElfW2(LIBELFBITS,Phdr)) - 1)) == 0))
112*7304104dSAndroid Build Coastguard Worker /* Simply use the mapped data. */
113*7304104dSAndroid Build Coastguard Worker elf->state.ELFW(elf,LIBELFBITS).phdr = file_phdr;
114*7304104dSAndroid Build Coastguard Worker else
115*7304104dSAndroid Build Coastguard Worker {
116*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Phdr) *notcvt;
117*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Phdr) *phdr;
118*7304104dSAndroid Build Coastguard Worker
119*7304104dSAndroid Build Coastguard Worker /* Allocate memory for the program headers. We know the number
120*7304104dSAndroid Build Coastguard Worker of entries from the ELF header. */
121*7304104dSAndroid Build Coastguard Worker phdr = elf->state.ELFW(elf,LIBELFBITS).phdr =
122*7304104dSAndroid Build Coastguard Worker (ElfW2(LIBELFBITS,Phdr) *) malloc (size);
123*7304104dSAndroid Build Coastguard Worker if (elf->state.ELFW(elf,LIBELFBITS).phdr == NULL)
124*7304104dSAndroid Build Coastguard Worker {
125*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_NOMEM);
126*7304104dSAndroid Build Coastguard Worker goto out;
127*7304104dSAndroid Build Coastguard Worker }
128*7304104dSAndroid Build Coastguard Worker elf->state.ELFW(elf,LIBELFBITS).phdr_flags |=
129*7304104dSAndroid Build Coastguard Worker ELF_F_MALLOCED | ELF_F_DIRTY;
130*7304104dSAndroid Build Coastguard Worker
131*7304104dSAndroid Build Coastguard Worker /* Now copy the data and at the same time convert the
132*7304104dSAndroid Build Coastguard Worker byte order. */
133*7304104dSAndroid Build Coastguard Worker
134*7304104dSAndroid Build Coastguard Worker if (ehdr->e_ident[EI_DATA] == MY_ELFDATA)
135*7304104dSAndroid Build Coastguard Worker {
136*7304104dSAndroid Build Coastguard Worker assert (! ALLOW_UNALIGNED);
137*7304104dSAndroid Build Coastguard Worker memcpy (phdr, file_phdr, size);
138*7304104dSAndroid Build Coastguard Worker }
139*7304104dSAndroid Build Coastguard Worker else
140*7304104dSAndroid Build Coastguard Worker {
141*7304104dSAndroid Build Coastguard Worker bool copy = ! (ALLOW_UNALIGNED
142*7304104dSAndroid Build Coastguard Worker || ((uintptr_t) file_phdr
143*7304104dSAndroid Build Coastguard Worker & (__alignof__ (ElfW2(LIBELFBITS,Phdr))
144*7304104dSAndroid Build Coastguard Worker - 1)) == 0);
145*7304104dSAndroid Build Coastguard Worker if (! copy)
146*7304104dSAndroid Build Coastguard Worker notcvt = file_phdr;
147*7304104dSAndroid Build Coastguard Worker else
148*7304104dSAndroid Build Coastguard Worker {
149*7304104dSAndroid Build Coastguard Worker notcvt = (ElfW2(LIBELFBITS,Phdr) *) malloc (size);
150*7304104dSAndroid Build Coastguard Worker if (unlikely (notcvt == NULL))
151*7304104dSAndroid Build Coastguard Worker {
152*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_NOMEM);
153*7304104dSAndroid Build Coastguard Worker goto out;
154*7304104dSAndroid Build Coastguard Worker }
155*7304104dSAndroid Build Coastguard Worker memcpy (notcvt, file_phdr, size);
156*7304104dSAndroid Build Coastguard Worker }
157*7304104dSAndroid Build Coastguard Worker
158*7304104dSAndroid Build Coastguard Worker for (size_t cnt = 0; cnt < phnum; ++cnt)
159*7304104dSAndroid Build Coastguard Worker {
160*7304104dSAndroid Build Coastguard Worker CONVERT_TO (phdr[cnt].p_type, notcvt[cnt].p_type);
161*7304104dSAndroid Build Coastguard Worker CONVERT_TO (phdr[cnt].p_offset, notcvt[cnt].p_offset);
162*7304104dSAndroid Build Coastguard Worker CONVERT_TO (phdr[cnt].p_vaddr, notcvt[cnt].p_vaddr);
163*7304104dSAndroid Build Coastguard Worker CONVERT_TO (phdr[cnt].p_paddr, notcvt[cnt].p_paddr);
164*7304104dSAndroid Build Coastguard Worker CONVERT_TO (phdr[cnt].p_filesz, notcvt[cnt].p_filesz);
165*7304104dSAndroid Build Coastguard Worker CONVERT_TO (phdr[cnt].p_memsz, notcvt[cnt].p_memsz);
166*7304104dSAndroid Build Coastguard Worker CONVERT_TO (phdr[cnt].p_flags, notcvt[cnt].p_flags);
167*7304104dSAndroid Build Coastguard Worker CONVERT_TO (phdr[cnt].p_align, notcvt[cnt].p_align);
168*7304104dSAndroid Build Coastguard Worker }
169*7304104dSAndroid Build Coastguard Worker
170*7304104dSAndroid Build Coastguard Worker if (copy)
171*7304104dSAndroid Build Coastguard Worker free (notcvt);
172*7304104dSAndroid Build Coastguard Worker }
173*7304104dSAndroid Build Coastguard Worker }
174*7304104dSAndroid Build Coastguard Worker }
175*7304104dSAndroid Build Coastguard Worker else if (likely (elf->fildes != -1))
176*7304104dSAndroid Build Coastguard Worker {
177*7304104dSAndroid Build Coastguard Worker /* Allocate memory for the program headers. We know the number
178*7304104dSAndroid Build Coastguard Worker of entries from the ELF header. */
179*7304104dSAndroid Build Coastguard Worker elf->state.ELFW(elf,LIBELFBITS).phdr =
180*7304104dSAndroid Build Coastguard Worker (ElfW2(LIBELFBITS,Phdr) *) malloc (size);
181*7304104dSAndroid Build Coastguard Worker if (elf->state.ELFW(elf,LIBELFBITS).phdr == NULL)
182*7304104dSAndroid Build Coastguard Worker {
183*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_NOMEM);
184*7304104dSAndroid Build Coastguard Worker goto out;
185*7304104dSAndroid Build Coastguard Worker }
186*7304104dSAndroid Build Coastguard Worker elf->state.ELFW(elf,LIBELFBITS).phdr_flags |= ELF_F_MALLOCED;
187*7304104dSAndroid Build Coastguard Worker
188*7304104dSAndroid Build Coastguard Worker /* Read the header. */
189*7304104dSAndroid Build Coastguard Worker ssize_t n = pread_retry (elf->fildes,
190*7304104dSAndroid Build Coastguard Worker elf->state.ELFW(elf,LIBELFBITS).phdr, size,
191*7304104dSAndroid Build Coastguard Worker elf->start_offset + ehdr->e_phoff);
192*7304104dSAndroid Build Coastguard Worker if (unlikely ((size_t) n != size))
193*7304104dSAndroid Build Coastguard Worker {
194*7304104dSAndroid Build Coastguard Worker /* Severe problems. We cannot read the data. */
195*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_READ_ERROR);
196*7304104dSAndroid Build Coastguard Worker free (elf->state.ELFW(elf,LIBELFBITS).phdr);
197*7304104dSAndroid Build Coastguard Worker elf->state.ELFW(elf,LIBELFBITS).phdr = NULL;
198*7304104dSAndroid Build Coastguard Worker goto out;
199*7304104dSAndroid Build Coastguard Worker }
200*7304104dSAndroid Build Coastguard Worker
201*7304104dSAndroid Build Coastguard Worker /* If the byte order of the file is not the same as the one
202*7304104dSAndroid Build Coastguard Worker of the host convert the data now. */
203*7304104dSAndroid Build Coastguard Worker if (ehdr->e_ident[EI_DATA] != MY_ELFDATA)
204*7304104dSAndroid Build Coastguard Worker {
205*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Phdr) *phdr
206*7304104dSAndroid Build Coastguard Worker = elf->state.ELFW(elf,LIBELFBITS).phdr;
207*7304104dSAndroid Build Coastguard Worker
208*7304104dSAndroid Build Coastguard Worker for (size_t cnt = 0; cnt < phnum; ++cnt)
209*7304104dSAndroid Build Coastguard Worker {
210*7304104dSAndroid Build Coastguard Worker CONVERT (phdr[cnt].p_type);
211*7304104dSAndroid Build Coastguard Worker CONVERT (phdr[cnt].p_offset);
212*7304104dSAndroid Build Coastguard Worker CONVERT (phdr[cnt].p_vaddr);
213*7304104dSAndroid Build Coastguard Worker CONVERT (phdr[cnt].p_paddr);
214*7304104dSAndroid Build Coastguard Worker CONVERT (phdr[cnt].p_filesz);
215*7304104dSAndroid Build Coastguard Worker CONVERT (phdr[cnt].p_memsz);
216*7304104dSAndroid Build Coastguard Worker CONVERT (phdr[cnt].p_flags);
217*7304104dSAndroid Build Coastguard Worker CONVERT (phdr[cnt].p_align);
218*7304104dSAndroid Build Coastguard Worker }
219*7304104dSAndroid Build Coastguard Worker }
220*7304104dSAndroid Build Coastguard Worker }
221*7304104dSAndroid Build Coastguard Worker else
222*7304104dSAndroid Build Coastguard Worker {
223*7304104dSAndroid Build Coastguard Worker /* The file descriptor was already enabled and not all data was
224*7304104dSAndroid Build Coastguard Worker read. */
225*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_FD_DISABLED);
226*7304104dSAndroid Build Coastguard Worker goto out;
227*7304104dSAndroid Build Coastguard Worker }
228*7304104dSAndroid Build Coastguard Worker
229*7304104dSAndroid Build Coastguard Worker result = elf->state.ELFW(elf,LIBELFBITS).phdr;
230*7304104dSAndroid Build Coastguard Worker }
231*7304104dSAndroid Build Coastguard Worker
232*7304104dSAndroid Build Coastguard Worker out:
233*7304104dSAndroid Build Coastguard Worker return result;
234*7304104dSAndroid Build Coastguard Worker }
235*7304104dSAndroid Build Coastguard Worker
ElfW2(LIBELFBITS,Phdr)236*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Phdr) *
237*7304104dSAndroid Build Coastguard Worker elfw2(LIBELFBITS,getphdr) (Elf *elf)
238*7304104dSAndroid Build Coastguard Worker {
239*7304104dSAndroid Build Coastguard Worker ElfW2(LIBELFBITS,Phdr) *result;
240*7304104dSAndroid Build Coastguard Worker
241*7304104dSAndroid Build Coastguard Worker if (elf == NULL)
242*7304104dSAndroid Build Coastguard Worker return NULL;
243*7304104dSAndroid Build Coastguard Worker
244*7304104dSAndroid Build Coastguard Worker if (unlikely (elf->kind != ELF_K_ELF))
245*7304104dSAndroid Build Coastguard Worker {
246*7304104dSAndroid Build Coastguard Worker __libelf_seterrno (ELF_E_INVALID_HANDLE);
247*7304104dSAndroid Build Coastguard Worker return NULL;
248*7304104dSAndroid Build Coastguard Worker }
249*7304104dSAndroid Build Coastguard Worker
250*7304104dSAndroid Build Coastguard Worker /* If the program header entry has already been filled in the code
251*7304104dSAndroid Build Coastguard Worker * in getphdr_wrlock must already have been run. So the class is
252*7304104dSAndroid Build Coastguard Worker * set, too. No need to waste any more time here. */
253*7304104dSAndroid Build Coastguard Worker result = elf->state.ELFW(elf,LIBELFBITS).phdr;
254*7304104dSAndroid Build Coastguard Worker if (likely (result != NULL))
255*7304104dSAndroid Build Coastguard Worker return result;
256*7304104dSAndroid Build Coastguard Worker
257*7304104dSAndroid Build Coastguard Worker rwlock_wrlock (elf->lock);
258*7304104dSAndroid Build Coastguard Worker result = __elfw2(LIBELFBITS,getphdr_wrlock) (elf);
259*7304104dSAndroid Build Coastguard Worker rwlock_unlock (elf->lock);
260*7304104dSAndroid Build Coastguard Worker
261*7304104dSAndroid Build Coastguard Worker return result;
262*7304104dSAndroid Build Coastguard Worker }
263*7304104dSAndroid Build Coastguard Worker INTDEF(elfw2(LIBELFBITS,getphdr))
264