mirror of
https://github.com/fadden/ciderpress.git
synced 2024-11-27 08:49:20 +00:00
144 lines
5.0 KiB
C++
144 lines
5.0 KiB
C++
/*
|
|
* CiderPress
|
|
* Copyright (C) 2007 by faddenSoft, LLC. All Rights Reserved.
|
|
* See the file LICENSE for distribution terms.
|
|
*/
|
|
/*
|
|
* Directory dump.
|
|
*/
|
|
#include "StdAfx.h"
|
|
#include "Directory.h"
|
|
#include "../diskimg/DiskImgDetail.h"
|
|
#include "../app/FileNameConv.h"
|
|
|
|
using namespace DiskImgLib;
|
|
|
|
/*
|
|
* Decide whether or not we want to handle this file.
|
|
*/
|
|
void ReformatDirectory::Examine(ReformatHolder* pHolder)
|
|
{
|
|
ReformatHolder::ReformatApplies applies = ReformatHolder::kApplicNot;
|
|
|
|
if (pHolder->GetFileType() == kTypeDIR)
|
|
applies = ReformatHolder::kApplicProbably;
|
|
|
|
pHolder->SetApplic(ReformatHolder::kReformatProDOSDirectory, applies,
|
|
ReformatHolder::kApplicNot, ReformatHolder::kApplicNot);
|
|
}
|
|
|
|
|
|
/*
|
|
* Convert a ProDOS directory into a format resembling BASIC.System's
|
|
* 80-column "catalog" command.
|
|
*/
|
|
int ReformatDirectory::Process(const ReformatHolder* pHolder,
|
|
ReformatHolder::ReformatID id, ReformatHolder::ReformatPart part,
|
|
ReformatOutput* pOutput)
|
|
{
|
|
const uint8_t* srcBuf = pHolder->GetSourceBuf(part);
|
|
long srcLen = pHolder->GetSourceLen(part);
|
|
fUseRTF = false;
|
|
|
|
if (srcLen < 512 || (srcLen % 512) != 0) {
|
|
LOGI("ReformatDirectory: invalid len %d", srcLen);
|
|
return -1;
|
|
}
|
|
|
|
BufPrintf(" NAME TYPE BLOCKS MODIFIED "
|
|
" CREATED ENDFILE SUBTYPE\r\n\r\n");
|
|
|
|
PrintDirEntries(srcBuf, srcLen, false);
|
|
|
|
BufPrintf("\r\nDeleted entries:\r\n");
|
|
PrintDirEntries(srcBuf, srcLen, true);
|
|
|
|
if ((srcBuf[0x04] & 0xf0) == 0xf0) {
|
|
/* this is the volume directory */
|
|
BufPrintf("\r\nTotal blocks: %d\n", srcBuf[0x29] | srcBuf[0x2a] << 8);
|
|
}
|
|
|
|
SetResultBuffer(pOutput);
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* Print all of the entries in the directory.
|
|
*
|
|
* The "showDeleted" flag determines whether we show active entries or
|
|
* deleted entries.
|
|
*/
|
|
void ReformatDirectory::PrintDirEntries(const uint8_t* srcBuf,
|
|
long srcLen, bool showDeleted)
|
|
{
|
|
const int kEntriesPerBlock = 0x0d; // expected value for entries per blk
|
|
const int kEntryLength = 0x27; // expected value for dir entry len
|
|
const int kMaxFileName = 15;
|
|
const uint8_t* pDirEntry;
|
|
int blockIdx;
|
|
int entryIdx;
|
|
|
|
for (blockIdx = 0; blockIdx < srcLen / 512; blockIdx++) {
|
|
pDirEntry = srcBuf + 512*blockIdx + 4; // skip 4 bytes of prev/next
|
|
|
|
for (entryIdx = 0; entryIdx < kEntriesPerBlock;
|
|
entryIdx++, pDirEntry += kEntryLength)
|
|
{
|
|
/* skip directory header; should probably check entries_per_block */
|
|
if (blockIdx == 0 && entryIdx == 0)
|
|
continue;
|
|
|
|
if ((showDeleted && (pDirEntry[0x00] & 0xf0) == 0) ||
|
|
(!showDeleted && (pDirEntry[0x00] & 0xf0) != 0))
|
|
{
|
|
if (pDirEntry[0x01] == 0) /* never-used entry */
|
|
continue;
|
|
|
|
int nameLen = pDirEntry[0x00] & 0x0f;
|
|
if (nameLen == 0) {
|
|
/* scan for it */
|
|
while (pDirEntry[0x01 + nameLen] != 0 && nameLen < kMaxFileName)
|
|
nameLen++;
|
|
}
|
|
|
|
char fileName[kMaxFileName +1];
|
|
strncpy(fileName, (const char*)&pDirEntry[0x01], kMaxFileName);
|
|
fileName[nameLen] = '\0';
|
|
|
|
CString createStrW, modStrW;
|
|
A2FileProDOS::ProDate prodosDateTime;
|
|
|
|
prodosDateTime = pDirEntry[0x18] | pDirEntry[0x19] << 8 |
|
|
pDirEntry[0x1a] << 16 | pDirEntry[0x1b] << 24;
|
|
FormatDate(A2FileProDOS::ConvertProDate(prodosDateTime), &createStrW);
|
|
CStringA createStr(createStrW);
|
|
prodosDateTime = pDirEntry[0x21] | pDirEntry[0x22] << 8 |
|
|
pDirEntry[0x23] << 16 | pDirEntry[0x24] << 24;
|
|
FormatDate(A2FileProDOS::ConvertProDate(prodosDateTime), &modStrW);
|
|
CStringA modStr(modStrW);
|
|
|
|
char lockedFlag = '*';
|
|
if (pDirEntry[0x1e] & 0x80)
|
|
lockedFlag = ' ';
|
|
|
|
CStringA auxTypeStr;
|
|
uint16_t auxType = pDirEntry[0x1f] | pDirEntry[0x20] << 8;
|
|
if (pDirEntry[0x10] == 0x06) // bin
|
|
auxTypeStr.Format("A=$%04X", auxType);
|
|
else if (pDirEntry[0x10] == 0x04) // txt
|
|
auxTypeStr.Format("R=%5d", auxType);
|
|
|
|
BufPrintf("%c%-15s %-3ls %6d %16s %16s %8d %s\r\n",
|
|
lockedFlag,
|
|
fileName,
|
|
PathProposal::FileTypeString(pDirEntry[0x10]),
|
|
pDirEntry[0x13] | pDirEntry[0x14] << 8,
|
|
(LPCSTR) modStr,
|
|
(LPCSTR) createStr,
|
|
pDirEntry[0x15] | pDirEntry[0x16] << 8 | pDirEntry[0x17] << 16,
|
|
(LPCSTR) auxTypeStr);
|
|
}
|
|
}
|
|
}
|
|
}
|