forked from 96boards/l-loader
-
Notifications
You must be signed in to change notification settings - Fork 15
/
Copy pathgen_loader.py
executable file
·268 lines (238 loc) · 9.69 KB
/
gen_loader.py
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
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
#!/usr/bin/env python
import os
import os.path
import sys, getopt
import binascii
import struct
import string
class generator(object):
#
# struct l_loader_head {
# unsigned int first_instr;
# unsigned char magic[16]; @ BOOTMAGICNUMBER!
# unsigned int l_loader_start;
# unsigned int l_loader_end;
# };
file_header = [0, 0, 0, 0, 0, 0, 0]
#
# struct entry_head {
# unsigned char magic[8]; @ ENTY
# unsigned char name[8]; @ loader/bl1
# unsigned int start_lba;
# unsigned int count_lba;
# unsigned int flag; @ boot partition or not
# };
s1_entry_name = ['loader', 'bl1']
s2_entry_name = ['primary', 'second']
block_size = 512
stage = 0
# set in self.add()
idx = 0
# set in self.parse()
ptable_lba = 0
stable_lba = 0
# file pointer
p_entry = 0
p_file = 0
def __init__(self, out_img):
try:
self.fp = open(out_img, "wb+")
except IOError, e:
print "*** file open error:", e
sys.exit(3)
else:
self.entry_hd = [[0 for col in range(7)] for row in range(5)]
def __del__(self):
self.fp.close()
# parse partition from the primary ptable
def parse(self, fname):
try:
fptable = open(fname, "rb")
except IOError, e:
print "*** file open error:", e
sys.exit(3)
else:
# skip the first block in primary partition table
# that is MBR protection information
fptable.read(self.block_size)
# check whether it's a primary paritition table
data = struct.unpack("8s", fptable.read(8))
efi_magic = 'EFI PART'
if cmp("EFI PART", data[0]):
print "It's not partition table image."
fptable.close()
sys.exit(4)
# skip 16 bytes
fptable.read(16)
# get lba of both primary partition table and secondary partition table
data = struct.unpack("QQQQ", fptable.read(32))
self.ptable_lba = data[0] - 1
self.stable_lba = data[3] + 1
# skip 24 bytes
fptable.read(24)
data = struct.unpack("i", fptable.read(4))
pentries = data[0]
# skip the reset in this block
fptable.read(self.block_size - 84)
for i in range(1, pentries):
# name is encoded as UTF-16
d0,lba,d2,name = struct.unpack("32sQ16s72s", fptable.read(128))
plainname = unicode(name, "utf-16")
if (not cmp(plainname[0:7], 'l-loader'[0:7])):
print 'bl1_lba: ', lba
self.bl1_lba = lba
sys.exit(1)
fptable.close()
def add(self, lba, fname):
try:
fsize = os.path.getsize(fname)
except IOError, e:
print "*** file open error:", e
sys.exit(4)
else:
blocks = (fsize + self.block_size - 1) / self.block_size
if (self.stage == 1):
# Boot Area1 in eMMC
bootp = 1
if self.idx == 0:
self.p_entry = 28
elif (self.stage == 2):
# User Data Area in eMMC
bootp = 0
# create an empty block only for stage2
# This empty block is used to store entry head
print 'p_file: ', self.p_file, 'p_entry: ', self.p_entry
if self.idx == 0:
self.fp.seek(self.p_file)
for i in range (0, self.block_size):
zero = struct.pack('x')
self.fp.write(zero)
self.p_file += self.block_size
self.p_entry = 0
else:
print "wrong stage ", stage, "is specified"
sys.exit(4)
# Maybe the file size isn't aligned. So pad it.
if (self.idx == 0) and (self.stage == 1):
if fsize > 2048:
print 'loader size exceeds 2KB. file size: ', fsize
sys.exit(4)
else:
left_bytes = 2048 - fsize
else:
left_bytes = fsize % self.block_size
if left_bytes:
left_bytes = self.block_size - left_bytes
print 'lba: ', lba, 'blocks: ', blocks, 'bootp: ', bootp, 'fname: ', fname
# write images
fimg = open(fname, "rb")
for i in range (0, blocks):
buf = fimg.read(self.block_size)
self.fp.seek(self.p_file)
self.fp.write(buf)
# p_file is the file pointer of the new binary file
# At last, it means the total block size of the new binary file
self.p_file += self.block_size
if (self.idx == 0) and (self.stage == 1):
self.p_file = 2048
print 'p_file: ', self.p_file, 'last block is ', fsize % self.block_size, 'bytes', ' tell: ', self.fp.tell(), 'left_bytes: ', left_bytes
if left_bytes:
for i in range (0, left_bytes):
zero = struct.pack('x')
self.fp.write(zero)
print 'p_file: ', self.p_file, ' pad to: ', self.fp.tell()
# write entry information at the header
if self.stage == 1:
byte = struct.pack('8s8siii', 'ENTRYHDR', self.s1_entry_name[self.idx], lba, blocks, bootp)
elif self.stage == 2:
byte = struct.pack('8s8siii', 'ENTRYHDR', self.s2_entry_name[self.idx], lba, blocks, bootp)
self.fp.seek(self.p_entry)
self.fp.write(byte)
self.p_entry += 28
self.idx += 1
fimg.close()
def hex2(self, data):
return data > 0 and hex(data) or hex(data & 0xffffffff)
def end(self):
if self.stage == 1:
self.fp.seek(20)
start,end = struct.unpack("ii", self.fp.read(8))
print "start: ", self.hex2(start), 'end: ', self.hex2(end)
end = start + self.p_file
print "start: ", self.hex2(start), 'end: ', self.hex2(end)
self.fp.seek(24)
byte = struct.pack('i', end)
self.fp.write(byte)
self.fp.close()
def create_stage1(self, img_loader, img_bl1, output_img):
print '+-----------------------------------------------------------+'
print ' Input Images:'
print ' loader: ', img_loader
print ' bl1: ', img_bl1
print ' Ouput Image: ', output_img
print '+-----------------------------------------------------------+\n'
self.stage = 1
# The first 2KB is reserved
# The next 2KB is for loader image
self.add(4, img_loader) # img_loader doesn't exist in partition table
print 'self.idx: ', self.idx
# bl1.bin starts from 4KB
self.add(8, img_bl1) # img_bl1 doesn't exist in partition table
def create_stage2(self, img_prm_ptable, img_sec_ptable, output_img):
print '+-----------------------------------------------------------+'
print ' Input Images:'
print ' primary partition table: ', img_prm_ptable
print ' secondary partition table: ', img_sec_ptable
print ' Ouput Image: ', output_img
print '+-----------------------------------------------------------+\n'
self.stage = 2
self.parse(img_prm_ptable)
self.add(self.ptable_lba, img_prm_ptable)
if (cmp(img_sec_ptable, 'secondary partition table')):
# Doesn't match. It means that secondary ptable is specified.
self.add(self.stable_lba, img_sec_ptable)
else:
print 'Don\'t need secondary partition table'
def main(argv):
stage1 = 0
stage2 = 0
img_prm_ptable = "primary partition table"
img_sec_ptable = "secondary partition table"
try:
opts, args = getopt.getopt(argv,"ho:",["img_loader=","img_bl1=","img_prm_ptable=","img_sec_ptable="])
except getopt.GetoptError:
print 'gen_loader.py -o <l-loader.bin> --img_loader <l-loader> --img_bl1 <bl1.bin> --img_prm_ptable <prm_ptable.img> --img_sec_ptable <sec_ptable.img>'
sys.exit(2)
for opt, arg in opts:
if opt == '-h':
print 'gen_loader.py -o <l-loader.bin> --img_loader <l-loader> --img_bl1 <bl1.bin> --img_prm_ptable <prm_ptable.img> --img_sec_ptable <sec_ptable.img>'
sys.exit(1)
elif opt == '-o':
output_img = arg
elif opt in ("--img_loader"):
img_loader = arg
stage1 = 1
elif opt in ("--img_bl1"):
img_bl1 = arg
stage1 = 1
elif opt in ("--img_prm_ptable"):
img_prm_ptable = arg
stage2 = 1
elif opt in ("--img_sec_ptable"):
img_sec_ptable = arg
loader = generator(output_img)
loader.idx = 0
if (stage1 == 1) and (stage2 == 1):
print 'There are only loader & BL1 in stage1.'
print 'And there are primary partition table, secondary partition table and FIP in stage2.'
sys.exit(1)
elif (stage1 == 0) and (stage2 == 0):
print 'No input images are specified.'
sys.exit(1)
elif stage1 == 1:
loader.create_stage1(img_loader, img_bl1, output_img)
elif stage2 == 1:
loader.create_stage2(img_prm_ptable, img_sec_ptable, output_img)
loader.end()
if __name__ == "__main__":
main(sys.argv[1:])