elf.scm 42 KB

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