Branch data Line data Source code
1 : : /* Create descriptor for processing file.
2 : : Copyright (C) 1998-2010, 2012, 2014, 2015, 2016 Red Hat, Inc.
3 : : Copyright (C) 2021, 2022 Mark J. Wielaard <mark@klomp.org>
4 : : This file is part of elfutils.
5 : : Written by Ulrich Drepper <drepper@redhat.com>, 1998.
6 : :
7 : : This file is free software; you can redistribute it and/or modify
8 : : it under the terms of either
9 : :
10 : : * the GNU Lesser General Public License as published by the Free
11 : : Software Foundation; either version 3 of the License, or (at
12 : : your option) any later version
13 : :
14 : : or
15 : :
16 : : * the GNU General Public License as published by the Free
17 : : Software Foundation; either version 2 of the License, or (at
18 : : your option) any later version
19 : :
20 : : or both in parallel, as here.
21 : :
22 : : elfutils is distributed in the hope that it will be useful, but
23 : : WITHOUT ANY WARRANTY; without even the implied warranty of
24 : : MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
25 : : General Public License for more details.
26 : :
27 : : You should have received copies of the GNU General Public License and
28 : : the GNU Lesser General Public License along with this program. If
29 : : not, see <http://www.gnu.org/licenses/>. */
30 : :
31 : : #ifdef HAVE_CONFIG_H
32 : : # include <config.h>
33 : : #endif
34 : :
35 : : #include <assert.h>
36 : : #include <ctype.h>
37 : : #include <errno.h>
38 : : #include <fcntl.h>
39 : : #include <stdbool.h>
40 : : #include <stddef.h>
41 : : #include <string.h>
42 : : #include <sys/stat.h>
43 : :
44 : : #include "libelfP.h"
45 : : #include "common.h"
46 : :
47 : :
48 : : /* Create descriptor for archive in memory. */
49 : : static inline Elf *
50 : 314 : file_read_ar (int fildes, void *map_address, off_t offset, size_t maxsize,
51 : : Elf_Cmd cmd, Elf *parent)
52 : : {
53 : 314 : Elf *elf;
54 : :
55 : : /* Create a descriptor. */
56 : 314 : elf = allocate_elf (fildes, map_address, offset, maxsize, cmd, parent,
57 : : ELF_K_AR, 0);
58 [ + - ]: 314 : if (elf != NULL)
59 : : {
60 : : /* We don't read all the symbol tables in advance. All this will
61 : : happen on demand. */
62 : 314 : elf->state.ar.offset = offset + SARMAG;
63 : :
64 : 314 : elf->state.ar.elf_ar_hdr.ar_rawname = elf->state.ar.raw_name;
65 : : }
66 : :
67 : 314 : return elf;
68 : : }
69 : :
70 : :
71 : : static size_t
72 : 31248 : get_shnum (void *map_address, unsigned char *e_ident, int fildes,
73 : : int64_t offset, size_t maxsize)
74 : : {
75 : 31248 : size_t result;
76 : 31248 : union
77 : : {
78 : : Elf32_Ehdr *e32;
79 : : Elf64_Ehdr *e64;
80 : : void *p;
81 : : } ehdr;
82 : 31248 : union
83 : : {
84 : : Elf32_Ehdr e32;
85 : : Elf64_Ehdr e64;
86 : : } ehdr_mem;
87 : 31248 : bool is32 = e_ident[EI_CLASS] == ELFCLASS32;
88 : :
89 [ + - ]: 31248 : if ((is32 && maxsize < sizeof (Elf32_Ehdr))
90 [ - + ]: 31248 : || (!is32 && maxsize < sizeof (Elf64_Ehdr)))
91 : : {
92 : 0 : __libelf_seterrno (ELF_E_INVALID_ELF);
93 : 0 : return (size_t) -1l;
94 : : }
95 : :
96 : : /* Make the ELF header available. */
97 [ + + ]: 31248 : if (e_ident[EI_DATA] == MY_ELFDATA
98 : : && (ALLOW_UNALIGNED
99 : : || (((size_t) e_ident
100 : : & ((is32 ? __alignof__ (Elf32_Ehdr) : __alignof__ (Elf64_Ehdr))
101 : : - 1)) == 0)))
102 : : ehdr.p = e_ident;
103 : : else
104 : : {
105 : : /* We already read the ELF header. We have to copy the header
106 : : since we possibly modify the data here and the caller
107 : : expects the memory it passes in to be preserved. */
108 : 1668 : ehdr.p = &ehdr_mem;
109 : :
110 [ + + ]: 1668 : if (is32)
111 : : {
112 : 790 : if (ALLOW_UNALIGNED)
113 : : {
114 : 790 : ehdr_mem.e32.e_shnum = ((Elf32_Ehdr *) e_ident)->e_shnum;
115 : 790 : ehdr_mem.e32.e_shoff = ((Elf32_Ehdr *) e_ident)->e_shoff;
116 : : }
117 : : else
118 : : memcpy (&ehdr_mem, e_ident, sizeof (Elf32_Ehdr));
119 : :
120 : 790 : if (e_ident[EI_DATA] != MY_ELFDATA)
121 : : {
122 : 790 : CONVERT (ehdr_mem.e32.e_shnum);
123 : 790 : CONVERT (ehdr_mem.e32.e_shoff);
124 : : }
125 : : }
126 : : else
127 : : {
128 : 878 : if (ALLOW_UNALIGNED)
129 : : {
130 : 878 : ehdr_mem.e64.e_shnum = ((Elf64_Ehdr *) e_ident)->e_shnum;
131 : 878 : ehdr_mem.e64.e_shoff = ((Elf64_Ehdr *) e_ident)->e_shoff;
132 : : }
133 : : else
134 : : memcpy (&ehdr_mem, e_ident, sizeof (Elf64_Ehdr));
135 : :
136 : 878 : if (e_ident[EI_DATA] != MY_ELFDATA)
137 : : {
138 : 878 : CONVERT (ehdr_mem.e64.e_shnum);
139 : 878 : CONVERT (ehdr_mem.e64.e_shoff);
140 : : }
141 : : }
142 : : }
143 : :
144 [ + + ]: 31248 : if (is32)
145 : : {
146 : : /* Get the number of sections from the ELF header. */
147 : 2208 : result = ehdr.e32->e_shnum;
148 : :
149 [ + + + + ]: 2208 : if (unlikely (result == 0) && ehdr.e32->e_shoff != 0)
150 : : {
151 [ + - ]: 36 : if (unlikely (ehdr.e32->e_shoff >= maxsize)
152 [ - + ]: 36 : || unlikely (maxsize - ehdr.e32->e_shoff < sizeof (Elf32_Shdr)))
153 : : /* Cannot read the first section header. */
154 : : return 0;
155 : :
156 [ + + ]: 36 : if (likely (map_address != NULL) && e_ident[EI_DATA] == MY_ELFDATA
157 [ + + ]: 28 : && (ALLOW_UNALIGNED
158 : : || (((size_t) ((char *) (map_address + ehdr.e32->e_shoff
159 : : + offset)))
160 : : & (__alignof__ (Elf32_Shdr) - 1)) == 0))
161 : : /* We can directly access the memory. */
162 : 22 : result = ((Elf32_Shdr *) ((char *) map_address + ehdr.e32->e_shoff
163 : 22 : + offset))->sh_size;
164 : : else
165 : : {
166 : 14 : Elf32_Word size;
167 : 14 : ssize_t r;
168 : :
169 [ + + ]: 14 : if (likely (map_address != NULL))
170 : : /* gcc will optimize the memcpy to a simple memory
171 : : access while taking care of alignment issues. */
172 : 20 : memcpy (&size, ((char *) map_address
173 : : + ehdr.e32->e_shoff
174 : 6 : + offset
175 : 6 : + offsetof (Elf32_Shdr, sh_size)),
176 : : sizeof (Elf32_Word));
177 : : else
178 [ - + ]: 8 : if (unlikely ((r = pread_retry (fildes, &size,
179 : : sizeof (Elf32_Word),
180 : : offset + ehdr.e32->e_shoff
181 : : + offsetof (Elf32_Shdr,
182 : : sh_size)))
183 : : != sizeof (Elf32_Word)))
184 : : {
185 [ # # ]: 0 : if (r < 0)
186 : 0 : __libelf_seterrno (ELF_E_INVALID_FILE);
187 : : else
188 : 0 : __libelf_seterrno (ELF_E_INVALID_ELF);
189 : 0 : return (size_t) -1l;
190 : : }
191 : :
192 [ + + ]: 14 : if (e_ident[EI_DATA] != MY_ELFDATA)
193 : 6 : CONVERT (size);
194 : :
195 : 14 : result = size;
196 : : }
197 : : }
198 : :
199 : : /* If the section headers were truncated, pretend none were there. */
200 [ - + ]: 2208 : if (ehdr.e32->e_shoff > maxsize
201 [ - + ]: 2208 : || maxsize - ehdr.e32->e_shoff < sizeof (Elf32_Shdr) * result)
202 : 6 : result = 0;
203 : : }
204 : : else
205 : : {
206 : : /* Get the number of sections from the ELF header. */
207 : 29040 : result = ehdr.e64->e_shnum;
208 : :
209 [ + + + + ]: 29040 : if (unlikely (result == 0) && ehdr.e64->e_shoff != 0)
210 : : {
211 [ + - ]: 14 : if (unlikely (ehdr.e64->e_shoff >= maxsize)
212 [ + - ]: 14 : || unlikely (ehdr.e64->e_shoff + sizeof (Elf64_Shdr) > maxsize))
213 : : /* Cannot read the first section header. */
214 : 0 : return 0;
215 : :
216 : 14 : Elf64_Xword size;
217 [ + - ]: 14 : if (likely (map_address != NULL) && e_ident[EI_DATA] == MY_ELFDATA
218 [ + + ]: 14 : && (ALLOW_UNALIGNED
219 : : || (((size_t) ((char *) (map_address + ehdr.e64->e_shoff
220 : : + offset)))
221 : : & (__alignof__ (Elf64_Shdr) - 1)) == 0))
222 : : /* We can directly access the memory. */
223 : 8 : size = ((Elf64_Shdr *) ((char *) map_address + ehdr.e64->e_shoff
224 : 8 : + offset))->sh_size;
225 : : else
226 : : {
227 : 6 : ssize_t r;
228 [ + - ]: 6 : if (likely (map_address != NULL))
229 : : /* gcc will optimize the memcpy to a simple memory
230 : : access while taking care of alignment issues. */
231 : 12 : memcpy (&size, ((char *) map_address
232 : : + ehdr.e64->e_shoff
233 : 6 : + offset
234 : 6 : + offsetof (Elf64_Shdr, sh_size)),
235 : : sizeof (Elf64_Xword));
236 : : else
237 [ # # ]: 0 : if (unlikely ((r = pread_retry (fildes, &size,
238 : : sizeof (Elf64_Xword),
239 : : offset + ehdr.e64->e_shoff
240 : : + offsetof (Elf64_Shdr,
241 : : sh_size)))
242 : : != sizeof (Elf64_Xword)))
243 : : {
244 [ # # ]: 0 : if (r < 0)
245 : 0 : __libelf_seterrno (ELF_E_INVALID_FILE);
246 : : else
247 : 0 : __libelf_seterrno (ELF_E_INVALID_ELF);
248 : 0 : return (size_t) -1l;
249 : : }
250 : :
251 [ + - ]: 6 : if (e_ident[EI_DATA] != MY_ELFDATA)
252 : 6 : CONVERT (size);
253 : : }
254 : :
255 : : /* Although sh_size is an Elf64_Xword and can contain a 64bit
256 : : value, we only expect an 32bit value max. GElf_Word is
257 : : 32bit unsigned. */
258 [ - + ]: 14 : if (size > ~((GElf_Word) 0))
259 : : {
260 : : /* Invalid value, it is too large. */
261 : 0 : __libelf_seterrno (ELF_E_INVALID_ELF);
262 : 0 : return (size_t) -1l;
263 : : }
264 : :
265 : 14 : result = size;
266 : : }
267 : :
268 : : /* If the section headers were truncated, pretend none were there. */
269 [ + + ]: 29040 : if (ehdr.e64->e_shoff > maxsize
270 [ - + ]: 29034 : || maxsize - ehdr.e64->e_shoff < sizeof (Elf64_Shdr) * result)
271 : 6 : result = 0;
272 : : }
273 : :
274 : : return result;
275 : : }
276 : :
277 : :
278 : : /* Create descriptor for ELF file in memory. */
279 : : static Elf *
280 : 31248 : file_read_elf (int fildes, void *map_address, unsigned char *e_ident,
281 : : int64_t offset, size_t maxsize, Elf_Cmd cmd, Elf *parent)
282 : : {
283 : : /* Verify the binary is of the class we can handle. */
284 [ + - - + ]: 31248 : if (unlikely ((e_ident[EI_CLASS] != ELFCLASS32
285 : : && e_ident[EI_CLASS] != ELFCLASS64)
286 : : /* We also can only handle two encodings. */
287 : : || (e_ident[EI_DATA] != ELFDATA2LSB
288 : : && e_ident[EI_DATA] != ELFDATA2MSB)))
289 : : {
290 : : /* Cannot handle this. */
291 : 0 : __libelf_seterrno (ELF_E_INVALID_ELF);
292 : 0 : return NULL;
293 : : }
294 : :
295 : : /* Determine the number of sections. Returns -1 and sets libelf errno
296 : : if the file handle or elf file is invalid. Returns zero if there
297 : : are no section headers (or they cannot be read). */
298 : 31248 : size_t scncnt = get_shnum (map_address, e_ident, fildes, offset, maxsize);
299 [ - + ]: 31247 : if (scncnt == (size_t) -1l)
300 : : /* Could not determine the number of sections. */
301 : : return NULL;
302 : :
303 : : /* Check for too many sections. */
304 [ + + ]: 31247 : if (e_ident[EI_CLASS] == ELFCLASS32)
305 : : {
306 [ - + ]: 2208 : if (scncnt > SIZE_MAX / (sizeof (Elf_Scn) + sizeof (Elf32_Shdr)))
307 : : {
308 : 0 : __libelf_seterrno (ELF_E_INVALID_ELF);
309 : 0 : return NULL;
310 : : }
311 : : }
312 [ - + ]: 29039 : else if (scncnt > SIZE_MAX / (sizeof (Elf_Scn) + sizeof (Elf64_Shdr)))
313 : : {
314 : 0 : __libelf_seterrno (ELF_E_INVALID_ELF);
315 : 0 : return NULL;
316 : : }
317 : :
318 : : /* We can now allocate the memory. Even if there are no section headers,
319 : : we allocate space for a zeroth section in case we need it later. */
320 [ + + ]: 31247 : const size_t scnmax = (scncnt ?: (cmd == ELF_C_RDWR || cmd == ELF_C_RDWR_MMAP)
321 : 388 : ? 1 : 0);
322 : 31247 : Elf *elf = allocate_elf (fildes, map_address, offset, maxsize, cmd, parent,
323 : : ELF_K_ELF, scnmax * sizeof (Elf_Scn));
324 [ - + ]: 31248 : if (elf == NULL)
325 : : /* Not enough memory. allocate_elf will have set libelf errno. */
326 : : return NULL;
327 : :
328 [ - + ]: 31248 : assert ((unsigned int) scncnt == scncnt);
329 : 31248 : assert (offsetof (struct Elf, state.elf32.scns)
330 : : == offsetof (struct Elf, state.elf64.scns));
331 : 31248 : elf->state.elf32.scns.cnt = scncnt;
332 : 31248 : elf->state.elf32.scns.max = scnmax;
333 : :
334 : : /* Some more or less arbitrary value. */
335 : 31248 : elf->state.elf.scnincr = 10;
336 : :
337 : : /* Make the class easily available. */
338 : 31248 : elf->class = e_ident[EI_CLASS];
339 : :
340 [ + + ]: 31248 : if (e_ident[EI_CLASS] == ELFCLASS32)
341 : : {
342 : : /* This pointer might not be directly usable if the alignment is
343 : : not sufficient for the architecture. */
344 : 2208 : uintptr_t ehdr = (uintptr_t) map_address + offset;
345 : :
346 : : /* This is a 32-bit binary. */
347 [ + + ]: 2208 : if (map_address != NULL && e_ident[EI_DATA] == MY_ELFDATA
348 [ + + ]: 1316 : && (ALLOW_UNALIGNED
349 : : || (ehdr & (__alignof__ (Elf32_Ehdr) - 1)) == 0))
350 : : {
351 : : /* We can use the mmapped memory. */
352 : 860 : elf->state.elf32.ehdr = (Elf32_Ehdr *) ehdr;
353 : : }
354 : : else
355 : : {
356 : : /* Copy the ELF header. */
357 [ + + ]: 1348 : elf->state.elf32.ehdr = memcpy (&elf->state.elf32.ehdr_mem, e_ident,
358 : : sizeof (Elf32_Ehdr));
359 : :
360 [ + + ]: 1348 : if (e_ident[EI_DATA] != MY_ELFDATA)
361 : : {
362 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_type);
363 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_machine);
364 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_version);
365 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_entry);
366 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_phoff);
367 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_shoff);
368 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_flags);
369 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_ehsize);
370 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_phentsize);
371 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_phnum);
372 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_shentsize);
373 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_shnum);
374 : 790 : CONVERT (elf->state.elf32.ehdr_mem.e_shstrndx);
375 : : }
376 : : }
377 : :
378 : : /* Don't precache the phdr pointer here.
379 : : elf32_getphdr will validate it against the size when asked. */
380 : :
381 : 2208 : Elf32_Off e_shoff = elf->state.elf32.ehdr->e_shoff;
382 [ + + + + ]: 2208 : if (map_address != NULL && e_ident[EI_DATA] == MY_ELFDATA
383 : : && cmd != ELF_C_READ_MMAP /* We need a copy to be able to write. */
384 [ + + ]: 860 : && (ALLOW_UNALIGNED
385 : : || (((ehdr + e_shoff) & (__alignof__ (Elf32_Shdr) - 1)) == 0)))
386 : : {
387 [ + + - + ]: 482 : if (unlikely (scncnt > 0 && e_shoff >= maxsize)
388 [ - + ]: 482 : || unlikely (maxsize - e_shoff
389 : : < scncnt * sizeof (Elf32_Shdr)))
390 : : {
391 : 0 : free_and_out:
392 : 0 : free (elf);
393 : 0 : __libelf_seterrno (ELF_E_INVALID_ELF);
394 : 0 : return NULL;
395 : : }
396 : :
397 [ + + ]: 482 : if (scncnt > 0)
398 : 460 : elf->state.elf32.shdr = (Elf32_Shdr *) (ehdr + e_shoff);
399 : :
400 [ + + ]: 602994 : for (size_t cnt = 0; cnt < scncnt; ++cnt)
401 : : {
402 : 602512 : elf->state.elf32.scns.data[cnt].index = cnt;
403 : 602512 : elf->state.elf32.scns.data[cnt].elf = elf;
404 : 602512 : elf->state.elf32.scns.data[cnt].shdr.e32 =
405 : 602512 : &elf->state.elf32.shdr[cnt];
406 [ + + ]: 602512 : if (likely (elf->state.elf32.shdr[cnt].sh_offset < maxsize)
407 [ + + ]: 602508 : && likely (elf->state.elf32.shdr[cnt].sh_size
408 : : <= maxsize - elf->state.elf32.shdr[cnt].sh_offset))
409 : 602494 : elf->state.elf32.scns.data[cnt].rawdata_base =
410 : 602494 : elf->state.elf32.scns.data[cnt].data_base =
411 : : ((char *) map_address + offset
412 : 602494 : + elf->state.elf32.shdr[cnt].sh_offset);
413 : 602512 : elf->state.elf32.scns.data[cnt].list = &elf->state.elf32.scns;
414 : :
415 : : /* If this is a section with an extended index add a
416 : : reference in the section which uses the extended
417 : : index. */
418 [ - + ]: 602512 : if (elf->state.elf32.shdr[cnt].sh_type == SHT_SYMTAB_SHNDX
419 [ # # ]: 0 : && elf->state.elf32.shdr[cnt].sh_link < scncnt)
420 : 0 : elf->state.elf32.scns.data[elf->state.elf32.shdr[cnt].sh_link].shndx_index
421 : 0 : = cnt;
422 : :
423 : : /* Set the own shndx_index field in case it has not yet
424 : : been set. */
425 [ + - ]: 602512 : if (elf->state.elf32.scns.data[cnt].shndx_index == 0)
426 : 602512 : elf->state.elf32.scns.data[cnt].shndx_index = -1;
427 : : }
428 : : }
429 : : else
430 : : {
431 [ + + ]: 4038938 : for (size_t cnt = 0; cnt < scncnt; ++cnt)
432 : : {
433 : 4037212 : elf->state.elf32.scns.data[cnt].index = cnt;
434 : 4037212 : elf->state.elf32.scns.data[cnt].elf = elf;
435 : 4037212 : elf->state.elf32.scns.data[cnt].list = &elf->state.elf32.scns;
436 : : }
437 : : }
438 : :
439 : : /* So far only one block with sections. */
440 : 2208 : elf->state.elf32.scns_last = &elf->state.elf32.scns;
441 : : }
442 : : else
443 : : {
444 : : /* This pointer might not be directly usable if the alignment is
445 : : not sufficient for the architecture. */
446 : 29040 : uintptr_t ehdr = (uintptr_t) map_address + offset;
447 : :
448 : : /* This is a 64-bit binary. */
449 [ + + ]: 29040 : if (map_address != NULL && e_ident[EI_DATA] == MY_ELFDATA
450 [ + + ]: 13650 : && (ALLOW_UNALIGNED
451 : : || (ehdr & (__alignof__ (Elf64_Ehdr) - 1)) == 0))
452 : : {
453 : : /* We can use the mmapped memory. */
454 : 13122 : elf->state.elf64.ehdr = (Elf64_Ehdr *) ehdr;
455 : : }
456 : : else
457 : : {
458 : : /* Copy the ELF header. */
459 [ + + ]: 15918 : elf->state.elf64.ehdr = memcpy (&elf->state.elf64.ehdr_mem, e_ident,
460 : : sizeof (Elf64_Ehdr));
461 : :
462 [ + + ]: 15918 : if (e_ident[EI_DATA] != MY_ELFDATA)
463 : : {
464 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_type);
465 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_machine);
466 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_version);
467 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_entry);
468 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_phoff);
469 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_shoff);
470 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_flags);
471 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_ehsize);
472 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_phentsize);
473 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_phnum);
474 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_shentsize);
475 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_shnum);
476 : 878 : CONVERT (elf->state.elf64.ehdr_mem.e_shstrndx);
477 : : }
478 : : }
479 : :
480 : : /* Don't precache the phdr pointer here.
481 : : elf64_getphdr will validate it against the size when asked. */
482 : :
483 : 29040 : Elf64_Off e_shoff = elf->state.elf64.ehdr->e_shoff;
484 [ + + + + ]: 29040 : if (map_address != NULL && e_ident[EI_DATA] == MY_ELFDATA
485 : : && cmd != ELF_C_READ_MMAP /* We need a copy to be able to write. */
486 [ + + ]: 13122 : && (ALLOW_UNALIGNED
487 : : || (((ehdr + e_shoff) & (__alignof__ (Elf64_Shdr) - 1)) == 0)))
488 : : {
489 [ + - ]: 11782 : if (unlikely (scncnt > 0 && e_shoff >= maxsize)
490 [ - + ]: 11782 : || unlikely (maxsize - e_shoff
491 : : < scncnt * sizeof (Elf64_Shdr)))
492 : 0 : goto free_and_out;
493 : :
494 [ + + ]: 11782 : if (scncnt > 0)
495 : 11718 : elf->state.elf64.shdr = (Elf64_Shdr *) (ehdr + e_shoff);
496 : :
497 [ + + ]: 1073812 : for (size_t cnt = 0; cnt < scncnt; ++cnt)
498 : : {
499 : 1062030 : elf->state.elf64.scns.data[cnt].index = cnt;
500 : 1062030 : elf->state.elf64.scns.data[cnt].elf = elf;
501 : 1062030 : elf->state.elf64.scns.data[cnt].shdr.e64 =
502 : 1062030 : &elf->state.elf64.shdr[cnt];
503 [ + + ]: 1062030 : if (likely (elf->state.elf64.shdr[cnt].sh_offset < maxsize)
504 [ + + ]: 1061746 : && likely (elf->state.elf64.shdr[cnt].sh_size
505 : : <= maxsize - elf->state.elf64.shdr[cnt].sh_offset))
506 : 1061632 : elf->state.elf64.scns.data[cnt].rawdata_base =
507 : 1061632 : elf->state.elf64.scns.data[cnt].data_base =
508 : : ((char *) map_address + offset
509 : 1061632 : + elf->state.elf64.shdr[cnt].sh_offset);
510 : 1062030 : elf->state.elf64.scns.data[cnt].list = &elf->state.elf64.scns;
511 : :
512 : : /* If this is a section with an extended index add a
513 : : reference in the section which uses the extended
514 : : index. */
515 [ - + ]: 1062030 : if (elf->state.elf64.shdr[cnt].sh_type == SHT_SYMTAB_SHNDX
516 [ # # ]: 0 : && elf->state.elf64.shdr[cnt].sh_link < scncnt)
517 : 0 : elf->state.elf64.scns.data[elf->state.elf64.shdr[cnt].sh_link].shndx_index
518 : 0 : = cnt;
519 : :
520 : : /* Set the own shndx_index field in case it has not yet
521 : : been set. */
522 [ + - ]: 1062030 : if (elf->state.elf64.scns.data[cnt].shndx_index == 0)
523 : 1062030 : elf->state.elf64.scns.data[cnt].shndx_index = -1;
524 : : }
525 : : }
526 : : else
527 : : {
528 [ + + ]: 3164456 : for (size_t cnt = 0; cnt < scncnt; ++cnt)
529 : : {
530 : 3147198 : elf->state.elf64.scns.data[cnt].index = cnt;
531 : 3147198 : elf->state.elf64.scns.data[cnt].elf = elf;
532 : 3147198 : elf->state.elf64.scns.data[cnt].list = &elf->state.elf64.scns;
533 : : }
534 : : }
535 : :
536 : : /* So far only one block with sections. */
537 : 29040 : elf->state.elf64.scns_last = &elf->state.elf64.scns;
538 : : }
539 : :
540 : : return elf;
541 : : }
542 : :
543 : :
544 : : Elf *
545 : : internal_function
546 : 15722 : __libelf_read_mmaped_file (int fildes, void *map_address, int64_t offset,
547 : : size_t maxsize, Elf_Cmd cmd, Elf *parent)
548 : : {
549 : : /* We have to find out what kind of file this is. We handle ELF
550 : : files and archives. To find out what we have we must look at the
551 : : header. The header for an ELF file is EI_NIDENT bytes in size,
552 : : the header for an archive file SARMAG bytes long. */
553 : 15722 : unsigned char *e_ident = (unsigned char *) map_address + offset;
554 : :
555 : : /* See what kind of object we have here. */
556 : 15722 : Elf_Kind kind = determine_kind (e_ident, maxsize);
557 : :
558 [ + + + ]: 15722 : switch (kind)
559 : : {
560 : 14966 : case ELF_K_ELF:
561 : 14966 : return file_read_elf (fildes, map_address, e_ident, offset, maxsize,
562 : : cmd, parent);
563 : :
564 : 60 : case ELF_K_AR:
565 : 60 : return file_read_ar (fildes, map_address, offset, maxsize, cmd, parent);
566 : :
567 : : default:
568 : 696 : break;
569 : : }
570 : :
571 : : /* This case is easy. Since we cannot do anything with this file
572 : : create a dummy descriptor. */
573 : 696 : return allocate_elf (fildes, map_address, offset, maxsize, cmd, parent,
574 : : ELF_K_NONE, 0);
575 : : }
576 : :
577 : :
578 : : static Elf *
579 : 17010 : read_unmmaped_file (int fildes, int64_t offset, size_t maxsize, Elf_Cmd cmd,
580 : : Elf *parent)
581 : : {
582 : : /* We have to find out what kind of file this is. We handle ELF
583 : : files and archives. To find out what we have we must read the
584 : : header. The identification header for an ELF file is EI_NIDENT
585 : : bytes in size, but we read the whole ELF header since we will
586 : : need it anyway later. For archives the header in SARMAG bytes
587 : : long. Read the maximum of these numbers.
588 : :
589 : : XXX We have to change this for the extended `ar' format some day.
590 : :
591 : : Use a union to ensure alignment. We might later access the
592 : : memory as a ElfXX_Ehdr. */
593 : 17010 : union
594 : : {
595 : : Elf64_Ehdr ehdr;
596 : : unsigned char header[MAX (sizeof (Elf64_Ehdr), SARMAG)];
597 : : } mem;
598 : :
599 : : /* Read the head of the file. */
600 : 34020 : ssize_t nread = pread_retry (fildes, mem.header,
601 : 17010 : MIN (MAX (sizeof (Elf64_Ehdr), SARMAG),
602 : : maxsize),
603 : : offset);
604 [ - + ]: 17010 : if (unlikely (nread == -1))
605 : : {
606 : : /* We cannot even read the head of the file. Maybe FILDES is associated
607 : : with an unseekable device. This is nothing we can handle. */
608 : 0 : __libelf_seterrno (ELF_E_INVALID_FILE);
609 : 0 : return NULL;
610 : : }
611 : :
612 : : /* See what kind of object we have here. */
613 : 17010 : Elf_Kind kind = determine_kind (mem.header, nread);
614 : :
615 [ + + + ]: 17010 : switch (kind)
616 : : {
617 : 254 : case ELF_K_AR:
618 : 254 : return file_read_ar (fildes, NULL, offset, maxsize, cmd, parent);
619 : :
620 : 16282 : case ELF_K_ELF:
621 : : /* Make sure at least the ELF header is contained in the file. */
622 [ + - ]: 16282 : if ((size_t) nread >= (mem.header[EI_CLASS] == ELFCLASS32
623 [ + + ]: 16282 : ? sizeof (Elf32_Ehdr) : sizeof (Elf64_Ehdr)))
624 : 16282 : return file_read_elf (fildes, NULL, mem.header, offset, maxsize, cmd,
625 : : parent);
626 : 474 : FALLTHROUGH;
627 : :
628 : : default:
629 : 474 : break;
630 : : }
631 : :
632 : : /* This case is easy. Since we cannot do anything with this file
633 : : create a dummy descriptor. */
634 : 474 : return allocate_elf (fildes, NULL, offset, maxsize, cmd, parent,
635 : : ELF_K_NONE, 0);
636 : : }
637 : :
638 : :
639 : : /* Open a file for reading. If possible we will try to mmap() the file. */
640 : : static struct Elf *
641 : 32526 : read_file (int fildes, int64_t offset, size_t maxsize,
642 : : Elf_Cmd cmd, Elf *parent)
643 : : {
644 : 32526 : void *map_address = NULL;
645 : 32526 : int use_mmap = (cmd == ELF_C_READ_MMAP || cmd == ELF_C_RDWR_MMAP
646 : : || cmd == ELF_C_WRITE_MMAP
647 : 32526 : || cmd == ELF_C_READ_MMAP_PRIVATE);
648 : :
649 [ + + ]: 32526 : if (parent == NULL)
650 : : {
651 [ + - ]: 17988 : if (maxsize == ~((size_t) 0))
652 : : {
653 : : /* We don't know in the moment how large the file is.
654 : : Determine it now. */
655 : 17988 : struct stat st;
656 : :
657 [ + - ]: 17988 : if (fstat (fildes, &st) == 0
658 : : && (sizeof (size_t) >= sizeof (st.st_size)
659 : : || st.st_size <= ~((size_t) 0)))
660 : 17988 : maxsize = (size_t) st.st_size;
661 : : }
662 : : }
663 : : else
664 : : {
665 : : /* The parent is already loaded. Use it. */
666 [ - + ]: 14538 : assert (maxsize != ~((size_t) 0));
667 : : }
668 : :
669 [ + + ]: 32526 : if (use_mmap)
670 : : {
671 [ + + ]: 15538 : if (parent == NULL)
672 : : {
673 : : /* We try to map the file ourself. */
674 [ + + ]: 28390 : map_address = mmap (NULL, maxsize, (cmd == ELF_C_READ_MMAP
675 : : ? PROT_READ
676 : : : PROT_READ|PROT_WRITE),
677 : 15342 : cmd == ELF_C_READ_MMAP_PRIVATE
678 [ + + ]: 15342 : || cmd == ELF_C_READ_MMAP
679 : : ? MAP_PRIVATE : MAP_SHARED,
680 : : fildes, offset);
681 : :
682 [ + + ]: 15342 : if (map_address == MAP_FAILED)
683 : : map_address = NULL;
684 : : }
685 : : else
686 : : {
687 : 196 : map_address = parent->map_address;
688 : : }
689 : : }
690 : :
691 : : /* If we have the file in memory optimize the access. */
692 [ + - ]: 15516 : if (map_address != NULL)
693 : : {
694 [ - + ]: 15516 : assert (map_address != MAP_FAILED);
695 : :
696 : 15516 : struct Elf *result = __libelf_read_mmaped_file (fildes, map_address,
697 : : offset, maxsize, cmd,
698 : : parent);
699 : :
700 : : /* If something went wrong during the initialization unmap the
701 : : memory if we mmaped here. */
702 [ - + ]: 15516 : if (result == NULL
703 [ # # ]: 0 : && (parent == NULL
704 [ # # ]: 0 : || parent->map_address != map_address))
705 : 0 : munmap (map_address, maxsize);
706 [ + + ]: 15516 : else if (parent == NULL)
707 : : /* Remember that we mmap()ed the memory. */
708 : 15320 : result->flags |= ELF_F_MMAPPED;
709 : :
710 : 15516 : return result;
711 : : }
712 : :
713 : : /* Otherwise we have to do it the hard way. We read as much as necessary
714 : : from the file whenever we need information which is not available. */
715 : 17010 : return read_unmmaped_file (fildes, offset, maxsize, cmd, parent);
716 : : }
717 : :
718 : :
719 : : /* Find the entry with the long names for the content of this archive. */
720 : : static const char *
721 : 204 : read_long_names (Elf *elf)
722 : : {
723 : 204 : off_t offset = SARMAG; /* This is the first entry. */
724 : 408 : struct ar_hdr hdrm;
725 : 408 : struct ar_hdr *hdr;
726 : 408 : char *newp;
727 : 408 : size_t len;
728 : :
729 : 612 : while (1)
730 : 204 : {
731 [ - + ]: 408 : if (elf->map_address != NULL)
732 : : {
733 [ # # ]: 0 : if ((size_t) offset > elf->maximum_size
734 [ # # ]: 0 : || elf->maximum_size - offset < sizeof (struct ar_hdr))
735 : 0 : return NULL;
736 : :
737 : : /* The data is mapped. */
738 : 0 : hdr = (struct ar_hdr *) (elf->map_address + offset);
739 : : }
740 : : else
741 : : {
742 : : /* Read the header from the file. */
743 [ + - ]: 408 : if (unlikely (pread_retry (elf->fildes, &hdrm, sizeof (hdrm),
744 : : elf->start_offset + offset)
745 : : != sizeof (hdrm)))
746 : : return NULL;
747 : :
748 : : hdr = &hdrm;
749 : : }
750 : :
751 : : /* The ar_size is given as a fixed size decimal string, right
752 : : padded with spaces. Make sure we read it properly even if
753 : : there is no terminating space. */
754 : 408 : char buf[sizeof (hdr->ar_size) + 1];
755 : 408 : const char *string = hdr->ar_size;
756 [ - + ]: 408 : if (hdr->ar_size[sizeof (hdr->ar_size) - 1] != ' ')
757 : : {
758 : 0 : *((char *) mempcpy (buf, hdr->ar_size, sizeof (hdr->ar_size))) = '\0';
759 : 0 : string = buf;
760 : : }
761 : :
762 : : /* atol expects to see at least one digit.
763 : : It also cannot be negative (-). */
764 [ + - ]: 408 : if (!isdigit(string[0]))
765 : : return NULL;
766 : 408 : len = atol (string);
767 : :
768 [ + + ]: 408 : if (memcmp (hdr->ar_name, "// ", 16) == 0)
769 : : break;
770 : :
771 : 204 : offset += sizeof (struct ar_hdr) + ((len + 1) & ~1l);
772 : : }
773 : :
774 : : /* Sanity check len early if we can. */
775 [ - + ]: 204 : if (elf->map_address != NULL)
776 : : {
777 [ # # ]: 0 : if (len > elf->maximum_size - offset - sizeof (struct ar_hdr))
778 : : return NULL;
779 : : }
780 : :
781 : : /* Due to the stupid format of the long name table entry (which are not
782 : : NUL terminted) we have to provide an appropriate representation anyhow.
783 : : Therefore we always make a copy which has the appropriate form. */
784 : 204 : newp = malloc (len);
785 [ + - ]: 204 : if (newp != NULL)
786 : : {
787 : 204 : char *runp;
788 : :
789 [ - + ]: 204 : if (elf->map_address != NULL)
790 : : {
791 : : /* Simply copy it over. */
792 : 0 : elf->state.ar.long_names = (char *) memcpy (newp,
793 : : elf->map_address + offset
794 : 0 : + sizeof (struct ar_hdr),
795 : : len);
796 : : }
797 : : else
798 : : {
799 [ - + ]: 204 : if (unlikely ((size_t) pread_retry (elf->fildes, newp, len,
800 : : elf->start_offset + offset
801 : : + sizeof (struct ar_hdr))
802 : : != len))
803 : : {
804 : : /* We were not able to read all data. */
805 : 0 : free (newp);
806 : 0 : elf->state.ar.long_names = NULL;
807 : 0 : return NULL;
808 : : }
809 : 204 : elf->state.ar.long_names = newp;
810 : : }
811 : :
812 : 204 : elf->state.ar.long_names_len = len;
813 : :
814 : : /* Now NUL-terminate the strings. */
815 : 204 : runp = newp;
816 : 8364 : while (1)
817 : : {
818 : 8364 : char *startp = runp;
819 : 8364 : runp = (char *) memchr (runp, '/', newp + len - runp);
820 [ + + ]: 8364 : if (runp == NULL)
821 : : {
822 : : /* This was the last entry. Clear any left overs. */
823 : 204 : memset (startp, '\0', newp + len - startp);
824 : : break;
825 : : }
826 : :
827 : : /* NUL-terminate the string. */
828 : 8160 : *runp++ = '\0';
829 : :
830 : : /* A sanity check. Somebody might have generated invalid
831 : : archive. */
832 [ + - ]: 8160 : if (runp >= newp + len)
833 : : break;
834 : : }
835 : : }
836 : :
837 : : return newp;
838 : : }
839 : :
840 : :
841 : : /* Read the next archive header. */
842 : : int
843 : : internal_function
844 : 14578 : __libelf_next_arhdr_wrlock (Elf *elf)
845 : : {
846 : 14578 : struct ar_hdr *ar_hdr;
847 : 14578 : Elf_Arhdr *elf_ar_hdr;
848 : :
849 [ + + ]: 14578 : if (elf->map_address != NULL)
850 : : {
851 : : /* See whether this entry is in the file. */
852 [ + - + + ]: 236 : if (unlikely ((size_t) elf->state.ar.offset
853 : : > elf->start_offset + elf->maximum_size
854 : : || (elf->start_offset + elf->maximum_size
855 : : - elf->state.ar.offset) < sizeof (struct ar_hdr)))
856 : : {
857 : : /* This record is not anymore in the file. */
858 : 40 : __libelf_seterrno (ELF_E_RANGE);
859 : 40 : return -1;
860 : : }
861 : 196 : ar_hdr = (struct ar_hdr *) (elf->map_address + elf->state.ar.offset);
862 : : }
863 : : else
864 : : {
865 : 14342 : ar_hdr = &elf->state.ar.ar_hdr;
866 : :
867 [ - + ]: 14342 : if (unlikely (pread_retry (elf->fildes, ar_hdr, sizeof (struct ar_hdr),
868 : : elf->state.ar.offset)
869 : : != sizeof (struct ar_hdr)))
870 : : {
871 : : /* Something went wrong while reading the file. */
872 : 0 : __libelf_seterrno (ELF_E_RANGE);
873 : 0 : return -1;
874 : : }
875 : : }
876 : :
877 : : /* One little consistency check. */
878 [ - + ]: 14538 : if (unlikely (memcmp (ar_hdr->ar_fmag, ARFMAG, 2) != 0))
879 : : {
880 : : /* This is no valid archive. */
881 : 0 : __libelf_seterrno (ELF_E_ARCHIVE_FMAG);
882 : 0 : return -1;
883 : : }
884 : :
885 : : /* Copy the raw name over to a NUL terminated buffer. */
886 [ + + ]: 14538 : *((char *) mempcpy (elf->state.ar.raw_name, ar_hdr->ar_name, 16)) = '\0';
887 : :
888 : 14538 : elf_ar_hdr = &elf->state.ar.elf_ar_hdr;
889 : :
890 : : /* Now convert the `struct ar_hdr' into `Elf_Arhdr'.
891 : : Determine whether this is a special entry. */
892 [ + + ]: 14538 : if (ar_hdr->ar_name[0] == '/')
893 : : {
894 [ + + ]: 4156 : if (ar_hdr->ar_name[1] == ' '
895 [ + - ]: 240 : && memcmp (ar_hdr->ar_name, "/ ", 16) == 0)
896 : : /* This is the index. */
897 : 240 : elf_ar_hdr->ar_name = memcpy (elf->state.ar.ar_name, "/", 2);
898 [ + + ]: 3916 : else if (ar_hdr->ar_name[1] == 'S'
899 [ + - ]: 4 : && memcmp (ar_hdr->ar_name, "/SYM64/ ", 16) == 0)
900 : : /* 64-bit index. */
901 : 4 : elf_ar_hdr->ar_name = memcpy (elf->state.ar.ar_name, "/SYM64/", 8);
902 [ + + ]: 3912 : else if (ar_hdr->ar_name[1] == '/'
903 [ + - ]: 232 : && memcmp (ar_hdr->ar_name, "// ", 16) == 0)
904 : : /* This is the array with the long names. */
905 : 232 : elf_ar_hdr->ar_name = memcpy (elf->state.ar.ar_name, "//", 3);
906 [ + - ]: 3680 : else if (likely (isdigit (ar_hdr->ar_name[1])))
907 : : {
908 : 3680 : size_t offset;
909 : :
910 : : /* This is a long name. First we have to read the long name
911 : : table, if this hasn't happened already. */
912 [ + + - + ]: 3680 : if (unlikely (elf->state.ar.long_names == NULL
913 : : && read_long_names (elf) == NULL))
914 : : {
915 : : /* No long name table although it is reference. The archive is
916 : : broken. */
917 : 0 : __libelf_seterrno (ELF_E_INVALID_ARCHIVE);
918 : 0 : return -1;
919 : : }
920 : :
921 : 3680 : offset = atol (ar_hdr->ar_name + 1);
922 [ - + ]: 3680 : if (unlikely (offset >= elf->state.ar.long_names_len))
923 : : {
924 : : /* The index in the long name table is larger than the table. */
925 : 0 : __libelf_seterrno (ELF_E_INVALID_ARCHIVE);
926 : 0 : return -1;
927 : : }
928 : 3680 : elf_ar_hdr->ar_name = elf->state.ar.long_names + offset;
929 : : }
930 : : else
931 : : {
932 : : /* This is none of the known special entries. */
933 : 0 : __libelf_seterrno (ELF_E_INVALID_ARCHIVE);
934 : 0 : return -1;
935 : : }
936 : : }
937 : : else
938 : : {
939 : 10382 : char *endp;
940 : :
941 : : /* It is a normal entry. Copy over the name. */
942 : 10382 : endp = (char *) memccpy (elf->state.ar.ar_name, ar_hdr->ar_name,
943 : : '/', 16);
944 [ + - ]: 10382 : if (endp != NULL)
945 : 10382 : endp[-1] = '\0';
946 : : else
947 : : {
948 : : /* In the old BSD style of archive, there is no / terminator.
949 : : Instead, there is space padding at the end of the name. */
950 : : size_t i = 15;
951 : 0 : do
952 : 0 : elf->state.ar.ar_name[i] = '\0';
953 [ # # # # ]: 0 : while (i > 0 && elf->state.ar.ar_name[--i] == ' ');
954 : : }
955 : :
956 : 10382 : elf_ar_hdr->ar_name = elf->state.ar.ar_name;
957 : : }
958 : :
959 [ - + ]: 14538 : if (unlikely (ar_hdr->ar_size[0] == ' '))
960 : : /* Something is really wrong. We cannot live without a size for
961 : : the member since it will not be possible to find the next
962 : : archive member. */
963 : : {
964 : 0 : __libelf_seterrno (ELF_E_INVALID_ARCHIVE);
965 : 0 : return -1;
966 : : }
967 : :
968 : : /* Since there are no specialized functions to convert ASCII to
969 : : time_t, uid_t, gid_t, mode_t, and off_t we use either atol or
970 : : atoll depending on the size of the types. We are also prepared
971 : : for the case where the whole field in the `struct ar_hdr' is
972 : : filled in which case we cannot simply use atol/l but instead have
973 : : to create a temporary copy. Note that all fields use decimal
974 : : encoding, except ar_mode which uses octal. */
975 : :
976 : : #define INT_FIELD(FIELD) \
977 : : do \
978 : : { \
979 : : char buf[sizeof (ar_hdr->FIELD) + 1]; \
980 : : const char *string = ar_hdr->FIELD; \
981 : : if (ar_hdr->FIELD[sizeof (ar_hdr->FIELD) - 1] != ' ') \
982 : : { \
983 : : *((char *) mempcpy (buf, ar_hdr->FIELD, sizeof (ar_hdr->FIELD))) \
984 : : = '\0'; \
985 : : string = buf; \
986 : : } \
987 : : if (sizeof (elf_ar_hdr->FIELD) <= sizeof (long int)) \
988 : : elf_ar_hdr->FIELD = (__typeof (elf_ar_hdr->FIELD)) atol (string); \
989 : : else \
990 : : elf_ar_hdr->FIELD = (__typeof (elf_ar_hdr->FIELD)) atoll (string); \
991 : : } \
992 : : while (0)
993 : :
994 : : #define OCT_FIELD(FIELD) \
995 : : do \
996 : : { \
997 : : char buf[sizeof (ar_hdr->FIELD) + 1]; \
998 : : const char *string = ar_hdr->FIELD; \
999 : : if (ar_hdr->FIELD[sizeof (ar_hdr->FIELD) - 1] != ' ') \
1000 : : { \
1001 : : *((char *) mempcpy (buf, ar_hdr->FIELD, sizeof (ar_hdr->FIELD))) \
1002 : : = '\0'; \
1003 : : string = buf; \
1004 : : } \
1005 : : if (sizeof (elf_ar_hdr->FIELD) <= sizeof (long int)) \
1006 : : elf_ar_hdr->FIELD \
1007 : : = (__typeof (elf_ar_hdr->FIELD)) strtol (string, NULL, 8); \
1008 : : else \
1009 : : elf_ar_hdr->FIELD \
1010 : : = (__typeof (elf_ar_hdr->FIELD)) strtoll (string, NULL, 8); \
1011 : : } \
1012 : : while (0)
1013 : :
1014 [ - + ]: 14538 : INT_FIELD (ar_date);
1015 [ - + ]: 14538 : INT_FIELD (ar_uid);
1016 [ - + ]: 14538 : INT_FIELD (ar_gid);
1017 [ - + ]: 14538 : OCT_FIELD (ar_mode);
1018 [ - + ]: 14538 : INT_FIELD (ar_size);
1019 : :
1020 [ - + ]: 14538 : if (elf_ar_hdr->ar_size < 0)
1021 : : {
1022 : 0 : __libelf_seterrno (ELF_E_INVALID_ARCHIVE);
1023 : 0 : return -1;
1024 : : }
1025 : :
1026 : : /* Truncated file? */
1027 : 14538 : size_t maxsize;
1028 : 14538 : maxsize = (elf->start_offset + elf->maximum_size
1029 : 14538 : - elf->state.ar.offset - sizeof (struct ar_hdr));
1030 [ - + ]: 14538 : if ((size_t) elf_ar_hdr->ar_size > maxsize)
1031 : 0 : elf_ar_hdr->ar_size = maxsize;
1032 : :
1033 : : return 0;
1034 : : }
1035 : :
1036 : :
1037 : : /* We were asked to return a clone of an existing descriptor. This
1038 : : function must be called with the lock on the parent descriptor
1039 : : being held. */
1040 : : static Elf *
1041 : 14540 : dup_elf (int fildes, Elf_Cmd cmd, Elf *ref)
1042 : : {
1043 : 14540 : struct Elf *result;
1044 : :
1045 [ + + ]: 14540 : if (fildes == -1)
1046 : : /* Allow the user to pass -1 as the file descriptor for the new file. */
1047 : 20 : fildes = ref->fildes;
1048 : : /* The file descriptor better should be the same. If it was disconnected
1049 : : already (using `elf_cntl') we do not test it. */
1050 [ + - - + ]: 14520 : else if (unlikely (ref->fildes != -1 && fildes != ref->fildes))
1051 : : {
1052 : 0 : __libelf_seterrno (ELF_E_FD_MISMATCH);
1053 : 0 : return NULL;
1054 : : }
1055 : :
1056 : : /* The mode must allow reading. I.e., a descriptor creating with a
1057 : : command different then ELF_C_READ, ELF_C_WRITE and ELF_C_RDWR is
1058 : : not allowed. */
1059 [ + + + - : 14540 : if (unlikely (ref->cmd != ELF_C_READ && ref->cmd != ELF_C_READ_MMAP
+ - + - +
- - + ]
1060 : : && ref->cmd != ELF_C_WRITE && ref->cmd != ELF_C_WRITE_MMAP
1061 : : && ref->cmd != ELF_C_RDWR && ref->cmd != ELF_C_RDWR_MMAP
1062 : : && ref->cmd != ELF_C_READ_MMAP_PRIVATE))
1063 : : {
1064 : 0 : __libelf_seterrno (ELF_E_INVALID_OP);
1065 : 0 : return NULL;
1066 : : }
1067 : :
1068 : : /* Now it is time to distinguish between reading normal files and
1069 : : archives. Normal files can easily be handled be incrementing the
1070 : : reference counter and return the same descriptor. */
1071 [ - + ]: 14540 : if (ref->kind != ELF_K_AR)
1072 : : {
1073 : 0 : ++ref->ref_count;
1074 : 0 : return ref;
1075 : : }
1076 : :
1077 : : /* This is an archive. We must create a descriptor for the archive
1078 : : member the internal pointer of the archive file descriptor is
1079 : : pointing to. First read the header of the next member if this
1080 : : has not happened already. */
1081 [ + + ]: 14540 : if (ref->state.ar.elf_ar_hdr.ar_name == NULL
1082 [ + + ]: 272 : && __libelf_next_arhdr_wrlock (ref) != 0)
1083 : : /* Something went wrong. Maybe there is no member left. */
1084 : : return NULL;
1085 : :
1086 : : /* We have all the information we need about the next archive member.
1087 : : Now create a descriptor for it. */
1088 : 29076 : result = read_file (fildes, ref->state.ar.offset + sizeof (struct ar_hdr),
1089 : 14538 : ref->state.ar.elf_ar_hdr.ar_size, cmd, ref);
1090 : :
1091 : : /* Enlist this new descriptor in the list of children. */
1092 [ + - ]: 14538 : if (result != NULL)
1093 : : {
1094 : 14538 : result->next = ref->state.ar.children;
1095 : 14538 : ref->state.ar.children = result;
1096 : : }
1097 : :
1098 : : return result;
1099 : : }
1100 : :
1101 : :
1102 : : /* Return descriptor for empty file ready for writing. */
1103 : : static struct Elf *
1104 : 1044 : write_file (int fd, Elf_Cmd cmd)
1105 : : {
1106 : : /* We simply create an empty `Elf' structure. */
1107 : : #define NSCNSALLOC 10
1108 : 1044 : Elf *result = allocate_elf (fd, NULL, 0, 0, cmd, NULL, ELF_K_ELF,
1109 : : NSCNSALLOC * sizeof (Elf_Scn));
1110 : :
1111 [ + - ]: 1044 : if (result != NULL)
1112 : : {
1113 : : /* We have to write to the file in any case. */
1114 : 1044 : result->flags = ELF_F_DIRTY;
1115 : :
1116 : : /* Some more or less arbitrary value. */
1117 : 1044 : result->state.elf.scnincr = NSCNSALLOC;
1118 : :
1119 : : /* We have allocated room for some sections. */
1120 : 1044 : assert (offsetof (struct Elf, state.elf32.scns)
1121 : : == offsetof (struct Elf, state.elf64.scns));
1122 : 1044 : result->state.elf.scns_last = &result->state.elf32.scns;
1123 : 1044 : result->state.elf32.scns.max = NSCNSALLOC;
1124 : : }
1125 : :
1126 : 1044 : return result;
1127 : : }
1128 : :
1129 : : /* Lock if necessary before dup an archive. */
1130 : : static inline Elf *
1131 : 14540 : lock_dup_elf (int fildes, Elf_Cmd cmd, Elf *ref)
1132 : : {
1133 : : /* We need wrlock to dup an archive. */
1134 : 14540 : if (ref->kind == ELF_K_AR)
1135 : : {
1136 : 14540 : rwlock_unlock (ref->lock);
1137 : 14540 : rwlock_wrlock (ref->lock);
1138 : : }
1139 : : /* Duplicate the descriptor. */
1140 : 14540 : return dup_elf (fildes, cmd, ref);
1141 : : }
1142 : :
1143 : : /* Return a descriptor for the file belonging to FILDES. */
1144 : : Elf *
1145 : 33608 : elf_begin (int fildes, Elf_Cmd cmd, Elf *ref)
1146 : : {
1147 : 33608 : Elf *retval;
1148 : :
1149 [ - + ]: 33608 : if (unlikely (__libelf_version != EV_CURRENT))
1150 : : {
1151 : : /* Version wasn't set so far. */
1152 : 0 : __libelf_seterrno (ELF_E_NO_VERSION);
1153 : 0 : return NULL;
1154 : : }
1155 : :
1156 [ + + ]: 33608 : if (ref != NULL)
1157 : : /* Make sure the descriptor is not suddenly going away. */
1158 : : rwlock_rdlock (ref->lock);
1159 [ - + - - ]: 19032 : else if (unlikely (fcntl (fildes, F_GETFD) == -1 && errno == EBADF))
1160 : : {
1161 : : /* We cannot do anything productive without a file descriptor. */
1162 : 0 : __libelf_seterrno (ELF_E_INVALID_FILE);
1163 : 0 : return NULL;
1164 : : }
1165 : :
1166 [ + + + + : 33608 : switch (cmd)
+ - ]
1167 : : {
1168 : : case ELF_C_NULL:
1169 : : /* We simply return a NULL pointer. */
1170 : : retval = NULL;
1171 : : break;
1172 : :
1173 : 13056 : case ELF_C_READ_MMAP_PRIVATE:
1174 : : /* If we have a reference it must also be opened this way. */
1175 [ + + - + ]: 13056 : if (unlikely (ref != NULL && ref->cmd != ELF_C_READ_MMAP_PRIVATE))
1176 : : {
1177 : 0 : __libelf_seterrno (ELF_E_INVALID_CMD);
1178 : 0 : retval = NULL;
1179 : 0 : break;
1180 : : }
1181 : 32274 : FALLTHROUGH;
1182 : :
1183 : : case ELF_C_READ:
1184 : : case ELF_C_READ_MMAP:
1185 [ + + ]: 32274 : if (ref != NULL)
1186 : 14540 : retval = lock_dup_elf (fildes, cmd, ref);
1187 : : else
1188 : : /* Create descriptor for existing file. */
1189 : 17742 : retval = read_file (fildes, 0, ~((size_t) 0), cmd, NULL);
1190 : : break;
1191 : :
1192 : 246 : case ELF_C_RDWR:
1193 : : case ELF_C_RDWR_MMAP:
1194 : : /* If we have a REF object it must also be opened using this
1195 : : command. */
1196 [ - + ]: 246 : if (ref != NULL)
1197 : : {
1198 [ # # # # : 0 : if (unlikely (ref->cmd != ELF_C_RDWR && ref->cmd != ELF_C_RDWR_MMAP
# # ]
1199 : : && ref->cmd != ELF_C_WRITE
1200 : : && ref->cmd != ELF_C_WRITE_MMAP))
1201 : : {
1202 : : /* This is not ok. REF must also be opened for writing. */
1203 : 0 : __libelf_seterrno (ELF_E_INVALID_CMD);
1204 : 0 : retval = NULL;
1205 : : }
1206 : : else
1207 : 0 : retval = lock_dup_elf (fildes, cmd, ref);
1208 : : }
1209 : : else
1210 : : /* Create descriptor for existing file. */
1211 : 246 : retval = read_file (fildes, 0, ~((size_t) 0), cmd, NULL);
1212 : : break;
1213 : :
1214 : 1044 : case ELF_C_WRITE:
1215 : : case ELF_C_WRITE_MMAP:
1216 : : /* We ignore REF and prepare a descriptor to write a new file. */
1217 : 1044 : retval = write_file (fildes, cmd);
1218 : 1044 : break;
1219 : :
1220 : 0 : default:
1221 : 0 : __libelf_seterrno (ELF_E_INVALID_CMD);
1222 : 0 : retval = NULL;
1223 : 0 : break;
1224 : : }
1225 : :
1226 : : /* Release the lock. */
1227 : : if (ref != NULL)
1228 : : rwlock_unlock (ref->lock);
1229 : :
1230 : : return retval;
1231 : : }
1232 : : INTDEF(elf_begin)
|