/src/libbpf/elfutils/libelf/gelf_xlate.c
Line | Count | Source (jump to first uncovered line) |
1 | | /* Transformation functions for ELF data types. |
2 | | Copyright (C) 1998,1999,2000,2002,2004,2005,2006,2007,2015 Red Hat, Inc. |
3 | | Copyright (C) 2022 Mark J. Wielaard <mark@klomp.org> |
4 | | This file is part of elfutils. |
5 | | Written by Ulrich Drepper <drepper@redhat.com>, 1998. |
6 | | |
7 | | This file is free software; you can redistribute it and/or modify |
8 | | it under the terms of either |
9 | | |
10 | | * the GNU Lesser General Public License as published by the Free |
11 | | Software Foundation; either version 3 of the License, or (at |
12 | | your option) any later version |
13 | | |
14 | | or |
15 | | |
16 | | * the GNU General Public License as published by the Free |
17 | | Software Foundation; either version 2 of the License, or (at |
18 | | your option) any later version |
19 | | |
20 | | or both in parallel, as here. |
21 | | |
22 | | elfutils is distributed in the hope that it will be useful, but |
23 | | WITHOUT ANY WARRANTY; without even the implied warranty of |
24 | | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
25 | | General Public License for more details. |
26 | | |
27 | | You should have received copies of the GNU General Public License and |
28 | | the GNU Lesser General Public License along with this program. If |
29 | | not, see <http://www.gnu.org/licenses/>. */ |
30 | | |
31 | | #ifdef HAVE_CONFIG_H |
32 | | # include <config.h> |
33 | | #endif |
34 | | |
35 | | #include <stdint.h> |
36 | | #include <string.h> |
37 | | #include <stdlib.h> |
38 | | |
39 | | #include "libelfP.h" |
40 | | |
41 | | #ifndef LIBELFBITS |
42 | | # define LIBELFBITS 32 |
43 | | #endif |
44 | | |
45 | | |
46 | | /* Well, what shall I say. Nothing to do here. */ |
47 | | #define elf_cvt_Byte(dest, src, n) \ |
48 | | (__builtin_constant_p (n) && (n) == 1 \ |
49 | | ? (void) (*((char *) (dest)) = *((char *) (src))) \ |
50 | | : Elf32_cvt_Byte (dest, src, n)) |
51 | | static void |
52 | | (elf_cvt_Byte) (void *dest, const void *src, size_t n, |
53 | | int encode __attribute__ ((unused))) |
54 | 0 | { |
55 | 0 | if (n != 0) |
56 | 0 | memmove (dest, src, n); |
57 | 0 | } |
58 | | |
59 | | |
60 | | /* We'll optimize the definition of the conversion functions here a |
61 | | bit. We need only functions for 16, 32, and 64 bits. The |
62 | | functions referenced in the table will be aliases for one of these |
63 | | functions. Which one is decided by the ELFxx_FSZ_type. */ |
64 | | |
65 | | #if ALLOW_UNALIGNED |
66 | | |
67 | | #define FETCH(Bits, ptr) (*(const uint##Bits##_t *) ptr) |
68 | 210k | #define STORE(Bits, ptr, val) (*(uint##Bits##_t *) ptr = val) |
69 | | |
70 | | #else |
71 | | |
72 | | union unaligned |
73 | | { |
74 | | uint16_t u16; |
75 | | uint32_t u32; |
76 | | uint64_t u64; |
77 | | } attribute_packed; |
78 | | |
79 | | #define FETCH(Bits, ptr) (((const union unaligned *) ptr)->u##Bits) |
80 | | #define STORE(Bits, ptr, val) (((union unaligned *) ptr)->u##Bits = val) |
81 | | |
82 | | #endif |
83 | | |
84 | | /* Now define the conversion functions for the basic types. We use here |
85 | | the fact that file and memory types are the same and that we have the |
86 | | ELFxx_FSZ_* macros. |
87 | | |
88 | | At the same time we define inline functions which we will use to |
89 | | convert the complex types. */ |
90 | | #define FUNDAMENTAL(NAME, Name, Bits) \ |
91 | | INLINE2 (ELFW2(Bits,FSZ_##NAME), ElfW2(Bits,cvt_##Name), ElfW2(Bits,Name)) |
92 | | #define INLINE2(Bytes, FName, TName) \ |
93 | | INLINE3 (Bytes, FName, TName) |
94 | | #define INLINE3(Bytes, FName, TName) \ |
95 | | static inline void FName##1 (void *dest, const void *ptr) \ |
96 | 210k | { \ |
97 | 210k | switch (Bytes) \ |
98 | 210k | { \ |
99 | 85.4k | case 2: STORE (16, dest, bswap_16 (FETCH (16, ptr))); break; \ |
100 | 59.0k | case 4: STORE (32, dest, bswap_32 (FETCH (32, ptr))); break; \ |
101 | 66.5k | case 8: STORE (64, dest, bswap_64 (FETCH (64, ptr))); break; \ |
102 | 0 | default: \ |
103 | 0 | abort (); \ |
104 | 210k | } \ |
105 | 210k | } \ Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Addr1 Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Sword1 Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Half1 gelf_xlate.c:Elf32_cvt_Word1 Line | Count | Source | 96 | 4.44k | { \ | 97 | 4.44k | switch (Bytes) \ | 98 | 4.44k | { \ | 99 | 0 | case 2: STORE (16, dest, bswap_16 (FETCH (16, ptr))); break; \ | 100 | 4.44k | case 4: STORE (32, dest, bswap_32 (FETCH (32, ptr))); break; \ | 101 | 0 | case 8: STORE (64, dest, bswap_64 (FETCH (64, ptr))); break; \ | 102 | 0 | default: \ | 103 | 0 | abort (); \ | 104 | 4.44k | } \ | 105 | 4.44k | } \ |
Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Off1 Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Xword1 Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Sxword1 gelf_xlate.c:Elf64_cvt_Addr1 Line | Count | Source | 96 | 35.5k | { \ | 97 | 35.5k | switch (Bytes) \ | 98 | 35.5k | { \ | 99 | 0 | case 2: STORE (16, dest, bswap_16 (FETCH (16, ptr))); break; \ | 100 | 0 | case 4: STORE (32, dest, bswap_32 (FETCH (32, ptr))); break; \ | 101 | 35.5k | case 8: STORE (64, dest, bswap_64 (FETCH (64, ptr))); break; \ | 102 | 0 | default: \ | 103 | 0 | abort (); \ | 104 | 35.5k | } \ | 105 | 35.5k | } \ |
gelf_xlate.c:Elf64_cvt_Xword1 Line | Count | Source | 96 | 30.6k | { \ | 97 | 30.6k | switch (Bytes) \ | 98 | 30.6k | { \ | 99 | 0 | case 2: STORE (16, dest, bswap_16 (FETCH (16, ptr))); break; \ | 100 | 0 | case 4: STORE (32, dest, bswap_32 (FETCH (32, ptr))); break; \ | 101 | 30.6k | case 8: STORE (64, dest, bswap_64 (FETCH (64, ptr))); break; \ | 102 | 0 | default: \ | 103 | 0 | abort (); \ | 104 | 30.6k | } \ | 105 | 30.6k | } \ |
gelf_xlate.c:Elf64_cvt_Half1 Line | Count | Source | 96 | 85.4k | { \ | 97 | 85.4k | switch (Bytes) \ | 98 | 85.4k | { \ | 99 | 85.4k | case 2: STORE (16, dest, bswap_16 (FETCH (16, ptr))); break; \ | 100 | 0 | case 4: STORE (32, dest, bswap_32 (FETCH (32, ptr))); break; \ | 101 | 0 | case 8: STORE (64, dest, bswap_64 (FETCH (64, ptr))); break; \ | 102 | 0 | default: \ | 103 | 0 | abort (); \ | 104 | 85.4k | } \ | 105 | 85.4k | } \ |
gelf_xlate.c:Elf64_cvt_Word1 Line | Count | Source | 96 | 54.5k | { \ | 97 | 54.5k | switch (Bytes) \ | 98 | 54.5k | { \ | 99 | 0 | case 2: STORE (16, dest, bswap_16 (FETCH (16, ptr))); break; \ | 100 | 54.5k | case 4: STORE (32, dest, bswap_32 (FETCH (32, ptr))); break; \ | 101 | 0 | case 8: STORE (64, dest, bswap_64 (FETCH (64, ptr))); break; \ | 102 | 0 | default: \ | 103 | 0 | abort (); \ | 104 | 54.5k | } \ | 105 | 54.5k | } \ |
Unexecuted instantiation: gelf_xlate.c:Elf64_cvt_Off1 gelf_xlate.c:Elf64_cvt_Sxword1 Line | Count | Source | 96 | 382 | { \ | 97 | 382 | switch (Bytes) \ | 98 | 382 | { \ | 99 | 0 | case 2: STORE (16, dest, bswap_16 (FETCH (16, ptr))); break; \ | 100 | 0 | case 4: STORE (32, dest, bswap_32 (FETCH (32, ptr))); break; \ | 101 | 382 | case 8: STORE (64, dest, bswap_64 (FETCH (64, ptr))); break; \ | 102 | 0 | default: \ | 103 | 0 | abort (); \ | 104 | 382 | } \ | 105 | 382 | } \ |
Unexecuted instantiation: gelf_xlate.c:Elf64_cvt_Sword1 |
106 | | \ |
107 | | static void FName (void *dest, const void *ptr, size_t len, \ |
108 | | int encode __attribute__ ((unused))) \ |
109 | 587 | { \ |
110 | 587 | size_t n = len / sizeof (TName); \ |
111 | 587 | if (dest < ptr) \ |
112 | 56.4k | while (n-- > 0) \ |
113 | 56.0k | { \ |
114 | 56.0k | FName##1 (dest, ptr); \ |
115 | 56.0k | dest += Bytes; \ |
116 | 56.0k | ptr += Bytes; \ |
117 | 56.0k | } \ |
118 | 587 | else \ |
119 | 587 | { \ |
120 | 219 | dest += len; \ |
121 | 219 | ptr += len; \ |
122 | 21.3k | while (n-- > 0) \ |
123 | 21.1k | { \ |
124 | 21.1k | ptr -= Bytes; \ |
125 | 21.1k | dest -= Bytes; \ |
126 | 21.1k | FName##1 (dest, ptr); \ |
127 | 21.1k | } \ |
128 | 219 | } \ |
129 | 587 | } Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Addr Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Half Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Off Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Sword Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Word Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Xword Unexecuted instantiation: gelf_xlate.c:Elf32_cvt_Sxword gelf_xlate.c:Elf64_cvt_Addr Line | Count | Source | 109 | 91 | { \ | 110 | 91 | size_t n = len / sizeof (TName); \ | 111 | 91 | if (dest < ptr) \ | 112 | 2.86k | while (n-- > 0) \ | 113 | 2.82k | { \ | 114 | 2.82k | FName##1 (dest, ptr); \ | 115 | 2.82k | dest += Bytes; \ | 116 | 2.82k | ptr += Bytes; \ | 117 | 2.82k | } \ | 118 | 91 | else \ | 119 | 91 | { \ | 120 | 43 | dest += len; \ | 121 | 43 | ptr += len; \ | 122 | 4.09k | while (n-- > 0) \ | 123 | 4.04k | { \ | 124 | 4.04k | ptr -= Bytes; \ | 125 | 4.04k | dest -= Bytes; \ | 126 | 4.04k | FName##1 (dest, ptr); \ | 127 | 4.04k | } \ | 128 | 43 | } \ | 129 | 91 | } |
gelf_xlate.c:Elf64_cvt_Half Line | Count | Source | 109 | 184 | { \ | 110 | 184 | size_t n = len / sizeof (TName); \ | 111 | 184 | if (dest < ptr) \ | 112 | 40.5k | while (n-- > 0) \ | 113 | 40.3k | { \ | 114 | 40.3k | FName##1 (dest, ptr); \ | 115 | 40.3k | dest += Bytes; \ | 116 | 40.3k | ptr += Bytes; \ | 117 | 40.3k | } \ | 118 | 184 | else \ | 119 | 184 | { \ | 120 | 60 | dest += len; \ | 121 | 60 | ptr += len; \ | 122 | 10.2k | while (n-- > 0) \ | 123 | 10.1k | { \ | 124 | 10.1k | ptr -= Bytes; \ | 125 | 10.1k | dest -= Bytes; \ | 126 | 10.1k | FName##1 (dest, ptr); \ | 127 | 10.1k | } \ | 128 | 60 | } \ | 129 | 184 | } |
Unexecuted instantiation: gelf_xlate.c:Elf64_cvt_Off Unexecuted instantiation: gelf_xlate.c:Elf64_cvt_Sword gelf_xlate.c:Elf64_cvt_Word Line | Count | Source | 109 | 312 | { \ | 110 | 312 | size_t n = len / sizeof (TName); \ | 111 | 312 | if (dest < ptr) \ | 112 | 13.0k | while (n-- > 0) \ | 113 | 12.8k | { \ | 114 | 12.8k | FName##1 (dest, ptr); \ | 115 | 12.8k | dest += Bytes; \ | 116 | 12.8k | ptr += Bytes; \ | 117 | 12.8k | } \ | 118 | 312 | else \ | 119 | 312 | { \ | 120 | 116 | dest += len; \ | 121 | 116 | ptr += len; \ | 122 | 7.03k | while (n-- > 0) \ | 123 | 6.91k | { \ | 124 | 6.91k | ptr -= Bytes; \ | 125 | 6.91k | dest -= Bytes; \ | 126 | 6.91k | FName##1 (dest, ptr); \ | 127 | 6.91k | } \ | 128 | 116 | } \ | 129 | 312 | } |
Unexecuted instantiation: gelf_xlate.c:Elf64_cvt_Xword Unexecuted instantiation: gelf_xlate.c:Elf64_cvt_Sxword |
130 | | |
131 | | |
132 | | /* Now the tricky part: define the transformation functions for the |
133 | | complex types. We will use the definitions of the types in |
134 | | abstract.h. */ |
135 | | #define START(Bits, Name, EName) \ |
136 | | static void \ |
137 | | ElfW2 (Bits, cvt_##Name) (void *dest, const void *src, size_t len, \ |
138 | | int encode __attribute__ ((unused))) \ |
139 | 2.20k | { ElfW2(Bits, Name) *tdest = (ElfW2(Bits, Name) *) dest; \ |
140 | 2.20k | ElfW2(Bits, Name) *tsrc = (ElfW2(Bits, Name) *) src; \ |
141 | 2.20k | size_t sz = sizeof (ElfW2(Bits, Name)); \ |
142 | 2.20k | size_t n; \ |
143 | 38.9k | for (n = len / sz; n > 0; ++tdest, ++tsrc, --n) { |
144 | | #define END(Bits, Name) \ |
145 | 36.7k | } \ |
146 | 2.20k | if (len % sz > 0) /* Cannot convert partial structures, just copy. */ \ |
147 | 2.20k | memmove (dest, src, len % sz); \ |
148 | 2.20k | } |
149 | | #define TYPE_EXTRA(Code) |
150 | 52.7k | #define TYPE_XLATE(Code) Code |
151 | 132k | #define TYPE_NAME(Type, Name) TYPE_NAME2 (Type, Name) |
152 | 132k | #define TYPE_NAME2(Type, Name) Type##1 (&tdest->Name, &tsrc->Name); |
153 | | #define TYPE(Name, Bits) TYPE2 (Name, Bits) |
154 | | #define TYPE2(Name, Bits) TYPE3 (Name##Bits) |
155 | | #define TYPE3(Name) Name (cvt_) |
156 | | |
157 | | /* Signal that we are generating conversion functions. */ |
158 | | #define GENERATE_CONVERSION |
159 | | |
160 | | /* First generate the 32-bit conversion functions. */ |
161 | | #define LIBELFBITS 32 |
162 | | #include "gelf_xlate.h" |
163 | | |
164 | | /* Now generate the 64-bit conversion functions. */ |
165 | | #define LIBELFBITS 64 |
166 | | #include "gelf_xlate.h" |
167 | | |
168 | | |
169 | | /* We have a few functions which we must create by hand since the sections |
170 | | do not contain records of only one type. */ |
171 | | #include "version_xlate.h" |
172 | | #include "gnuhash_xlate.h" |
173 | | #include "note_xlate.h" |
174 | | #include "chdr_xlate.h" |
175 | | |
176 | | |
177 | | /* Now the externally visible table with the function pointers. */ |
178 | | const xfct_t __elf_xfctstom[ELFCLASSNUM - 1][ELF_T_NUM] = |
179 | | { |
180 | | [ELFCLASS32 - 1] = { |
181 | | #define define_xfcts(Bits) \ |
182 | | [ELF_T_BYTE] = elf_cvt_Byte, \ |
183 | | [ELF_T_ADDR] = ElfW2(Bits, cvt_Addr), \ |
184 | | [ELF_T_DYN] = ElfW2(Bits, cvt_Dyn), \ |
185 | | [ELF_T_EHDR] = ElfW2(Bits, cvt_Ehdr), \ |
186 | | [ELF_T_HALF] = ElfW2(Bits, cvt_Half), \ |
187 | | [ELF_T_OFF] = ElfW2(Bits, cvt_Off), \ |
188 | | [ELF_T_PHDR] = ElfW2(Bits, cvt_Phdr), \ |
189 | | [ELF_T_RELA] = ElfW2(Bits, cvt_Rela), \ |
190 | | [ELF_T_REL] = ElfW2(Bits, cvt_Rel), \ |
191 | | [ELF_T_SHDR] = ElfW2(Bits, cvt_Shdr), \ |
192 | | [ELF_T_SWORD] = ElfW2(Bits, cvt_Sword), \ |
193 | | [ELF_T_SYM] = ElfW2(Bits, cvt_Sym), \ |
194 | | [ELF_T_WORD] = ElfW2(Bits, cvt_Word), \ |
195 | | [ELF_T_XWORD] = ElfW2(Bits, cvt_Xword), \ |
196 | | [ELF_T_SXWORD] = ElfW2(Bits, cvt_Sxword), \ |
197 | | [ELF_T_VDEF] = elf_cvt_Verdef, \ |
198 | | [ELF_T_VDAUX] = elf_cvt_Verdef, \ |
199 | | [ELF_T_VNEED] = elf_cvt_Verneed, \ |
200 | | [ELF_T_VNAUX] = elf_cvt_Verneed, \ |
201 | | [ELF_T_NHDR] = elf_cvt_note4, \ |
202 | | [ELF_T_NHDR8] = elf_cvt_note8, \ |
203 | | [ELF_T_SYMINFO] = ElfW2(Bits, cvt_Syminfo), \ |
204 | | [ELF_T_MOVE] = ElfW2(Bits, cvt_Move), \ |
205 | | [ELF_T_LIB] = ElfW2(Bits, cvt_Lib), \ |
206 | | [ELF_T_AUXV] = ElfW2(Bits, cvt_auxv_t), \ |
207 | | [ELF_T_CHDR] = ElfW2(Bits, cvt_chdr) |
208 | | define_xfcts (32), |
209 | | [ELF_T_GNUHASH] = Elf32_cvt_Word |
210 | | }, |
211 | | [ELFCLASS64 - 1] = { |
212 | | define_xfcts (64), |
213 | | [ELF_T_GNUHASH] = elf_cvt_gnuhash |
214 | | } |
215 | | }; |