Coverage Report

Created: 2025-11-06 06:34

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/src/elfutils/libelf/elf_getdata_rawchunk.c
Line
Count
Source
1
/* Return converted data from raw chunk of ELF file.
2
   Copyright (C) 2007, 2014, 2015 Red Hat, Inc.
3
   Copyright (C) 2022, 2023 Mark J. Wielaard <mark@klomp.org>
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 <assert.h>
35
#include <errno.h>
36
#include <string.h>
37
38
#include "libelfP.h"
39
#include "common.h"
40
#include "eu-search.h"
41
42
static int
43
chunk_compare (const void *a, const void *b)
44
4.27k
{
45
4.27k
  Elf_Data_Chunk *da = (Elf_Data_Chunk *)a;
46
4.27k
  Elf_Data_Chunk *db = (Elf_Data_Chunk *)b;
47
48
4.27k
  if (da->offset != db->offset)
49
1.01k
    return da->offset - db->offset;
50
51
3.25k
  if (da->data.d.d_size != db->data.d.d_size)
52
588
    return da->data.d.d_size - db->data.d.d_size;
53
54
2.67k
  return da->data.d.d_type - db->data.d.d_type;
55
3.25k
}
56
57
Elf_Data *
58
elf_getdata_rawchunk (Elf *elf, int64_t offset, size_t size, Elf_Type type)
59
12.8k
{
60
12.8k
  if (unlikely (elf == NULL))
61
0
    return NULL;
62
63
12.8k
  if (unlikely (elf->kind != ELF_K_ELF))
64
0
    {
65
      /* No valid descriptor.  */
66
0
      __libelf_seterrno (ELF_E_INVALID_HANDLE);
67
0
      return NULL;
68
0
    }
69
70
12.8k
  if (unlikely (offset < 0 || (uint64_t) offset > elf->maximum_size
71
12.8k
    || elf->maximum_size - (uint64_t) offset < size))
72
73
596
    {
74
      /* Invalid request.  */
75
596
      __libelf_seterrno (ELF_E_INVALID_OP);
76
596
      return NULL;
77
596
    }
78
79
12.2k
  if (type >= ELF_T_NUM)
80
0
    {
81
0
      __libelf_seterrno (ELF_E_UNKNOWN_TYPE);
82
0
      return NULL;
83
0
    }
84
85
  /* Get the raw bytes from the file.  */
86
12.2k
  void *rawchunk;
87
12.2k
  int flags = 0;
88
12.2k
  Elf_Data *result = NULL;
89
90
12.2k
  rwlock_wrlock (elf->lock);
91
92
  /* Maybe we already got this chunk?  */
93
12.2k
  Elf_Data_Chunk key;
94
12.2k
  key.offset = offset;
95
12.2k
  key.data.d.d_size = size;
96
12.2k
  key.data.d.d_type = type;
97
12.2k
  Elf_Data_Chunk **found
98
12.2k
    = eu_tsearch_nolock (&key, &elf->state.elf.rawchunk_tree, &chunk_compare);
99
100
12.2k
  if (found == NULL)
101
0
    goto nomem;
102
103
  /* Existing entry.  */
104
12.2k
  if (*found != &key && *found != NULL)
105
2.57k
    {
106
2.57k
      result = &(*found)->data.d;
107
2.57k
      goto out;
108
2.57k
    }
109
110
  /* New entry.  Note that *found will point to the newly inserted
111
     (dummy) key.  We'll replace it with a real rawchunk when that is
112
     setup.  Make sure to tdelete the dummy key if anything goes
113
     wrong.  */
114
115
9.68k
  size_t align = __libelf_type_align (elf->class, type);
116
9.68k
  if (elf->map_address != NULL)
117
9.68k
    {
118
    /* If the file is mmap'ed we can use it directly, if aligned for type.  */
119
9.68k
      char *rawdata = elf->map_address + elf->start_offset + offset;
120
9.68k
      if (((uintptr_t) rawdata & (align - 1)) == 0)
121
8.59k
  rawchunk = rawdata;
122
1.08k
      else
123
1.08k
  {
124
    /* We allocate the memory and memcpy it to get aligned data. */
125
1.08k
    rawchunk = malloc (size);
126
1.08k
    if (rawchunk == NULL)
127
0
      goto nomem;
128
1.08k
    memcpy (rawchunk, rawdata, size);
129
1.08k
    flags = ELF_F_MALLOCED;
130
1.08k
  }
131
9.68k
    }
132
0
  else
133
0
    {
134
      /* We allocate the memory and read the data from the file.  */
135
0
      rawchunk = malloc (size);
136
0
      if (rawchunk == NULL)
137
0
  {
138
0
  nomem:
139
0
    eu_tdelete_nolock (&key, &elf->state.elf.rawchunk_tree,
140
0
           &chunk_compare);
141
0
    __libelf_seterrno (ELF_E_NOMEM);
142
0
    goto out;
143
0
  }
144
145
      /* Read the file content.  */
146
0
      if (unlikely ((size_t) pread_retry (elf->fildes, rawchunk, size,
147
0
            elf->start_offset + offset)
148
0
        != size))
149
0
  {
150
    /* Something went wrong.  */
151
0
    eu_tdelete_nolock (&key, &elf->state.elf.rawchunk_tree,
152
0
           &chunk_compare);
153
0
    free (rawchunk);
154
0
    __libelf_seterrno (ELF_E_READ_ERROR);
155
0
    goto out;
156
0
  }
157
158
0
      flags = ELF_F_MALLOCED;
159
0
    }
160
161
  /* Copy and/or convert the data as needed for aligned native-order access.  */
162
9.68k
  void *buffer;
163
9.68k
  if (elf->state.elf32.ehdr->e_ident[EI_DATA] == MY_ELFDATA)
164
1.80k
    {
165
1.80k
      if (((uintptr_t) rawchunk & (align - 1)) == 0)
166
  /* No need to copy, we can use the raw data.  */
167
1.80k
  buffer = rawchunk;
168
0
      else
169
0
  {
170
    /* A malloc'd block is always sufficiently aligned.  */
171
0
    assert (flags == 0);
172
173
0
    buffer = malloc (size);
174
0
    if (unlikely (buffer == NULL))
175
0
      goto nomem;
176
0
    flags = ELF_F_MALLOCED;
177
178
    /* The copy will be appropriately aligned for direct access.  */
179
0
    memcpy (buffer, rawchunk, size);
180
181
0
    free (rawchunk);
182
0
  }
183
1.80k
    }
184
7.88k
  else
185
7.88k
    {
186
7.88k
      if (flags)
187
363
  buffer = rawchunk;
188
7.51k
      else
189
7.51k
  {
190
7.51k
    buffer = malloc (size);
191
7.51k
    if (unlikely (buffer == NULL))
192
0
      goto nomem;
193
7.51k
    flags = ELF_F_MALLOCED;
194
7.51k
  }
195
196
      /* Call the conversion function.  */
197
7.88k
      (*__elf_xfctstom[elf->class - 1][type])(buffer, rawchunk, size, 0);
198
199
7.88k
      if (!flags)
200
0
  free (rawchunk);
201
7.88k
    }
202
203
  /* Allocate the dummy container to point at this buffer.  */
204
9.68k
  Elf_Data_Chunk *chunk = calloc (1, sizeof *chunk);
205
9.68k
  if (chunk == NULL)
206
0
    {
207
0
      if (flags)
208
0
  free (buffer);
209
0
      goto nomem;
210
0
    }
211
212
9.68k
  chunk->dummy_scn.elf = elf;
213
9.68k
  chunk->dummy_scn.flags = flags;
214
9.68k
  chunk->data.s = &chunk->dummy_scn;
215
9.68k
  chunk->data.d.d_buf = buffer;
216
9.68k
  chunk->data.d.d_size = size;
217
9.68k
  chunk->data.d.d_type = type;
218
9.68k
  chunk->data.d.d_align = align;
219
9.68k
  chunk->data.d.d_version = EV_CURRENT;
220
9.68k
  chunk->offset = offset;
221
222
9.68k
  *found = chunk;
223
9.68k
  result = &chunk->data.d;
224
225
12.2k
 out:
226
12.2k
  rwlock_unlock (elf->lock);
227
12.2k
  return result;
228
9.68k
}