Coverage Report

Created: 2025-04-11 06:16

/src/elfutils/libdwfl/open.c
Line
Count
Source (jump to first uncovered line)
1
/* Decompression support for libdwfl: zlib (gzip), bzlib (bzip2) or lzma (xz).
2
   Copyright (C) 2009, 2016 Red Hat, Inc.
3
   Copyright (C) 2022 Google LLC
4
   This file is part of elfutils.
5
6
   This file is free software; you can redistribute it and/or modify
7
   it under the terms of either
8
9
     * the GNU Lesser General Public License as published by the Free
10
       Software Foundation; either version 3 of the License, or (at
11
       your option) any later version
12
13
   or
14
15
     * the GNU General Public License as published by the Free
16
       Software Foundation; either version 2 of the License, or (at
17
       your option) any later version
18
19
   or both in parallel, as here.
20
21
   elfutils is distributed in the hope that it will be useful, but
22
   WITHOUT ANY WARRANTY; without even the implied warranty of
23
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24
   General Public License for more details.
25
26
   You should have received copies of the GNU General Public License and
27
   the GNU Lesser General Public License along with this program.  If
28
   not, see <http://www.gnu.org/licenses/>.  */
29
30
#ifdef HAVE_CONFIG_H
31
# include <config.h>
32
#endif
33
34
#include "libelfP.h"
35
#include "libdwflP.h"
36
37
#if !USE_BZLIB
38
535
# define __libdw_bunzip2(...) DWFL_E_BADELF
39
#endif
40
41
#if !USE_LZMA
42
535
# define __libdw_unlzma(...)  DWFL_E_BADELF
43
#endif
44
45
#if !USE_ZSTD
46
535
# define __libdw_unzstd(...)  DWFL_E_BADELF
47
#endif
48
49
/* Consumes and replaces *ELF only on success.  */
50
static Dwfl_Error
51
decompress (int fd __attribute__ ((unused)), Elf **elf)
52
4.24k
{
53
4.24k
  Dwfl_Error error = DWFL_E_BADELF;
54
  /* ELF cannot be decompressed, if there is no file descriptor.  */
55
4.24k
  if (fd == -1)
56
0
    return error;
57
4.24k
  void *buffer = NULL;
58
4.24k
  size_t size = 0;
59
60
4.24k
  const off_t offset = (*elf)->start_offset;
61
4.24k
  void *const mapped = ((*elf)->map_address == NULL ? NULL
62
4.24k
      : (*elf)->map_address + offset);
63
4.24k
  const size_t mapped_size = (*elf)->maximum_size;
64
4.24k
  if (mapped_size == 0)
65
0
    return error;
66
67
4.24k
  error = __libdw_gunzip (fd, offset, mapped, mapped_size, &buffer, &size);
68
4.24k
  if (error == DWFL_E_BADELF)
69
535
    error = __libdw_bunzip2 (fd, offset, mapped, mapped_size, &buffer, &size);
70
4.24k
  if (error == DWFL_E_BADELF)
71
535
    error = __libdw_unlzma (fd, offset, mapped, mapped_size, &buffer, &size);
72
4.24k
  if (error == DWFL_E_BADELF)
73
535
    error = __libdw_unzstd (fd, offset, mapped, mapped_size, &buffer, &size);
74
75
4.24k
  if (error == DWFL_E_NOERROR)
76
3.23k
    {
77
3.23k
      if (unlikely (size == 0))
78
638
  {
79
638
    error = DWFL_E_BADELF;
80
638
    free (buffer);
81
638
  }
82
2.60k
      else
83
2.60k
  {
84
2.60k
    Elf *memelf = elf_memory (buffer, size);
85
2.60k
    if (memelf == NULL)
86
8
      {
87
8
        error = DWFL_E_LIBELF;
88
8
        free (buffer);
89
8
      }
90
2.59k
    else
91
2.59k
      {
92
2.59k
        memelf->flags |= ELF_F_MALLOCED;
93
2.59k
        elf_end (*elf);
94
2.59k
        *elf = memelf;
95
2.59k
      }
96
2.60k
  }
97
3.23k
    }
98
1.00k
  else
99
1.00k
    free (buffer);
100
101
4.24k
  return error;
102
4.24k
}
103
104
static Dwfl_Error
105
what_kind (int fd, Elf **elfp, Elf_Kind *kind, bool *may_close_fd)
106
19.2k
{
107
19.2k
  Dwfl_Error error = DWFL_E_NOERROR;
108
19.2k
  *kind = elf_kind (*elfp);
109
19.2k
  if (unlikely (*kind == ELF_K_NONE))
110
4.85k
    {
111
4.85k
      if (unlikely (*elfp == NULL))
112
606
  error = DWFL_E_LIBELF;
113
4.24k
      else
114
4.24k
  {
115
4.24k
    error = decompress (fd, elfp);
116
4.24k
    if (error == DWFL_E_NOERROR)
117
2.59k
      {
118
2.59k
        *may_close_fd = true;
119
2.59k
        *kind = elf_kind (*elfp);
120
2.59k
      }
121
4.24k
  }
122
4.85k
    }
123
19.2k
  return error;
124
19.2k
}
125
126
static Dwfl_Error
127
libdw_open_elf (int *fdp, Elf **elfp, bool close_on_fail, bool archive_ok,
128
    bool never_close_fd, bool bad_elf_ok, bool use_elfp)
129
19.0k
{
130
19.0k
  bool may_close_fd = false;
131
132
19.0k
  Elf *elf =
133
19.0k
      use_elfp ? *elfp : elf_begin (*fdp, ELF_C_READ_MMAP_PRIVATE, NULL);
134
135
19.0k
  Elf_Kind kind;
136
19.0k
  Dwfl_Error error = what_kind (*fdp, &elf, &kind, &may_close_fd);
137
19.0k
  if (error == DWFL_E_BADELF)
138
1.06k
    {
139
      /* It's not an ELF file or a compressed file.
140
   See if it's an image with a header preceding the real file.  */
141
142
1.06k
      off_t offset = elf->start_offset;
143
1.06k
      error = __libdw_image_header (*fdp, &offset,
144
1.06k
            (elf->map_address == NULL ? NULL
145
1.06k
             : elf->map_address + offset),
146
1.06k
            elf->maximum_size);
147
1.06k
      if (error == DWFL_E_NOERROR)
148
166
  {
149
    /* Pure evil.  libelf needs some better interfaces.  */
150
166
    elf->kind = ELF_K_AR;
151
166
    elf->state.ar.elf_ar_hdr.ar_name = "libdwfl is faking you out";
152
166
    elf->state.ar.elf_ar_hdr.ar_size = elf->maximum_size - offset;
153
166
    elf->state.ar.offset = offset - sizeof (struct ar_hdr);
154
166
    Elf *subelf = elf_begin (-1, elf->cmd, elf);
155
166
    elf->kind = ELF_K_NONE;
156
166
    if (unlikely (subelf == NULL))
157
1
      error = DWFL_E_LIBELF;
158
165
    else
159
165
      {
160
165
        subelf->parent = NULL;
161
165
        subelf->flags |= elf->flags & (ELF_F_MMAPPED | ELF_F_MALLOCED);
162
165
        elf->flags &= ~(ELF_F_MMAPPED | ELF_F_MALLOCED);
163
165
        elf_end (elf);
164
165
        elf = subelf;
165
165
        error = what_kind (*fdp, &elf, &kind, &may_close_fd);
166
165
      }
167
166
  }
168
1.06k
    }
169
170
19.0k
  if (error == DWFL_E_NOERROR
171
19.0k
      && kind != ELF_K_ELF
172
19.0k
      && !(archive_ok && kind == ELF_K_AR))
173
589
    error = DWFL_E_BADELF;
174
175
  /* This basically means, we keep a ELF_K_NONE Elf handle and return it.  */
176
19.0k
  if (bad_elf_ok && error == DWFL_E_BADELF)
177
0
    error = DWFL_E_NOERROR;
178
179
19.0k
  if (error != DWFL_E_NOERROR)
180
2.68k
    {
181
2.68k
      elf_end (elf);
182
2.68k
      elf = NULL;
183
2.68k
    }
184
185
19.0k
  if (! never_close_fd
186
19.0k
      && error == DWFL_E_NOERROR ? may_close_fd : close_on_fail)
187
4.69k
    {
188
4.69k
      close (*fdp);
189
4.69k
      *fdp = -1;
190
4.69k
    }
191
192
19.0k
  *elfp = elf;
193
19.0k
  return error;
194
19.0k
}
195
196
Dwfl_Error internal_function
197
__libdw_open_file (int *fdp, Elf **elfp, bool close_on_fail, bool archive_ok)
198
19.0k
{
199
19.0k
  return libdw_open_elf (fdp, elfp, close_on_fail, archive_ok, false, false,
200
19.0k
       false);
201
19.0k
}
202
203
Dwfl_Error internal_function
204
__libdw_open_elf_memory (char *data, size_t size, Elf **elfp, bool archive_ok)
205
0
{
206
  /* It is ok to use `fd == -1` here, because libelf uses it as a value for
207
     "no file opened" and code supports working with this value, and also
208
     `never_close_fd == false` is passed to prevent closing non-existent file.
209
     The only caveat is in `decompress` method, which doesn't support
210
     decompressing from memory, so reading compressed zImage using this method
211
     won't work.  */
212
0
  int fd = -1;
213
0
  *elfp = elf_memory (data, size);
214
0
  if (unlikely(*elfp == NULL))
215
0
    {
216
0
      return DWFL_E_LIBELF;
217
0
    }
218
0
  return libdw_open_elf (&fd, elfp, false, archive_ok, true, false, true);
219
0
}
220
221
Dwfl_Error internal_function
222
__libdw_open_elf (int fd, Elf **elfp)
223
0
{
224
0
  return libdw_open_elf (&fd, elfp, false, true, true, true, false);
225
0
}