/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 | } |