Merge remote-tracking branch 'origin/stable-2.0'
[bpt/guile.git] / module / system / vm / elf.scm
1 ;;; Guile ELF reader and writer
2
3 ;; Copyright (C) 2011, 2012, 2013 Free Software Foundation, Inc.
4
5 ;;;; This library is free software; you can redistribute it and/or
6 ;;;; modify it under the terms of the GNU Lesser General Public
7 ;;;; License as published by the Free Software Foundation; either
8 ;;;; version 3 of the License, or (at your option) any later version.
9 ;;;;
10 ;;;; This library is distributed in the hope that it will be useful,
11 ;;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
12 ;;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 ;;;; Lesser General Public License for more details.
14 ;;;;
15 ;;;; You should have received a copy of the GNU Lesser General Public
16 ;;;; License along with this library; if not, write to the Free Software
17 ;;;; Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
18
19 ;;; Commentary:
20 ;;;
21 ;;; A module to read and write Executable and Linking Format (ELF)
22 ;;; files.
23 ;;;
24 ;;; This module exports a number of record types that represent the
25 ;;; various parts that make up ELF files. Fundamentally this is the
26 ;;; main header, the segment headers (program headers), and the section
27 ;;; headers. It also exports bindings for symbolic constants and
28 ;;; utilities to parse and write special kinds of ELF sections.
29 ;;;
30 ;;; See elf(5) for more information on ELF.
31 ;;;
32 ;;; Code:
33
34 (define-module (system vm elf)
35 #:use-module (rnrs bytevectors)
36 #:use-module (system foreign)
37 #:use-module (system base target)
38 #:use-module (srfi srfi-9)
39 #:use-module (ice-9 receive)
40 #:use-module (ice-9 vlist)
41 #:export (has-elf-header?
42
43 (make-elf* . make-elf)
44 elf?
45 elf-bytes elf-word-size elf-byte-order
46 elf-abi elf-type elf-machine-type
47 elf-entry elf-phoff elf-shoff elf-flags elf-ehsize
48 elf-phentsize elf-phnum elf-shentsize elf-shnum elf-shstrndx
49
50 elf-header-len elf-header-shoff-offset
51 write-elf-header
52
53 (make-elf-segment* . make-elf-segment)
54 elf-segment?
55 elf-segment-index
56 elf-segment-type elf-segment-offset elf-segment-vaddr
57 elf-segment-paddr elf-segment-filesz elf-segment-memsz
58 elf-segment-flags elf-segment-align
59
60 elf-program-header-len write-elf-program-header
61
62 PT_NULL PT_LOAD PT_DYNAMIC PT_INTERP PT_NOTE PT_SHLIB
63 PT_PHDR PT_TLS PT_NUM PT_LOOS PT_GNU_EH_FRAME PT_GNU_STACK
64 PT_GNU_RELRO
65
66 PF_R PF_W PF_X
67
68 (make-elf-section* . make-elf-section)
69 elf-section?
70 elf-section-index
71 elf-section-name elf-section-type elf-section-flags
72 elf-section-addr elf-section-offset elf-section-size
73 elf-section-link elf-section-info elf-section-addralign
74 elf-section-entsize
75
76 elf-section-header-len elf-section-header-addr-offset
77 elf-section-header-offset-offset
78 write-elf-section-header
79
80 (make-elf-symbol* . make-elf-symbol)
81 elf-symbol?
82 elf-symbol-name elf-symbol-value elf-symbol-size
83 elf-symbol-info elf-symbol-other elf-symbol-shndx
84 elf-symbol-binding elf-symbol-type elf-symbol-visibility
85
86 elf-symbol-len write-elf-symbol
87
88 SHN_UNDEF
89
90 SHT_NULL SHT_PROGBITS SHT_SYMTAB SHT_STRTAB SHT_RELA
91 SHT_HASH SHT_DYNAMIC SHT_NOTE SHT_NOBITS SHT_REL SHT_SHLIB
92 SHT_DYNSYM SHT_INIT_ARRAY SHT_FINI_ARRAY SHT_PREINIT_ARRAY
93 SHT_GROUP SHT_SYMTAB_SHNDX SHT_NUM SHT_LOOS SHT_HIOS
94 SHT_LOPROC SHT_HIPROC SHT_LOUSER SHT_HIUSER
95
96 SHF_WRITE SHF_ALLOC SHF_EXECINSTR SHF_MERGE SHF_STRINGS
97 SHF_INFO_LINK SHF_LINK_ORDER SHF_OS_NONCONFORMING SHF_GROUP
98 SHF_TLS
99
100 DT_NULL DT_NEEDED DT_PLTRELSZ DT_PLTGOT DT_HASH DT_STRTAB
101 DT_SYMTAB DT_RELA DT_RELASZ DT_RELAENT DT_STRSZ DT_SYMENT
102 DT_INIT DT_FINI DT_SONAME DT_RPATH DT_SYMBOLIC DT_REL
103 DT_RELSZ DT_RELENT DT_PLTREL DT_DEBUG DT_TEXTREL DT_JMPREL
104 DT_BIND_NOW DT_INIT_ARRAY DT_FINI_ARRAY DT_INIT_ARRAYSZ
105 DT_FINI_ARRAYSZ DT_RUNPATH DT_FLAGS DT_ENCODING
106 DT_PREINIT_ARRAY DT_PREINIT_ARRAYSZ DT_NUM DT_LOGUILE
107 DT_GUILE_GC_ROOT DT_GUILE_GC_ROOT_SZ DT_GUILE_ENTRY
108 DT_GUILE_VM_VERSION DT_HIGUILE DT_LOOS DT_HIOS DT_LOPROC
109 DT_HIPROC
110
111 string-table-ref
112
113 STB_LOCAL STB_GLOBAL STB_WEAK STB_NUM STB_LOOS STB_GNU
114 STB_HIOS STB_LOPROC STB_HIPROC
115
116 STT_NOTYPE STT_OBJECT STT_FUNC STT_SECTION STT_FILE
117 STT_COMMON STT_TLS STT_NUM STT_LOOS STT_GNU STT_HIOS
118 STT_LOPROC STT_HIPROC
119
120 STV_DEFAULT STV_INTERNAL STV_HIDDEN STV_PROTECTED
121
122 NT_GNU_ABI_TAG NT_GNU_HWCAP NT_GNU_BUILD_ID NT_GNU_GOLD_VERSION
123
124 parse-elf
125 elf-segment elf-segments
126 elf-section elf-sections elf-section-by-name elf-sections-by-name
127 elf-symbol-table-len elf-symbol-table-ref
128
129 parse-elf-note
130 elf-note-name elf-note-desc elf-note-type))
131
132 ;; #define EI_NIDENT 16
133
134 ;; typedef struct {
135 ;; unsigned char e_ident[EI_NIDENT];
136 ;; uint16_t e_type;
137 ;; uint16_t e_machine;
138 ;; uint32_t e_version;
139 ;; ElfN_Addr e_entry;
140 ;; ElfN_Off e_phoff;
141 ;; ElfN_Off e_shoff;
142 ;; uint32_t e_flags;
143 ;; uint16_t e_ehsize;
144 ;; uint16_t e_phentsize;
145 ;; uint16_t e_phnum;
146 ;; uint16_t e_shentsize;
147 ;; uint16_t e_shnum;
148 ;; uint16_t e_shstrndx;
149 ;; } ElfN_Ehdr;
150
151 (define elf32-header-len 52)
152 (define elf64-header-len 64)
153 (define (elf-header-len word-size)
154 (case word-size
155 ((4) elf32-header-len)
156 ((8) elf64-header-len)
157 (else (error "invalid word size" word-size))))
158 (define (elf-header-shoff-offset word-size)
159 (case word-size
160 ((4) 32)
161 ((8) 40)
162 (else (error "bad word size" word-size))))
163
164 (define ELFCLASS32 1) ; 32-bit objects
165 (define ELFCLASS64 2) ; 64-bit objects
166
167 (define ELFDATA2LSB 1) ; 2's complement, little endian
168 (define ELFDATA2MSB 2) ; 2's complement, big endian
169
170 (define EV_CURRENT 1) ; Current version
171
172 (define ELFOSABI_STANDALONE 255) ; Standalone (embedded) application
173
174 (define ET_DYN 3) ; Shared object file
175
176 ;;
177 ;; Machine types
178 ;;
179 ;; Just a sampling of these values. We could include more, but the
180 ;; important thing is to recognize architectures for which we have a
181 ;; native compiler. Recognizing more common machine types is icing on
182 ;; the cake.
183 ;;
184 (define EM_NONE 0) ; No machine
185 (define EM_SPARC 2) ; SUN SPARC
186 (define EM_386 3) ; Intel 80386
187 (define EM_MIPS 8) ; MIPS R3000 big-endian
188 (define EM_PPC 20) ; PowerPC
189 (define EM_PPC64 21) ; PowerPC 64-bit
190 (define EM_ARM 40) ; ARM
191 (define EM_SH 42) ; Hitachi SH
192 (define EM_SPARCV9 43) ; SPARC v9 64-bit
193 (define EM_IA_64 50) ; Intel Merced
194 (define EM_X86_64 62) ; AMD x86-64 architecture
195
196 (define cpu-mapping (make-hash-table))
197 (for-each (lambda (pair)
198 (hashq-set! cpu-mapping (car pair) (cdr pair)))
199 `((none . ,EM_NONE)
200 (sparc . ,EM_SPARC) ; FIXME: map 64-bit to SPARCV9 ?
201 (i386 . ,EM_386)
202 (mips . ,EM_MIPS)
203 (ppc . ,EM_PPC)
204 (ppc64 . ,EM_PPC64)
205 (arm . ,EM_ARM) ; FIXME: there are more arm cpu variants
206 (sh . ,EM_SH) ; FIXME: there are more sh cpu variants
207 (ia64 . ,EM_IA_64)
208 (x86_64 . ,EM_X86_64)))
209
210 (define SHN_UNDEF 0)
211
212 (define host-machine-type
213 (hashq-ref cpu-mapping
214 (string->symbol (car (string-split %host-type #\-)))
215 EM_NONE))
216
217 (define host-word-size
218 (sizeof '*))
219
220 (define host-byte-order
221 (native-endianness))
222
223 (define (has-elf-header? bv)
224 (and
225 ;; e_ident
226 (>= (bytevector-length bv) 16)
227 (= (bytevector-u8-ref bv 0) #x7f)
228 (= (bytevector-u8-ref bv 1) (char->integer #\E))
229 (= (bytevector-u8-ref bv 2) (char->integer #\L))
230 (= (bytevector-u8-ref bv 3) (char->integer #\F))
231 (cond
232 ((= (bytevector-u8-ref bv 4) ELFCLASS32)
233 (>= (bytevector-length bv) elf32-header-len))
234 ((= (bytevector-u8-ref bv 4) ELFCLASS64)
235 (>= (bytevector-length bv) elf64-header-len))
236 (else #f))
237 (or (= (bytevector-u8-ref bv 5) ELFDATA2LSB)
238 (= (bytevector-u8-ref bv 5) ELFDATA2MSB))
239 (= (bytevector-u8-ref bv 6) EV_CURRENT)
240 ;; Look at ABI later.
241 (= (bytevector-u8-ref bv 8) 0) ; ABI version
242 ;; The rest of the e_ident is padding.
243
244 ;; e_version
245 (let ((byte-order (if (= (bytevector-u8-ref bv 5) ELFDATA2LSB)
246 (endianness little)
247 (endianness big))))
248 (= (bytevector-u32-ref bv 20 byte-order) EV_CURRENT))))
249
250 (define-record-type <elf>
251 (make-elf bytes word-size byte-order abi type machine-type
252 entry phoff shoff flags ehsize
253 phentsize phnum shentsize shnum shstrndx)
254 elf?
255 (bytes elf-bytes)
256 (word-size elf-word-size)
257 (byte-order elf-byte-order)
258 (abi elf-abi)
259 (type elf-type)
260 (machine-type elf-machine-type)
261 (entry elf-entry)
262 (phoff elf-phoff)
263 (shoff elf-shoff)
264 (flags elf-flags)
265 (ehsize elf-ehsize)
266 (phentsize elf-phentsize)
267 (phnum elf-phnum)
268 (shentsize elf-shentsize)
269 (shnum elf-shnum)
270 (shstrndx elf-shstrndx))
271
272 (define* (make-elf* #:key (bytes #f)
273 (byte-order (target-endianness))
274 (word-size (target-word-size))
275 (abi ELFOSABI_STANDALONE)
276 (type ET_DYN)
277 (machine-type EM_NONE)
278 (entry 0)
279 (phoff (elf-header-len word-size))
280 (shoff -1)
281 (flags 0)
282 (ehsize (elf-header-len word-size))
283 (phentsize (elf-program-header-len word-size))
284 (phnum 0)
285 (shentsize (elf-section-header-len word-size))
286 (shnum 0)
287 (shstrndx SHN_UNDEF))
288 (make-elf bytes word-size byte-order abi type machine-type
289 entry phoff shoff flags ehsize
290 phentsize phnum shentsize shnum shstrndx))
291
292 (define (parse-elf32 bv byte-order)
293 (make-elf bv 4 byte-order
294 (bytevector-u8-ref bv 7)
295 (bytevector-u16-ref bv 16 byte-order)
296 (bytevector-u16-ref bv 18 byte-order)
297 (bytevector-u32-ref bv 24 byte-order)
298 (bytevector-u32-ref bv 28 byte-order)
299 (bytevector-u32-ref bv 32 byte-order)
300 (bytevector-u32-ref bv 36 byte-order)
301 (bytevector-u16-ref bv 40 byte-order)
302 (bytevector-u16-ref bv 42 byte-order)
303 (bytevector-u16-ref bv 44 byte-order)
304 (bytevector-u16-ref bv 46 byte-order)
305 (bytevector-u16-ref bv 48 byte-order)
306 (bytevector-u16-ref bv 50 byte-order)))
307
308 (define (write-elf-ident bv class data abi)
309 (bytevector-u8-set! bv 0 #x7f)
310 (bytevector-u8-set! bv 1 (char->integer #\E))
311 (bytevector-u8-set! bv 2 (char->integer #\L))
312 (bytevector-u8-set! bv 3 (char->integer #\F))
313 (bytevector-u8-set! bv 4 class)
314 (bytevector-u8-set! bv 5 data)
315 (bytevector-u8-set! bv 6 EV_CURRENT)
316 (bytevector-u8-set! bv 7 abi)
317 (bytevector-u8-set! bv 8 0) ; ABI version
318 (bytevector-u8-set! bv 9 0) ; Pad to 16 bytes.
319 (bytevector-u8-set! bv 10 0)
320 (bytevector-u8-set! bv 11 0)
321 (bytevector-u8-set! bv 12 0)
322 (bytevector-u8-set! bv 13 0)
323 (bytevector-u8-set! bv 14 0)
324 (bytevector-u8-set! bv 15 0))
325
326 (define (write-elf32-header bv elf)
327 (let ((byte-order (elf-byte-order elf)))
328 (write-elf-ident bv ELFCLASS32
329 (case byte-order
330 ((little) ELFDATA2LSB)
331 ((big) ELFDATA2MSB)
332 (else (error "unknown endianness" byte-order)))
333 (elf-abi elf))
334 (bytevector-u16-set! bv 16 (elf-type elf) byte-order)
335 (bytevector-u16-set! bv 18 (elf-machine-type elf) byte-order)
336 (bytevector-u32-set! bv 20 EV_CURRENT byte-order)
337 (bytevector-u32-set! bv 24 (elf-entry elf) byte-order)
338 (bytevector-u32-set! bv 28 (elf-phoff elf) byte-order)
339 (bytevector-u32-set! bv 32 (elf-shoff elf) byte-order)
340 (bytevector-u32-set! bv 36 (elf-flags elf) byte-order)
341 (bytevector-u16-set! bv 40 (elf-ehsize elf) byte-order)
342 (bytevector-u16-set! bv 42 (elf-phentsize elf) byte-order)
343 (bytevector-u16-set! bv 44 (elf-phnum elf) byte-order)
344 (bytevector-u16-set! bv 46 (elf-shentsize elf) byte-order)
345 (bytevector-u16-set! bv 48 (elf-shnum elf) byte-order)
346 (bytevector-u16-set! bv 50 (elf-shstrndx elf) byte-order)))
347
348 (define (parse-elf64 bv byte-order)
349 (make-elf bv 8 byte-order
350 (bytevector-u8-ref bv 7)
351 (bytevector-u16-ref bv 16 byte-order)
352 (bytevector-u16-ref bv 18 byte-order)
353 (bytevector-u64-ref bv 24 byte-order)
354 (bytevector-u64-ref bv 32 byte-order)
355 (bytevector-u64-ref bv 40 byte-order)
356 (bytevector-u32-ref bv 48 byte-order)
357 (bytevector-u16-ref bv 52 byte-order)
358 (bytevector-u16-ref bv 54 byte-order)
359 (bytevector-u16-ref bv 56 byte-order)
360 (bytevector-u16-ref bv 58 byte-order)
361 (bytevector-u16-ref bv 60 byte-order)
362 (bytevector-u16-ref bv 62 byte-order)))
363
364 (define (write-elf64-header bv elf)
365 (let ((byte-order (elf-byte-order elf)))
366 (write-elf-ident bv ELFCLASS64
367 (case byte-order
368 ((little) ELFDATA2LSB)
369 ((big) ELFDATA2MSB)
370 (else (error "unknown endianness" byte-order)))
371 (elf-abi elf))
372 (bytevector-u16-set! bv 16 (elf-type elf) byte-order)
373 (bytevector-u16-set! bv 18 (elf-machine-type elf) byte-order)
374 (bytevector-u32-set! bv 20 EV_CURRENT byte-order)
375 (bytevector-u64-set! bv 24 (elf-entry elf) byte-order)
376 (bytevector-u64-set! bv 32 (elf-phoff elf) byte-order)
377 (bytevector-u64-set! bv 40 (elf-shoff elf) byte-order)
378 (bytevector-u32-set! bv 48 (elf-flags elf) byte-order)
379 (bytevector-u16-set! bv 52 (elf-ehsize elf) byte-order)
380 (bytevector-u16-set! bv 54 (elf-phentsize elf) byte-order)
381 (bytevector-u16-set! bv 56 (elf-phnum elf) byte-order)
382 (bytevector-u16-set! bv 58 (elf-shentsize elf) byte-order)
383 (bytevector-u16-set! bv 60 (elf-shnum elf) byte-order)
384 (bytevector-u16-set! bv 62 (elf-shstrndx elf) byte-order)))
385
386 (define (parse-elf bv)
387 (cond
388 ((has-elf-header? bv)
389 (let ((class (bytevector-u8-ref bv 4))
390 (byte-order (let ((data (bytevector-u8-ref bv 5)))
391 (cond
392 ((= data ELFDATA2LSB) (endianness little))
393 ((= data ELFDATA2MSB) (endianness big))
394 (else (error "unhandled byte order" data))))))
395 (cond
396 ((= class ELFCLASS32) (parse-elf32 bv byte-order))
397 ((= class ELFCLASS64) (parse-elf64 bv byte-order))
398 (else (error "unhandled class" class)))))
399 (else
400 (error "Invalid ELF" bv))))
401
402 (define* (write-elf-header bv elf)
403 ((case (elf-word-size elf)
404 ((4) write-elf32-header)
405 ((8) write-elf64-header)
406 (else (error "unknown word size" (elf-word-size elf))))
407 bv elf))
408
409 ;;
410 ;; Segment types
411 ;;
412 (define PT_NULL 0) ; Program header table entry unused
413 (define PT_LOAD 1) ; Loadable program segment
414 (define PT_DYNAMIC 2) ; Dynamic linking information
415 (define PT_INTERP 3) ; Program interpreter
416 (define PT_NOTE 4) ; Auxiliary information
417 (define PT_SHLIB 5) ; Reserved
418 (define PT_PHDR 6) ; Entry for header table itself
419 (define PT_TLS 7) ; Thread-local storage segment
420 (define PT_NUM 8) ; Number of defined types
421 (define PT_LOOS #x60000000) ; Start of OS-specific
422 (define PT_GNU_EH_FRAME #x6474e550) ; GCC .eh_frame_hdr segment
423 (define PT_GNU_STACK #x6474e551) ; Indicates stack executability
424 (define PT_GNU_RELRO #x6474e552) ; Read-only after relocation
425
426 ;;
427 ;; Segment flags
428 ;;
429 (define PF_X (ash 1 0)) ; Segment is executable
430 (define PF_W (ash 1 1)) ; Segment is writable
431 (define PF_R (ash 1 2)) ; Segment is readable
432
433 (define-record-type <elf-segment>
434 (make-elf-segment index type offset vaddr paddr filesz memsz flags align)
435 elf-segment?
436 (index elf-segment-index)
437 (type elf-segment-type)
438 (offset elf-segment-offset)
439 (vaddr elf-segment-vaddr)
440 (paddr elf-segment-paddr)
441 (filesz elf-segment-filesz)
442 (memsz elf-segment-memsz)
443 (flags elf-segment-flags)
444 (align elf-segment-align))
445
446 (define* (make-elf-segment* #:key (index -1) (type PT_LOAD) (offset 0) (vaddr 0)
447 (paddr 0) (filesz 0) (memsz filesz)
448 (flags (logior PF_W PF_R))
449 (align 8))
450 (make-elf-segment index type offset vaddr paddr filesz memsz flags align))
451
452 ;; typedef struct {
453 ;; uint32_t p_type;
454 ;; Elf32_Off p_offset;
455 ;; Elf32_Addr p_vaddr;
456 ;; Elf32_Addr p_paddr;
457 ;; uint32_t p_filesz;
458 ;; uint32_t p_memsz;
459 ;; uint32_t p_flags;
460 ;; uint32_t p_align;
461 ;; } Elf32_Phdr;
462
463 (define (parse-elf32-program-header index bv offset byte-order)
464 (if (<= (+ offset 32) (bytevector-length bv))
465 (make-elf-segment index
466 (bytevector-u32-ref bv offset byte-order)
467 (bytevector-u32-ref bv (+ offset 4) byte-order)
468 (bytevector-u32-ref bv (+ offset 8) byte-order)
469 (bytevector-u32-ref bv (+ offset 12) byte-order)
470 (bytevector-u32-ref bv (+ offset 16) byte-order)
471 (bytevector-u32-ref bv (+ offset 20) byte-order)
472 (bytevector-u32-ref bv (+ offset 24) byte-order)
473 (bytevector-u32-ref bv (+ offset 28) byte-order))
474 (error "corrupt ELF (offset out of range)" offset)))
475
476 (define (write-elf32-program-header bv offset byte-order seg)
477 (bytevector-u32-set! bv offset (elf-segment-type seg) byte-order)
478 (bytevector-u32-set! bv (+ offset 4) (elf-segment-offset seg) byte-order)
479 (bytevector-u32-set! bv (+ offset 8) (elf-segment-vaddr seg) byte-order)
480 (bytevector-u32-set! bv (+ offset 12) (elf-segment-paddr seg) byte-order)
481 (bytevector-u32-set! bv (+ offset 16) (elf-segment-filesz seg) byte-order)
482 (bytevector-u32-set! bv (+ offset 20) (elf-segment-memsz seg) byte-order)
483 (bytevector-u32-set! bv (+ offset 24) (elf-segment-flags seg) byte-order)
484 (bytevector-u32-set! bv (+ offset 28) (elf-segment-align seg) byte-order))
485
486
487 ;; typedef struct {
488 ;; uint32_t p_type;
489 ;; uint32_t p_flags;
490 ;; Elf64_Off p_offset;
491 ;; Elf64_Addr p_vaddr;
492 ;; Elf64_Addr p_paddr;
493 ;; uint64_t p_filesz;
494 ;; uint64_t p_memsz;
495 ;; uint64_t p_align;
496 ;; } Elf64_Phdr;
497
498 ;; NB: position of `flags' is different!
499
500 (define (parse-elf64-program-header index bv offset byte-order)
501 (if (<= (+ offset 56) (bytevector-length bv))
502 (make-elf-segment index
503 (bytevector-u32-ref bv offset byte-order)
504 (bytevector-u64-ref bv (+ offset 8) byte-order)
505 (bytevector-u64-ref bv (+ offset 16) byte-order)
506 (bytevector-u64-ref bv (+ offset 24) byte-order)
507 (bytevector-u64-ref bv (+ offset 32) byte-order)
508 (bytevector-u64-ref bv (+ offset 40) byte-order)
509 (bytevector-u32-ref bv (+ offset 4) byte-order)
510 (bytevector-u64-ref bv (+ offset 48) byte-order))
511 (error "corrupt ELF (offset out of range)" offset)))
512
513 (define (write-elf64-program-header bv offset byte-order seg)
514 (bytevector-u32-set! bv offset (elf-segment-type seg) byte-order)
515 (bytevector-u64-set! bv (+ offset 8) (elf-segment-offset seg) byte-order)
516 (bytevector-u64-set! bv (+ offset 16) (elf-segment-vaddr seg) byte-order)
517 (bytevector-u64-set! bv (+ offset 24) (elf-segment-paddr seg) byte-order)
518 (bytevector-u64-set! bv (+ offset 32) (elf-segment-filesz seg) byte-order)
519 (bytevector-u64-set! bv (+ offset 40) (elf-segment-memsz seg) byte-order)
520 (bytevector-u32-set! bv (+ offset 4) (elf-segment-flags seg) byte-order)
521 (bytevector-u64-set! bv (+ offset 48) (elf-segment-align seg) byte-order))
522
523 (define (write-elf-program-header bv offset byte-order word-size seg)
524 ((case word-size
525 ((4) write-elf32-program-header)
526 ((8) write-elf64-program-header)
527 (else (error "invalid word size" word-size)))
528 bv offset byte-order seg))
529
530 (define (elf-program-header-len word-size)
531 (case word-size
532 ((4) 32)
533 ((8) 56)
534 (else (error "bad word size" word-size))))
535
536 (define (elf-segment elf n)
537 (if (not (< -1 n (elf-phnum elf)))
538 (error "bad segment number" n))
539 ((case (elf-word-size elf)
540 ((4) parse-elf32-program-header)
541 ((8) parse-elf64-program-header)
542 (else (error "unhandled pointer size")))
543 (elf-bytes elf)
544 (+ (elf-phoff elf) (* n (elf-phentsize elf)))
545 (elf-byte-order elf)))
546
547 (define (elf-segments elf)
548 (let lp ((n (elf-phnum elf)) (out '()))
549 (if (zero? n)
550 out
551 (lp (1- n) (cons (elf-segment elf (1- n)) out)))))
552
553 (define-record-type <elf-section>
554 (make-elf-section index name type flags
555 addr offset size link info addralign entsize)
556 elf-section?
557 (index elf-section-index)
558 (name elf-section-name)
559 (type elf-section-type)
560 (flags elf-section-flags)
561 (addr elf-section-addr)
562 (offset elf-section-offset)
563 (size elf-section-size)
564 (link elf-section-link)
565 (info elf-section-info)
566 (addralign elf-section-addralign)
567 (entsize elf-section-entsize))
568
569 (define* (make-elf-section* #:key (index SHN_UNDEF) (name 0) (type SHT_PROGBITS)
570 (flags SHF_ALLOC) (addr 0) (offset 0) (size 0)
571 (link 0) (info 0) (addralign 8) (entsize 0))
572 (make-elf-section index name type flags addr offset size link info addralign
573 entsize))
574
575 ;; typedef struct {
576 ;; uint32_t sh_name;
577 ;; uint32_t sh_type;
578 ;; uint32_t sh_flags;
579 ;; Elf32_Addr sh_addr;
580 ;; Elf32_Off sh_offset;
581 ;; uint32_t sh_size;
582 ;; uint32_t sh_link;
583 ;; uint32_t sh_info;
584 ;; uint32_t sh_addralign;
585 ;; uint32_t sh_entsize;
586 ;; } Elf32_Shdr;
587
588 (define (parse-elf32-section-header index bv offset byte-order)
589 (if (<= (+ offset 40) (bytevector-length bv))
590 (make-elf-section index
591 (bytevector-u32-ref bv offset byte-order)
592 (bytevector-u32-ref bv (+ offset 4) byte-order)
593 (bytevector-u32-ref bv (+ offset 8) byte-order)
594 (bytevector-u32-ref bv (+ offset 12) byte-order)
595 (bytevector-u32-ref bv (+ offset 16) byte-order)
596 (bytevector-u32-ref bv (+ offset 20) byte-order)
597 (bytevector-u32-ref bv (+ offset 24) byte-order)
598 (bytevector-u32-ref bv (+ offset 28) byte-order)
599 (bytevector-u32-ref bv (+ offset 32) byte-order)
600 (bytevector-u32-ref bv (+ offset 36) byte-order))
601 (error "corrupt ELF (offset out of range)" offset)))
602
603 (define (write-elf32-section-header bv offset byte-order sec)
604 (bytevector-u32-set! bv offset (elf-section-name sec) byte-order)
605 (bytevector-u32-set! bv (+ offset 4) (elf-section-type sec) byte-order)
606 (bytevector-u32-set! bv (+ offset 8) (elf-section-flags sec) byte-order)
607 (bytevector-u32-set! bv (+ offset 12) (elf-section-addr sec) byte-order)
608 (bytevector-u32-set! bv (+ offset 16) (elf-section-offset sec) byte-order)
609 (bytevector-u32-set! bv (+ offset 20) (elf-section-size sec) byte-order)
610 (bytevector-u32-set! bv (+ offset 24) (elf-section-link sec) byte-order)
611 (bytevector-u32-set! bv (+ offset 28) (elf-section-info sec) byte-order)
612 (bytevector-u32-set! bv (+ offset 32) (elf-section-addralign sec) byte-order)
613 (bytevector-u32-set! bv (+ offset 36) (elf-section-entsize sec) byte-order))
614
615
616 ;; typedef struct {
617 ;; uint32_t sh_name;
618 ;; uint32_t sh_type;
619 ;; uint64_t sh_flags;
620 ;; Elf64_Addr sh_addr;
621 ;; Elf64_Off sh_offset;
622 ;; uint64_t sh_size;
623 ;; uint32_t sh_link;
624 ;; uint32_t sh_info;
625 ;; uint64_t sh_addralign;
626 ;; uint64_t sh_entsize;
627 ;; } Elf64_Shdr;
628
629 (define (elf-section-header-len word-size)
630 (case word-size
631 ((4) 40)
632 ((8) 64)
633 (else (error "bad word size" word-size))))
634
635 (define (elf-section-header-addr-offset word-size)
636 (case word-size
637 ((4) 12)
638 ((8) 16)
639 (else (error "bad word size" word-size))))
640
641 (define (elf-section-header-offset-offset word-size)
642 (case word-size
643 ((4) 16)
644 ((8) 24)
645 (else (error "bad word size" word-size))))
646
647 (define (parse-elf64-section-header index bv offset byte-order)
648 (if (<= (+ offset 64) (bytevector-length bv))
649 (make-elf-section index
650 (bytevector-u32-ref bv offset byte-order)
651 (bytevector-u32-ref bv (+ offset 4) byte-order)
652 (bytevector-u64-ref bv (+ offset 8) byte-order)
653 (bytevector-u64-ref bv (+ offset 16) byte-order)
654 (bytevector-u64-ref bv (+ offset 24) byte-order)
655 (bytevector-u64-ref bv (+ offset 32) byte-order)
656 (bytevector-u32-ref bv (+ offset 40) byte-order)
657 (bytevector-u32-ref bv (+ offset 44) byte-order)
658 (bytevector-u64-ref bv (+ offset 48) byte-order)
659 (bytevector-u64-ref bv (+ offset 56) byte-order))
660 (error "corrupt ELF (offset out of range)" offset)))
661
662 (define (write-elf64-section-header bv offset byte-order sec)
663 (bytevector-u32-set! bv offset (elf-section-name sec) byte-order)
664 (bytevector-u32-set! bv (+ offset 4) (elf-section-type sec) byte-order)
665 (bytevector-u64-set! bv (+ offset 8) (elf-section-flags sec) byte-order)
666 (bytevector-u64-set! bv (+ offset 16) (elf-section-addr sec) byte-order)
667 (bytevector-u64-set! bv (+ offset 24) (elf-section-offset sec) byte-order)
668 (bytevector-u64-set! bv (+ offset 32) (elf-section-size sec) byte-order)
669 (bytevector-u32-set! bv (+ offset 40) (elf-section-link sec) byte-order)
670 (bytevector-u32-set! bv (+ offset 44) (elf-section-info sec) byte-order)
671 (bytevector-u64-set! bv (+ offset 48) (elf-section-addralign sec) byte-order)
672 (bytevector-u64-set! bv (+ offset 56) (elf-section-entsize sec) byte-order))
673
674 (define (elf-section elf n)
675 (if (not (< -1 n (elf-shnum elf)))
676 (error "bad section number" n))
677 ((case (elf-word-size elf)
678 ((4) parse-elf32-section-header)
679 ((8) parse-elf64-section-header)
680 (else (error "unhandled pointer size")))
681 n
682 (elf-bytes elf)
683 (+ (elf-shoff elf) (* n (elf-shentsize elf)))
684 (elf-byte-order elf)))
685
686 (define (write-elf-section-header bv offset byte-order word-size sec)
687 ((case word-size
688 ((4) write-elf32-section-header)
689 ((8) write-elf64-section-header)
690 (else (error "invalid word size" word-size)))
691 bv offset byte-order sec))
692
693 (define (elf-sections elf)
694 (let lp ((n (elf-shnum elf)) (out '()))
695 (if (zero? n)
696 out
697 (lp (1- n) (cons (elf-section elf (1- n)) out)))))
698
699 ;;
700 ;; Section Types
701 ;;
702 (define SHT_NULL 0) ; Section header table entry unused
703 (define SHT_PROGBITS 1) ; Program data
704 (define SHT_SYMTAB 2) ; Symbol table
705 (define SHT_STRTAB 3) ; String table
706 (define SHT_RELA 4) ; Relocation entries with addends
707 (define SHT_HASH 5) ; Symbol hash table
708 (define SHT_DYNAMIC 6) ; Dynamic linking information
709 (define SHT_NOTE 7) ; Notes
710 (define SHT_NOBITS 8) ; Program space with no data (bss)
711 (define SHT_REL 9) ; Relocation entries, no addends
712 (define SHT_SHLIB 10) ; Reserved
713 (define SHT_DYNSYM 11) ; Dynamic linker symbol table
714 (define SHT_INIT_ARRAY 14) ; Array of constructors
715 (define SHT_FINI_ARRAY 15) ; Array of destructors
716 (define SHT_PREINIT_ARRAY 16) ; Array of pre-constructors
717 (define SHT_GROUP 17) ; Section group
718 (define SHT_SYMTAB_SHNDX 18) ; Extended section indeces
719 (define SHT_NUM 19) ; Number of defined types.
720 (define SHT_LOOS #x60000000) ; Start OS-specific.
721 (define SHT_HIOS #x6fffffff) ; End OS-specific type
722 (define SHT_LOPROC #x70000000) ; Start of processor-specific
723 (define SHT_HIPROC #x7fffffff) ; End of processor-specific
724 (define SHT_LOUSER #x80000000) ; Start of application-specific
725 (define SHT_HIUSER #x8fffffff) ; End of application-specific
726
727 ;;
728 ;; Section Flags
729 ;;
730 (define SHF_WRITE (ash 1 0)) ; Writable
731 (define SHF_ALLOC (ash 1 1)) ; Occupies memory during execution
732 (define SHF_EXECINSTR (ash 1 2)) ; Executable
733 (define SHF_MERGE (ash 1 4)) ; Might be merged
734 (define SHF_STRINGS (ash 1 5)) ; Contains nul-terminated strings
735 (define SHF_INFO_LINK (ash 1 6)) ; `sh_info' contains SHT index
736 (define SHF_LINK_ORDER (ash 1 7)) ; Preserve order after combining
737 (define SHF_OS_NONCONFORMING (ash 1 8)) ; Non-standard OS specific handling required
738 (define SHF_GROUP (ash 1 9)) ; Section is member of a group.
739 (define SHF_TLS (ash 1 10)) ; Section hold thread-local data.
740
741 ;;
742 ;; Dynamic entry types. The DT_GUILE types are non-standard.
743 ;;
744 (define DT_NULL 0) ; Marks end of dynamic section
745 (define DT_NEEDED 1) ; Name of needed library
746 (define DT_PLTRELSZ 2) ; Size in bytes of PLT relocs
747 (define DT_PLTGOT 3) ; Processor defined value
748 (define DT_HASH 4) ; Address of symbol hash table
749 (define DT_STRTAB 5) ; Address of string table
750 (define DT_SYMTAB 6) ; Address of symbol table
751 (define DT_RELA 7) ; Address of Rela relocs
752 (define DT_RELASZ 8) ; Total size of Rela relocs
753 (define DT_RELAENT 9) ; Size of one Rela reloc
754 (define DT_STRSZ 10) ; Size of string table
755 (define DT_SYMENT 11) ; Size of one symbol table entry
756 (define DT_INIT 12) ; Address of init function
757 (define DT_FINI 13) ; Address of termination function
758 (define DT_SONAME 14) ; Name of shared object
759 (define DT_RPATH 15) ; Library search path (deprecated)
760 (define DT_SYMBOLIC 16) ; Start symbol search here
761 (define DT_REL 17) ; Address of Rel relocs
762 (define DT_RELSZ 18) ; Total size of Rel relocs
763 (define DT_RELENT 19) ; Size of one Rel reloc
764 (define DT_PLTREL 20) ; Type of reloc in PLT
765 (define DT_DEBUG 21) ; For debugging ; unspecified
766 (define DT_TEXTREL 22) ; Reloc might modify .text
767 (define DT_JMPREL 23) ; Address of PLT relocs
768 (define DT_BIND_NOW 24) ; Process relocations of object
769 (define DT_INIT_ARRAY 25) ; Array with addresses of init fct
770 (define DT_FINI_ARRAY 26) ; Array with addresses of fini fct
771 (define DT_INIT_ARRAYSZ 27) ; Size in bytes of DT_INIT_ARRAY
772 (define DT_FINI_ARRAYSZ 28) ; Size in bytes of DT_FINI_ARRAY
773 (define DT_RUNPATH 29) ; Library search path
774 (define DT_FLAGS 30) ; Flags for the object being loaded
775 (define DT_ENCODING 32) ; Start of encoded range
776 (define DT_PREINIT_ARRAY 32) ; Array with addresses of preinit fc
777 (define DT_PREINIT_ARRAYSZ 33) ; size in bytes of DT_PREINIT_ARRAY
778 (define DT_NUM 34) ; Number used
779 (define DT_LOGUILE #x37146000) ; Start of Guile-specific
780 (define DT_GUILE_GC_ROOT #x37146000) ; Offset of GC roots
781 (define DT_GUILE_GC_ROOT_SZ #x37146001) ; Size in machine words of GC roots
782 (define DT_GUILE_ENTRY #x37146002) ; Address of entry thunk
783 (define DT_GUILE_VM_VERSION #x37146003) ; Bytecode version
784 (define DT_HIGUILE #x37146fff) ; End of Guile-specific
785 (define DT_LOOS #x6000000d) ; Start of OS-specific
786 (define DT_HIOS #x6ffff000) ; End of OS-specific
787 (define DT_LOPROC #x70000000) ; Start of processor-specific
788 (define DT_HIPROC #x7fffffff) ; End of processor-specific
789
790
791 (define (string-table-ref bv offset)
792 (let lp ((end offset))
793 (if (zero? (bytevector-u8-ref bv end))
794 (let ((out (make-bytevector (- end offset))))
795 (bytevector-copy! bv offset out 0 (- end offset))
796 (utf8->string out))
797 (lp (1+ end)))))
798
799 (define (elf-section-by-name elf name)
800 (let ((off (elf-section-offset (elf-section elf (elf-shstrndx elf)))))
801 (let lp ((n (elf-shnum elf)))
802 (and (> n 0)
803 (let ((section (elf-section elf (1- n))))
804 (if (equal? (string-table-ref (elf-bytes elf)
805 (+ off (elf-section-name section)))
806 name)
807 section
808 (lp (1- n))))))))
809
810 (define (elf-sections-by-name elf)
811 (let* ((sections (elf-sections elf))
812 (off (elf-section-offset (list-ref sections (elf-shstrndx elf)))))
813 (map (lambda (section)
814 (cons (string-table-ref (elf-bytes elf)
815 (+ off (elf-section-name section)))
816 section))
817 sections)))
818
819 (define-record-type <elf-symbol>
820 (make-elf-symbol name value size info other shndx)
821 elf-symbol?
822 (name elf-symbol-name)
823 (value elf-symbol-value)
824 (size elf-symbol-size)
825 (info elf-symbol-info)
826 (other elf-symbol-other)
827 (shndx elf-symbol-shndx))
828
829 (define* (make-elf-symbol* #:key (name 0) (value 0) (size 0)
830 (binding STB_LOCAL) (type STT_NOTYPE)
831 (info (logior (ash binding 4) type))
832 (visibility STV_DEFAULT) (other visibility)
833 (shndx SHN_UNDEF))
834 (make-elf-symbol name value size info other shndx))
835
836 ;; typedef struct {
837 ;; uint32_t st_name;
838 ;; Elf32_Addr st_value;
839 ;; uint32_t st_size;
840 ;; unsigned char st_info;
841 ;; unsigned char st_other;
842 ;; uint16_t st_shndx;
843 ;; } Elf32_Sym;
844
845 (define (elf-symbol-len word-size)
846 (case word-size
847 ((4) 16)
848 ((8) 24)
849 (else (error "bad word size" word-size))))
850
851 (define (parse-elf32-symbol bv offset stroff byte-order)
852 (if (<= (+ offset 16) (bytevector-length bv))
853 (make-elf-symbol (let ((name (bytevector-u32-ref bv offset byte-order)))
854 (if stroff
855 (string-table-ref bv (+ stroff name))
856 name))
857 (bytevector-u32-ref bv (+ offset 4) byte-order)
858 (bytevector-u32-ref bv (+ offset 8) byte-order)
859 (bytevector-u8-ref bv (+ offset 12))
860 (bytevector-u8-ref bv (+ offset 13))
861 (bytevector-u16-ref bv (+ offset 14) byte-order))
862 (error "corrupt ELF (offset out of range)" offset)))
863
864 (define (write-elf32-symbol bv offset byte-order sym)
865 (bytevector-u32-set! bv offset (elf-symbol-name sym) byte-order)
866 (bytevector-u32-set! bv (+ offset 4) (elf-symbol-value sym) byte-order)
867 (bytevector-u32-set! bv (+ offset 8) (elf-symbol-size sym) byte-order)
868 (bytevector-u8-set! bv (+ offset 12) (elf-symbol-info sym))
869 (bytevector-u8-set! bv (+ offset 13) (elf-symbol-other sym))
870 (bytevector-u16-set! bv (+ offset 14) (elf-symbol-shndx sym) byte-order))
871
872 ;; typedef struct {
873 ;; uint32_t st_name;
874 ;; unsigned char st_info;
875 ;; unsigned char st_other;
876 ;; uint16_t st_shndx;
877 ;; Elf64_Addr st_value;
878 ;; uint64_t st_size;
879 ;; } Elf64_Sym;
880
881 (define (parse-elf64-symbol bv offset stroff byte-order)
882 (if (<= (+ offset 24) (bytevector-length bv))
883 (make-elf-symbol (let ((name (bytevector-u32-ref bv offset byte-order)))
884 (if stroff
885 (string-table-ref bv (+ stroff name))
886 name))
887 (bytevector-u64-ref bv (+ offset 8) byte-order)
888 (bytevector-u64-ref bv (+ offset 16) byte-order)
889 (bytevector-u8-ref bv (+ offset 4))
890 (bytevector-u8-ref bv (+ offset 5))
891 (bytevector-u16-ref bv (+ offset 6) byte-order))
892 (error "corrupt ELF (offset out of range)" offset)))
893
894 (define (write-elf64-symbol bv offset byte-order sym)
895 (bytevector-u32-set! bv offset (elf-symbol-name sym) byte-order)
896 (bytevector-u8-set! bv (+ offset 4) (elf-symbol-info sym))
897 (bytevector-u8-set! bv (+ offset 5) (elf-symbol-other sym))
898 (bytevector-u16-set! bv (+ offset 6) (elf-symbol-shndx sym) byte-order)
899 (bytevector-u64-set! bv (+ offset 8) (elf-symbol-value sym) byte-order)
900 (bytevector-u64-set! bv (+ offset 16) (elf-symbol-size sym) byte-order))
901
902 (define (write-elf-symbol bv offset byte-order word-size sym)
903 ((case word-size
904 ((4) write-elf32-symbol)
905 ((8) write-elf64-symbol)
906 (else (error "invalid word size" word-size)))
907 bv offset byte-order sym))
908
909 (define (elf-symbol-table-len section)
910 (let ((len (elf-section-size section))
911 (entsize (elf-section-entsize section)))
912 (unless (and (not (zero? entsize)) (zero? (modulo len entsize)))
913 (error "bad symbol table" section))
914 (/ len entsize)))
915
916 (define* (elf-symbol-table-ref elf section n #:optional strtab)
917 (let ((bv (elf-bytes elf))
918 (byte-order (elf-byte-order elf))
919 (stroff (and strtab (elf-section-offset strtab)))
920 (base (elf-section-offset section))
921 (len (elf-section-size section))
922 (entsize (elf-section-entsize section)))
923 (unless (<= (* (1+ n) entsize) len)
924 (error "out of range symbol table access" section n))
925 (case (elf-word-size elf)
926 ((4)
927 (unless (<= 16 entsize)
928 (error "bad entsize for symbol table" section))
929 (parse-elf32-symbol bv (+ base (* n entsize)) stroff byte-order))
930 ((8)
931 (unless (<= 24 entsize)
932 (error "bad entsize for symbol table" section))
933 (parse-elf64-symbol bv (+ base (* n entsize)) stroff byte-order))
934 (else (error "bad word size" elf)))))
935
936 ;; Legal values for ST_BIND subfield of st_info (symbol binding).
937
938 (define STB_LOCAL 0) ; Local symbol
939 (define STB_GLOBAL 1) ; Global symbol
940 (define STB_WEAK 2) ; Weak symbol
941 (define STB_NUM 3) ; Number of defined types.
942 (define STB_LOOS 10) ; Start of OS-specific
943 (define STB_GNU_UNIQUE 10) ; Unique symbol.
944 (define STB_HIOS 12) ; End of OS-specific
945 (define STB_LOPROC 13) ; Start of processor-specific
946 (define STB_HIPROC 15) ; End of processor-specific
947
948 ;; Legal values for ST_TYPE subfield of st_info (symbol type).
949
950 (define STT_NOTYPE 0) ; Symbol type is unspecified
951 (define STT_OBJECT 1) ; Symbol is a data object
952 (define STT_FUNC 2) ; Symbol is a code object
953 (define STT_SECTION 3) ; Symbol associated with a section
954 (define STT_FILE 4) ; Symbol's name is file name
955 (define STT_COMMON 5) ; Symbol is a common data object
956 (define STT_TLS 6) ; Symbol is thread-local data objec
957 (define STT_NUM 7) ; Number of defined types.
958 (define STT_LOOS 10) ; Start of OS-specific
959 (define STT_GNU_IFUNC 10) ; Symbol is indirect code object
960 (define STT_HIOS 12) ; End of OS-specific
961 (define STT_LOPROC 13) ; Start of processor-specific
962 (define STT_HIPROC 15) ; End of processor-specific
963
964 ;; Symbol visibility specification encoded in the st_other field.
965
966 (define STV_DEFAULT 0) ; Default symbol visibility rules
967 (define STV_INTERNAL 1) ; Processor specific hidden class
968 (define STV_HIDDEN 2) ; Sym unavailable in other modules
969 (define STV_PROTECTED 3) ; Not preemptible, not exported
970
971 (define (elf-symbol-binding sym)
972 (ash (elf-symbol-info sym) -4))
973
974 (define (elf-symbol-type sym)
975 (logand (elf-symbol-info sym) #xf))
976
977 (define (elf-symbol-visibility sym)
978 (logand (elf-symbol-other sym) #x3))
979
980 (define NT_GNU_ABI_TAG 1)
981 (define NT_GNU_HWCAP 2)
982 (define NT_GNU_BUILD_ID 3)
983 (define NT_GNU_GOLD_VERSION 4)
984
985 (define-record-type <elf-note>
986 (make-elf-note name desc type)
987 elf-note?
988 (name elf-note-name)
989 (desc elf-note-desc)
990 (type elf-note-type))
991
992 (define (parse-elf-note elf section)
993 (let ((bv (elf-bytes elf))
994 (byte-order (elf-byte-order elf))
995 (offset (elf-section-offset section)))
996 (unless (<= (+ offset 12) (bytevector-length bv))
997 (error "corrupt ELF (offset out of range)" offset))
998 (let ((namesz (bytevector-u32-ref bv offset byte-order))
999 (descsz (bytevector-u32-ref bv (+ offset 4) byte-order))
1000 (type (bytevector-u32-ref bv (+ offset 8) byte-order)))
1001 (unless (<= (+ offset 12 namesz descsz) (bytevector-length bv))
1002 (error "corrupt ELF (offset out of range)" offset))
1003 (let ((name (make-bytevector (1- namesz)))
1004 (desc (make-bytevector descsz)))
1005 (bytevector-copy! bv (+ offset 12) name 0 (1- namesz))
1006 (bytevector-copy! bv (+ offset 12 namesz) desc 0 descsz)
1007 (make-elf-note (utf8->string name) desc type)))))