source: mainline/tools/mkfat.py@ ea906c29

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since ea906c29 was 67435b1, checked in by Vojtech Horky <vojtechhorky@…>, 13 years ago

Make tools work with Python 3 again

  • Property mode set to 100755
File size: 15.6 KB
RevLine 
[5749372]1#!/usr/bin/env python
2#
3# Copyright (c) 2008 Martin Decky
4# All rights reserved.
5#
6# Redistribution and use in source and binary forms, with or without
7# modification, are permitted provided that the following conditions
8# are met:
9#
10# - Redistributions of source code must retain the above copyright
11# notice, this list of conditions and the following disclaimer.
12# - Redistributions in binary form must reproduce the above copyright
13# notice, this list of conditions and the following disclaimer in the
14# documentation and/or other materials provided with the distribution.
15# - The name of the author may not be used to endorse or promote products
16# derived from this software without specific prior written permission.
17#
18# THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
19# IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
20# OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
21# IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
22# INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
23# NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
24# DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
25# THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
26# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
27# THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
28#
[3c80f2b]29
[5749372]30"""
31FAT creator
32"""
33
34import sys
35import os
36import random
37import xstruct
[04619ba]38import array
[cc1a727]39from imgutil import *
[700eaa9]40
[04619ba]41def subtree_size(root, cluster_size, dirent_size):
[700eaa9]42 "Recursive directory walk and calculate size"
43
44 size = 0
[95b730c2]45 files = 2
[700eaa9]46
[cc1a727]47 for item in listdir_items(root):
48 if item.is_file:
49 size += align_up(item.size, cluster_size)
[700eaa9]50 files += 1
[cc1a727]51 elif item.is_dir:
52 size += subtree_size(item.path, cluster_size, dirent_size)
[700eaa9]53 files += 1
54
[04619ba]55 return size + align_up(files * dirent_size, cluster_size)
[700eaa9]56
57def root_entries(root):
58 "Return number of root directory entries"
59
60 return len(os.listdir(root))
61
[cc1a727]62def write_file(item, outf, cluster_size, data_start, fat, reserved_clusters):
[04619ba]63 "Store the contents of a file"
64
65 prev = -1
[95b730c2]66 first = 0
[04619ba]67
[cc1a727]68 for data in chunks(item, cluster_size):
[04619ba]69 empty_cluster = fat.index(0)
70 fat[empty_cluster] = 0xffff
[95b730c2]71
[04619ba]72 if (prev != -1):
73 fat[prev] = empty_cluster
74 else:
75 first = empty_cluster
76
77 prev = empty_cluster
78
[95b730c2]79 outf.seek(data_start + (empty_cluster - reserved_clusters) * cluster_size)
[04619ba]80 outf.write(data)
81
[cc1a727]82 return first, item.size
[04619ba]83
[95b730c2]84def write_directory(directory, outf, cluster_size, data_start, fat, reserved_clusters, dirent_size, empty_cluster):
85 "Store the contents of a directory"
86
87 length = len(directory)
88 size = length * dirent_size
89 prev = -1
90 first = 0
91
92 i = 0
93 rd = 0;
94 while (rd < size):
95 if (prev != -1):
96 empty_cluster = fat.index(0)
97 fat[empty_cluster] = 0xffff
98 fat[prev] = empty_cluster
99 else:
100 first = empty_cluster
101
102 prev = empty_cluster
103
[28f4adb]104 data = bytes()
[95b730c2]105 data_len = 0
106 while ((i < length) and (data_len < cluster_size)):
107 if (i == 0):
108 directory[i].cluster = empty_cluster
109
110 data += directory[i].pack()
111 data_len += dirent_size
112 i += 1
113
114 outf.seek(data_start + (empty_cluster - reserved_clusters) * cluster_size)
115 outf.write(data)
116 rd += len(data)
117
118 return first, size
119
[04619ba]120DIR_ENTRY = """little:
121 char name[8] /* file name */
122 char ext[3] /* file extension */
123 uint8_t attr /* file attributes */
[a248234]124 uint8_t lcase /* file name case (NT extension) */
[04619ba]125 uint8_t ctime_fine /* create time (fine resolution) */
126 uint16_t ctime /* create time */
127 uint16_t cdate /* create date */
128 uint16_t adate /* access date */
129 padding[2] /* EA-index */
130 uint16_t mtime /* modification time */
131 uint16_t mdate /* modification date */
132 uint16_t cluster /* first cluster */
133 uint32_t size /* file size */
134"""
135
[95b730c2]136DOT_DIR_ENTRY = """little:
137 uint8_t signature /* 0x2e signature */
138 char name[7] /* empty */
139 char ext[3] /* empty */
140 uint8_t attr /* file attributes */
141 padding[1] /* reserved for NT */
142 uint8_t ctime_fine /* create time (fine resolution) */
143 uint16_t ctime /* create time */
144 uint16_t cdate /* create date */
145 uint16_t adate /* access date */
146 padding[2] /* EA-index */
147 uint16_t mtime /* modification time */
148 uint16_t mdate /* modification date */
149 uint16_t cluster /* first cluster */
150 uint32_t size /* file size */
151"""
152
153DOTDOT_DIR_ENTRY = """little:
154 uint8_t signature[2] /* 0x2e signature */
155 char name[6] /* empty */
156 char ext[3] /* empty */
157 uint8_t attr /* file attributes */
158 padding[1] /* reserved for NT */
159 uint8_t ctime_fine /* create time (fine resolution) */
160 uint16_t ctime /* create time */
161 uint16_t cdate /* create date */
162 uint16_t adate /* access date */
163 padding[2] /* EA-index */
164 uint16_t mtime /* modification time */
165 uint16_t mdate /* modification date */
166 uint16_t cluster /* first cluster */
167 uint32_t size /* file size */
168"""
169
[7751ff1]170LFN_DIR_ENTRY = """little:
171 uint8_t seq /* sequence number */
172 char name1[10] /* first part of the name */
173 uint8_t attr /* attributes */
174 uint8_t rec_type /* LFN record type */
175 uint8_t checksum /* LFN checksum */
176 char name2[12] /* second part of the name */
177 uint16_t cluster /* cluster */
178 char name3[4] /* third part of the name */
[a9a6d8d]179"""
180
[7751ff1]181lchars = set(['A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J',
182 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T',
183 'U', 'V', 'W', 'X', 'Y', 'Z',
184 '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
185 '!', '#', '$', '%', '&', '\'', '(', ')', '-', '@',
186 '^', '_', '`', '{', '}', '~', '.'])
[a9a6d8d]187
[7751ff1]188def fat_lchars(name):
189 "Filter FAT legal characters"
190
[67435b1]191 filtered_name = b''
[87608a5]192 filtered = False
[7751ff1]193
[87608a5]194 for char in name.encode('ascii', 'replace').upper():
195 if char in lchars:
196 filtered_name += char
197 else:
198 filtered_name += b'_'
199 filtered = True
[7751ff1]200
[87608a5]201 return (filtered_name, filtered)
[a9a6d8d]202
[7751ff1]203def fat_name83(name, name83_list):
204 "Create a 8.3 name for the given name"
[95b730c2]205
[87608a5]206 ascii_name, lfn = fat_lchars(name)
[67435b1]207 # Splitting works only on strings, not on bytes
208 ascii_parts = ascii_name.decode('utf8').split('.')
[7751ff1]209
210 short_name = ''
211 short_ext = ''
212
213 if len(ascii_name) > 11:
[a9a6d8d]214 lfn = True
[7751ff1]215
216 if len(ascii_parts) > 0:
217 short_name = ascii_parts[0]
218 if len(short_name) > 8:
[a9a6d8d]219 lfn = True
[7751ff1]220
221 if len(ascii_parts) > 1:
222 short_ext = ascii_parts[-1]
223 if len(short_ext) > 3:
[a9a6d8d]224 lfn = True
[7751ff1]225
226 if len(ascii_parts) > 2:
[a9a6d8d]227 lfn = True
[7751ff1]228
229 if lfn == False:
230 name83_list.append(short_name + '.' + short_ext)
231 return (short_name.ljust(8)[0:8], short_ext.ljust(3)[0:3], False)
232
[a9a6d8d]233 # For filenames with multiple extensions, we treat the last one
234 # as the actual extension. The rest of the filename is stripped
235 # of dots and concatenated to form the short name
[7751ff1]236 for part in ascii_parts[1:-1]:
237 short_name += part
238
239 for number in range(1, 999999):
240 number_str = ('~' + str(number)).upper()
241
242 if len(short_name) + len(number_str) > 8:
243 short_name = short_name[0:8 - len(number_str)]
244
245 short_name += number_str;
246
247 if not (short_name + '.' + short_ext) in name83_list:
[a9a6d8d]248 break
[7751ff1]249
250 name83_list.append(short_name + '.' + short_ext)
251 return (short_name.ljust(8)[0:8], short_ext.ljust(3)[0:3], True)
[a9a6d8d]252
[7751ff1]253def create_lfn_dirent(name, seq, checksum):
254 "Create LFN directory entry"
[042fbe0]255
[7751ff1]256 entry = xstruct.create(LFN_DIR_ENTRY)
257 name_rest = name[26:]
[042fbe0]258
[7751ff1]259 if len(name_rest) > 0:
260 entry.seq = seq
261 else:
262 entry.seq = seq | 0x40
263
264 entry.name1 = name[0:10]
265 entry.name2 = name[10:22]
266 entry.name3 = name[22:26]
267
268 entry.attr = 0x0F
269 entry.rec_type = 0
270 entry.checksum = checksum
271 entry.cluster = 0
272
273 return (entry, name_rest)
[a9a6d8d]274
[7751ff1]275def lfn_checksum(name):
276 "Calculate LFN checksum"
277
278 checksum = 0
279 for i in range(0, 11):
280 checksum = (((checksum & 1) << 7) + (checksum >> 1) + ord(name[i])) & 0xFF
281
282 return checksum
[04619ba]283
[7751ff1]284def create_dirent(name, name83_list, directory, cluster, size):
285 short_name, short_ext, lfn = fat_name83(name, name83_list)
[04619ba]286
[a9a6d8d]287 dir_entry = xstruct.create(DIR_ENTRY)
[04619ba]288
[7751ff1]289 dir_entry.name = short_name
290 dir_entry.ext = short_ext
291
[04619ba]292 if (directory):
293 dir_entry.attr = 0x30
294 else:
295 dir_entry.attr = 0x20
296
[a248234]297 dir_entry.lcase = 0x18
[04619ba]298 dir_entry.ctime_fine = 0 # FIXME
299 dir_entry.ctime = 0 # FIXME
300 dir_entry.cdate = 0 # FIXME
301 dir_entry.adate = 0 # FIXME
302 dir_entry.mtime = 0 # FIXME
303 dir_entry.mdate = 0 # FIXME
304 dir_entry.cluster = cluster
[95b730c2]305
306 if (directory):
307 dir_entry.size = 0
308 else:
309 dir_entry.size = size
310
[a9a6d8d]311 if not lfn:
312 return [dir_entry]
313
[7751ff1]314 long_name = name.encode('utf_16_le')
315 entries = [dir_entry]
[a9a6d8d]316
[7751ff1]317 seq = 1
318 checksum = lfn_checksum(dir_entry.name + dir_entry.ext)
319
320 while len(long_name) > 0:
321 long_entry, long_name = create_lfn_dirent(long_name, seq, checksum)
322 entries.append(long_entry)
323 seq += 1
324
325 entries.reverse()
[a9a6d8d]326 return entries
[95b730c2]327
328def create_dot_dirent(empty_cluster):
329 dir_entry = xstruct.create(DOT_DIR_ENTRY)
330
331 dir_entry.signature = 0x2e
[432f68a]332 dir_entry.name = b' '
333 dir_entry.ext = b' '
[95b730c2]334 dir_entry.attr = 0x10
335
336 dir_entry.ctime_fine = 0 # FIXME
337 dir_entry.ctime = 0 # FIXME
338 dir_entry.cdate = 0 # FIXME
339 dir_entry.adate = 0 # FIXME
340 dir_entry.mtime = 0 # FIXME
341 dir_entry.mdate = 0 # FIXME
342 dir_entry.cluster = empty_cluster
343 dir_entry.size = 0
344
345 return dir_entry
346
347def create_dotdot_dirent(parent_cluster):
348 dir_entry = xstruct.create(DOTDOT_DIR_ENTRY)
349
350 dir_entry.signature = [0x2e, 0x2e]
[432f68a]351 dir_entry.name = b' '
352 dir_entry.ext = b' '
[95b730c2]353 dir_entry.attr = 0x10
354
355 dir_entry.ctime_fine = 0 # FIXME
356 dir_entry.ctime = 0 # FIXME
357 dir_entry.cdate = 0 # FIXME
358 dir_entry.adate = 0 # FIXME
359 dir_entry.mtime = 0 # FIXME
360 dir_entry.mdate = 0 # FIXME
361 dir_entry.cluster = parent_cluster
362 dir_entry.size = 0
[04619ba]363
364 return dir_entry
365
[95b730c2]366def recursion(head, root, outf, cluster_size, root_start, data_start, fat, reserved_clusters, dirent_size, parent_cluster):
[04619ba]367 "Recursive directory walk"
368
369 directory = []
[7751ff1]370 name83_list = []
[95b730c2]371
[7751ff1]372 if not head:
[95b730c2]373 # Directory cluster preallocation
374 empty_cluster = fat.index(0)
[7751ff1]375 fat[empty_cluster] = 0xFFFF
[95b730c2]376
377 directory.append(create_dot_dirent(empty_cluster))
378 directory.append(create_dotdot_dirent(parent_cluster))
379 else:
380 empty_cluster = 0
381
[7751ff1]382 for item in listdir_items(root):
[cc1a727]383 if item.is_file:
384 rv = write_file(item, outf, cluster_size, data_start, fat, reserved_clusters)
[7751ff1]385 directory.extend(create_dirent(item.name, name83_list, False, rv[0], rv[1]))
[cc1a727]386 elif item.is_dir:
387 rv = recursion(False, item.path, outf, cluster_size, root_start, data_start, fat, reserved_clusters, dirent_size, empty_cluster)
[7751ff1]388 directory.extend(create_dirent(item.name, name83_list, True, rv[0], rv[1]))
[04619ba]389
[7751ff1]390 if head:
[04619ba]391 outf.seek(root_start)
392 for dir_entry in directory:
393 outf.write(dir_entry.pack())
[95b730c2]394 else:
395 return write_directory(directory, outf, cluster_size, data_start, fat, reserved_clusters, dirent_size, empty_cluster)
[04619ba]396
[5749372]397BOOT_SECTOR = """little:
398 uint8_t jmp[3] /* jump instruction */
399 char oem[8] /* OEM string */
400 uint16_t sector /* bytes per sector */
401 uint8_t cluster /* sectors per cluster */
402 uint16_t reserved /* reserved sectors */
403 uint8_t fats /* number of FATs */
404 uint16_t rootdir /* root directory entries */
405 uint16_t sectors /* total number of sectors */
406 uint8_t descriptor /* media descriptor */
407 uint16_t fat_sectors /* sectors per single FAT */
408 uint16_t track_sectors /* sectors per track */
409 uint16_t heads /* number of heads */
410 uint32_t hidden /* hidden sectors */
411 uint32_t sectors_big /* total number of sectors (if sectors == 0) */
412
413 /* Extended BIOS Parameter Block */
414 uint8_t drive /* physical drive number */
415 padding[1] /* reserved (current head) */
416 uint8_t extboot_signature /* extended boot signature */
417 uint32_t serial /* serial number */
418 char label[11] /* volume label */
419 char fstype[8] /* filesystem type */
420 padding[448] /* boot code */
421 uint8_t boot_signature[2] /* boot signature */
422"""
423
[8bc6fcf]424EMPTY_SECTOR = """little:
[04619ba]425 padding[512] /* empty sector data */
426"""
427
428FAT_ENTRY = """little:
429 uint16_t next /* FAT16 entry */
[8bc6fcf]430"""
431
[5749372]432def usage(prname):
433 "Print usage syntax"
[28f4adb]434 print(prname + " <EXTRA_BYTES> <PATH> <IMAGE>")
[5749372]435
436def main():
[14f2100]437 if (len(sys.argv) < 4):
[5749372]438 usage(sys.argv[0])
439 return
440
[14f2100]441 if (not sys.argv[1].isdigit()):
[28f4adb]442 print("<EXTRA_BYTES> must be a number")
[14f2100]443 return
444
445 extra_bytes = int(sys.argv[1])
446
[67435b1]447 path = os.path.abspath(sys.argv[2])
[5749372]448 if (not os.path.isdir(path)):
[28f4adb]449 print("<PATH> must be a directory")
[5749372]450 return
451
[c4702798]452 fat16_clusters = 4096
[04619ba]453
[700eaa9]454 sector_size = 512
455 cluster_size = 4096
[04619ba]456 dirent_size = 32
457 fatent_size = 2
458 fat_count = 2
459 reserved_clusters = 2
460
[f4057f5]461 # Make sure the filesystem is large enough for FAT16
[14f2100]462 size = subtree_size(path, cluster_size, dirent_size) + reserved_clusters * cluster_size + extra_bytes
[28f4adb]463 while (size // cluster_size < fat16_clusters):
[ab579fa]464 if (cluster_size > sector_size):
[28f4adb]465 cluster_size = cluster_size // 2
[14f2100]466 size = subtree_size(path, cluster_size, dirent_size) + reserved_clusters * cluster_size + extra_bytes
[04619ba]467 else:
468 size = fat16_clusters * cluster_size + reserved_clusters * cluster_size
[700eaa9]469
[04619ba]470 root_size = align_up(root_entries(path) * dirent_size, cluster_size)
[700eaa9]471
[28f4adb]472 fat_size = align_up(align_up(size, cluster_size) // cluster_size * fatent_size, sector_size)
[04619ba]473
[28f4adb]474 sectors = (cluster_size + fat_count * fat_size + root_size + size) // sector_size
[04619ba]475 root_start = cluster_size + fat_count * fat_size
476 data_start = root_start + root_size
[700eaa9]477
[28f4adb]478 outf = open(sys.argv[3], "wb")
[5749372]479
480 boot_sector = xstruct.create(BOOT_SECTOR)
481 boot_sector.jmp = [0xEB, 0x3C, 0x90]
[28f4adb]482 boot_sector.oem = b'MSDOS5.0'
[700eaa9]483 boot_sector.sector = sector_size
[28f4adb]484 boot_sector.cluster = cluster_size // sector_size
485 boot_sector.reserved = cluster_size // sector_size
[04619ba]486 boot_sector.fats = fat_count
[28f4adb]487 boot_sector.rootdir = root_size // dirent_size
[de4a1cf]488 if (sectors <= 65535):
489 boot_sector.sectors = sectors
490 else:
491 boot_sector.sectors = 0
[5749372]492 boot_sector.descriptor = 0xF8
[28f4adb]493 boot_sector.fat_sectors = fat_size // sector_size
[700eaa9]494 boot_sector.track_sectors = 63
495 boot_sector.heads = 6
[5749372]496 boot_sector.hidden = 0
[de4a1cf]497 if (sectors > 65535):
498 boot_sector.sectors_big = sectors
499 else:
500 boot_sector.sectors_big = 0
[5749372]501
[700eaa9]502 boot_sector.drive = 0x80
[5749372]503 boot_sector.extboot_signature = 0x29
[0951495]504 boot_sector.serial = random.randint(0, 0x7fffffff)
[28f4adb]505 boot_sector.label = b'HELENOS'
506 boot_sector.fstype = b'FAT16 '
[5749372]507 boot_sector.boot_signature = [0x55, 0xAA]
508
509 outf.write(boot_sector.pack())
510
[8bc6fcf]511 empty_sector = xstruct.create(EMPTY_SECTOR)
512
[04619ba]513 # Reserved sectors
[28f4adb]514 for i in range(1, cluster_size // sector_size):
[8bc6fcf]515 outf.write(empty_sector.pack())
516
517 # FAT tables
[04619ba]518 for i in range(0, fat_count):
[28f4adb]519 for j in range(0, fat_size // sector_size):
[8bc6fcf]520 outf.write(empty_sector.pack())
521
522 # Root directory
[28f4adb]523 for i in range(0, root_size // sector_size):
[8bc6fcf]524 outf.write(empty_sector.pack())
525
526 # Data
[28f4adb]527 for i in range(0, size // sector_size):
[8bc6fcf]528 outf.write(empty_sector.pack())
529
[28f4adb]530 fat = array.array('L', [0] * (fat_size // fatent_size))
[04619ba]531 fat[0] = 0xfff8
532 fat[1] = 0xffff
533
[95b730c2]534 recursion(True, path, outf, cluster_size, root_start, data_start, fat, reserved_clusters, dirent_size, 0)
[04619ba]535
536 # Store FAT
537 fat_entry = xstruct.create(FAT_ENTRY)
538 for i in range(0, fat_count):
539 outf.seek(cluster_size + i * fat_size)
[28f4adb]540 for j in range(0, fat_size // fatent_size):
[04619ba]541 fat_entry.next = fat[j]
542 outf.write(fat_entry.pack())
543
[5749372]544 outf.close()
[7751ff1]545
[5749372]546if __name__ == '__main__':
547 main()
Note: See TracBrowser for help on using the repository browser.