Coverage Report

Created: 2025-07-18 06:08

/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
534
# define __libdw_bunzip2(...) DWFL_E_BADELF
39
#endif
40
41
#if !USE_LZMA
42
534
# define __libdw_unlzma(...)  DWFL_E_BADELF
43
#endif
44
45
#if !USE_ZSTD
46
534
# 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.57k
{
53
4.57k
  Dwfl_Error error = DWFL_E_BADELF;
54
  /* ELF cannot be decompressed, if there is no file descriptor.  */
55
4.57k
  if (fd == -1)
56
0
    return error;
57
4.57k
  void *buffer = NULL;
58
4.57k
  size_t size = 0;
59
60
4.57k
  const off_t offset = (*elf)->start_offset;
61
4.57k
  void *const mapped = ((*elf)->map_address == NULL ? NULL
62
4.57k
      : (*elf)->map_address + offset);
63
4.57k
  const size_t mapped_size = (*elf)->maximum_size;
64
4.57k
  if (mapped_size == 0)
65
0
    return error;
66
67
4.57k
  error = __libdw_gunzip (fd, offset, mapped, mapped_size, &buffer, &size);
68
4.57k
  if (error == DWFL_E_BADELF)
69
534
    error = __libdw_bunzip2 (fd, offset, mapped, mapped_size, &buffer, &size);
70
4.57k
  if (error == DWFL_E_BADELF)
71
534
    error = __libdw_unlzma (fd, offset, mapped, mapped_size, &buffer, &size);
72
4.57k
  if (error == DWFL_E_BADELF)
73
534
    error = __libdw_unzstd (fd, offset, mapped, mapped_size, &buffer, &size);
74
75
4.57k
  if (error == DWFL_E_NOERROR)
76
3.47k
    {
77
3.47k
      if (unlikely (size == 0))
78
693
  {
79
693
    error = DWFL_E_BADELF;
80
693
    free (buffer);
81
693
  }
82
2.78k
      else
83
2.78k
  {
84
2.78k
    Elf *memelf = elf_memory (buffer, size);
85
2.78k
    if (memelf == NULL)
86
2
      {
87
2
        error = DWFL_E_LIBELF;
88
2
        free (buffer);
89
2
      }
90
2.78k
    else
91
2.78k
      {
92
2.78k
        memelf->flags |= ELF_F_MALLOCED;
93
2.78k
        elf_end (*elf);
94
2.78k
        *elf = memelf;
95
2.78k
      }
96
2.78k
  }
97
3.47k
    }
98
1.09k
  else
99
1.09k
    free (buffer);
100
101
4.57k
  return error;
102
4.57k
}
103
104
static Dwfl_Error
105
what_kind (int fd, Elf **elfp, Elf_Kind *kind, bool *may_close_fd)
106
19.7k
{
107
19.7k
  Dwfl_Error error = DWFL_E_NOERROR;
108
19.7k
  *kind = elf_kind (*elfp);
109
19.7k
  if (unlikely (*kind == ELF_K_NONE))
110
5.14k
    {
111
5.14k
      if (unlikely (*elfp == NULL))
112
568
  error = DWFL_E_LIBELF;
113
4.57k
      else
114
4.57k
  {
115
4.57k
    error = decompress (fd, elfp);
116
4.57k
    if (error == DWFL_E_NOERROR)
117
2.78k
      {
118
2.78k
        *may_close_fd = true;
119
2.78k
        *kind = elf_kind (*elfp);
120
2.78k
      }
121
4.57k
  }
122
5.14k
    }
123
19.7k
  return error;
124
19.7k
}
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.5k
{
130
19.5k
  bool may_close_fd = false;
131
132
19.5k
  Elf *elf =
133
19.5k
      use_elfp ? *elfp : elf_begin (*fdp, ELF_C_READ_MMAP_PRIVATE, NULL);
134
135
19.5k
  Elf_Kind kind;
136
19.5k
  Dwfl_Error error = what_kind (*fdp, &elf, &kind, &may_close_fd);
137
19.5k
  if (error == DWFL_E_BADELF)
138
1.11k
    {
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.11k
      off_t offset = elf->start_offset;
143
1.11k
      error = __libdw_image_header (*fdp, &offset,
144
1.11k
            (elf->map_address == NULL ? NULL
145
1.11k
             : elf->map_address + offset),
146
1.11k
            elf->maximum_size);
147
1.11k
      if (error == DWFL_E_NOERROR)
148
178
  {
149
    /* Pure evil.  libelf needs some better interfaces.  */
150
178
    elf->kind = ELF_K_AR;
151
178
    elf->state.ar.elf_ar_hdr.ar_name = "libdwfl is faking you out";
152
178
    elf->state.ar.elf_ar_hdr.ar_size = elf->maximum_size - offset;
153
178
    elf->state.ar.offset = offset - sizeof (struct ar_hdr);
154
178
    Elf *subelf = elf_begin (-1, elf->cmd, elf);
155
178
    elf->kind = ELF_K_NONE;
156
178
    if (unlikely (subelf == NULL))
157
1
      error = DWFL_E_LIBELF;
158
177
    else
159
177
      {
160
177
        subelf->parent = NULL;
161
177
        subelf->flags |= elf->flags & (ELF_F_MMAPPED | ELF_F_MALLOCED);
162
177
        elf->flags &= ~(ELF_F_MMAPPED | ELF_F_MALLOCED);
163
177
        elf_end (elf);
164
177
        elf = subelf;
165
177
        error = what_kind (*fdp, &elf, &kind, &may_close_fd);
166
177
      }
167
178
  }
168
1.11k
    }
169
170
19.5k
  if (error == DWFL_E_NOERROR
171
19.5k
      && kind != ELF_K_ELF
172
19.5k
      && !(archive_ok && kind == ELF_K_AR))
173
657
    error = DWFL_E_BADELF;
174
175
  /* This basically means, we keep a ELF_K_NONE Elf handle and return it.  */
176
19.5k
  if (bad_elf_ok && error == DWFL_E_BADELF)
177
0
    error = DWFL_E_NOERROR;
178
179
19.5k
  if (error != DWFL_E_NOERROR)
180
2.84k
    {
181
2.84k
      elf_end (elf);
182
2.84k
      elf = NULL;
183
2.84k
    }
184
185
19.5k
  if (! never_close_fd
186
19.5k
      && error == DWFL_E_NOERROR ? may_close_fd : close_on_fail)
187
4.96k
    {
188
4.96k
      close (*fdp);
189
4.96k
      *fdp = -1;
190
4.96k
    }
191
192
19.5k
  *elfp = elf;
193
19.5k
  return error;
194
19.5k
}
195
196
Dwfl_Error internal_function
197
__libdw_open_file (int *fdp, Elf **elfp, bool close_on_fail, bool archive_ok)
198
19.5k
{
199
19.5k
  return libdw_open_elf (fdp, elfp, close_on_fail, archive_ok, false, false,
200
19.5k
       false);
201
19.5k
}
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
}