mirror of
https://github.com/fadden/ciderpress.git
synced 2025-01-02 15:30:02 +00:00
1663 lines
68 KiB
C++
1663 lines
68 KiB
C++
/*
|
|
* CiderPress
|
|
* Copyright (C) 2009 by CiderPress authors. All Rights Reserved.
|
|
* Copyright (C) 2007 by faddenSoft, LLC. All Rights Reserved.
|
|
* See the file LICENSE for distribution terms.
|
|
*/
|
|
/*
|
|
* Public declarations for the DiskImg library.
|
|
*
|
|
* Everything is wrapped in the "DiskImgLib" namespace. Either prefix
|
|
* all references with "DiskImgLib::", or add "using namespace DiskImgLib"
|
|
* to all C++ source files that make use of it.
|
|
*
|
|
* Under Linux, this should be compiled with -D_FILE_OFFSET_BITS=64.
|
|
*
|
|
* These classes are not thread-safe with respect to access to a single
|
|
* disk image. Writing to the same disk image from multiple threads
|
|
* simultaneously is bound to end in disaster. Simultaneous access to
|
|
* different objects will work, though modifying the same disk image
|
|
* file from multiple objects will lead to unpredictable results.
|
|
*/
|
|
#ifndef DISKIMG_DISKIMG_H
|
|
#define DISKIMG_DISKIMG_H
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
#include <assert.h>
|
|
|
|
//#define EXCISE_GPL_CODE
|
|
|
|
/* Windows DLL stuff */
|
|
#ifdef _WIN32
|
|
# ifdef DISKIMG_EXPORTS
|
|
# define DISKIMG_API __declspec(dllexport)
|
|
# else
|
|
# define DISKIMG_API __declspec(dllimport)
|
|
# endif
|
|
#else
|
|
# define DISKIMG_API
|
|
#endif
|
|
|
|
namespace DiskImgLib {
|
|
|
|
/* compiled-against versions; call DiskImg::GetVersion for linked-against */
|
|
#define kDiskImgVersionMajor 4
|
|
#define kDiskImgVersionMinor 6
|
|
#define kDiskImgVersionBug 0
|
|
|
|
|
|
/*
|
|
* Errors from the various disk image classes.
|
|
*/
|
|
typedef enum DIError {
|
|
kDIErrNone = 0,
|
|
|
|
/* I/O request errors (should renumber/rename to match GS/OS errors?) */
|
|
kDIErrAccessDenied = -10,
|
|
kDIErrVWAccessForbidden = -11, // write access to volume forbidden
|
|
kDIErrSharingViolation = -12, // file is in use and not shareable
|
|
kDIErrNoExclusiveAccess = -13, // couldn't get exclusive access
|
|
kDIErrWriteProtected = -14, // disk is write protected
|
|
kDIErrCDROMNotSupported = -15, // access to CD-ROM drives not supptd
|
|
kDIErrASPIFailure = -16, // generic ASPI failure result
|
|
kDIErrSPTIFailure = -17, // generic SPTI failure result
|
|
kDIErrSCSIFailure = -18, // generic SCSI failure result
|
|
kDIErrDeviceNotReady = -19, // floppy or CD-ROM drive has no media
|
|
|
|
kDIErrFileNotFound = -20,
|
|
kDIErrForkNotFound = -21, // requested fork does not exist
|
|
kDIErrAlreadyOpen = -22, // already open, can't open a 2nd time
|
|
kDIErrFileOpen = -23, // file is open, can't delete it
|
|
kDIErrNotReady = -24,
|
|
kDIErrFileExists = -25, // file already exists
|
|
kDIErrDirectoryExists = -26, // directory already exists
|
|
|
|
kDIErrEOF = -30, // end-of-file reached
|
|
kDIErrReadFailed = -31,
|
|
kDIErrWriteFailed = -32,
|
|
kDIErrDataUnderrun = -33, // tried to read off end of the image
|
|
kDIErrDataOverrun = -34, // tried to write off end of the image
|
|
kDIErrGenericIO = -35, // generic I/O error
|
|
|
|
kDIErrOddLength = -40, // image size not multiple of sectors
|
|
kDIErrUnrecognizedFileFmt = -41, // file format just not recognized
|
|
kDIErrBadFileFormat = -42, // filename ext doesn't match contents
|
|
kDIErrUnsupportedFileFmt = -43, // recognized but not supported
|
|
kDIErrUnsupportedPhysicalFmt = -44, // (same)
|
|
kDIErrUnsupportedFSFmt = -45, // (and again)
|
|
kDIErrBadOrdering = -46, // requested sector ordering no good
|
|
kDIErrFilesystemNotFound = -47, // requested filesystem isn't there
|
|
kDIErrUnsupportedAccess = -48, // e.g. read sectors from blocks-only
|
|
kDIErrUnsupportedImageFeature = -49, // e.g. FDI image w/Amiga sectors
|
|
|
|
kDIErrInvalidTrack = -50, // request for invalid track number
|
|
kDIErrInvalidSector = -51, // request for invalid sector number
|
|
kDIErrInvalidBlock = -52, // request for invalid block number
|
|
kDIErrInvalidIndex = -53, // request with an invalid index
|
|
|
|
kDIErrDirectoryLoop = -60, // directory chain points into itself
|
|
kDIErrFileLoop = -61, // file sector or block alloc loops
|
|
kDIErrBadDiskImage = -62, // the FS on the disk image is damaged
|
|
kDIErrBadFile = -63, // bad file on disk image
|
|
kDIErrBadDirectory = -64, // bad dir on disk image
|
|
kDIErrBadPartition = -65, // bad partition on multi-part format
|
|
|
|
kDIErrFileArchive = -70, // file archive, not disk archive
|
|
kDIErrUnsupportedCompression = -71, // compression method is not supported
|
|
kDIErrBadChecksum = -72, // image file's checksum is bad
|
|
kDIErrBadCompressedData = -73, // data can't even be unpacked
|
|
kDIErrBadArchiveStruct = -74, // bad archive structure
|
|
|
|
kDIErrBadNibbleSectors = -80, // can't read sectors from this image
|
|
kDIErrSectorUnreadable = -81, // requested sector not readable
|
|
kDIErrInvalidDiskByte = -82, // invalid byte for encoding type
|
|
kDIErrBadRawData = -83, // couldn't get correct nibbles
|
|
|
|
kDIErrInvalidFileName = -90, // tried to create file with bad name
|
|
kDIErrDiskFull = -91, // no space left on disk
|
|
kDIErrVolumeDirFull = -92, // no more entries in volume dir
|
|
kDIErrInvalidCreateReq = -93, // CreateImage request was flawed
|
|
kDIErrTooBig = -94, // larger than we want to handle
|
|
|
|
/* higher-level errors */
|
|
kDIErrGeneric = -101,
|
|
kDIErrInternal = -102,
|
|
kDIErrMalloc = -103,
|
|
kDIErrInvalidArg = -104,
|
|
kDIErrNotSupported = -105, // feature not currently supported
|
|
kDIErrCancelled = -106, // an operation was cancelled by user
|
|
|
|
kDIErrNufxLibInitFailed = -110,
|
|
} DIError;
|
|
|
|
/* return a string describing the error */
|
|
DISKIMG_API const char* DIStrError(DIError dierr);
|
|
|
|
|
|
/* exact definition of off_t varies, so just define our own */
|
|
#ifdef _ULONGLONG_
|
|
typedef LONGLONG di_off_t;
|
|
#else
|
|
typedef off_t di_off_t;
|
|
#endif
|
|
|
|
/* common definition of "whence" for seeks */
|
|
typedef enum DIWhence {
|
|
kSeekSet = SEEK_SET,
|
|
kSeekCur = SEEK_CUR,
|
|
kSeekEnd = SEEK_END
|
|
};
|
|
|
|
/* try to load ASPI under Win2K; if successful, SPTI should be disabled */
|
|
const bool kAlwaysTryASPI = false;
|
|
/* ASPI device "filenames" look like "ASPI:x:y:z\" */
|
|
DISKIMG_API extern const char* kASPIDev;
|
|
|
|
/* some nibble-encoding constants */
|
|
const int kTrackLenNib525 = 6656;
|
|
const int kTrackLenNb2525 = 6384;
|
|
const int kTrackLenTrackStar525 = 6525; // max len of data in TS image
|
|
const int kTrackAllocSize = 6656; // max 5.25 nibble track len; for buffers
|
|
const int kTrackCount525 = 35; // expected #of tracks on 5.25 img
|
|
const int kMaxNibbleTracks525 = 40; // max #of tracks on 5.25 nibble img
|
|
const int kDefaultNibbleVolumeNum = 254;
|
|
const int kBlockSize = 512; // block size for DiskImg interfaces
|
|
const int kSectorSize = 256; // sector size (1/2 block)
|
|
const int kD13Length = 256 * 13 * 35; // length of a .d13 image
|
|
|
|
/* largest expanse we allow access to on a volume (8GB in 512-byte blocks) */
|
|
const long kVolumeMaxBlocks = 8*1024*(1024*1024 / kBlockSize);
|
|
|
|
/* largest .gz file we'll open (uncompressed size) */
|
|
const long kGzipMax = 32*1024*1024;
|
|
|
|
/* forward and external class definitions */
|
|
class DiskFS;
|
|
class A2File;
|
|
class A2FileDescr;
|
|
class GenericFD;
|
|
class OuterWrapper;
|
|
class ImageWrapper;
|
|
class CircularBufferAccess;
|
|
class ASPI;
|
|
class LinearBitmap;
|
|
|
|
|
|
/*
|
|
* Library-global data functions.
|
|
*
|
|
* This class is just a namespace clumper. Do not instantiate.
|
|
*/
|
|
class DISKIMG_API Global {
|
|
public:
|
|
// one-time DLL initialization; use SetDebugMsgHandler first
|
|
static DIError AppInit(void);
|
|
// one-time DLL cleanup
|
|
static DIError AppCleanup(void);
|
|
|
|
// return the DiskImg version number
|
|
static void GetVersion(long* pMajor, long* pMinor, long* pBug);
|
|
|
|
static bool GetAppInitCalled(void) { return fAppInitCalled; }
|
|
static bool GetHasSPTI(void);
|
|
static bool GetHasASPI(void);
|
|
|
|
// return a pointer to our global ASPI instance, or NULL
|
|
static ASPI* GetASPI(void) { return fpASPI; }
|
|
// shortcut for fpASPI->GetVersion()
|
|
static unsigned long GetASPIVersion(void);
|
|
|
|
// pointer to the debug message handler
|
|
typedef void (*DebugMsgHandler)(const char* file, int line, const char* msg);
|
|
static DebugMsgHandler gDebugMsgHandler;
|
|
|
|
static DebugMsgHandler SetDebugMsgHandler(DebugMsgHandler handler);
|
|
static void PrintDebugMsg(const char* file, int line, const char* fmt, ...)
|
|
#if defined(__GNUC__)
|
|
__attribute__ ((format(printf, 3, 4)))
|
|
#endif
|
|
;
|
|
|
|
private:
|
|
// no instantiation allowed
|
|
Global(void) {}
|
|
~Global(void) {}
|
|
|
|
// make sure app calls AppInit
|
|
static bool fAppInitCalled;
|
|
|
|
static ASPI* fpASPI;
|
|
};
|
|
|
|
extern bool gAllowWritePhys0; // ugh -- see Win32BlockIO.cpp
|
|
|
|
|
|
/*
|
|
* Disk I/O class, roughly equivalent to a GS/OS disk device driver.
|
|
*
|
|
* Abstracts away the file's source (file on disk, file in memory) and
|
|
* storage format (DOS order, ProDOS order, nibble). Will also cope
|
|
* with common disk compression and wrapper formats (Mac DiskCopy, 2MG,
|
|
* ShrinkIt, etc) if handed a file on disk.
|
|
*
|
|
* Images may be embedded within other images, e.g. UNIDOS and storage
|
|
* type $04 pascal volumes.
|
|
*
|
|
* THOUGHT: we need a list(?) of pointers from here back to the DiskFS
|
|
* so that any modifications here will "wake" the DiskFS and sub-volumes.
|
|
* We also need a "dirty" flag so things like CloseNufx can know not to
|
|
* re-do work when Closing after a Flush. Also DiskFS can alert us to
|
|
* any locally cached stuff, and we can tell them to flush everything.
|
|
* (Possibly useful when doing disk updates, so stuff can be trivially
|
|
* un-done. Currently CiderPress checks the filename manually after
|
|
* each write, but that's generally less reliable than having the knowledge
|
|
* contained in the DiskImg.)
|
|
*
|
|
* THOUGHT: need a ReadRawTrack that gets raw nibblized data. For a
|
|
* nibblized image it returns the data, for a sector image it generates
|
|
* the raw data.
|
|
*
|
|
* THOUGHT: we could reduce the risk of problems and increase performance
|
|
* for physical media with a "copy on write" scheme. We'd create a shadow
|
|
* array of modified blocks, and write them at Flush time. This would
|
|
* provide an instantaneous "revert" feature, and prevent formats like
|
|
* DiskCopy42 (which has a CRC in its header) from being inconsistent for
|
|
* long stretches.
|
|
*/
|
|
class DISKIMG_API DiskImg {
|
|
public:
|
|
// create DiskImg object
|
|
DiskImg(void);
|
|
virtual ~DiskImg(void);
|
|
|
|
/*
|
|
* Types describing an image file.
|
|
*
|
|
* The file itself is described by an external parameter ("file source")
|
|
* that is either the name of the file, a memory buffer, or an EFD
|
|
* (EmbeddedFileDescriptor).
|
|
*/
|
|
typedef enum { // format of the "wrapper wrapper"
|
|
kOuterFormatUnknown = 0,
|
|
kOuterFormatNone = 1, // (plain)
|
|
kOuterFormatCompress = 2, // .xx.Z
|
|
kOuterFormatGzip = 3, // .xx.gz
|
|
kOuterFormatBzip2 = 4, // .xx.bz2
|
|
kOuterFormatZip = 10, // .zip
|
|
} OuterFormat;
|
|
typedef enum { // format of the image "wrapper"
|
|
kFileFormatUnknown = 0,
|
|
kFileFormatUnadorned = 1, // .po, .do, ,nib, .raw, .d13
|
|
kFileFormat2MG = 2, // .2mg, .2img, $e0/0130
|
|
kFileFormatDiskCopy42 = 3, // .dsk/.disk, maybe .dc
|
|
kFileFormatDiskCopy60 = 4, // .dc6 (often just raw format)
|
|
kFileFormatDavex = 5, // $e0/8004
|
|
kFileFormatSim2eHDV = 6, // .hdv
|
|
kFileFormatTrackStar = 7, // .app (40-track or 80-track)
|
|
kFileFormatFDI = 8, // .fdi (5.25" or 3.5")
|
|
kFileFormatNuFX = 20, // .shk, .sdk, .bxy
|
|
kFileFormatDDD = 21, // .ddd
|
|
kFileFormatDDDDeluxe = 22, // $DD, .ddd
|
|
} FileFormat;
|
|
typedef enum { // format of the image data stream
|
|
kPhysicalFormatUnknown = 0,
|
|
kPhysicalFormatSectors = 1, // sequential 256-byte sectors (13/16/32)
|
|
kPhysicalFormatNib525_6656 = 2, // 5.25" disk ".nib" (6656 bytes/track)
|
|
kPhysicalFormatNib525_6384 = 3, // 5.25" disk ".nb2" (6384 bytes/track)
|
|
kPhysicalFormatNib525_Var = 4, // 5.25" disk (variable len, e.g. ".app")
|
|
} PhysicalFormat;
|
|
typedef enum { // sector ordering for "sector" format images
|
|
kSectorOrderUnknown = 0,
|
|
kSectorOrderProDOS = 1, // written as series of ProDOS blocks
|
|
kSectorOrderDOS = 2, // written as series of DOS sectors
|
|
kSectorOrderCPM = 3, // written as series of 1K CP/M blocks
|
|
kSectorOrderPhysical = 4, // written as un-interleaved sectors
|
|
kSectorOrderMax, // (used for array sizing)
|
|
} SectorOrder;
|
|
typedef enum { // main filesystem format (based on NuFX enum)
|
|
kFormatUnknown = 0,
|
|
kFormatProDOS = 1,
|
|
kFormatDOS33 = 2,
|
|
kFormatDOS32 = 3,
|
|
kFormatPascal = 4,
|
|
kFormatMacHFS = 5,
|
|
kFormatMacMFS = 6,
|
|
kFormatLisa = 7,
|
|
kFormatCPM = 8,
|
|
//kFormatCharFST
|
|
kFormatMSDOS = 10, // any FAT filesystem
|
|
//kFormatHighSierra
|
|
kFormatISO9660 = 12,
|
|
//kFormatAppleShare
|
|
kFormatRDOS33 = 20, // 16-sector RDOS disk
|
|
kFormatRDOS32 = 21, // 13-sector RDOS disk
|
|
kFormatRDOS3 = 22, // 13-sector RDOS disk converted to 16
|
|
// "generic" formats *must* be in their own "decade"
|
|
kFormatGenericPhysicalOrd = 30, // unknown, but physical-sector-ordered
|
|
kFormatGenericProDOSOrd = 31, // unknown, but ProDOS-block-ordered
|
|
kFormatGenericDOSOrd = 32, // unknown, but DOS-sector-ordered
|
|
kFormatGenericCPMOrd = 33, // unknown, but CP/M-block-ordered
|
|
kFormatUNIDOS = 40, // two 400K DOS 3.3 volumes
|
|
kFormatOzDOS = 41, // two 400K DOS 3.3 volumes, weird order
|
|
kFormatCFFA4 = 42, // CFFA image with 4 or 6 partitions
|
|
kFormatCFFA8 = 43, // CFFA image with 8 partitions
|
|
kFormatMacPart = 44, // Macintosh-style partitioned disk
|
|
kFormatMicroDrive = 45, // ///SHH Systeme's MicroDrive format
|
|
kFormatFocusDrive = 46, // Parsons Engineering FocusDrive format
|
|
kFormatGutenberg = 47, // Gutenberg word processor format
|
|
|
|
// try to keep this in an unsigned char, e.g. for CP clipboard
|
|
} FSFormat;
|
|
|
|
/*
|
|
* Nibble encode/decode description. Use no pointers here, so we
|
|
* store as an array and resize at will.
|
|
*
|
|
* Should we define an enum to describe whether address and data
|
|
* headers are standard or some wacky variant?
|
|
*/
|
|
typedef enum {
|
|
kNibbleAddrPrologLen = 3, // d5 aa 96
|
|
kNibbleAddrEpilogLen = 3, // de aa eb
|
|
kNibbleDataPrologLen = 3, // d5 aa ad
|
|
kNibbleDataEpilogLen = 3, // de aa eb
|
|
};
|
|
typedef enum {
|
|
kNibbleEncUnknown = 0,
|
|
kNibbleEnc44,
|
|
kNibbleEnc53,
|
|
kNibbleEnc62,
|
|
} NibbleEnc;
|
|
typedef enum {
|
|
kNibbleSpecialNone = 0,
|
|
kNibbleSpecialMuse, // doubled sector numbers on tracks > 2
|
|
kNibbleSpecialSkipFirstAddrByte,
|
|
} NibbleSpecial;
|
|
typedef struct {
|
|
char description[32];
|
|
short numSectors; // 13 or 16 (or 18?)
|
|
|
|
unsigned char addrProlog[kNibbleAddrPrologLen];
|
|
unsigned char addrEpilog[kNibbleAddrEpilogLen];
|
|
unsigned char addrChecksumSeed;
|
|
bool addrVerifyChecksum;
|
|
bool addrVerifyTrack;
|
|
int addrEpilogVerifyCount;
|
|
|
|
unsigned char dataProlog[kNibbleDataPrologLen];
|
|
unsigned char dataEpilog[kNibbleDataEpilogLen];
|
|
unsigned char dataChecksumSeed;
|
|
bool dataVerifyChecksum;
|
|
int dataEpilogVerifyCount;
|
|
|
|
NibbleEnc encoding;
|
|
NibbleSpecial special;
|
|
} NibbleDescr;
|
|
|
|
|
|
static inline bool IsSectorFormat(PhysicalFormat fmt) {
|
|
return (fmt == kPhysicalFormatSectors);
|
|
}
|
|
static inline bool IsNibbleFormat(PhysicalFormat fmt) {
|
|
return (fmt == kPhysicalFormatNib525_6656 ||
|
|
fmt == kPhysicalFormatNib525_6384 ||
|
|
fmt == kPhysicalFormatNib525_Var);
|
|
}
|
|
|
|
// file is on disk; stuff like 2MG headers will be identified and stripped
|
|
DIError OpenImage(const char* filename, char fssep, bool readOnly);
|
|
// file is in memory; provide a pointer to the data start and buffer size
|
|
DIError OpenImage(const void* buffer, long length, bool readOnly);
|
|
// file is a range of blocks on an open block-oriented disk image
|
|
DIError OpenImage(DiskImg* pParent, long firstBlock, long numBlocks);
|
|
// file is a range of tracks/sectors on an open sector-oriented disk image
|
|
DIError OpenImage(DiskImg* pParent, long firstTrack, long firstSector,
|
|
long numSectors);
|
|
|
|
// create a new, blank image file
|
|
DIError CreateImage(const char* pathName, const char* storageName,
|
|
OuterFormat outerFormat, FileFormat fileFormat,
|
|
PhysicalFormat physical, const NibbleDescr* pNibbleDescr,
|
|
SectorOrder order, FSFormat format,
|
|
long numBlocks, bool skipFormat);
|
|
DIError CreateImage(const char* pathName, const char* storageName,
|
|
OuterFormat outerFormat, FileFormat fileFormat,
|
|
PhysicalFormat physical, const NibbleDescr* pNibbleDescr,
|
|
SectorOrder order, FSFormat format,
|
|
long numTracks, long numSectPerTrack, bool skipFormat);
|
|
|
|
// flush any changes to disk; slow recompress only for "kFlushAll"
|
|
typedef enum { kFlushUnknown=0, kFlushFastOnly=1, kFlushAll=2 } FlushMode;
|
|
DIError FlushImage(FlushMode mode);
|
|
// close the image, freeing up any resources in use
|
|
DIError CloseImage(void);
|
|
// raise/lower refCnt (may want to track pointers someday)
|
|
void AddDiskFS(DiskFS* pDiskFS) { fDiskFSRefCnt++; }
|
|
void RemoveDiskFS(DiskFS* pDiskFS) {
|
|
assert(fDiskFSRefCnt > 0);
|
|
fDiskFSRefCnt--;
|
|
}
|
|
|
|
// (re-)format this image in the specified FS format
|
|
DIError FormatImage(FSFormat format, const char* volName);
|
|
// reset all blocks/sectors to zeroes
|
|
DIError ZeroImage(void);
|
|
|
|
// configure for paired sectors (OzDOS)
|
|
void SetPairedSectors(bool enable, int idx);
|
|
|
|
// identify sector ordering and disk format
|
|
// (may want a version that takes "hints" for special disks?)
|
|
DIError AnalyzeImage(void);
|
|
// figure out what FS and sector ordering is on the disk image
|
|
void AnalyzeImageFS(void);
|
|
bool ShowAsBlocks(void) const;
|
|
// overrule the analyzer (generally not recommended) -- does not
|
|
// override FileFormat, which is very reliable
|
|
DIError OverrideFormat(PhysicalFormat physical, FSFormat format,
|
|
SectorOrder order);
|
|
|
|
// Create a DiskFS that matches this DiskImg. Must be called after
|
|
// AnalayzeImage, or you will always get a DiskFSUnknown. The DiskFS
|
|
// must be freed with "delete" when no longer needed.
|
|
DiskFS* OpenAppropriateDiskFS(bool allowUnknown = false);
|
|
|
|
// Add information or a warning to the list of notes. Use linefeeds to
|
|
// indicate line breaks. This is currently append-only.
|
|
typedef enum { kNoteInfo, kNoteWarning } NoteType;
|
|
void AddNote(NoteType type, const char* fmt, ...)
|
|
#if defined(__GNUC__)
|
|
__attribute__ ((format(printf, 3, 4)))
|
|
#endif
|
|
;
|
|
const char* GetNotes(void) const;
|
|
|
|
// simple accessors
|
|
OuterFormat GetOuterFormat(void) const { return fOuterFormat; }
|
|
FileFormat GetFileFormat(void) const { return fFileFormat; }
|
|
PhysicalFormat GetPhysicalFormat(void) const { return fPhysical; }
|
|
SectorOrder GetSectorOrder(void) const { return fOrder; }
|
|
FSFormat GetFSFormat(void) const { return fFormat; }
|
|
long GetNumTracks(void) const { return fNumTracks; }
|
|
int GetNumSectPerTrack(void) const { return fNumSectPerTrack; }
|
|
long GetNumBlocks(void) const { return fNumBlocks; }
|
|
bool GetReadOnly(void) const { return fReadOnly; }
|
|
bool GetDirtyFlag(void) const { return fDirty; }
|
|
|
|
// set read-only flag; don't use this (open with correct setting;
|
|
// this was added as safety hack for the volume copier)
|
|
void SetReadOnly(bool val) { fReadOnly = val; }
|
|
|
|
// read a 256-byte sector
|
|
// NOTE to self: this function should not be available for odd-sized
|
|
// volumes, e.g. a ProDOS /RAM or /RAM5 stored with Davex. Need some way
|
|
// to communicate that to disk editor so it knows to grey-out the
|
|
// selection checkbox and/or not use "show as sectors" as default.
|
|
virtual DIError ReadTrackSector(long track, int sector, void* buf) {
|
|
return ReadTrackSectorSwapped(track, sector, buf, fOrder,
|
|
fFileSysOrder);
|
|
}
|
|
DIError ReadTrackSectorSwapped(long track, int sector,
|
|
void* buf, SectorOrder imageOrder, SectorOrder fsOrder);
|
|
// write a 256-byte sector
|
|
virtual DIError WriteTrackSector(long track, int sector, const void* buf);
|
|
|
|
// read a 512-byte block
|
|
virtual DIError ReadBlock(long block, void* buf) {
|
|
return ReadBlockSwapped(block, buf, fOrder, fFileSysOrder);
|
|
}
|
|
DIError ReadBlockSwapped(long block, void* buf, SectorOrder imageOrder,
|
|
SectorOrder fsOrder);
|
|
// read multiple blocks
|
|
virtual DIError ReadBlocks(long startBlock, int numBlocks, void* buf);
|
|
// check our virtual bad block map
|
|
bool CheckForBadBlocks(long startBlock, int numBlocks);
|
|
// write a 512-byte block
|
|
virtual DIError WriteBlock(long block, const void* buf);
|
|
// write multiple blocks
|
|
virtual DIError WriteBlocks(long startBlock, int numBlocks, const void* buf);
|
|
|
|
// read an entire nibblized track
|
|
virtual DIError ReadNibbleTrack(long track, unsigned char* buf,
|
|
long* pTrackLen);
|
|
// write a track; trackLen must be <= those in image
|
|
virtual DIError WriteNibbleTrack(long track, const unsigned char* buf,
|
|
long trackLen);
|
|
|
|
// save the current image as a 2MG file
|
|
//DIError Write2MG(const char* filename);
|
|
|
|
// need to treat the DOS volume number as meta-data for some disks
|
|
short GetDOSVolumeNum(void) const { return fDOSVolumeNum; }
|
|
void SetDOSVolumeNum(short val) { fDOSVolumeNum = val; }
|
|
enum { kVolumeNumNotSet = -1 };
|
|
|
|
// some simple getters
|
|
bool GetHasSectors(void) const { return fHasSectors; }
|
|
bool GetHasBlocks(void) const { return fHasBlocks; }
|
|
bool GetHasNibbles(void) const { return fHasNibbles; }
|
|
bool GetIsEmbedded(void) const { return fpParentImg != NULL; }
|
|
|
|
// return the current NibbleDescr
|
|
const NibbleDescr* GetNibbleDescr(void) const { return fpNibbleDescr; }
|
|
// set the NibbleDescr; we do this by copying the entry into our table
|
|
// (could improve by doing memcmp on available entries?)
|
|
void SetNibbleDescr(int idx);
|
|
void SetCustomNibbleDescr(const NibbleDescr* pDescr);
|
|
const NibbleDescr* GetNibbleDescrTable(int* pCount) const {
|
|
*pCount = fNumNibbleDescrEntries;
|
|
return fpNibbleDescrTable;
|
|
}
|
|
|
|
// set the NuFX compression type, used when compressing or re-compressing;
|
|
// must be set before image is opened or created
|
|
void SetNuFXCompressionType(int val) { fNuFXCompressType = val; }
|
|
|
|
/*
|
|
* Set up a progress callback to use when scanning a disk volume. Pass
|
|
* NULL for "func" to disable.
|
|
*
|
|
* The callback function is expected to return "true" if all is well.
|
|
* If it returns false, kDIErrCancelled will eventually come back.
|
|
*/
|
|
typedef bool (*ScanProgressCallback)(void* cookie, const char* str,
|
|
int count);
|
|
void SetScanProgressCallback(ScanProgressCallback func, void* cookie);
|
|
/* update status dialog during disk scan; called from DiskFS code */
|
|
bool UpdateScanProgress(const char* newStr);
|
|
|
|
/*
|
|
* Static utility functions.
|
|
*/
|
|
// returns "true" if the files on this image have DOS structure, i.e.
|
|
// simple file types and high-ASCII text files
|
|
static bool UsesDOSFileStructure(FSFormat format) {
|
|
return (format == kFormatDOS33 ||
|
|
format == kFormatDOS32 ||
|
|
format == kFormatGutenberg ||
|
|
format == kFormatUNIDOS ||
|
|
format == kFormatOzDOS ||
|
|
format == kFormatRDOS33 ||
|
|
format == kFormatRDOS32 ||
|
|
format == kFormatRDOS3);
|
|
}
|
|
// returns "true" if we can open files on the specified filesystem
|
|
static bool CanOpenFiles(FSFormat format) {
|
|
return (format == kFormatProDOS ||
|
|
format == kFormatDOS33 ||
|
|
format == kFormatDOS32 ||
|
|
format == kFormatPascal ||
|
|
format == kFormatCPM ||
|
|
format == kFormatRDOS33 ||
|
|
format == kFormatRDOS32 ||
|
|
format == kFormatRDOS3);
|
|
}
|
|
// returns "true" if we can create subdirectories on this filesystem
|
|
static bool IsHierarchical(FSFormat format) {
|
|
return (format == kFormatProDOS ||
|
|
format == kFormatMacHFS ||
|
|
format == kFormatMSDOS);
|
|
}
|
|
// returns "true" if we can create resource forks on this filesystem
|
|
static bool HasResourceForks(FSFormat format) {
|
|
return (format == kFormatProDOS ||
|
|
format == kFormatMacHFS);
|
|
}
|
|
// returns "true" if the format is one of the "generics"
|
|
static bool IsGenericFormat(FSFormat format) {
|
|
return (format / 10 == DiskImg::kFormatGenericDOSOrd / 10);
|
|
}
|
|
|
|
/* this must match DiskImg::kStdNibbleDescrs table */
|
|
typedef enum StdNibbleDescr {
|
|
kNibbleDescrDOS33Std = 0,
|
|
kNibbleDescrDOS33Patched,
|
|
kNibbleDescrDOS33IgnoreChecksum,
|
|
kNibbleDescrDOS32Std,
|
|
kNibbleDescrDOS32Patched,
|
|
kNibbleDescrMuse32,
|
|
kNibbleDescrRDOS33,
|
|
kNibbleDescrRDOS32,
|
|
kNibbleDescrCustom, // slot for custom entry
|
|
|
|
kNibbleDescrMAX // must be last
|
|
};
|
|
static const NibbleDescr* GetStdNibbleDescr(StdNibbleDescr idx);
|
|
// call this once, at DLL initialization time
|
|
static void CalcNibbleInvTables(void);
|
|
// calculate block number from cyl/head/sect on 3.5" disk
|
|
static int CylHeadSect35ToBlock(int cyl, int head, int sect);
|
|
// unpack nibble data from a 3.5" disk track
|
|
static DIError UnpackNibbleTrack35(const unsigned char* nibbleBuf,
|
|
long nibbleLen, unsigned char* outputBuf, int cyl, int head,
|
|
LinearBitmap* pBadBlockMap);
|
|
// compute the #of sectors per track for cylinder N (0-79)
|
|
static int SectorsPerTrack35(int cylinder);
|
|
|
|
// get the order in which we test for sector ordering
|
|
static void GetSectorOrderArray(SectorOrder* orderArray, SectorOrder first);
|
|
|
|
// utility function used by HFS filename normalizer; available to apps
|
|
static inline unsigned char MacToASCII(unsigned char uch) {
|
|
if (uch < 0x20)
|
|
return '?';
|
|
else if (uch < 0x80)
|
|
return uch;
|
|
else
|
|
return kMacHighASCII[uch - 0x80];
|
|
}
|
|
|
|
// Allow write access to physical disk 0. This is usually the boot disk,
|
|
// but with some BIOS the first IDE drive is always physical 0 even if
|
|
// you're booting from SATA. This only has meaning under Win32.
|
|
static void SetAllowWritePhys0(bool val);
|
|
|
|
/*
|
|
* Get string constants for enumerated values.
|
|
*/
|
|
typedef struct { int format; const char* str; } ToStringLookup;
|
|
static const char* ToStringCommon(int format, const ToStringLookup* pTable,
|
|
int tableSize);
|
|
static const char* ToString(OuterFormat format);
|
|
static const char* ToString(FileFormat format);
|
|
static const char* ToString(PhysicalFormat format);
|
|
static const char* ToString(SectorOrder format);
|
|
static const char* ToString(FSFormat format);
|
|
|
|
private:
|
|
/*
|
|
* Fundamental disk image identification.
|
|
*/
|
|
OuterFormat fOuterFormat; // e.g. gzip
|
|
FileFormat fFileFormat;
|
|
PhysicalFormat fPhysical;
|
|
const NibbleDescr* fpNibbleDescr; // only used for "nibble" images
|
|
SectorOrder fOrder; // only used for "sector" images
|
|
FSFormat fFormat;
|
|
|
|
/*
|
|
* This affects how the DiskImg responds to requests for reading
|
|
* a track or sector.
|
|
*
|
|
* "fFileSysOrder", together with with "fOrder", determines how
|
|
* sector numbers are translated. It describes the order that the
|
|
* DiskFS filesystem expects things to be in. If the image isn't
|
|
* sector-addressable, then it is assumed to be in linear block order.
|
|
*
|
|
* If "fSectorPairing" is set, the DiskImg treats the disk as if
|
|
* it were in OzDOS format, with one sector from two different
|
|
* volumes in a single 512-byte block.
|
|
*/
|
|
SectorOrder fFileSysOrder;
|
|
bool fSectorPairing;
|
|
int fSectorPairOffset; // which image (should be 0 or 1)
|
|
|
|
/*
|
|
* Internal state.
|
|
*/
|
|
GenericFD* fpOuterGFD; // outer wrapper, if any (.gz only)
|
|
GenericFD* fpWrapperGFD; // Apple II image file
|
|
GenericFD* fpDataGFD; // raw Apple II data
|
|
OuterWrapper* fpOuterWrapper; // needed for outer .gz wrapper
|
|
ImageWrapper* fpImageWrapper; // disk image wrapper (2MG, SHK, etc)
|
|
DiskImg* fpParentImg; // set for embedded volumes
|
|
short fDOSVolumeNum; // specified by some wrapper formats
|
|
di_off_t fOuterLength; // total len of file
|
|
di_off_t fWrappedLength; // len of file after Outer wrapper removed
|
|
di_off_t fLength; // len of disk image (w/o wrappers)
|
|
bool fExpandable; // ProDOS .hdv can expand
|
|
bool fReadOnly; // allow writes to this image?
|
|
bool fDirty; // have we modified this image?
|
|
//bool fIsEmbedded; // is this image embedded in another?
|
|
|
|
bool fHasSectors; // image is sector-addressable
|
|
bool fHasBlocks; // image is block-addressable
|
|
bool fHasNibbles; // image is nibble-addressable
|
|
|
|
long fNumTracks; // for sector-addressable images
|
|
int fNumSectPerTrack; // (ditto)
|
|
long fNumBlocks; // for 512-byte block-addressable images
|
|
|
|
unsigned char* fNibbleTrackBuf; // allocated on heap
|
|
int fNibbleTrackLoaded; // track currently in buffer
|
|
|
|
int fNuFXCompressType; // used when compressing a NuFX image
|
|
|
|
char* fNotes; // warnings and FYIs about DiskImg/DiskFS
|
|
|
|
LinearBitmap* fpBadBlockMap; // used for 3.5" nibble images
|
|
|
|
int fDiskFSRefCnt; // #of DiskFS objects pointing at us
|
|
|
|
/*
|
|
* NibbleDescr entries. There are several standard ones, and we want
|
|
* to allow applications to define additional ones.
|
|
*/
|
|
NibbleDescr* fpNibbleDescrTable;
|
|
int fNumNibbleDescrEntries;
|
|
|
|
/* static table of default values */
|
|
static const NibbleDescr kStdNibbleDescrs[];
|
|
|
|
DIError CreateImageCommon(const char* pathName, const char* storageName,
|
|
bool skipFormat);
|
|
DIError ValidateCreateFormat(void) const;
|
|
DIError FormatSectors(GenericFD* pGFD, bool quickFormat) const;
|
|
//DIError FormatBlocks(GenericFD* pGFD) const;
|
|
|
|
DIError CopyBytesOut(void* buf, di_off_t offset, int size) const;
|
|
DIError CopyBytesIn(const void* buf, di_off_t offset, int size);
|
|
DIError AnalyzeImageFile(const char* pathName, char fssep);
|
|
// Figure out the sector ordering for this filesystem, so we can decide
|
|
// how the sectors need to be re-arranged when we're reading them.
|
|
SectorOrder CalcFSSectorOrder(void) const;
|
|
// Handle sector order calculations.
|
|
DIError CalcSectorAndOffset(long track, int sector, SectorOrder ImageOrder,
|
|
SectorOrder fsOrder, di_off_t* pOffset, int* pNewSector);
|
|
inline bool IsLinearBlocks(SectorOrder imageOrder, SectorOrder fsOrder);
|
|
|
|
/*
|
|
* Progress update during the filesystem scan. This only exists in the
|
|
* topmost DiskImg. (This is arguably more appropriate in DiskFS, but
|
|
* DiskFS objects don't have a notion of "parent" and are somewhat more
|
|
* ephemeral.)
|
|
*/
|
|
ScanProgressCallback fpScanProgressCallback;
|
|
void* fScanProgressCookie;
|
|
int fScanCount;
|
|
char fScanMsg[128];
|
|
time_t fScanLastMsgWhen;
|
|
|
|
/*
|
|
* 5.25" nibble image access.
|
|
*/
|
|
enum {
|
|
kDataSize62 = 343, // 342 bytes + checksum byte
|
|
kChunkSize62 = 86, // (0x56)
|
|
|
|
kDataSize53 = 411, // 410 bytes + checksum byte
|
|
kChunkSize53 = 51, // (0x33)
|
|
kThreeSize = (kChunkSize53 * 3) + 1, // same as 410 - 256
|
|
};
|
|
DIError ReadNibbleSector(long track, int sector, void* buf,
|
|
const NibbleDescr* pNibbleDescr);
|
|
DIError WriteNibbleSector(long track, int sector, const void* buf,
|
|
const NibbleDescr* pNibbleDescr);
|
|
void DumpNibbleDescr(const NibbleDescr* pNibDescr) const;
|
|
int GetNibbleTrackLength(long track) const;
|
|
int GetNibbleTrackOffset(long track) const;
|
|
int GetNibbleTrackFormatLength(void) const {
|
|
/* return length to use when formatting for 16 sectors */
|
|
if (fPhysical == kPhysicalFormatNib525_6656)
|
|
return kTrackLenNib525;
|
|
else if (fPhysical == kPhysicalFormatNib525_6384)
|
|
return kTrackLenNb2525;
|
|
else if (fPhysical == kPhysicalFormatNib525_Var) {
|
|
if (fFileFormat == kFileFormatTrackStar ||
|
|
fFileFormat == kFileFormatFDI)
|
|
{
|
|
return kTrackLenNb2525; // use minimum possible
|
|
}
|
|
}
|
|
assert(false);
|
|
return -1;
|
|
}
|
|
int GetNibbleTrackAllocLength(void) const {
|
|
/* return length to allocate when creating an image */
|
|
if (fPhysical == kPhysicalFormatNib525_Var &&
|
|
(fFileFormat == kFileFormatTrackStar ||
|
|
fFileFormat == kFileFormatFDI))
|
|
{
|
|
// use maximum possible
|
|
return kTrackLenTrackStar525;
|
|
}
|
|
return GetNibbleTrackFormatLength();
|
|
}
|
|
DIError LoadNibbleTrack(long track, long* pTrackLen);
|
|
DIError SaveNibbleTrack(void);
|
|
int FindNibbleSectorStart(const CircularBufferAccess& buffer,
|
|
int track, int sector, const NibbleDescr* pNibbleDescr, int* pVol);
|
|
void DecodeAddr(const CircularBufferAccess& buffer, int offset,
|
|
short* pVol, short* pTrack, short* pSector, short* pChksum);
|
|
inline unsigned int ConvFrom44(unsigned char val1, unsigned char val2) {
|
|
return ((val1 << 1) | 0x01) & val2;
|
|
}
|
|
DIError DecodeNibbleData(const CircularBufferAccess& buffer, int idx,
|
|
unsigned char* sctBuf, const NibbleDescr* pNibbleDescr);
|
|
void EncodeNibbleData(const CircularBufferAccess& buffer, int idx,
|
|
const unsigned char* sctBuf, const NibbleDescr* pNibbleDescr) const;
|
|
DIError DecodeNibble62(const CircularBufferAccess& buffer, int idx,
|
|
unsigned char* sctBuf, const NibbleDescr* pNibbleDescr);
|
|
void EncodeNibble62(const CircularBufferAccess& buffer, int idx,
|
|
const unsigned char* sctBuf, const NibbleDescr* pNibbleDescr) const;
|
|
DIError DecodeNibble53(const CircularBufferAccess& buffer, int idx,
|
|
unsigned char* sctBuf, const NibbleDescr* pNibbleDescr);
|
|
void EncodeNibble53(const CircularBufferAccess& buffer, int idx,
|
|
const unsigned char* sctBuf, const NibbleDescr* pNibbleDescr) const;
|
|
int TestNibbleTrack(int track, const NibbleDescr* pNibbleDescr, int* pVol);
|
|
DIError AnalyzeNibbleData(void);
|
|
inline unsigned char Conv44(unsigned short val, bool first) const {
|
|
if (first)
|
|
return (val >> 1) | 0xaa;
|
|
else
|
|
return val | 0xaa;
|
|
}
|
|
DIError FormatNibbles(GenericFD* pGFD) const;
|
|
|
|
static const unsigned char kMacHighASCII[];
|
|
|
|
/*
|
|
* 3.5" nibble access
|
|
*/
|
|
static int FindNextSector35(const CircularBufferAccess& buffer, int start,
|
|
int cyl, int head, int* pSector);
|
|
static bool DecodeNibbleSector35(const CircularBufferAccess& buffer,
|
|
int start, unsigned char* sectorBuf, unsigned char* readChecksum,
|
|
unsigned char* calcChecksum);
|
|
static bool UnpackChecksum35(const CircularBufferAccess& buffer,
|
|
int offset, unsigned char* checksumBuf);
|
|
static void EncodeNibbleSector35(const unsigned char* sectorData,
|
|
unsigned char* outBuf);
|
|
|
|
/* static data tables */
|
|
static unsigned char kDiskBytes53[32];
|
|
static unsigned char kDiskBytes62[64];
|
|
static unsigned char kInvDiskBytes53[256];
|
|
static unsigned char kInvDiskBytes62[256];
|
|
enum { kInvInvalidValue = 0xff };
|
|
|
|
private: // some C++ stuff to block behavior we don't support
|
|
DiskImg& operator=(const DiskImg&);
|
|
DiskImg(const DiskImg&);
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
* Disk filesystem class, roughly equivalent to a GS/OS FST. This is an
|
|
* abstract base class.
|
|
*
|
|
* Static functions know how to access a DiskImg and figure out what kind
|
|
* of image we have. Once known, the appropriate sub-class can be
|
|
* instantiated.
|
|
*
|
|
* We maintain a linear list of files to make it easy for applications to
|
|
* traverse the full set of files. Sometimes this makes it hard for us to
|
|
* update internally (especially HFS). With some minor cleverness it
|
|
* should be possible to switch to a tree structure while retaining the
|
|
* linear "get next" API. This would be a big help for ProDOS and HFS.
|
|
*
|
|
* NEED: some notification mechanism for changes to files and/or block
|
|
* editing of the disk (especially with regard to open sub-volumes). If
|
|
* a disk volume open for file-by-file viewing is modified with the disk
|
|
* editor, we should close the file when the disk editor exits.
|
|
*
|
|
* NEED? disk utilities, such as "disk crunch", for Pascal volumes. Could
|
|
* be written externally, but might as well keep fs knowledge embedded.
|
|
*
|
|
* MISSING: there is no way to override the image analyzer when working
|
|
* with sub-volumes. Actually, there is; it just has to happen *after*
|
|
* the DiskFS has been created. We should provide an approach that either
|
|
* stifles the DiskFS creation, or allows us to override and replace the
|
|
* internal DiskFS so we can pop up a sub-volume list, show what we *think*
|
|
* is there, and then let the user pick a volume and pick overrides (mainly
|
|
* for use in the disk editor). Not sure if we want the changes to "stick";
|
|
* we probably do. Q: does the "scan for sub-volumes" attribute propagate
|
|
* recursively to each sub-sub-volume? Probably.
|
|
*
|
|
* NOTE to self: should make "test" functions more lax when called
|
|
* from here, on the assumption that the caller is knowledgeable. Perhaps
|
|
* an independent "strictness" variable that can be cranked down through
|
|
* multiple calls to AnalyzeImage??
|
|
*/
|
|
class DISKIMG_API DiskFS {
|
|
public:
|
|
/*
|
|
* Information about volume usage.
|
|
*
|
|
* Each "chunk" is a track/sector on a DOS disk or a block on a ProDOS
|
|
* or Pascal disk. CP/M really ought to use 1K blocks, but for
|
|
* convenience we're just using 512-byte blocks (it's up to the CP/M
|
|
* code to set two "chunks" per block).
|
|
*
|
|
* NOTE: the current DOS/ProDOS/Pascal code is sloppy when it comes to
|
|
* keeping this structure up to date. HFS doesn't use it at all. This
|
|
* has always been a low-priority feature.
|
|
*/
|
|
class DISKIMG_API VolumeUsage {
|
|
public:
|
|
VolumeUsage(void) {
|
|
fByBlocks = false;
|
|
fTotalChunks = -1;
|
|
fNumSectors = -1;
|
|
//fFreeChunks = -1;
|
|
fList = NULL;
|
|
fListSize = -1;
|
|
}
|
|
~VolumeUsage(void) {
|
|
delete[] fList;
|
|
}
|
|
|
|
/*
|
|
* These values MUST fit in five bits.
|
|
*
|
|
* Suggested disk map colors:
|
|
* 0 = unknown (color-key pink)
|
|
* 1 = conflict (medium-red)
|
|
* 2 = boot loader (dark-blue)
|
|
* 3 = volume dir (light-blue)
|
|
* 4 = subdir (medium-blue)
|
|
* 5 = user data (medium-green)
|
|
* 6 = user index blocks (light-green)
|
|
* 7 = embedded filesys (yellow)
|
|
*
|
|
* THOUGHT: Add flag for I/O error (nibble images) -- requires
|
|
* automatic disk verify pass. (Or maybe could be done manually
|
|
* on request?)
|
|
*
|
|
* unused --> black
|
|
* marked-used-but-not-used --> dark-red
|
|
* used-but-not-marked-used --> orange
|
|
*/
|
|
typedef enum {
|
|
kChunkPurposeUnknown = 0,
|
|
kChunkPurposeConflict = 1, // two or more different things
|
|
kChunkPurposeSystem = 2, // boot blocks, volume bitmap
|
|
kChunkPurposeVolumeDir = 3, // volume dir (or only dir)
|
|
kChunkPurposeSubdir = 4, // ProDOS sub-directory
|
|
kChunkPurposeUserData = 5, // file on this filesystem
|
|
kChunkPurposeFileStruct = 6, // index blocks, T/S lists
|
|
kChunkPurposeEmbedded = 7, // embedded filesystem
|
|
// how about: outside range claimed by disk, e.g. fTotalBlocks on
|
|
// 800K ProDOS disk in a 32MB CFFA volume?
|
|
} ChunkPurpose;
|
|
|
|
typedef struct ChunkState {
|
|
bool isUsed;
|
|
bool isMarkedUsed;
|
|
ChunkPurpose purpose; // only valid if isUsed is set
|
|
} ChunkState;
|
|
|
|
// initialize, configuring for either blocks or sectors
|
|
DIError Create(long numBlocks);
|
|
DIError Create(long numTracks, long numSectors);
|
|
bool GetInitialized(void) const { return (fList != NULL); }
|
|
|
|
// return the number of chunks on this disk
|
|
long GetNumChunks(void) const { return fTotalChunks; }
|
|
|
|
// return the number of unallocated chunks, taking into account
|
|
// both the free-chunk bitmap (if any) and the actual usage
|
|
long GetActualFreeChunks(void) const;
|
|
|
|
// return the state of the specified chunk
|
|
DIError GetChunkState(long block, ChunkState* pState) const;
|
|
DIError GetChunkState(long track, long sector,
|
|
ChunkState* pState) const;
|
|
|
|
// set the state of a particular chunk (should only be done by
|
|
// the DiskFS sub-classes)
|
|
DIError SetChunkState(long block, const ChunkState* pState);
|
|
DIError SetChunkState(long track, long sector,
|
|
const ChunkState* pState);
|
|
|
|
void Dump(void) const; // debugging
|
|
|
|
private:
|
|
DIError GetChunkStateIdx(int idx, ChunkState* pState) const;
|
|
DIError SetChunkStateIdx(int idx, const ChunkState* pState);
|
|
inline char StateToChar(ChunkState* pState) const;
|
|
|
|
/*
|
|
* Chunk state is stored as a set of bits in one byte:
|
|
*
|
|
* 0-4: how is block used (only has meaning if bit 6 is set)
|
|
* 5: for nibble images, indicates the block or sector is unreadable
|
|
* 6: is block used by something (0=no, 1=yes)
|
|
* 7: is block marked "in use" by system map (0=no, 1=yes)
|
|
*
|
|
* [ Consider reducing "purpose" to 0-3 and adding bad block bit for
|
|
* nibble images and physical media.]
|
|
*/
|
|
enum {
|
|
kChunkPurposeMask = 0x1f, // ChunkPurpose enum
|
|
kChunkDamagedFlag = 0x20,
|
|
kChunkUsedFlag = 0x40,
|
|
kChunkMarkedUsedFlag = 0x80,
|
|
};
|
|
|
|
bool fByBlocks;
|
|
long fTotalChunks;
|
|
long fNumSectors; // only valid if !fByBlocks
|
|
//long fFreeChunks;
|
|
unsigned char* fList;
|
|
int fListSize;
|
|
}; // end of VolumeUsage class
|
|
|
|
/*
|
|
* List of sub-volumes. The SubVolume owns the DiskImg and DiskFS
|
|
* that are handed to it, so they can be deleted when the SubVolume
|
|
* is deleted as part of destroying the parent.
|
|
*/
|
|
class SubVolume {
|
|
public:
|
|
SubVolume(void) : fpDiskImg(NULL), fpDiskFS(NULL),
|
|
fpPrev(NULL), fpNext(NULL) {}
|
|
~SubVolume(void) {
|
|
delete fpDiskFS; // must close first; may need flush to DiskImg
|
|
delete fpDiskImg;
|
|
}
|
|
|
|
void Create(DiskImg* pDiskImg, DiskFS* pDiskFS) {
|
|
assert(pDiskImg != NULL);
|
|
assert(pDiskFS != NULL);
|
|
fpDiskImg = pDiskImg;
|
|
fpDiskFS = pDiskFS;
|
|
}
|
|
|
|
DiskImg* GetDiskImg(void) const { return fpDiskImg; }
|
|
DiskFS* GetDiskFS(void) const { return fpDiskFS; }
|
|
|
|
SubVolume* GetPrev(void) const { return fpPrev; }
|
|
void SetPrev(SubVolume* pSubVol) { fpPrev = pSubVol; }
|
|
SubVolume* GetNext(void) const { return fpNext; }
|
|
void SetNext(SubVolume* pSubVol) { fpNext = pSubVol; }
|
|
|
|
private:
|
|
DiskImg* fpDiskImg;
|
|
DiskFS* fpDiskFS;
|
|
|
|
SubVolume* fpPrev;
|
|
SubVolume* fpNext;
|
|
}; // end of SubVolume class
|
|
|
|
|
|
|
|
/*
|
|
* Start of DiskFS declarations.
|
|
*/
|
|
public:
|
|
typedef enum SubScanMode {
|
|
kScanSubUnknown = 0,
|
|
kScanSubDisabled,
|
|
kScanSubEnabled,
|
|
kScanSubContainerOnly,
|
|
} SubScanMode;
|
|
|
|
|
|
DiskFS(void) {
|
|
fpA2Head = fpA2Tail = NULL;
|
|
fpSubVolumeHead = fpSubVolumeTail = NULL;
|
|
fpImg = NULL;
|
|
fScanForSubVolumes = kScanSubDisabled;
|
|
|
|
fParmTable[kParm_CreateUnique] = 0;
|
|
fParmTable[kParmProDOS_AllowLowerCase] = 1;
|
|
fParmTable[kParmProDOS_AllocSparse] = 1;
|
|
}
|
|
virtual ~DiskFS(void) {
|
|
DeleteSubVolumeList();
|
|
DeleteFileList();
|
|
SetDiskImg(NULL);
|
|
}
|
|
|
|
/*
|
|
* Static FSFormat-analysis functions, called by the DiskImg AnalyzeImage
|
|
* function. Capable of determining with a high degree of accuracy
|
|
* what format the disk is in, yet remaining flexible enough to
|
|
* function correctly with variations (like DOS3.3 disks with
|
|
* truncated TOCs and ProDOS images from hard drives).
|
|
*
|
|
* The "pOrder" and "pFormat" arguments are in/out. Set them to the
|
|
* appropriate "unknown" enum value on entry. If something is known
|
|
* of the order or format, put that in instead; in some cases this will
|
|
* bias the proceedings, which is useful for efficiency and for making
|
|
* overrides work correctly.
|
|
*
|
|
* On success, these return kDIErrNone and set "*pOrder". On failure,
|
|
* they return nonzero and leave "*pOrder" unmodified.
|
|
*
|
|
* Each DiskFS sub-class should declare a TestFS function. It's not
|
|
* virtual void here because, since it's called before the DiskFS is
|
|
* created, it must be static.
|
|
*/
|
|
typedef enum FSLeniency { kLeniencyNot, kLeniencyVery } FSLeniency;
|
|
//static DIError TestFS(const DiskImg* pImg, DiskImg::SectorOrder* pOrder,
|
|
// DiskImg::FSFormat* pFormat, FSLeniency leniency);
|
|
|
|
/*
|
|
* Load the disk contents and (if enabled) scan for sub-volumes.
|
|
*
|
|
* If "headerOnly" is set, we just do a quick scan of the volume header
|
|
* to get basic information. The deep file scan is skipped (but can
|
|
* be done later). (Sub-classes can choose to ignore the flag and
|
|
* always do the full scan; this is an optimization.) Guaranteed to
|
|
* set the volume name and volume block/sector count.
|
|
*
|
|
* If a progress callback is set up, this can return with a "cancelled"
|
|
* result, which should not be treated as a failure.
|
|
*/
|
|
typedef enum { kInitUnknown = 0, kInitHeaderOnly, kInitFull } InitMode;
|
|
virtual DIError Initialize(DiskImg* pImg, InitMode initMode) = 0;
|
|
|
|
/*
|
|
* Format the disk with the appropriate filesystem, creating all filesystem
|
|
* structures and (when appropriate) boot blocks.
|
|
*/
|
|
virtual DIError Format(DiskImg* pDiskImg, const char* volName)
|
|
{ return kDIErrNotSupported; }
|
|
|
|
/*
|
|
* Pass in a full path to normalize, and a buffer to copy the output
|
|
* into. On entry "pNormalizedBufLen" holds the length of the buffer.
|
|
* On exit, it holds the size of the buffer required to hold the
|
|
* normalized string. If the buffer is NULL or isn't big enough, no part
|
|
* of the normalized path will be copied into the buffer, and a specific
|
|
* error (kDIErrDataOverrun) will be returned.
|
|
*/
|
|
virtual DIError NormalizePath(const char* path, char fssep,
|
|
char* normalizedBuf, int* pNormalizedBufLen)
|
|
{ return kDIErrNotSupported; }
|
|
|
|
|
|
/*
|
|
* Create a file. The CreateParms struct specifies the full set of file
|
|
* details. To remain FS-agnostic, use the NufxLib constants
|
|
* (kNuStorageDirectory, kNuAccessUnlocked, etc). They match up with
|
|
* their ProDOS equivalents, and I promise to make them work right.
|
|
*
|
|
* On success, the file exists as a fully-formed, zero-length file. A
|
|
* pointer to the relevant A2File structure is returned.
|
|
*/
|
|
enum {
|
|
/* valid values for CreateParms; must match ProDOS enum */
|
|
kStorageSeedling = 1,
|
|
kStorageExtended = 5,
|
|
kStorageDirectory = 13,
|
|
};
|
|
typedef struct CreateParms {
|
|
const char* pathName; // full pathname for file on disk image
|
|
char fssep;
|
|
int storageType; // determines normal, subdir, or forked
|
|
long fileType;
|
|
long auxType;
|
|
int access;
|
|
time_t createWhen;
|
|
time_t modWhen;
|
|
} CreateParms;
|
|
virtual DIError CreateFile(const CreateParms* pParms, A2File** ppNewFile)
|
|
{ return kDIErrNotSupported; }
|
|
|
|
/*
|
|
* Delete a file from the disk.
|
|
*/
|
|
virtual DIError DeleteFile(A2File* pFile)
|
|
{ return kDIErrNotSupported; }
|
|
|
|
/*
|
|
* Rename a file.
|
|
*/
|
|
virtual DIError RenameFile(A2File* pFile, const char* newName)
|
|
{ return kDIErrNotSupported; }
|
|
|
|
/*
|
|
* Alter file attributes.
|
|
*/
|
|
virtual DIError SetFileInfo(A2File* pFile, long fileType, long auxType,
|
|
long accessFlags)
|
|
{ return kDIErrNotSupported; }
|
|
|
|
/*
|
|
* Rename a volume. Also works for changing the disk volume number.
|
|
*/
|
|
virtual DIError RenameVolume(const char* newName)
|
|
{ return kDIErrNotSupported; }
|
|
|
|
|
|
// Accessor
|
|
DiskImg* GetDiskImg(void) const { return fpImg; }
|
|
|
|
// Test file and volume names (and volume numbers)
|
|
// [these need to be static functions for some things... hmm]
|
|
//virtual bool IsValidFileName(const char* name) const { return false; }
|
|
//virtual bool IsValidVolumeName(const char* name) const { return false; }
|
|
|
|
// Return the disk volume name, or NULL if there isn't one.
|
|
virtual const char* GetVolumeName(void) const = 0;
|
|
|
|
// Return a printable string identifying the FS type and volume
|
|
virtual const char* GetVolumeID(void) const = 0;
|
|
|
|
// Return the "bare" volume name. For formats where the volume name
|
|
// is actually a number (e.g. DOS 3.3), this returns just the number.
|
|
// For formats without a volume name or number (e.g. CP/M), this returns
|
|
// NULL, indicating that any attempt to change the volume name will fail.
|
|
virtual const char* GetBareVolumeName(void) const = 0;
|
|
|
|
// Returns "false" if we only support read-only access to this FS type
|
|
virtual bool GetReadWriteSupported(void) const = 0;
|
|
|
|
// Returns "true" if the filesystem shows evidence of damage.
|
|
virtual bool GetFSDamaged(void) const = 0;
|
|
|
|
// Returns number of blocks recognized by the filesystem, or -1 if the
|
|
// FS isn't block-oriented (e.g. DOS 3.2/3.3)
|
|
virtual long GetFSNumBlocks(void) const { return -1; }
|
|
|
|
// Get the next file in the list. Start by passing in NULL to get the
|
|
// head of the list. Returns NULL when the end of the list is reached.
|
|
A2File* GetNextFile(A2File* pCurrent) const;
|
|
|
|
// Get a count of the files and directories on this disk.
|
|
long GetFileCount(void) const;
|
|
|
|
/*
|
|
* Find a file by case-insensitive pathname. Assumes fssep=':'. The
|
|
* compare function can be overridden for systems like HFS, where "case
|
|
* insensitive" has a different meaning because of the native
|
|
* character set.
|
|
*
|
|
* The A2File* returned should not be deleted.
|
|
*/
|
|
typedef int (*StringCompareFunc)(const char* str1, const char* str2);
|
|
A2File* GetFileByName(const char* pathName, StringCompareFunc func = NULL);
|
|
|
|
// This controls scanning for sub-volumes; must be set before Initialize().
|
|
SubScanMode GetScanForSubVolumes(void) const { return fScanForSubVolumes; }
|
|
void SetScanForSubVolumes(SubScanMode val) { fScanForSubVolumes = val; }
|
|
|
|
// some constants for non-ProDOS filesystems to use
|
|
enum { kFileAccessLocked = 0x01, kFileAccessUnlocked = 0xc3 };
|
|
|
|
|
|
/*
|
|
* We use this as a filename separator character (i.e. between pathname
|
|
* components) in all filenames. It's useful to standardize on this
|
|
* so that behavior is consistent across all disk and archive formats.
|
|
*
|
|
* The choice of ':' is good because it's invalid in filenames on
|
|
* Windows, Mac OS, GS/OS, and pretty much anywhere else we could be
|
|
* running except for UNIX. It's valid under DOS 3.3, but since you
|
|
* can't have subdirectories under DOS there's no risk of confusion.
|
|
*/
|
|
enum { kDIFssep = ':' };
|
|
|
|
|
|
/*
|
|
* Return the volume use map. This is a non-const function because
|
|
* it might need to do a "just-in-time" usage map update. It returns
|
|
* const to keep non-DiskFS classes from altering the map.
|
|
*/
|
|
const VolumeUsage* GetVolumeUsageMap(void) {
|
|
if (fVolumeUsage.GetInitialized())
|
|
return &fVolumeUsage;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Return the total space and free space, in either blocks or sectors
|
|
* as appropriate. "*pUnitSize" will be kBlockSize or kSectorSize.
|
|
*/
|
|
virtual DIError GetFreeSpaceCount(long* pTotalUnits, long* pFreeUnits,
|
|
int* pUnitSize) const = 0;
|
|
|
|
|
|
/*
|
|
* Get the next volume in the list. Start by passing in NULL to get the
|
|
* head of the list. Returns NULL when the end of the list is reached.
|
|
*/
|
|
SubVolume* GetNextSubVolume(const SubVolume* pCurrent) const;
|
|
|
|
/*
|
|
* Set some parameters to tell the DiskFS how to operate. Some of
|
|
* these are FS-specific, some may be general.
|
|
*
|
|
* Parameters are set in the current object and all sub-volume objects.
|
|
*
|
|
* The enum is part of the interface and must be rigidly defined, but
|
|
* it is also used to size an array so it can't be too sparse.
|
|
*/
|
|
typedef enum DiskFSParameter {
|
|
kParmUnknown = 0,
|
|
|
|
kParm_CreateUnique = 1, // make new filenames unique
|
|
|
|
kParmProDOS_AllowLowerCase = 10, // allow lower case and spaces
|
|
kParmProDOS_AllocSparse = 11, // don't store empty blocks
|
|
|
|
kParmMax // must be last entry
|
|
} DiskFSParameter;
|
|
long GetParameter(DiskFSParameter parm);
|
|
void SetParameter(DiskFSParameter parm, long val);
|
|
|
|
/*
|
|
* Flush changed data.
|
|
*
|
|
* The individual filesystems shouldn't generally do any caching; if
|
|
* they do, we would want a virtual "FlushFS()" that gets called by
|
|
* Flush. The better answer is to cache in DiskImg, which works for
|
|
* everything, and knows if the underlying storage is already in RAM.
|
|
*
|
|
* For the most part this just needs to recursively flush the DiskImg
|
|
* objects in all of the sub-volumes and then the current volume. This
|
|
* is a no-op in most cases, but if the archive is compressed this will
|
|
* cause a new, compressed archive to be created.
|
|
*
|
|
* The "mode" value determines whether or not we do "heavy" flushes. It's
|
|
* very handy to be able to do "slow" flushes for anything that is being
|
|
* written directly to disk (as opposed to being run through Deflate),
|
|
* so that the UI doesn't indicate that they're partially written when
|
|
* in fact they're fully written.
|
|
*/
|
|
DIError Flush(DiskImg::FlushMode mode);
|
|
|
|
/*
|
|
* Set the read-only flag on our DiskImg and those of our subvolumes.
|
|
* Used to ensure that a DiskFS with un-flushed data can be deleted
|
|
* without corrupting the volume.
|
|
*/
|
|
void SetAllReadOnly(bool val);
|
|
|
|
// debug dump
|
|
void DumpFileList(void);
|
|
|
|
protected:
|
|
/*
|
|
* Set the DiskImg pointer. Updates the reference count in DiskImg.
|
|
*/
|
|
void SetDiskImg(DiskImg* pImg);
|
|
|
|
// once added, we own the pDiskImg and the pDiskFS (DO NOT pass the
|
|
// same DiskImg or DiskFS in more than once!). Note this copies the
|
|
// fParmTable and other stuff (fScanForSubVolumes) from parent to child.
|
|
void AddSubVolumeToList(DiskImg* pDiskImg, DiskFS* pDiskFS);
|
|
// add files to fpA2Head/fpA2Tail
|
|
void AddFileToList(A2File* pFile);
|
|
// only need for hierarchical filesystems; insert file after pPrev
|
|
void InsertFileInList(A2File* pFile, A2File* pPrev);
|
|
// delete an entry
|
|
void DeleteFileFromList(A2File* pFile);
|
|
|
|
// scan for damaged or suspicious files
|
|
void ScanForDamagedFiles(bool* pDamaged, bool* pSuspicious);
|
|
|
|
// pointer to the DiskImg structure underlying this filesystem
|
|
DiskImg* fpImg;
|
|
|
|
VolumeUsage fVolumeUsage;
|
|
SubScanMode fScanForSubVolumes;
|
|
|
|
|
|
private:
|
|
A2File* SkipSubdir(A2File* pSubdir);
|
|
void CopyInheritables(DiskFS* pNewFS);
|
|
void DeleteFileList(void);
|
|
void DeleteSubVolumeList(void);
|
|
|
|
long fParmTable[kParmMax]; // for DiskFSParameter
|
|
|
|
A2File* fpA2Head;
|
|
A2File* fpA2Tail;
|
|
SubVolume* fpSubVolumeHead;
|
|
SubVolume* fpSubVolumeTail;
|
|
|
|
private:
|
|
DiskFS& operator=(const DiskFS&);
|
|
DiskFS(const DiskFS&);
|
|
};
|
|
|
|
|
|
/*
|
|
* Apple II file class, representing a file on an Apple II volume. This is an
|
|
* abstract base class.
|
|
*
|
|
* There is a different sub-class for each filesystem type. The A2File object
|
|
* encapsulates all of the knowledge required to read a file from a disk
|
|
* image.
|
|
*
|
|
* The prev/next pointers, used to maintain a linked list of files, are only
|
|
* accessible from DiskFS functions. At some point we may want to rearrange
|
|
* the way this is handled, e.g. by not maintaining a list at all, so it's
|
|
* important that everything go through DiskFS requests.
|
|
*
|
|
* The FSFormat is made an explicit member, because sub-classes may not
|
|
* understand exactly where the file came from (e.g. was it DOS3.2 or
|
|
* DOS 3.3). Somebody might care.
|
|
*
|
|
*
|
|
* NOTE TO SELF: open files need to be generalized. Right now the internal
|
|
* implementations only allow a single open, which is okay for our purposes
|
|
* but bad for a general FS implementation. As it stands, you can't even
|
|
* open both forks at the same time on ProDOS/HFS.
|
|
*
|
|
* UMMM: The handling of "damaged" files could be more consistent.
|
|
*/
|
|
class DISKIMG_API A2File {
|
|
public:
|
|
friend class DiskFS;
|
|
|
|
A2File(DiskFS* pDiskFS) : fpDiskFS(pDiskFS) {
|
|
fpPrev = fpNext = NULL;
|
|
fFileQuality = kQualityGood;
|
|
}
|
|
virtual ~A2File(void) {}
|
|
|
|
/*
|
|
* All Apple II files have certain characteristics, of which ProDOS
|
|
* is roughly a superset. (Yes, you can have HFS on a IIgs, but
|
|
* all that fancy creator type stuff is decidedly Mac-centric. Still,
|
|
* we want to assume 4-byte file and aux types.)
|
|
*
|
|
* NEED: something distinguishing between files and disk images?
|
|
*
|
|
* NOTE: there is no guarantee that GetPathName will return unique values
|
|
* (duplicates are possible). We don't guarantee that you won't get an
|
|
* empty string back (it's valid to have an empty filename in the dir in
|
|
* DOS 3.3, and it's possible for other filesystems to be damaged). The
|
|
* pathname may receive some minor sanitizing, e.g. removal or conversion
|
|
* of high ASCII and control characters, but some filesystems (like HFS)
|
|
* make use of them.
|
|
*
|
|
* We do guarantee that the contents of subdirectories are grouped
|
|
* together. This makes it much easier to construct a hierarchy out of
|
|
* the linear list. This becomes important when dynamically adding
|
|
* files to a disk.
|
|
*/
|
|
virtual const char* GetFileName(void) const = 0; // name of this file
|
|
virtual const char* GetPathName(void) const = 0; // full path
|
|
virtual char GetFssep(void) const = 0; // '\0' if none
|
|
virtual long GetFileType(void) const = 0;
|
|
virtual long GetAuxType(void) const = 0;
|
|
virtual long GetAccess(void) const = 0; // ProDOS-style perms
|
|
virtual time_t GetCreateWhen(void) const = 0;
|
|
virtual time_t GetModWhen(void) const = 0;
|
|
virtual di_off_t GetDataLength(void) const = 0; // len of data fork
|
|
virtual di_off_t GetDataSparseLength(void) const = 0; // len w/o sparse areas
|
|
virtual di_off_t GetRsrcLength(void) const = 0; // len or -1 if no rsrc
|
|
virtual di_off_t GetRsrcSparseLength(void) const = 0; // len or -1 if no rsrc
|
|
virtual DiskImg::FSFormat GetFSFormat(void) const {
|
|
return fpDiskFS->GetDiskImg()->GetFSFormat();
|
|
}
|
|
virtual bool IsDirectory(void) const { return false; }
|
|
virtual bool IsVolumeDirectory(void) const { return false; }
|
|
|
|
/*
|
|
* Open a file. Treat the A2FileDescr like an fd.
|
|
*/
|
|
virtual DIError Open(A2FileDescr** ppOpenFile, bool readOnly,
|
|
bool rsrcFork = false) = 0;
|
|
|
|
/*
|
|
* This is called by the A2FileDescr object when somebody invokes Close().
|
|
* The A2File object should remove the A2FileDescr from its list of open
|
|
* files and delete the storage. The implementation must not call the
|
|
* A2FileDescr's Close function, since that would cause a recursive loop.
|
|
*
|
|
* This should not be called by an application.
|
|
*/
|
|
virtual void CloseDescr(A2FileDescr* pOpenFile) = 0;
|
|
|
|
/*
|
|
* This is only useful for hierarchical filesystems like ProDOS,
|
|
* where the order of items in the linear list is significant. It
|
|
* allows an unambiguous determination of which subdir a file resides
|
|
* in, even if somebody has sector-edited the filesystem so that two
|
|
* subdirs have the same name. (It's also a bit speedier to compare
|
|
* than pathname substrings would be.)
|
|
*/
|
|
virtual A2File* GetParent(void) const { return NULL; }
|
|
|
|
/*
|
|
* Returns "true" if either fork of the file is open, "false" if not.
|
|
*/
|
|
virtual bool IsFileOpen(void) const = 0;
|
|
|
|
virtual void Dump(void) const = 0; // debugging
|
|
|
|
typedef enum FileQuality {
|
|
kQualityUnknown = 0,
|
|
kQualityGood,
|
|
kQualitySuspicious,
|
|
kQualityDamaged,
|
|
} FileQuality;
|
|
virtual FileQuality GetQuality(void) const { return fFileQuality; }
|
|
virtual void SetQuality(FileQuality quality);
|
|
virtual void ResetQuality(void);
|
|
|
|
DiskFS* GetDiskFS(void) const { return fpDiskFS; }
|
|
|
|
protected:
|
|
DiskFS* fpDiskFS;
|
|
virtual void SetParent(A2File* pParent) { /* do nothing */ }
|
|
|
|
private:
|
|
A2File* GetPrev(void) const { return fpPrev; }
|
|
void SetPrev(A2File* pFile) { fpPrev = pFile; }
|
|
A2File* GetNext(void) const { return fpNext; }
|
|
void SetNext(A2File* pFile) { fpNext = pFile; }
|
|
|
|
// Set when file structure is damaged and application should not try
|
|
// to open the file.
|
|
FileQuality fFileQuality;
|
|
|
|
A2File* fpPrev;
|
|
A2File* fpNext;
|
|
|
|
|
|
private:
|
|
A2File& operator=(const A2File&);
|
|
A2File(const A2File&);
|
|
};
|
|
|
|
|
|
/*
|
|
* Abstract representation of an open file. This contains all active state
|
|
* and all information required to read and write a file.
|
|
*
|
|
* Do not delete these objects; instead, invoke the Close method, so that they
|
|
* can be removed from the parents' list of open files.
|
|
* TODO: consider making the destructor "protected"
|
|
*/
|
|
class DISKIMG_API A2FileDescr {
|
|
public:
|
|
A2FileDescr(A2File* pFile) : fpFile(pFile), fProgressUpdateFunc(NULL) {}
|
|
virtual ~A2FileDescr(void) { fpFile = NULL; /*paranoia*/ }
|
|
|
|
virtual DIError Read(void* buf, size_t len, size_t* pActual = NULL) = 0;
|
|
virtual DIError Write(const void* buf, size_t len, size_t* pActual = NULL) = 0;
|
|
virtual DIError Seek(di_off_t offset, DIWhence whence) = 0;
|
|
virtual di_off_t Tell(void) = 0;
|
|
virtual DIError Close(void) = 0;
|
|
|
|
virtual DIError GetStorage(long sectorIdx, long* pTrack, long* pSector)
|
|
const = 0;
|
|
virtual DIError GetStorage(long blockIdx, long* pBlock)
|
|
const = 0;
|
|
virtual long GetSectorCount(void) const = 0;
|
|
virtual long GetBlockCount(void) const = 0;
|
|
|
|
virtual DIError Rewind(void) { return Seek(0, kSeekSet); }
|
|
|
|
A2File* GetFile(void) const { return fpFile; }
|
|
|
|
/*
|
|
* Progress update callback mechanism. Pass in the length or (for writes)
|
|
* expected length of the file. This invokes the callback with the
|
|
* lengths and some pointers.
|
|
*
|
|
* If the progress callback returns "true", progress continues. If it
|
|
* returns "false", the read or write function will return with
|
|
* kDIErrCancelled.
|
|
*/
|
|
typedef bool (*ProgressUpdater)(A2FileDescr* pFile, di_off_t max,
|
|
di_off_t current, void* vState);
|
|
void SetProgressUpdater(ProgressUpdater func, di_off_t max, void* state) {
|
|
fProgressUpdateFunc = func;
|
|
fProgressUpdateMax = max;
|
|
fProgressUpdateState = state;
|
|
}
|
|
void ClearProgressUpdater(void) {
|
|
fProgressUpdateFunc = NULL;
|
|
}
|
|
|
|
protected:
|
|
A2File* fpFile;
|
|
|
|
/*
|
|
* Internal utility functions for mapping blocks to sectors and vice-versa.
|
|
*/
|
|
virtual void TrackSectorToBlock(long track, long sector, long* pBlock,
|
|
bool* pSecondHalf) const
|
|
{
|
|
int numSectPerTrack = fpFile->GetDiskFS()->GetDiskImg()->GetNumSectPerTrack();
|
|
assert(track < fpFile->GetDiskFS()->GetDiskImg()->GetNumTracks());
|
|
assert(sector < numSectPerTrack);
|
|
long dblBlock = track * numSectPerTrack + sector;
|
|
*pBlock = dblBlock / 2;
|
|
*pSecondHalf = (dblBlock & 0x01) != 0;
|
|
}
|
|
virtual void BlockToTrackSector(long block, bool secondHalf, long* pTrack,
|
|
long* pSector) const
|
|
{
|
|
assert(block < fpFile->GetDiskFS()->GetDiskImg()->GetNumBlocks());
|
|
int numSectPerTrack = fpFile->GetDiskFS()->GetDiskImg()->GetNumSectPerTrack();
|
|
int dblBlock = block * 2;
|
|
if (secondHalf)
|
|
dblBlock++;
|
|
*pTrack = dblBlock / numSectPerTrack;
|
|
*pSector = dblBlock % numSectPerTrack;
|
|
}
|
|
|
|
/*
|
|
* Call this from FS-specific read/write functions on successful
|
|
* completion (and perhaps more often for filesystems with potentially
|
|
* large files, e.g. ProDOS/HFS).
|
|
*
|
|
* Test the return value; if "false", user wishes to cancel the op, and
|
|
* long read or write calls should return immediately.
|
|
*/
|
|
inline bool UpdateProgress(di_off_t current) {
|
|
if (fProgressUpdateFunc != NULL) {
|
|
return (*fProgressUpdateFunc)(this, fProgressUpdateMax, current,
|
|
fProgressUpdateState);
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
private:
|
|
A2FileDescr& operator=(const A2FileDescr&);
|
|
A2FileDescr(const A2FileDescr&);
|
|
|
|
/* storage for progress update goodies */
|
|
ProgressUpdater fProgressUpdateFunc;
|
|
di_off_t fProgressUpdateMax;
|
|
void* fProgressUpdateState;
|
|
};
|
|
|
|
}; // namespace DiskImgLib
|
|
|
|
#endif /*DISKIMG_DISKIMG_H*/
|