forked from Ferada/binary-types
-
Notifications
You must be signed in to change notification settings - Fork 2
/
example.lisp
146 lines (133 loc) · 4.62 KB
/
example.lisp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
;;;;------------------------------------------------------------------
;;;;
;;;; Copyright (C) 200120001999,
;;;; Department of Computer Science, University of Tromsø, Norway
;;;;
;;;; Filename: example.lisp
;;;; Description:
;;;; Author: Frode Vatvedt Fjeld <[email protected]>
;;;; Created at: Wed Dec 8 15:15:06 1999
;;;; Distribution: See the accompanying file COPYING.
;;;;
;;;; $Id: example.lisp,v 1.1.1.1 2004/01/13 11:13:13 ffjeld Exp $
;;;;
;;;;------------------------------------------------------------------
(defpackage "EXAMPLE"
(:use "COMMON-LISP" "BINARY-TYPES")
(:export run))
(in-package "EXAMPLE")
;;; ELF basic types
(define-unsigned word 4)
(define-signed sword 4)
(define-unsigned addr 4)
(define-unsigned off 4)
(define-unsigned half 2)
;;; Mapping from ELF symbols to BT:*ENDIAN* values
(defun elf-data-to-endian (elf-data)
(ecase elf-data
((elf-data-2lsb) :little-endian)
((elf-data-2msb) :big-endian)))
(defconstant +ELF-MAGIC+ '(#x7f #\E #\L #\F))
;;; ELF file header structure
(define-binary-class elf-header ()
((e-ident
:binary-type (define-binary-struct e-ident ()
(ei-magic nil :binary-type
(define-binary-struct ei-magic ()
(ei-mag0 0 :binary-type u8)
(ei-mag1 #\null :binary-type char8)
(ei-mag2 #\null :binary-type char8)
(ei-mag3 #\null :binary-type char8)))
(ei-class nil :binary-type
(define-enum ei-class (u8)
elf-class-none 0
elf-class-32 1
elf-class-64 2))
(ei-data nil :binary-type
(define-enum ei-data (u8)
elf-data-none 0
elf-data-2lsb 1
elf-data-2msb 2))
(ei-version 0 :binary-type u8)
(padding nil :binary-type 1)
(ei-name "" :binary-type
(define-null-terminated-string ei-name 8))))
(e-type
:binary-type (define-enum e-type (half)
et-none 0
et-rel 1
et-exec 2
et-dyn 3
et-core 4
et-loproc #xff00
et-hiproc #xffff))
(e-machine
:binary-type (define-enum e-machine (half)
em-none 0
em-m32 1
em-sparc 2
em-386 3
em-68k 4
em-88k 5
em-860 7
em-mips 8))
(e-version :binary-type word)
(e-entry :binary-type addr)
(e-phoff :binary-type off)
(e-shoff :binary-type off)
(e-flags :binary-type word)
(e-ehsize :binary-type half)
(e-phentsize :binary-type half)
(e-phnum :binary-type half)
(e-shentsize :binary-type half)
(e-shnum :binary-type half)
(e-shstrndx :binary-type half)))
(define-condition elf32-reader-error (error)
((stream :initarg :stream :reader elf32-parse-error-stream)
(message :initarg :message :reader elf32-parse-error-message))
(:report (lambda (condition stream)
(princ (elf32-parse-error-message condition)
stream))))
(define-condition elf32-wrong-magic (elf32-reader-error)
((magic :initarg :magic :reader elf32-wrong-magic-magic)))
(define-condition elf32-wrong-class (elf32-reader-error)
((class :initarg :class :reader elf32-wrong-class-class)))
(defun read-elf-file-header (stream)
"Returns an ELF-HEADER and the file's endianess."
(let ((header (read-binary 'elf-header stream :stop 'e-type)))
(with-slots (ei-data ei-class ei-magic)
(slot-value header 'e-ident)
(let* ((binary-types:*endian* (elf-data-to-endian ei-data))
(magic (mapcar #'(lambda (slot-name)
(slot-value ei-magic slot-name))
(binary-record-slot-names 'ei-magic))))
;; Check that file is in fact 32-bit ELF
(unless (equal +ELF-MAGIC+ magic)
(error 'elf32-wrong-magic
:stream stream
:message (format nil "file doesn't match ELF-MAGIC: ~A" magic)
:magic magic))
(unless (eq 'elf-class-32 ei-class)
(error 'elf32-wrong-class
:stream stream
:message (format nil "file is not 32-bit ELF (~A)" ei-class)
:class ei-class))
;; Read the rest of the file-header and merge it with what
;; we've allready got.
(let ((rest (read-binary 'elf-header stream :start 'e-type)))
(dolist (slot-name (binary-record-slot-names 'elf-header))
(unless (slot-boundp header slot-name)
(setf (slot-value header slot-name)
(slot-value rest slot-name))))
(values header binary-types:*endian*))))))
(defun run (path)
(with-binary-file (stream path :direction :input)
(let ((elf-header (read-elf-file-header stream)))
(format t "~&ELF header for \"~A\":~:{~&~12@A: ~S~}~%" path
(mapcar #'(lambda (slot-name)
(list slot-name
(slot-value elf-header slot-name)))
(binary-record-slot-names 'elf-header)))
elf-header)))
#+unix
(run "/bin/ls")