1*7304104dSAndroid Build Coastguard Worker /* Decompression support for libdwfl: zlib (gzip), bzlib (bzip2) or lzma (xz).
2*7304104dSAndroid Build Coastguard Worker Copyright (C) 2009, 2016 Red Hat, Inc.
3*7304104dSAndroid Build Coastguard Worker Copyright (C) 2022 Google LLC
4*7304104dSAndroid Build Coastguard Worker This file is part of elfutils.
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 "libelfP.h"
35*7304104dSAndroid Build Coastguard Worker #include "libdwflP.h"
36*7304104dSAndroid Build Coastguard Worker
37*7304104dSAndroid Build Coastguard Worker #if !USE_BZLIB
38*7304104dSAndroid Build Coastguard Worker # define __libdw_bunzip2(...) DWFL_E_BADELF
39*7304104dSAndroid Build Coastguard Worker #endif
40*7304104dSAndroid Build Coastguard Worker
41*7304104dSAndroid Build Coastguard Worker #if !USE_LZMA
42*7304104dSAndroid Build Coastguard Worker # define __libdw_unlzma(...) DWFL_E_BADELF
43*7304104dSAndroid Build Coastguard Worker #endif
44*7304104dSAndroid Build Coastguard Worker
45*7304104dSAndroid Build Coastguard Worker #if !USE_ZSTD
46*7304104dSAndroid Build Coastguard Worker # define __libdw_unzstd(...) DWFL_E_BADELF
47*7304104dSAndroid Build Coastguard Worker #endif
48*7304104dSAndroid Build Coastguard Worker
49*7304104dSAndroid Build Coastguard Worker /* Consumes and replaces *ELF only on success. */
50*7304104dSAndroid Build Coastguard Worker static Dwfl_Error
decompress(int fd,Elf ** elf)51*7304104dSAndroid Build Coastguard Worker decompress (int fd __attribute__ ((unused)), Elf **elf)
52*7304104dSAndroid Build Coastguard Worker {
53*7304104dSAndroid Build Coastguard Worker Dwfl_Error error = DWFL_E_BADELF;
54*7304104dSAndroid Build Coastguard Worker /* ELF cannot be decompressed, if there is no file descriptor. */
55*7304104dSAndroid Build Coastguard Worker if (fd == -1)
56*7304104dSAndroid Build Coastguard Worker return error;
57*7304104dSAndroid Build Coastguard Worker void *buffer = NULL;
58*7304104dSAndroid Build Coastguard Worker size_t size = 0;
59*7304104dSAndroid Build Coastguard Worker
60*7304104dSAndroid Build Coastguard Worker const off_t offset = (*elf)->start_offset;
61*7304104dSAndroid Build Coastguard Worker void *const mapped = ((*elf)->map_address == NULL ? NULL
62*7304104dSAndroid Build Coastguard Worker : (*elf)->map_address + offset);
63*7304104dSAndroid Build Coastguard Worker const size_t mapped_size = (*elf)->maximum_size;
64*7304104dSAndroid Build Coastguard Worker if (mapped_size == 0)
65*7304104dSAndroid Build Coastguard Worker return error;
66*7304104dSAndroid Build Coastguard Worker
67*7304104dSAndroid Build Coastguard Worker error = __libdw_gunzip (fd, offset, mapped, mapped_size, &buffer, &size);
68*7304104dSAndroid Build Coastguard Worker if (error == DWFL_E_BADELF)
69*7304104dSAndroid Build Coastguard Worker error = __libdw_bunzip2 (fd, offset, mapped, mapped_size, &buffer, &size);
70*7304104dSAndroid Build Coastguard Worker if (error == DWFL_E_BADELF)
71*7304104dSAndroid Build Coastguard Worker error = __libdw_unlzma (fd, offset, mapped, mapped_size, &buffer, &size);
72*7304104dSAndroid Build Coastguard Worker if (error == DWFL_E_BADELF)
73*7304104dSAndroid Build Coastguard Worker error = __libdw_unzstd (fd, offset, mapped, mapped_size, &buffer, &size);
74*7304104dSAndroid Build Coastguard Worker
75*7304104dSAndroid Build Coastguard Worker if (error == DWFL_E_NOERROR)
76*7304104dSAndroid Build Coastguard Worker {
77*7304104dSAndroid Build Coastguard Worker if (unlikely (size == 0))
78*7304104dSAndroid Build Coastguard Worker {
79*7304104dSAndroid Build Coastguard Worker error = DWFL_E_BADELF;
80*7304104dSAndroid Build Coastguard Worker free (buffer);
81*7304104dSAndroid Build Coastguard Worker }
82*7304104dSAndroid Build Coastguard Worker else
83*7304104dSAndroid Build Coastguard Worker {
84*7304104dSAndroid Build Coastguard Worker Elf *memelf = elf_memory (buffer, size);
85*7304104dSAndroid Build Coastguard Worker if (memelf == NULL)
86*7304104dSAndroid Build Coastguard Worker {
87*7304104dSAndroid Build Coastguard Worker error = DWFL_E_LIBELF;
88*7304104dSAndroid Build Coastguard Worker free (buffer);
89*7304104dSAndroid Build Coastguard Worker }
90*7304104dSAndroid Build Coastguard Worker else
91*7304104dSAndroid Build Coastguard Worker {
92*7304104dSAndroid Build Coastguard Worker memelf->flags |= ELF_F_MALLOCED;
93*7304104dSAndroid Build Coastguard Worker elf_end (*elf);
94*7304104dSAndroid Build Coastguard Worker *elf = memelf;
95*7304104dSAndroid Build Coastguard Worker }
96*7304104dSAndroid Build Coastguard Worker }
97*7304104dSAndroid Build Coastguard Worker }
98*7304104dSAndroid Build Coastguard Worker else
99*7304104dSAndroid Build Coastguard Worker free (buffer);
100*7304104dSAndroid Build Coastguard Worker
101*7304104dSAndroid Build Coastguard Worker return error;
102*7304104dSAndroid Build Coastguard Worker }
103*7304104dSAndroid Build Coastguard Worker
104*7304104dSAndroid Build Coastguard Worker static Dwfl_Error
what_kind(int fd,Elf ** elfp,Elf_Kind * kind,bool * may_close_fd)105*7304104dSAndroid Build Coastguard Worker what_kind (int fd, Elf **elfp, Elf_Kind *kind, bool *may_close_fd)
106*7304104dSAndroid Build Coastguard Worker {
107*7304104dSAndroid Build Coastguard Worker Dwfl_Error error = DWFL_E_NOERROR;
108*7304104dSAndroid Build Coastguard Worker *kind = elf_kind (*elfp);
109*7304104dSAndroid Build Coastguard Worker if (unlikely (*kind == ELF_K_NONE))
110*7304104dSAndroid Build Coastguard Worker {
111*7304104dSAndroid Build Coastguard Worker if (unlikely (*elfp == NULL))
112*7304104dSAndroid Build Coastguard Worker error = DWFL_E_LIBELF;
113*7304104dSAndroid Build Coastguard Worker else
114*7304104dSAndroid Build Coastguard Worker {
115*7304104dSAndroid Build Coastguard Worker error = decompress (fd, elfp);
116*7304104dSAndroid Build Coastguard Worker if (error == DWFL_E_NOERROR)
117*7304104dSAndroid Build Coastguard Worker {
118*7304104dSAndroid Build Coastguard Worker *may_close_fd = true;
119*7304104dSAndroid Build Coastguard Worker *kind = elf_kind (*elfp);
120*7304104dSAndroid Build Coastguard Worker }
121*7304104dSAndroid Build Coastguard Worker }
122*7304104dSAndroid Build Coastguard Worker }
123*7304104dSAndroid Build Coastguard Worker return error;
124*7304104dSAndroid Build Coastguard Worker }
125*7304104dSAndroid Build Coastguard Worker
126*7304104dSAndroid Build Coastguard Worker static Dwfl_Error
libdw_open_elf(int * fdp,Elf ** elfp,bool close_on_fail,bool archive_ok,bool never_close_fd,bool bad_elf_ok,bool use_elfp)127*7304104dSAndroid Build Coastguard Worker libdw_open_elf (int *fdp, Elf **elfp, bool close_on_fail, bool archive_ok,
128*7304104dSAndroid Build Coastguard Worker bool never_close_fd, bool bad_elf_ok, bool use_elfp)
129*7304104dSAndroid Build Coastguard Worker {
130*7304104dSAndroid Build Coastguard Worker bool may_close_fd = false;
131*7304104dSAndroid Build Coastguard Worker
132*7304104dSAndroid Build Coastguard Worker Elf *elf =
133*7304104dSAndroid Build Coastguard Worker use_elfp ? *elfp : elf_begin (*fdp, ELF_C_READ_MMAP_PRIVATE, NULL);
134*7304104dSAndroid Build Coastguard Worker
135*7304104dSAndroid Build Coastguard Worker Elf_Kind kind;
136*7304104dSAndroid Build Coastguard Worker Dwfl_Error error = what_kind (*fdp, &elf, &kind, &may_close_fd);
137*7304104dSAndroid Build Coastguard Worker if (error == DWFL_E_BADELF)
138*7304104dSAndroid Build Coastguard Worker {
139*7304104dSAndroid Build Coastguard Worker /* It's not an ELF file or a compressed file.
140*7304104dSAndroid Build Coastguard Worker See if it's an image with a header preceding the real file. */
141*7304104dSAndroid Build Coastguard Worker
142*7304104dSAndroid Build Coastguard Worker off_t offset = elf->start_offset;
143*7304104dSAndroid Build Coastguard Worker error = __libdw_image_header (*fdp, &offset,
144*7304104dSAndroid Build Coastguard Worker (elf->map_address == NULL ? NULL
145*7304104dSAndroid Build Coastguard Worker : elf->map_address + offset),
146*7304104dSAndroid Build Coastguard Worker elf->maximum_size);
147*7304104dSAndroid Build Coastguard Worker if (error == DWFL_E_NOERROR)
148*7304104dSAndroid Build Coastguard Worker {
149*7304104dSAndroid Build Coastguard Worker /* Pure evil. libelf needs some better interfaces. */
150*7304104dSAndroid Build Coastguard Worker elf->kind = ELF_K_AR;
151*7304104dSAndroid Build Coastguard Worker elf->state.ar.elf_ar_hdr.ar_name = "libdwfl is faking you out";
152*7304104dSAndroid Build Coastguard Worker elf->state.ar.elf_ar_hdr.ar_size = elf->maximum_size - offset;
153*7304104dSAndroid Build Coastguard Worker elf->state.ar.offset = offset - sizeof (struct ar_hdr);
154*7304104dSAndroid Build Coastguard Worker Elf *subelf = elf_begin (-1, elf->cmd, elf);
155*7304104dSAndroid Build Coastguard Worker elf->kind = ELF_K_NONE;
156*7304104dSAndroid Build Coastguard Worker if (unlikely (subelf == NULL))
157*7304104dSAndroid Build Coastguard Worker error = DWFL_E_LIBELF;
158*7304104dSAndroid Build Coastguard Worker else
159*7304104dSAndroid Build Coastguard Worker {
160*7304104dSAndroid Build Coastguard Worker subelf->parent = NULL;
161*7304104dSAndroid Build Coastguard Worker subelf->flags |= elf->flags & (ELF_F_MMAPPED | ELF_F_MALLOCED);
162*7304104dSAndroid Build Coastguard Worker elf->flags &= ~(ELF_F_MMAPPED | ELF_F_MALLOCED);
163*7304104dSAndroid Build Coastguard Worker elf_end (elf);
164*7304104dSAndroid Build Coastguard Worker elf = subelf;
165*7304104dSAndroid Build Coastguard Worker error = what_kind (*fdp, &elf, &kind, &may_close_fd);
166*7304104dSAndroid Build Coastguard Worker }
167*7304104dSAndroid Build Coastguard Worker }
168*7304104dSAndroid Build Coastguard Worker }
169*7304104dSAndroid Build Coastguard Worker
170*7304104dSAndroid Build Coastguard Worker if (error == DWFL_E_NOERROR
171*7304104dSAndroid Build Coastguard Worker && kind != ELF_K_ELF
172*7304104dSAndroid Build Coastguard Worker && !(archive_ok && kind == ELF_K_AR))
173*7304104dSAndroid Build Coastguard Worker error = DWFL_E_BADELF;
174*7304104dSAndroid Build Coastguard Worker
175*7304104dSAndroid Build Coastguard Worker /* This basically means, we keep a ELF_K_NONE Elf handle and return it. */
176*7304104dSAndroid Build Coastguard Worker if (bad_elf_ok && error == DWFL_E_BADELF)
177*7304104dSAndroid Build Coastguard Worker error = DWFL_E_NOERROR;
178*7304104dSAndroid Build Coastguard Worker
179*7304104dSAndroid Build Coastguard Worker if (error != DWFL_E_NOERROR)
180*7304104dSAndroid Build Coastguard Worker {
181*7304104dSAndroid Build Coastguard Worker elf_end (elf);
182*7304104dSAndroid Build Coastguard Worker elf = NULL;
183*7304104dSAndroid Build Coastguard Worker }
184*7304104dSAndroid Build Coastguard Worker
185*7304104dSAndroid Build Coastguard Worker if (! never_close_fd
186*7304104dSAndroid Build Coastguard Worker && error == DWFL_E_NOERROR ? may_close_fd : close_on_fail)
187*7304104dSAndroid Build Coastguard Worker {
188*7304104dSAndroid Build Coastguard Worker close (*fdp);
189*7304104dSAndroid Build Coastguard Worker *fdp = -1;
190*7304104dSAndroid Build Coastguard Worker }
191*7304104dSAndroid Build Coastguard Worker
192*7304104dSAndroid Build Coastguard Worker *elfp = elf;
193*7304104dSAndroid Build Coastguard Worker return error;
194*7304104dSAndroid Build Coastguard Worker }
195*7304104dSAndroid Build Coastguard Worker
196*7304104dSAndroid Build Coastguard Worker Dwfl_Error internal_function
__libdw_open_file(int * fdp,Elf ** elfp,bool close_on_fail,bool archive_ok)197*7304104dSAndroid Build Coastguard Worker __libdw_open_file (int *fdp, Elf **elfp, bool close_on_fail, bool archive_ok)
198*7304104dSAndroid Build Coastguard Worker {
199*7304104dSAndroid Build Coastguard Worker return libdw_open_elf (fdp, elfp, close_on_fail, archive_ok, false, false,
200*7304104dSAndroid Build Coastguard Worker false);
201*7304104dSAndroid Build Coastguard Worker }
202*7304104dSAndroid Build Coastguard Worker
203*7304104dSAndroid Build Coastguard Worker Dwfl_Error internal_function
__libdw_open_elf_memory(char * data,size_t size,Elf ** elfp,bool archive_ok)204*7304104dSAndroid Build Coastguard Worker __libdw_open_elf_memory (char *data, size_t size, Elf **elfp, bool archive_ok)
205*7304104dSAndroid Build Coastguard Worker {
206*7304104dSAndroid Build Coastguard Worker /* It is ok to use `fd == -1` here, because libelf uses it as a value for
207*7304104dSAndroid Build Coastguard Worker "no file opened" and code supports working with this value, and also
208*7304104dSAndroid Build Coastguard Worker `never_close_fd == false` is passed to prevent closing non-existent file.
209*7304104dSAndroid Build Coastguard Worker The only caveat is in `decompress` method, which doesn't support
210*7304104dSAndroid Build Coastguard Worker decompressing from memory, so reading compressed zImage using this method
211*7304104dSAndroid Build Coastguard Worker won't work. */
212*7304104dSAndroid Build Coastguard Worker int fd = -1;
213*7304104dSAndroid Build Coastguard Worker *elfp = elf_memory (data, size);
214*7304104dSAndroid Build Coastguard Worker if (unlikely(*elfp == NULL))
215*7304104dSAndroid Build Coastguard Worker {
216*7304104dSAndroid Build Coastguard Worker return DWFL_E_LIBELF;
217*7304104dSAndroid Build Coastguard Worker }
218*7304104dSAndroid Build Coastguard Worker return libdw_open_elf (&fd, elfp, false, archive_ok, true, false, true);
219*7304104dSAndroid Build Coastguard Worker }
220*7304104dSAndroid Build Coastguard Worker
221*7304104dSAndroid Build Coastguard Worker Dwfl_Error internal_function
__libdw_open_elf(int fd,Elf ** elfp)222*7304104dSAndroid Build Coastguard Worker __libdw_open_elf (int fd, Elf **elfp)
223*7304104dSAndroid Build Coastguard Worker {
224*7304104dSAndroid Build Coastguard Worker return libdw_open_elf (&fd, elfp, false, true, true, true, false);
225*7304104dSAndroid Build Coastguard Worker }
226