mirror of
https://github.com/fadden/ciderpress.git
synced 2025-01-02 15:30:02 +00:00
Fix handling of ProDOS disks with short volume directories
The ProDOS code was assuming that the volume directory was 4 blocks long. Some disks, such as the ProDOS 2.4.2 distribution disk, only use the first two blocks. CiderPress now scans the volume directory to determine the actual length. Fixes issue #32.
This commit is contained in:
parent
21151cd76a
commit
fb25998be5
@ -1596,6 +1596,7 @@ private:
|
|||||||
|
|
||||||
DIError Initialize(InitMode initMode);
|
DIError Initialize(InitMode initMode);
|
||||||
DIError LoadVolHeader(void);
|
DIError LoadVolHeader(void);
|
||||||
|
DIError DetermineVolDirLen(uint16_t nextBlock, uint16_t* pBlocksUsed);
|
||||||
void SetVolumeID(void);
|
void SetVolumeID(void);
|
||||||
void DumpVolHeader(void);
|
void DumpVolHeader(void);
|
||||||
DIError ScanVolBitmap(void);
|
DIError ScanVolBitmap(void);
|
||||||
|
@ -30,7 +30,7 @@
|
|||||||
|
|
||||||
const int kBlkSize = 512;
|
const int kBlkSize = 512;
|
||||||
const int kVolHeaderBlock = 2; // block where Volume Header resides
|
const int kVolHeaderBlock = 2; // block where Volume Header resides
|
||||||
const int kVolDirExpectedNumBlocks = 4; // customary #of volume header blocks
|
const int kFormatVolDirNumBlocks = 4; // #of volume header blocks for new volumes
|
||||||
const int kMinReasonableBlocks = 16; // min size for ProDOS volume
|
const int kMinReasonableBlocks = 16; // min size for ProDOS volume
|
||||||
const int kExpectedBitmapStart = 6; // block# where vol bitmap should start
|
const int kExpectedBitmapStart = 6; // block# where vol bitmap should start
|
||||||
const int kMaxCatalogIterations = 1024; // theoretical max is 32768?
|
const int kMaxCatalogIterations = 1024; // theoretical max is 32768?
|
||||||
@ -360,8 +360,11 @@ DIError DiskFSProDOS::LoadVolHeader(void)
|
|||||||
pEntry->fileName[nameLen] = '\0';
|
pEntry->fileName[nameLen] = '\0';
|
||||||
pEntry->fileType = kTypeDIR;
|
pEntry->fileType = kTypeDIR;
|
||||||
pEntry->keyPointer = kVolHeaderBlock;
|
pEntry->keyPointer = kVolHeaderBlock;
|
||||||
pEntry->blocksUsed = kVolDirExpectedNumBlocks;
|
dierr = DetermineVolDirLen(GetShortLE(&blkBuf[0x02]), &pEntry->blocksUsed);
|
||||||
pEntry->eof = kVolDirExpectedNumBlocks * 512;
|
if (dierr != kDIErrNone) {
|
||||||
|
goto bail;
|
||||||
|
}
|
||||||
|
pEntry->eof = pEntry->blocksUsed * 512;
|
||||||
pEntry->createWhen = GetLongLE(&blkBuf[0x1c]);
|
pEntry->createWhen = GetLongLE(&blkBuf[0x1c]);
|
||||||
pEntry->version = blkBuf[0x20];
|
pEntry->version = blkBuf[0x20];
|
||||||
pEntry->minVersion = blkBuf[0x21];
|
pEntry->minVersion = blkBuf[0x21];
|
||||||
@ -375,8 +378,47 @@ DIError DiskFSProDOS::LoadVolHeader(void)
|
|||||||
pFile->fSparseRsrcEof = -1;
|
pFile->fSparseRsrcEof = -1;
|
||||||
|
|
||||||
AddFileToList(pFile);
|
AddFileToList(pFile);
|
||||||
|
pFile = NULL;
|
||||||
|
|
||||||
bail:
|
bail:
|
||||||
|
delete pFile;
|
||||||
|
return dierr;
|
||||||
|
}
|
||||||
|
|
||||||
|
DIError DiskFSProDOS::DetermineVolDirLen(uint16_t nextBlock, uint16_t* pBlocksUsed) {
|
||||||
|
DIError dierr = kDIErrNone;
|
||||||
|
uint8_t blkBuf[kBlkSize];
|
||||||
|
uint16_t blocksUsed = 1;
|
||||||
|
int iterCount = 0;
|
||||||
|
|
||||||
|
// Traverse the volume directory chain, counting blocks. Normally this will have 4, but
|
||||||
|
// variations are possible.
|
||||||
|
while (nextBlock != 0) {
|
||||||
|
blocksUsed++;
|
||||||
|
|
||||||
|
if (nextBlock < 2 || nextBlock >= fpImg->GetNumBlocks()) {
|
||||||
|
LOGI(" ProDOS ERROR: invalid volume dir link block %u", nextBlock);
|
||||||
|
dierr = kDIErrInvalidBlock;
|
||||||
|
goto bail;
|
||||||
|
}
|
||||||
|
dierr = fpImg->ReadBlock(nextBlock, blkBuf);
|
||||||
|
if (dierr != kDIErrNone) {
|
||||||
|
goto bail;
|
||||||
|
}
|
||||||
|
|
||||||
|
nextBlock = GetShortLE(&blkBuf[0x02]);
|
||||||
|
|
||||||
|
// Watch for infinite loop.
|
||||||
|
iterCount++;
|
||||||
|
if (iterCount > fpImg->GetNumBlocks()) {
|
||||||
|
LOGI(" ProDOS ERROR: infinite vol directory loop found");
|
||||||
|
dierr = kDIErrDirectoryLoop;
|
||||||
|
goto bail;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bail:
|
||||||
|
*pBlocksUsed = blocksUsed;
|
||||||
return dierr;
|
return dierr;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -552,7 +594,7 @@ DIError DiskFSProDOS::ScanVolBitmap(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Generate an empty block use map.
|
* Generate an empty block use map. Used by disk formatter.
|
||||||
*/
|
*/
|
||||||
DIError DiskFSProDOS::CreateEmptyBlockMap(void)
|
DIError DiskFSProDOS::CreateEmptyBlockMap(void)
|
||||||
{
|
{
|
||||||
@ -569,7 +611,7 @@ DIError DiskFSProDOS::CreateEmptyBlockMap(void)
|
|||||||
*/
|
*/
|
||||||
long block;
|
long block;
|
||||||
long firstEmpty =
|
long firstEmpty =
|
||||||
kVolHeaderBlock + kVolDirExpectedNumBlocks + GetNumBitmapBlocks();
|
kVolHeaderBlock + kFormatVolDirNumBlocks + GetNumBitmapBlocks();
|
||||||
for (block = 0; block < firstEmpty; block++)
|
for (block = 0; block < firstEmpty; block++)
|
||||||
SetBlockUseEntry(block, true);
|
SetBlockUseEntry(block, true);
|
||||||
for ( ; block < fTotalBlocks; block++)
|
for ( ; block < fTotalBlocks; block++)
|
||||||
@ -1494,10 +1536,10 @@ DIError DiskFSProDOS::Format(DiskImg* pDiskImg, const char* volName)
|
|||||||
*/
|
*/
|
||||||
int i;
|
int i;
|
||||||
memset(blkBuf, 0, sizeof(blkBuf));
|
memset(blkBuf, 0, sizeof(blkBuf));
|
||||||
for (i = kVolHeaderBlock+1; i < kVolHeaderBlock+kVolDirExpectedNumBlocks; i++)
|
for (i = kVolHeaderBlock+1; i < kVolHeaderBlock+kFormatVolDirNumBlocks; i++)
|
||||||
{
|
{
|
||||||
PutShortLE(&blkBuf[0x00], i-1);
|
PutShortLE(&blkBuf[0x00], i-1);
|
||||||
if (i == kVolHeaderBlock+kVolDirExpectedNumBlocks-1)
|
if (i == kVolHeaderBlock+kFormatVolDirNumBlocks-1)
|
||||||
PutShortLE(&blkBuf[0x02], 0);
|
PutShortLE(&blkBuf[0x02], 0);
|
||||||
else
|
else
|
||||||
PutShortLE(&blkBuf[0x02], i+1);
|
PutShortLE(&blkBuf[0x02], i+1);
|
||||||
@ -1540,7 +1582,7 @@ DIError DiskFSProDOS::Format(DiskImg* pDiskImg, const char* volName)
|
|||||||
blkBuf[0x23] = 0x27; // entry_length: always $27
|
blkBuf[0x23] = 0x27; // entry_length: always $27
|
||||||
blkBuf[0x24] = 0x0d; // entries_per_block: always $0d
|
blkBuf[0x24] = 0x0d; // entries_per_block: always $0d
|
||||||
/* file_count is zero - does not include volume dir */
|
/* file_count is zero - does not include volume dir */
|
||||||
PutShortLE(&blkBuf[0x27], kVolHeaderBlock + kVolDirExpectedNumBlocks); // bit_map_pointer
|
PutShortLE(&blkBuf[0x27], kVolHeaderBlock + kFormatVolDirNumBlocks); // bit_map_pointer
|
||||||
PutShortLE(&blkBuf[0x29], (uint16_t) formatBlocks); // total_blocks
|
PutShortLE(&blkBuf[0x29], (uint16_t) formatBlocks); // total_blocks
|
||||||
dierr = fpImg->WriteBlock(kVolHeaderBlock, blkBuf);
|
dierr = fpImg->WriteBlock(kVolHeaderBlock, blkBuf);
|
||||||
if (dierr != kDIErrNone) {
|
if (dierr != kDIErrNone) {
|
||||||
@ -1558,8 +1600,7 @@ DIError DiskFSProDOS::Format(DiskImg* pDiskImg, const char* volName)
|
|||||||
|
|
||||||
/*
|
/*
|
||||||
* Generate the initial block usage map. The only entries in use are
|
* Generate the initial block usage map. The only entries in use are
|
||||||
* right at the start of the disk. When we finish, scan what we just
|
* right at the start of the disk.
|
||||||
* created into
|
|
||||||
*/
|
*/
|
||||||
CreateEmptyBlockMap();
|
CreateEmptyBlockMap();
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user