prodos: make structures BlockSources + BlockSinks

This commit is contained in:
Zellyn Hunter 2017-03-29 22:24:55 -04:00
parent 1d7be34b5c
commit 1758fc0f32
2 changed files with 58 additions and 17 deletions

View File

@ -33,6 +33,10 @@ type bitmapPart struct {
data disk.Block data disk.Block
} }
// Ensure that bitmapPart is valid BlockSource and BlockSink.
var _ disk.BlockSource = (*bitmapPart)(nil)
var _ disk.BlockSink = (*bitmapPart)(nil)
// FromBlock unmarshals a bitmapPart from a Block. // FromBlock unmarshals a bitmapPart from a Block.
func (bp *bitmapPart) FromBlock(block disk.Block) error { func (bp *bitmapPart) FromBlock(block disk.Block) error {
bp.data = block bp.data = block
@ -87,7 +91,8 @@ func (vbm VolumeBitMap) IsFree(block uint16) bool {
return vbm[blockIndex].data[blockByteIndex]&bit > 0 return vbm[blockIndex].data[blockByteIndex]&bit > 0
} }
// ReadVolumeBitMap // ReadVolumeBitMap reads the entire volume bitmap from a block
// device.
func ReadVolumeBitMap(bd disk.BlockDevice, startBlock uint16) (VolumeBitMap, error) { func ReadVolumeBitMap(bd disk.BlockDevice, startBlock uint16) (VolumeBitMap, error) {
blocks := bd.Blocks() / 4096 blocks := bd.Blocks() / 4096
vbm := NewVolumeBitMap(startBlock, blocks) vbm := NewVolumeBitMap(startBlock, blocks)
@ -145,6 +150,7 @@ func (dt DateTime) Validate(fieldDescription string) (errors []error) {
// VolumeDirectoryKeyBlock is the struct used to hold the ProDOS Volume Directory Key // VolumeDirectoryKeyBlock is the struct used to hold the ProDOS Volume Directory Key
// Block structure. See page 4-4 of Beneath Apple ProDOS. // Block structure. See page 4-4 of Beneath Apple ProDOS.
type VolumeDirectoryKeyBlock struct { type VolumeDirectoryKeyBlock struct {
blockBase
Prev uint16 // Pointer to previous block (always zero: the KeyBlock is the first Volume Directory block Prev uint16 // Pointer to previous block (always zero: the KeyBlock is the first Volume Directory block
Next uint16 // Pointer to next block in the Volume Directory Next uint16 // Pointer to next block in the Volume Directory
Header VolumeDirectoryHeader Header VolumeDirectoryHeader
@ -152,8 +158,12 @@ type VolumeDirectoryKeyBlock struct {
Extra byte // Trailing byte (so we don't lose it) Extra byte // Trailing byte (so we don't lose it)
} }
// Ensure that VolumeDirectoryKeyBlock is valid BlockSource and BlockSink.
var _ disk.BlockSource = (*VolumeDirectoryKeyBlock)(nil)
var _ disk.BlockSink = (*VolumeDirectoryKeyBlock)(nil)
// ToBlock marshals the VolumeDirectoryKeyBlock to a Block of bytes. // ToBlock marshals the VolumeDirectoryKeyBlock to a Block of bytes.
func (vdkb VolumeDirectoryKeyBlock) ToBlock() disk.Block { func (vdkb VolumeDirectoryKeyBlock) ToBlock() (disk.Block, error) {
var block disk.Block var block disk.Block
binary.LittleEndian.PutUint16(block[0x0:0x2], vdkb.Prev) binary.LittleEndian.PutUint16(block[0x0:0x2], vdkb.Prev)
binary.LittleEndian.PutUint16(block[0x2:0x4], vdkb.Next) binary.LittleEndian.PutUint16(block[0x2:0x4], vdkb.Next)
@ -162,11 +172,11 @@ func (vdkb VolumeDirectoryKeyBlock) ToBlock() disk.Block {
copyBytes(block[0x2b+i*0x27:0x2b+(i+1)*0x27], desc.toBytes()) copyBytes(block[0x2b+i*0x27:0x2b+(i+1)*0x27], desc.toBytes())
} }
block[511] = vdkb.Extra block[511] = vdkb.Extra
return block return block, nil
} }
// FromBlock unmarshals a Block of bytes into a VolumeDirectoryKeyBlock. // FromBlock unmarshals a Block of bytes into a VolumeDirectoryKeyBlock.
func (vdkb *VolumeDirectoryKeyBlock) FromBlock(block disk.Block) { func (vdkb *VolumeDirectoryKeyBlock) FromBlock(block disk.Block) error {
vdkb.Prev = binary.LittleEndian.Uint16(block[0x0:0x2]) vdkb.Prev = binary.LittleEndian.Uint16(block[0x0:0x2])
vdkb.Next = binary.LittleEndian.Uint16(block[0x2:0x4]) vdkb.Next = binary.LittleEndian.Uint16(block[0x2:0x4])
vdkb.Header.fromBytes(block[0x04:0x2b]) vdkb.Header.fromBytes(block[0x04:0x2b])
@ -174,6 +184,7 @@ func (vdkb *VolumeDirectoryKeyBlock) FromBlock(block disk.Block) {
vdkb.Descriptors[i].fromBytes(block[0x2b+i*0x27 : 0x2b+(i+1)*0x27]) vdkb.Descriptors[i].fromBytes(block[0x2b+i*0x27 : 0x2b+(i+1)*0x27])
} }
vdkb.Extra = block[511] vdkb.Extra = block[511]
return nil
} }
// Validate validates a VolumeDirectoryKeyBlock for valid values. // Validate validates a VolumeDirectoryKeyBlock for valid values.
@ -193,14 +204,19 @@ func (vdkb VolumeDirectoryKeyBlock) Validate() (errors []error) {
// VolumeDirectoryBlock is a normal (non-key) segment in the Volume Directory Header. // VolumeDirectoryBlock is a normal (non-key) segment in the Volume Directory Header.
type VolumeDirectoryBlock struct { type VolumeDirectoryBlock struct {
blockBase
Prev uint16 // Pointer to previous block in the Volume Directory. Prev uint16 // Pointer to previous block in the Volume Directory.
Next uint16 // Pointer to next block in the Volume Directory. Next uint16 // Pointer to next block in the Volume Directory.
Descriptors [13]FileDescriptor Descriptors [13]FileDescriptor
Extra byte // Trailing byte (so we don't lose it) Extra byte // Trailing byte (so we don't lose it)
} }
// Ensure that VolumeDirectoryBlock is valid BlockSource and BlockSink.
var _ disk.BlockSource = (*VolumeDirectoryBlock)(nil)
var _ disk.BlockSink = (*VolumeDirectoryBlock)(nil)
// ToBlock marshals a VolumeDirectoryBlock to a Block of bytes. // ToBlock marshals a VolumeDirectoryBlock to a Block of bytes.
func (vdb VolumeDirectoryBlock) ToBlock() disk.Block { func (vdb VolumeDirectoryBlock) ToBlock() (disk.Block, error) {
var block disk.Block var block disk.Block
binary.LittleEndian.PutUint16(block[0x0:0x2], vdb.Prev) binary.LittleEndian.PutUint16(block[0x0:0x2], vdb.Prev)
binary.LittleEndian.PutUint16(block[0x2:0x4], vdb.Next) binary.LittleEndian.PutUint16(block[0x2:0x4], vdb.Next)
@ -208,17 +224,18 @@ func (vdb VolumeDirectoryBlock) ToBlock() disk.Block {
copyBytes(block[0x04+i*0x27:0x04+(i+1)*0x27], desc.toBytes()) copyBytes(block[0x04+i*0x27:0x04+(i+1)*0x27], desc.toBytes())
} }
block[511] = vdb.Extra block[511] = vdb.Extra
return block return block, nil
} }
// FromBlock unmarshals a Block of bytes into a VolumeDirectoryBlock. // FromBlock unmarshals a Block of bytes into a VolumeDirectoryBlock.
func (vdb *VolumeDirectoryBlock) FromBlock(block disk.Block) { func (vdb *VolumeDirectoryBlock) FromBlock(block disk.Block) error {
vdb.Prev = binary.LittleEndian.Uint16(block[0x0:0x2]) vdb.Prev = binary.LittleEndian.Uint16(block[0x0:0x2])
vdb.Next = binary.LittleEndian.Uint16(block[0x2:0x4]) vdb.Next = binary.LittleEndian.Uint16(block[0x2:0x4])
for i := range vdb.Descriptors { for i := range vdb.Descriptors {
vdb.Descriptors[i].fromBytes(block[0x4+i*0x27 : 0x4+(i+1)*0x27]) vdb.Descriptors[i].fromBytes(block[0x4+i*0x27 : 0x4+(i+1)*0x27])
} }
vdb.Extra = block[511] vdb.Extra = block[511]
return nil
} }
// Validate validates a VolumeDirectoryBlock for valid values. // Validate validates a VolumeDirectoryBlock for valid values.
@ -392,6 +409,7 @@ func (i IndexBlock) Set(blockNum byte, block uint16) {
// SubdirectoryKeyBlock is the struct used to hold the first entry in // SubdirectoryKeyBlock is the struct used to hold the first entry in
// a subdirectory structure. // a subdirectory structure.
type SubdirectoryKeyBlock struct { type SubdirectoryKeyBlock struct {
blockBase
Prev uint16 // Pointer to previous block (always zero: the KeyBlock is the first Volume Directory block Prev uint16 // Pointer to previous block (always zero: the KeyBlock is the first Volume Directory block
Next uint16 // Pointer to next block in the Volume Directory Next uint16 // Pointer to next block in the Volume Directory
Header SubdirectoryHeader Header SubdirectoryHeader
@ -399,8 +417,12 @@ type SubdirectoryKeyBlock struct {
Extra byte // Trailing byte (so we don't lose it) Extra byte // Trailing byte (so we don't lose it)
} }
// Ensure that SubdirectoryKeyBlock is valid BlockSource and BlockSink.
var _ disk.BlockSource = (*SubdirectoryKeyBlock)(nil)
var _ disk.BlockSink = (*SubdirectoryKeyBlock)(nil)
// ToBlock marshals the SubdirectoryKeyBlock to a Block of bytes. // ToBlock marshals the SubdirectoryKeyBlock to a Block of bytes.
func (skb SubdirectoryKeyBlock) ToBlock() disk.Block { func (skb SubdirectoryKeyBlock) ToBlock() (disk.Block, error) {
var block disk.Block var block disk.Block
binary.LittleEndian.PutUint16(block[0x0:0x2], skb.Prev) binary.LittleEndian.PutUint16(block[0x0:0x2], skb.Prev)
binary.LittleEndian.PutUint16(block[0x2:0x4], skb.Next) binary.LittleEndian.PutUint16(block[0x2:0x4], skb.Next)
@ -409,11 +431,11 @@ func (skb SubdirectoryKeyBlock) ToBlock() disk.Block {
copyBytes(block[0x2b+i*0x27:0x2b+(i+1)*0x27], desc.toBytes()) copyBytes(block[0x2b+i*0x27:0x2b+(i+1)*0x27], desc.toBytes())
} }
block[511] = skb.Extra block[511] = skb.Extra
return block return block, nil
} }
// FromBlock unmarshals a Block of bytes into a SubdirectoryKeyBlock. // FromBlock unmarshals a Block of bytes into a SubdirectoryKeyBlock.
func (skb *SubdirectoryKeyBlock) FromBlock(block disk.Block) { func (skb *SubdirectoryKeyBlock) FromBlock(block disk.Block) error {
skb.Prev = binary.LittleEndian.Uint16(block[0x0:0x2]) skb.Prev = binary.LittleEndian.Uint16(block[0x0:0x2])
skb.Next = binary.LittleEndian.Uint16(block[0x2:0x4]) skb.Next = binary.LittleEndian.Uint16(block[0x2:0x4])
skb.Header.fromBytes(block[0x04:0x2b]) skb.Header.fromBytes(block[0x04:0x2b])
@ -421,6 +443,7 @@ func (skb *SubdirectoryKeyBlock) FromBlock(block disk.Block) {
skb.Descriptors[i].fromBytes(block[0x2b+i*0x27 : 0x2b+(i+1)*0x27]) skb.Descriptors[i].fromBytes(block[0x2b+i*0x27 : 0x2b+(i+1)*0x27])
} }
skb.Extra = block[511] skb.Extra = block[511]
return nil
} }
// Validate validates a SubdirectoryKeyBlock for valid values. // Validate validates a SubdirectoryKeyBlock for valid values.
@ -440,14 +463,19 @@ func (skb SubdirectoryKeyBlock) Validate() (errors []error) {
// SubdirectoryBlock is a normal (non-key) segment in a Subdirectory. // SubdirectoryBlock is a normal (non-key) segment in a Subdirectory.
type SubdirectoryBlock struct { type SubdirectoryBlock struct {
blockBase
Prev uint16 // Pointer to previous block in the Volume Directory. Prev uint16 // Pointer to previous block in the Volume Directory.
Next uint16 // Pointer to next block in the Volume Directory. Next uint16 // Pointer to next block in the Volume Directory.
Descriptors [13]FileDescriptor Descriptors [13]FileDescriptor
Extra byte // Trailing byte (so we don't lose it) Extra byte // Trailing byte (so we don't lose it)
} }
// Ensure that SubdirectoryBlock is valid BlockSource and BlockSink.
var _ disk.BlockSource = (*SubdirectoryBlock)(nil)
var _ disk.BlockSink = (*SubdirectoryBlock)(nil)
// ToBlock marshals a SubdirectoryBlock to a Block of bytes. // ToBlock marshals a SubdirectoryBlock to a Block of bytes.
func (sb SubdirectoryBlock) ToBlock() disk.Block { func (sb SubdirectoryBlock) ToBlock() (disk.Block, error) {
var block disk.Block var block disk.Block
binary.LittleEndian.PutUint16(block[0x0:0x2], sb.Prev) binary.LittleEndian.PutUint16(block[0x0:0x2], sb.Prev)
binary.LittleEndian.PutUint16(block[0x2:0x4], sb.Next) binary.LittleEndian.PutUint16(block[0x2:0x4], sb.Next)
@ -455,17 +483,18 @@ func (sb SubdirectoryBlock) ToBlock() disk.Block {
copyBytes(block[0x04+i*0x27:0x04+(i+1)*0x27], desc.toBytes()) copyBytes(block[0x04+i*0x27:0x04+(i+1)*0x27], desc.toBytes())
} }
block[511] = sb.Extra block[511] = sb.Extra
return block return block, nil
} }
// FromBlock unmarshals a Block of bytes into a SubdirectoryBlock. // FromBlock unmarshals a Block of bytes into a SubdirectoryBlock.
func (sb *SubdirectoryBlock) FromBlock(block disk.Block) { func (sb *SubdirectoryBlock) FromBlock(block disk.Block) error {
sb.Prev = binary.LittleEndian.Uint16(block[0x0:0x2]) sb.Prev = binary.LittleEndian.Uint16(block[0x0:0x2])
sb.Next = binary.LittleEndian.Uint16(block[0x2:0x4]) sb.Next = binary.LittleEndian.Uint16(block[0x2:0x4])
for i := range sb.Descriptors { for i := range sb.Descriptors {
sb.Descriptors[i].fromBytes(block[0x4+i*0x27 : 0x4+(i+1)*0x27]) sb.Descriptors[i].fromBytes(block[0x4+i*0x27 : 0x4+(i+1)*0x27])
} }
sb.Extra = block[511] sb.Extra = block[511]
return nil
} }
// Validate validates a SubdirectoryBlock for valid values. // Validate validates a SubdirectoryBlock for valid values.

View File

@ -20,7 +20,10 @@ func TestVolumeDirectoryKeyBlockMarshalRoundtrip(t *testing.T) {
b1 := randomBlock() b1 := randomBlock()
vdkb := &VolumeDirectoryKeyBlock{} vdkb := &VolumeDirectoryKeyBlock{}
vdkb.FromBlock(b1) vdkb.FromBlock(b1)
b2 := vdkb.ToBlock() b2, err := vdkb.ToBlock()
if err != nil {
t.Fatal(err)
}
if b1 != b2 { if b1 != b2 {
t.Fatalf("Blocks differ: %s", strings.Join(pretty.Diff(b1[:], b2[:]), "; ")) t.Fatalf("Blocks differ: %s", strings.Join(pretty.Diff(b1[:], b2[:]), "; "))
} }
@ -36,7 +39,10 @@ func TestVolumeDirectoryBlockMarshalRoundtrip(t *testing.T) {
b1 := randomBlock() b1 := randomBlock()
vdb := &VolumeDirectoryBlock{} vdb := &VolumeDirectoryBlock{}
vdb.FromBlock(b1) vdb.FromBlock(b1)
b2 := vdb.ToBlock() b2, err := vdb.ToBlock()
if err != nil {
t.Fatal(err)
}
if b1 != b2 { if b1 != b2 {
t.Fatalf("Blocks differ: %s", strings.Join(pretty.Diff(b1[:], b2[:]), "; ")) t.Fatalf("Blocks differ: %s", strings.Join(pretty.Diff(b1[:], b2[:]), "; "))
} }
@ -52,7 +58,10 @@ func TestSubdirectoryKeyBlockMarshalRoundtrip(t *testing.T) {
b1 := randomBlock() b1 := randomBlock()
skb := &SubdirectoryKeyBlock{} skb := &SubdirectoryKeyBlock{}
skb.FromBlock(b1) skb.FromBlock(b1)
b2 := skb.ToBlock() b2, err := skb.ToBlock()
if err != nil {
t.Fatal(err)
}
if b1 != b2 { if b1 != b2 {
t.Fatalf("Blocks differ: %s", strings.Join(pretty.Diff(b1[:], b2[:]), "; ")) t.Fatalf("Blocks differ: %s", strings.Join(pretty.Diff(b1[:], b2[:]), "; "))
} }
@ -68,7 +77,10 @@ func TestSubdirectoryBlockMarshalRoundtrip(t *testing.T) {
b1 := randomBlock() b1 := randomBlock()
sb := &SubdirectoryBlock{} sb := &SubdirectoryBlock{}
sb.FromBlock(b1) sb.FromBlock(b1)
b2 := sb.ToBlock() b2, err := sb.ToBlock()
if err != nil {
t.Fatal(err)
}
if b1 != b2 { if b1 != b2 {
t.Fatalf("Blocks differ: %s", strings.Join(pretty.Diff(b1[:], b2[:]), "; ")) t.Fatalf("Blocks differ: %s", strings.Join(pretty.Diff(b1[:], b2[:]), "; "))
} }