2018-09-24 12:27:07 +00:00
|
|
|
import struct
|
2018-12-09 07:20:59 +00:00
|
|
|
from macresources import Resource, make_file, parse_file
|
|
|
|
from . import btree, bitmanip
|
|
|
|
from .directory import AbstractFolder, Folder, File
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-07 15:12:21 +00:00
|
|
|
|
|
|
|
def _catalog_rec_sort(b):
|
2018-09-30 13:54:16 +00:00
|
|
|
order = [
|
|
|
|
0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
|
|
|
|
0x08, 0x09, 0x0a, 0x0b, 0x0c, 0x0d, 0x0e, 0x0f,
|
|
|
|
0x10, 0x11, 0x12, 0x13, 0x14, 0x15, 0x16, 0x17,
|
|
|
|
0x18, 0x19, 0x1a, 0x1b, 0x1c, 0x1d, 0x1e, 0x1f,
|
|
|
|
|
|
|
|
0x20, 0x22, 0x23, 0x28, 0x29, 0x2a, 0x2b, 0x2c,
|
|
|
|
0x2f, 0x30, 0x31, 0x32, 0x33, 0x34, 0x35, 0x36,
|
|
|
|
0x37, 0x38, 0x39, 0x3a, 0x3b, 0x3c, 0x3d, 0x3e,
|
|
|
|
0x3f, 0x40, 0x41, 0x42, 0x43, 0x44, 0x45, 0x46,
|
|
|
|
|
|
|
|
0x47, 0x48, 0x58, 0x5a, 0x5e, 0x60, 0x67, 0x69,
|
|
|
|
0x6b, 0x6d, 0x73, 0x75, 0x77, 0x79, 0x7b, 0x7f,
|
|
|
|
0x8d, 0x8f, 0x91, 0x93, 0x96, 0x98, 0x9f, 0xa1,
|
|
|
|
0xa3, 0xa5, 0xa8, 0xaa, 0xab, 0xac, 0xad, 0xae,
|
|
|
|
|
|
|
|
0x54, 0x48, 0x58, 0x5a, 0x5e, 0x60, 0x67, 0x69,
|
|
|
|
0x6b, 0x6d, 0x73, 0x75, 0x77, 0x79, 0x7b, 0x7f,
|
|
|
|
0x8d, 0x8f, 0x91, 0x93, 0x96, 0x98, 0x9f, 0xa1,
|
|
|
|
0xa3, 0xa5, 0xa8, 0xaf, 0xb0, 0xb1, 0xb2, 0xb3,
|
|
|
|
|
|
|
|
0x4c, 0x50, 0x5c, 0x62, 0x7d, 0x81, 0x9a, 0x55,
|
|
|
|
0x4a, 0x56, 0x4c, 0x4e, 0x50, 0x5c, 0x62, 0x64,
|
|
|
|
0x65, 0x66, 0x6f, 0x70, 0x71, 0x72, 0x7d, 0x89,
|
|
|
|
0x8a, 0x8b, 0x81, 0x83, 0x9c, 0x9d, 0x9e, 0x9a,
|
|
|
|
|
|
|
|
0xb4, 0xb5, 0xb6, 0xb7, 0xb8, 0xb9, 0xba, 0x95,
|
|
|
|
0xbb, 0xbc, 0xbd, 0xbe, 0xbf, 0xc0, 0x52, 0x85,
|
|
|
|
0xc1, 0xc2, 0xc3, 0xc4, 0xc5, 0xc6, 0xc7, 0xc8,
|
|
|
|
0xc9, 0xca, 0xcb, 0x57, 0x8c, 0xcc, 0x52, 0x85,
|
|
|
|
|
|
|
|
0xcd, 0xce, 0xcf, 0xd0, 0xd1, 0xd2, 0xd3, 0x26,
|
|
|
|
0x27, 0xd4, 0x20, 0x4a, 0x4e, 0x83, 0x87, 0x87,
|
|
|
|
0xd5, 0xd6, 0x24, 0x25, 0x2d, 0x2e, 0xd7, 0xd8,
|
|
|
|
0xa7, 0xd9, 0xda, 0xdb, 0xdc, 0xdd, 0xde, 0xdf,
|
|
|
|
|
|
|
|
0xe0, 0xe1, 0xe2, 0xe3, 0xe4, 0xe5, 0xe6, 0xe7,
|
|
|
|
0xe8, 0xe9, 0xea, 0xeb, 0xec, 0xed, 0xee, 0xef,
|
|
|
|
0xf0, 0xf1, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6, 0xf7,
|
|
|
|
0xf8, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff,
|
|
|
|
]
|
2018-10-03 07:46:03 +00:00
|
|
|
|
|
|
|
b = b[0] # we are only sorting keys!
|
|
|
|
|
|
|
|
return b[:4] + bytes(order[ch] for ch in b[5:])
|
|
|
|
|
2018-10-08 00:20:03 +00:00
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
def _suggest_allocblk_size(volsize, minalign):
|
|
|
|
min_nonalloc_blks = 6 # just for this estimation
|
|
|
|
retval = minalign
|
|
|
|
while volsize - min_nonalloc_blks*512 > retval*65536:
|
|
|
|
retval += minalign
|
|
|
|
return retval
|
2018-09-30 13:54:16 +00:00
|
|
|
|
|
|
|
|
2018-10-18 06:05:15 +00:00
|
|
|
def _get_every_extent(nblocks, firstrecord, cnid, xoflow, fork):
|
|
|
|
accum = 0
|
|
|
|
extlist = []
|
|
|
|
|
|
|
|
for a, b in btree.unpack_extent_record(firstrecord):
|
|
|
|
if not b: continue
|
|
|
|
accum += b
|
|
|
|
extlist.append((a, b))
|
|
|
|
|
|
|
|
while accum < nblocks:
|
|
|
|
nextrecord = xoflow[cnid, fork, accum]
|
|
|
|
for a, b in btree.unpack_extent_record(nextrecord):
|
|
|
|
if not b: continue
|
|
|
|
accum += b
|
|
|
|
extlist.append((a, b))
|
|
|
|
|
|
|
|
return extlist
|
|
|
|
|
|
|
|
|
2018-11-05 16:45:50 +00:00
|
|
|
def _encode_name(name, kind='file'):
|
|
|
|
longest = {'file': 31, 'vol': 27, 'bb': 15}[kind]
|
2018-10-23 09:51:37 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
encoded = name.encode('mac_roman')
|
|
|
|
except UnicodeEncodeError:
|
|
|
|
raise BadNameError(name)
|
|
|
|
except AttributeError:
|
|
|
|
pass
|
|
|
|
|
|
|
|
if not 1 <= len(encoded) <= longest or b':' in encoded:
|
|
|
|
raise BadNameError(name)
|
|
|
|
|
|
|
|
return encoded
|
|
|
|
|
|
|
|
|
2018-11-05 16:45:50 +00:00
|
|
|
def _bb_name(name):
|
|
|
|
return bitmanip.pstring(_encode_name(name)).ljust(16)
|
|
|
|
|
|
|
|
|
2018-10-07 15:12:21 +00:00
|
|
|
class _TempWrapper:
|
|
|
|
"""Volume uses this to store metadata while serialising"""
|
|
|
|
def __init__(self, of):
|
|
|
|
self.of = of
|
|
|
|
|
|
|
|
|
2018-10-23 09:51:37 +00:00
|
|
|
class OutOfSpaceError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
|
|
class BadNameError(Exception):
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
2018-12-09 07:20:59 +00:00
|
|
|
class Volume(AbstractFolder):
|
2018-09-30 13:54:16 +00:00
|
|
|
def __init__(self):
|
|
|
|
super().__init__()
|
|
|
|
|
2018-10-23 04:52:24 +00:00
|
|
|
self.crdate = self.mddate = self.bkdate = 0
|
2018-10-08 01:06:35 +00:00
|
|
|
self.name = 'Untitled'
|
2018-09-30 13:54:16 +00:00
|
|
|
|
|
|
|
def read(self, from_volume):
|
2018-10-07 15:47:23 +00:00
|
|
|
drSigWord, drCrDate, drLsMod, drAtrb, drNmFls, \
|
2018-09-30 13:54:16 +00:00
|
|
|
drVBMSt, drAllocPtr, drNmAlBlks, drAlBlkSiz, drClpSiz, drAlBlSt, \
|
2018-10-07 15:47:23 +00:00
|
|
|
drNxtCNID, drFreeBks, drVN, drVolBkUp, drVSeqNum, \
|
2018-09-30 13:54:16 +00:00
|
|
|
drWrCnt, drXTClpSiz, drCTClpSiz, drNmRtDirs, drFilCnt, drDirCnt, \
|
2018-10-07 15:47:23 +00:00
|
|
|
drFndrInfo, drVCSize, drVBMCSize, drCtlCSize, \
|
2018-10-07 15:12:21 +00:00
|
|
|
drXTFlSize, drXTExtRec, \
|
|
|
|
drCTFlSize, drCTExtRec, \
|
|
|
|
= struct.unpack_from('>2sLLHHHHHLLHLH28pLHLLLHLL32sHHHL12sL12s', from_volume, 1024)
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-23 04:52:24 +00:00
|
|
|
self.crdate, self.mddate, self.bkdate = drCrDate, drLsMod, drVolBkUp
|
2018-10-07 15:47:23 +00:00
|
|
|
|
2018-10-07 15:12:21 +00:00
|
|
|
block2offset = lambda block: 512*drAlBlSt + drAlBlkSiz*block
|
2018-10-18 06:05:15 +00:00
|
|
|
getextents = lambda extents: b''.join(from_volume[block2offset(firstblk):block2offset(firstblk+blkcnt)] for (firstblk, blkcnt) in extents)
|
|
|
|
getfork = lambda size, extrec1, cnid, fork: getextents(_get_every_extent((size+drAlBlkSiz-1)//drAlBlkSiz, extrec1, cnid, extoflow, fork))[:size]
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-07 15:12:21 +00:00
|
|
|
extoflow = {}
|
2018-10-18 06:05:15 +00:00
|
|
|
for rec in btree.dump_btree(getfork(drXTFlSize, drXTExtRec, 3, 'data')):
|
2018-09-30 13:54:16 +00:00
|
|
|
if rec[0] != 7: continue
|
2018-10-08 01:33:34 +00:00
|
|
|
xkrFkType, xkrFNum, xkrFABN, extrec = struct.unpack_from('>xBLH12s', rec)
|
|
|
|
if xkrFkType == 0xFF:
|
|
|
|
fork = 'rsrc'
|
|
|
|
elif xkrFkType == 0:
|
|
|
|
fork = 'data'
|
|
|
|
extoflow[xkrFNum, fork, xkrFABN] = extrec
|
2018-09-30 13:54:16 +00:00
|
|
|
|
|
|
|
cnids = {}
|
2018-10-08 01:06:35 +00:00
|
|
|
childlist = [] # list of (parent_cnid, child_name, child_object) tuples
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-14 01:11:09 +00:00
|
|
|
prev_key = None
|
2018-10-18 06:05:15 +00:00
|
|
|
for rec in btree.dump_btree(getfork(drCTFlSize, drCTExtRec, 4, 'data')):
|
2018-09-30 13:54:16 +00:00
|
|
|
# create a directory tree from the catalog file
|
|
|
|
rec_len = rec[0]
|
2018-10-03 07:46:03 +00:00
|
|
|
if rec_len == 0: continue
|
|
|
|
|
|
|
|
key = rec[2:1+rec_len]
|
2018-10-08 00:18:30 +00:00
|
|
|
val = rec[bitmanip.pad_up(1+rec_len, 2):]
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-14 01:11:09 +00:00
|
|
|
# if prev_key: # Uncomment this to test the sort order with 20% performance cost!
|
|
|
|
# if _catalog_rec_sort((prev_key,)) >= _catalog_rec_sort((key,)):
|
|
|
|
# raise ValueError('Sort error: %r, %r' % (prev_key, key))
|
|
|
|
# prev_key = key
|
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
ckrParID, namelen = struct.unpack_from('>LB', key)
|
2018-10-07 10:31:42 +00:00
|
|
|
ckrCName = key[5:5+namelen]
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
datatype = (None, 'dir', 'file', 'dthread', 'fthread')[val[0]]
|
2018-09-30 13:54:16 +00:00
|
|
|
datarec = val[2:]
|
|
|
|
|
2018-10-08 01:06:35 +00:00
|
|
|
# print(datatype + '\t' + repr(key))
|
|
|
|
# print('\t', datarec)
|
|
|
|
# print()
|
2018-10-03 07:46:03 +00:00
|
|
|
|
|
|
|
if datatype == 'dir':
|
2018-09-30 13:54:16 +00:00
|
|
|
dirFlags, dirVal, dirDirID, dirCrDat, dirMdDat, dirBkDat, dirUsrInfo, dirFndrInfo \
|
|
|
|
= struct.unpack_from('>HHLLLL16s16s', datarec)
|
|
|
|
|
|
|
|
f = Folder()
|
|
|
|
cnids[dirDirID] = f
|
2018-10-08 01:06:35 +00:00
|
|
|
childlist.append((ckrParID, ckrCName, f))
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-23 04:52:24 +00:00
|
|
|
f.crdate, f.mddate, f.bkdate = dirCrDat, dirMdDat, dirBkDat
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
elif datatype == 'file':
|
2018-09-30 13:54:16 +00:00
|
|
|
filFlags, filTyp, filUsrWds, filFlNum, \
|
|
|
|
filStBlk, filLgLen, filPyLen, \
|
|
|
|
filRStBlk, filRLgLen, filRPyLen, \
|
|
|
|
filCrDat, filMdDat, filBkDat, \
|
|
|
|
filFndrInfo, filClpSize, \
|
|
|
|
filExtRec, filRExtRec, \
|
|
|
|
= struct.unpack_from('>BB16sLHLLHLLLLL16sH12s12sxxxx', datarec)
|
|
|
|
|
|
|
|
f = File()
|
|
|
|
cnids[filFlNum] = f
|
2018-10-08 01:06:35 +00:00
|
|
|
childlist.append((ckrParID, ckrCName, f))
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-23 04:52:24 +00:00
|
|
|
f.crdate, f.mddate, f.bkdate = filCrDat, filMdDat, filBkDat
|
2018-09-30 13:54:16 +00:00
|
|
|
f.type, f.creator, f.flags, f.x, f.y = struct.unpack_from('>4s4sHHH', filUsrWds)
|
|
|
|
|
2018-10-18 06:05:15 +00:00
|
|
|
f.data = getfork(filLgLen, filExtRec, filFlNum, 'data')
|
|
|
|
f.rsrc = getfork(filRLgLen, filRExtRec, filFlNum, 'rsrc')
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-01 12:38:17 +00:00
|
|
|
# elif datatype == 3:
|
|
|
|
# print('dir thread:', rec)
|
|
|
|
# elif datatype == 4:
|
|
|
|
# print('fil thread:', rec)
|
|
|
|
|
2018-10-08 01:06:35 +00:00
|
|
|
for parent_cnid, child_name, child_obj in childlist:
|
|
|
|
if parent_cnid != 1:
|
|
|
|
parent_obj = cnids[parent_cnid]
|
|
|
|
parent_obj[child_name] = child_obj
|
2018-09-30 13:54:16 +00:00
|
|
|
|
|
|
|
self.update(cnids[2])
|
|
|
|
|
2018-11-03 12:29:00 +00:00
|
|
|
self.pop('Desktop', None)
|
|
|
|
self.pop('Desktop DB', None)
|
|
|
|
self.pop('Desktop DF', None)
|
|
|
|
|
2018-12-10 08:07:34 +00:00
|
|
|
def write(self, size=800*1024, align=512, desktopdb=True, bootable=True, startapp=None, sparse=False):
|
2018-10-03 07:46:03 +00:00
|
|
|
if align < 512 or align % 512:
|
|
|
|
raise ValueError('align must be multiple of 512')
|
|
|
|
|
|
|
|
if size < 400 * 1024 or size % 512:
|
|
|
|
raise ValueError('size must be a multiple of 512b and >= 800K')
|
|
|
|
|
2018-11-05 16:45:50 +00:00
|
|
|
drVN = _encode_name(self.name, 'vol')
|
2018-10-23 09:51:37 +00:00
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
# overall layout:
|
|
|
|
# 1. two boot blocks (offset=0)
|
|
|
|
# 2. one volume control block (offset=2)
|
|
|
|
# 3. some bitmap blocks (offset=3)
|
|
|
|
# 4. many allocation blocks
|
|
|
|
# 5. duplicate VCB (offset=-2)
|
|
|
|
# 6. unused block (offset=-1)
|
|
|
|
|
|
|
|
# so we will our best guess at these variables as we go:
|
|
|
|
# drNmAlBlks, drAlBlkSiz, drAlBlSt
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
# the smallest possible alloc block size
|
|
|
|
drAlBlkSiz = _suggest_allocblk_size(size, align)
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
# how many blocks will we use for the bitmap?
|
|
|
|
# (cheat by adding blocks to align the alloc area)
|
|
|
|
bitmap_blk_cnt = 0
|
|
|
|
while (size - (5+bitmap_blk_cnt)*512) // drAlBlkSiz > bitmap_blk_cnt*512*8:
|
|
|
|
bitmap_blk_cnt += 1
|
|
|
|
while (3+bitmap_blk_cnt)*512 % align:
|
|
|
|
bitmap_blk_cnt += 1
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-01 12:38:17 +00:00
|
|
|
# decide how many alloc blocks there will be
|
2018-10-03 07:46:03 +00:00
|
|
|
drNmAlBlks = (size - (5+bitmap_blk_cnt)*512) // drAlBlkSiz
|
2018-10-01 12:38:17 +00:00
|
|
|
blkaccum = []
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-23 09:51:37 +00:00
|
|
|
def accumulate(x):
|
|
|
|
blkaccum.extend(x)
|
|
|
|
if len(blkaccum) > drNmAlBlks:
|
|
|
|
raise OutOfSpaceError
|
|
|
|
|
2018-10-01 12:38:17 +00:00
|
|
|
# <<< put the empty extents overflow file in here >>>
|
2018-10-14 02:08:44 +00:00
|
|
|
extoflowfile = btree.make_btree([], bthKeyLen=7, blksize=drAlBlkSiz)
|
2018-10-03 07:46:03 +00:00
|
|
|
# also need to do some cleverness to ensure that this gets picked up...
|
|
|
|
drXTFlSize = len(extoflowfile)
|
|
|
|
drXTExtRec_Start = len(blkaccum)
|
2018-10-23 09:51:37 +00:00
|
|
|
accumulate(bitmanip.chunkify(extoflowfile, drAlBlkSiz))
|
2018-10-03 07:46:03 +00:00
|
|
|
drXTExtRec_Cnt = len(blkaccum) - drXTExtRec_Start
|
2018-10-01 12:38:17 +00:00
|
|
|
|
|
|
|
# write all the files in the volume
|
|
|
|
topwrap = _TempWrapper(self)
|
2018-10-07 15:47:23 +00:00
|
|
|
topwrap.path = (self.name,)
|
2018-10-01 12:38:17 +00:00
|
|
|
topwrap.cnid = 2
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-01 12:38:17 +00:00
|
|
|
godwrap = _TempWrapper(None)
|
|
|
|
godwrap.cnid = 1
|
|
|
|
|
2018-10-23 09:33:20 +00:00
|
|
|
root_dict_backup = self._prefdict
|
|
|
|
if desktopdb:
|
|
|
|
self._prefdict = dict(self._prefdict)
|
|
|
|
f = File()
|
|
|
|
f.type, f.creator = b'FNDR', b'ERIK'
|
|
|
|
f.flags = 0x4000 # invisible
|
|
|
|
f.rsrc = make_file([Resource(b'STR ', 0, data=b'\x0AFinder 1.0')])
|
|
|
|
self['Desktop'] = f
|
|
|
|
if size >= 2*1024*1024:
|
|
|
|
f = File()
|
|
|
|
f.type, f.creator = b'BTFL', b'DMGR'
|
|
|
|
f.flags = 0x4000
|
|
|
|
f.data = btree.make_btree([], bthKeyLen=37, blksize=drAlBlkSiz)
|
|
|
|
self['Desktop DB'] = f
|
|
|
|
f = File()
|
|
|
|
f.type, f.creator = b'DTFL', b'DMGR'
|
|
|
|
f.flags = 0x4000
|
|
|
|
self['Desktop DF'] = f
|
|
|
|
|
2018-11-05 16:45:50 +00:00
|
|
|
system_folder_cnid = 0
|
|
|
|
startapp_folder_cnid = 0
|
|
|
|
bootblocks = bytearray(1024)
|
|
|
|
|
2018-10-07 15:47:23 +00:00
|
|
|
path2wrap = {(): godwrap, (self.name,): topwrap}
|
2018-10-01 12:38:17 +00:00
|
|
|
drNxtCNID = 16
|
2018-10-07 15:47:23 +00:00
|
|
|
for path, obj in self.iter_paths():
|
|
|
|
path = (self.name,) + path
|
2018-10-01 12:38:17 +00:00
|
|
|
wrap = _TempWrapper(obj)
|
|
|
|
path2wrap[path] = wrap
|
|
|
|
wrap.path = path
|
|
|
|
wrap.cnid = drNxtCNID; drNxtCNID += 1
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-11-09 10:49:29 +00:00
|
|
|
if isinstance(obj, File) and obj.type.upper() == b'ZSYS':
|
2018-11-05 16:45:50 +00:00
|
|
|
try:
|
|
|
|
sysname = path[-1]
|
|
|
|
|
|
|
|
fellows = path2wrap[path[:-1]].of.items()
|
|
|
|
fndrname = next(n for (n, o) in fellows if isinstance(o, File) and o.type == b'FNDR')
|
|
|
|
|
|
|
|
sysresources = parse_file(obj.rsrc)
|
|
|
|
boot1 = next(r for r in sysresources if (r.type, r.id) == (b'boot', 1))
|
|
|
|
bb = bytearray(boot1.data)
|
|
|
|
if len(bb) != 1024: raise ValueError
|
|
|
|
|
|
|
|
bb[0x0A:0x1A] = _bb_name(sysname)
|
|
|
|
bb[0x1A:0x2A] = _bb_name(fndrname)
|
|
|
|
|
|
|
|
except:
|
|
|
|
pass
|
|
|
|
|
|
|
|
else:
|
|
|
|
bootblocks[:] = bb
|
|
|
|
system_folder_cnid = path2wrap[path[:-1]].cnid
|
2018-11-05 15:24:17 +00:00
|
|
|
|
2018-11-06 00:38:26 +00:00
|
|
|
if isinstance(obj, File) and startapp and path[1:] == tuple(startapp):
|
2018-11-05 16:45:50 +00:00
|
|
|
startapp_folder_cnid = path2wrap[path[:-1]].cnid
|
2018-11-05 15:46:58 +00:00
|
|
|
|
2018-09-30 13:54:16 +00:00
|
|
|
if isinstance(obj, File):
|
2018-10-03 07:58:39 +00:00
|
|
|
wrap.dfrk = wrap.rfrk = (0, 0)
|
2018-10-01 12:38:17 +00:00
|
|
|
if obj.data:
|
|
|
|
pre = len(blkaccum)
|
2018-10-23 09:51:37 +00:00
|
|
|
accumulate(bitmanip.chunkify(obj.data, drAlBlkSiz))
|
2018-10-03 07:58:39 +00:00
|
|
|
wrap.dfrk = (pre, len(blkaccum)-pre)
|
2018-10-01 12:38:17 +00:00
|
|
|
if obj.rsrc:
|
|
|
|
pre = len(blkaccum)
|
2018-10-23 09:51:37 +00:00
|
|
|
accumulate(bitmanip.chunkify(obj.rsrc, drAlBlkSiz))
|
2018-10-03 07:58:39 +00:00
|
|
|
wrap.rfrk = (pre, len(blkaccum)-pre)
|
2018-10-01 12:38:17 +00:00
|
|
|
|
2018-10-23 09:33:20 +00:00
|
|
|
self._prefdict = root_dict_backup
|
|
|
|
|
2018-10-01 12:38:17 +00:00
|
|
|
catalog = [] # (key, value) tuples
|
|
|
|
|
2018-10-07 10:31:42 +00:00
|
|
|
drFilCnt = 0
|
|
|
|
drDirCnt = -1 # to exclude the root directory
|
2018-10-01 12:38:17 +00:00
|
|
|
|
|
|
|
for path, wrap in path2wrap.items():
|
|
|
|
if wrap.cnid == 1: continue
|
|
|
|
|
|
|
|
obj = wrap.of
|
2018-11-05 16:45:50 +00:00
|
|
|
pstrname = bitmanip.pstring(_encode_name(path[-1], 'file'))
|
2018-10-01 12:38:17 +00:00
|
|
|
|
2018-10-08 01:06:35 +00:00
|
|
|
mainrec_key = struct.pack('>L', path2wrap[path[:-1]].cnid) + pstrname
|
2018-10-01 12:38:17 +00:00
|
|
|
|
|
|
|
if isinstance(wrap.of, File):
|
|
|
|
drFilCnt += 1
|
|
|
|
|
|
|
|
cdrType = 2
|
2018-10-03 08:28:02 +00:00
|
|
|
filFlags = 1 << 1 # file thread record exists, but is not locked, nor "file record is used"
|
2018-09-30 13:54:16 +00:00
|
|
|
filTyp = 0
|
|
|
|
filUsrWds = struct.pack('>4s4sHHHxxxxxx', obj.type, obj.creator, obj.flags, obj.x, obj.y)
|
2018-10-01 12:38:17 +00:00
|
|
|
filFlNum = wrap.cnid
|
2018-10-08 00:18:30 +00:00
|
|
|
filStBlk, filLgLen, filPyLen = wrap.dfrk[0], len(obj.data), bitmanip.pad_up(len(obj.data), drAlBlkSiz)
|
|
|
|
filRStBlk, filRLgLen, filRPyLen = wrap.rfrk[0], len(obj.rsrc), bitmanip.pad_up(len(obj.rsrc), drAlBlkSiz)
|
2018-10-23 04:52:24 +00:00
|
|
|
filCrDat, filMdDat, filBkDat = obj.crdate, obj.mddate, obj.bkdate
|
2018-09-30 13:54:16 +00:00
|
|
|
filFndrInfo = bytes(16) # todo must fix
|
|
|
|
filClpSize = 0 # todo must fix
|
2018-10-03 07:58:39 +00:00
|
|
|
filExtRec = struct.pack('>HHHHHH', *wrap.dfrk, 0, 0, 0, 0)
|
|
|
|
filRExtRec = struct.pack('>HHHHHH', *wrap.rfrk, 0, 0, 0, 0)
|
2018-09-30 13:54:16 +00:00
|
|
|
|
2018-10-01 12:38:17 +00:00
|
|
|
mainrec_val = struct.pack('>BxBB16sLHLLHLLLLL16sH12s12sxxxx',
|
|
|
|
cdrType, \
|
2018-09-30 13:54:16 +00:00
|
|
|
filFlags, filTyp, filUsrWds, filFlNum, \
|
|
|
|
filStBlk, filLgLen, filPyLen, \
|
|
|
|
filRStBlk, filRLgLen, filRPyLen, \
|
|
|
|
filCrDat, filMdDat, filBkDat, \
|
|
|
|
filFndrInfo, filClpSize, \
|
|
|
|
filExtRec, filRExtRec, \
|
2018-10-01 12:38:17 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
else: # assume directory
|
|
|
|
drDirCnt += 1
|
|
|
|
|
|
|
|
cdrType = 1
|
|
|
|
dirFlags = 0 # must fix
|
|
|
|
dirVal = len(wrap.of)
|
|
|
|
dirDirID = wrap.cnid
|
2018-10-23 04:52:24 +00:00
|
|
|
dirCrDat, dirMdDat, dirBkDat = obj.crdate, obj.mddate, obj.bkdate
|
2018-10-01 12:38:17 +00:00
|
|
|
dirUsrInfo = bytes(16)
|
|
|
|
dirFndrInfo = bytes(16)
|
2018-10-07 10:31:42 +00:00
|
|
|
mainrec_val = struct.pack('>BxHHLLLL16s16sxxxxxxxxxxxxxxxx',
|
2018-10-01 12:38:17 +00:00
|
|
|
cdrType, dirFlags, dirVal, dirDirID,
|
|
|
|
dirCrDat, dirMdDat, dirBkDat,
|
|
|
|
dirUsrInfo, dirFndrInfo,
|
|
|
|
)
|
|
|
|
|
|
|
|
catalog.append((mainrec_key, mainrec_val))
|
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
thdrec_key = struct.pack('>Lx', wrap.cnid)
|
2018-10-01 12:38:17 +00:00
|
|
|
thdrec_val_type = 4 if isinstance(wrap.of, File) else 3
|
2018-10-08 01:06:35 +00:00
|
|
|
thdrec_val = struct.pack('>BxxxxxxxxxL', thdrec_val_type, path2wrap[path[:-1]].cnid) + pstrname
|
2018-10-01 12:38:17 +00:00
|
|
|
|
|
|
|
catalog.append((thdrec_key, thdrec_val))
|
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
|
2018-10-01 12:38:17 +00:00
|
|
|
# now it is time to sort these records! fuck that shit...
|
2018-10-07 15:12:21 +00:00
|
|
|
catalog.sort(key=_catalog_rec_sort)
|
2018-10-14 02:08:44 +00:00
|
|
|
catalogfile = btree.make_btree(catalog, bthKeyLen=37, blksize=drAlBlkSiz)
|
2018-10-01 12:38:17 +00:00
|
|
|
# also need to do some cleverness to ensure that this gets picked up...
|
|
|
|
drCTFlSize = len(catalogfile)
|
|
|
|
drCTExtRec_Start = len(blkaccum)
|
2018-10-23 09:51:37 +00:00
|
|
|
accumulate(bitmanip.chunkify(catalogfile, drAlBlkSiz))
|
2018-10-01 12:38:17 +00:00
|
|
|
drCTExtRec_Cnt = len(blkaccum) - drCTExtRec_Start
|
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
if len(blkaccum) > drNmAlBlks:
|
|
|
|
raise ValueError('Does not fit!')
|
|
|
|
|
2018-10-01 12:38:17 +00:00
|
|
|
# Create the bitmap of free volume allocation blocks
|
2018-10-08 00:18:30 +00:00
|
|
|
bitmap = bitmanip.bits(bitmap_blk_cnt * 512 * 8, len(blkaccum))
|
2018-10-01 12:38:17 +00:00
|
|
|
|
2018-11-05 15:46:58 +00:00
|
|
|
# Set the startup app
|
2018-11-05 16:45:50 +00:00
|
|
|
if system_folder_cnid and startapp_folder_cnid:
|
|
|
|
try:
|
|
|
|
bootblocks[0x5A:0x6A] = _bb_name(startapp[-1])
|
|
|
|
except:
|
|
|
|
startapp_folder_cnid = 0
|
2018-11-05 15:24:17 +00:00
|
|
|
|
2018-10-01 12:38:17 +00:00
|
|
|
# Create the Volume Information Block
|
|
|
|
drSigWord = b'BD'
|
|
|
|
drNmFls = sum(isinstance(x, File) for x in self.values())
|
|
|
|
drNmRtDirs = sum(not isinstance(x, File) for x in self.values())
|
|
|
|
drVBMSt = 3 # first block of volume bitmap
|
2018-10-07 10:31:42 +00:00
|
|
|
drAllocPtr = 0
|
2018-10-01 12:38:17 +00:00
|
|
|
drClpSiz = drXTClpSiz = drCTClpSiz = drAlBlkSiz
|
2018-10-03 07:46:03 +00:00
|
|
|
drAlBlSt = 3 + bitmap_blk_cnt
|
2018-10-01 12:38:17 +00:00
|
|
|
drFreeBks = drNmAlBlks - len(blkaccum)
|
|
|
|
drWrCnt = 0 # ????volume write count
|
2018-10-03 08:28:02 +00:00
|
|
|
drVCSize = drVBMCSize = drCtlCSize = 0
|
2018-10-07 15:47:23 +00:00
|
|
|
drAtrb = 1<<8 # volume attributes (hwlock, swlock, CLEANUNMOUNT, badblocks)
|
|
|
|
drVolBkUp = 0 # date and time of last backup
|
|
|
|
drVSeqNum = 0 # volume backup sequence number
|
2018-11-05 16:45:50 +00:00
|
|
|
drFndrInfo = struct.pack('>LLL28x', system_folder_cnid, startapp_folder_cnid, startapp_folder_cnid)
|
2018-10-23 04:52:24 +00:00
|
|
|
drCrDate, drLsMod, drVolBkUp = self.crdate, self.mddate, self.bkdate
|
2018-10-01 12:38:17 +00:00
|
|
|
|
2018-10-03 07:46:03 +00:00
|
|
|
vib = struct.pack('>2sLLHHHHHLLHLH28pLHLLLHLL32sHHHLHHxxxxxxxxLHHxxxxxxxx',
|
2018-10-07 15:47:23 +00:00
|
|
|
drSigWord, drCrDate, drLsMod, drAtrb, drNmFls,
|
2018-10-01 12:38:17 +00:00
|
|
|
drVBMSt, drAllocPtr, drNmAlBlks, drAlBlkSiz, drClpSiz, drAlBlSt,
|
2018-10-07 15:47:23 +00:00
|
|
|
drNxtCNID, drFreeBks, drVN, drVolBkUp, drVSeqNum,
|
2018-10-01 12:38:17 +00:00
|
|
|
drWrCnt, drXTClpSiz, drCTClpSiz, drNmRtDirs, drFilCnt, drDirCnt,
|
2018-10-07 15:47:23 +00:00
|
|
|
drFndrInfo, drVCSize, drVBMCSize, drCtlCSize,
|
2018-10-01 12:38:17 +00:00
|
|
|
drXTFlSize, drXTExtRec_Start, drXTExtRec_Cnt,
|
|
|
|
drCTFlSize, drCTExtRec_Start, drCTExtRec_Cnt,
|
|
|
|
)
|
2018-10-03 07:46:03 +00:00
|
|
|
vib += bytes(512-len(vib))
|
2018-10-01 12:38:17 +00:00
|
|
|
|
|
|
|
assert all(len(x) == drAlBlkSiz for x in blkaccum)
|
2018-12-10 08:07:34 +00:00
|
|
|
left_elements = [bootblocks, vib, bitmap, *blkaccum]
|
|
|
|
|
|
|
|
unused_offset = sum(len(x) for x in left_elements)
|
|
|
|
unused_length = size - unused_offset - 2*512
|
|
|
|
|
|
|
|
right_elements = [vib, bytes(512)]
|
|
|
|
|
|
|
|
if sparse:
|
|
|
|
return b''.join(left_elements), unused_length, b''.join(right_elements)
|
|
|
|
else:
|
|
|
|
all_elements = left_elements
|
|
|
|
all_elements.append(bytes(unused_length))
|
|
|
|
all_elements.extend(right_elements)
|
|
|
|
return b''.join(all_elements)
|