scummvm/common/formats/prodos.h

216 lines
8.8 KiB
C
Raw Permalink Normal View History

/* ScummVM - Graphic Adventure Engine
*
* ScummVM is the legal property of its developers, whose names
* are too numerous to list here. Please refer to the COPYRIGHT
* file distributed with this source distribution.
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
#ifndef COMMON_PRODOS_H
#define COMMON_PRODOS_H
#include "common/memstream.h"
#include "common/file.h"
#include "common/debug.h"
#include "common/error.h"
2022-06-21 17:22:56 -04:00
/* Quick note about ProDOS:
* This disk code handles inactive, seedling, sapling, tree, and subdirectory files.
* It does *not* handle sparse files at the moment. If a sparse file exists, it may not
* be read correctly. It also does not do anything with Pascal files, but those should not
* matter for game engines anyway.
*/
namespace Common {
2022-06-21 17:22:56 -04:00
// These values define for ProDOS how to read the file entry, and also whether it's a keyblock (if it is a directory header, it's the keyblock of that directory)
enum FileType : char {
2022-07-12 01:09:34 -04:00
kFileTypeDead = 0,
kFileTypeSeed = 1,
kFileTypeSapling = 2,
kFileTypeTree = 3,
kFileTypePascal = 4,
kFileTypeSubDir = 0x0D,
kFileTypeSubHead = 0x0E,
kFileTypeVolHead = 0x0F
};
2022-06-21 17:22:56 -04:00
/* File extensions for all the ProDOS supported file types
* NOTE: ProDOS user defined files are F1-F8. If they are ever required,
* they can be added to this enum.
*/
enum FileExt {
2022-07-12 01:09:34 -04:00
kFileExtNull = 0,
kFileExtBad = 1,
kFileExtTxt = 4,
kFileExtBin = 6,
kFileExtGfx = 8,
kFileExtDir = 0xF,
kFileExtDB = 0x19,
kFileExtWord = 0x1A,
kFileExtSpread = 0x1B,
kFileExtSTART = 0xB3,
kFileExtPascal = 0xEF,
kFileExtPDCI = 0xF0,
kFileExtPDRes = 0xF9,
kFileExtIBProg = 0xFA,
kFileExtIBVar = 0xFB,
kFileExtAPSProg = 0xFC,
kFileExtAPSVar = 0xFD,
kFileExtEDASM = 0xFE,
kFileExtSYS = 0xFF
};
2022-06-21 17:22:56 -04:00
/* A ProDOS file simply contains meta data about the file and the ability to
* find and put together the data blocks that make up the file contents.
* This implements Common::ArchiveMember so that it can be used directly in
2022-06-21 17:22:56 -04:00
* the Archive methods in ProDOSDisk.
*/
2022-06-21 17:22:56 -04:00
class ProDOSFile : public Common::ArchiveMember {
public:
2022-07-12 01:09:34 -04:00
ProDOSFile(char name[16], uint8 type, uint16 tBlk, uint32 eof, uint16 bPtr, Common::File *disk);
~ProDOSFile() {}; // File does not need a destructor, because the file it reads from is a pointer to Disk, and Disk has a destructor
2022-07-12 01:09:34 -04:00
// -- These are the Common::ArchiveMember related functions --
Common::String getName() const override; // Returns _name
Common::SeekableReadStream *createReadStream() const override; // This is what the archive needs to create a file
void getDataBlock(byte *memOffset, int offset, int size) const; // Gets data up to the size of a single data block (512 bytes)
int parseIndexBlock(byte *memOffset, int blockNum, int cSize) const; // Uses getDataBlock() on every pointer in the index file, adding them to byte * memory block
2022-07-12 01:09:34 -04:00
// For debugging purposes, just prints the metadata
void printInfo();
private:
char _name[16];
uint8 _type; // As defined by enum FileType
uint16 _blockPtr; // Block index in volume of index block or data
uint16 _totalBlocks;
uint32 _eof; // End Of File, used generally as size (exception being sparse files)
Common::File *_disk; // This is a pointer because it is the same _disk as in ProDosDisk, passed to the file object
};
/* This class defines the entire disk volume. Upon using the open() method,
* it will parse the volume and add them to a hashmap where the file path
* returns a file object. This implements Common::Archive to allow regular
* file operations to work with it.
*/
2022-06-21 17:22:56 -04:00
class ProDOSDisk : public Common::Archive {
public:
static const int kBlockSize = 512; // A ProDOS block is always 512 bytes (should this be an enum?)
2022-07-12 01:09:34 -04:00
ProDOSDisk(const Common::String filename);
~ProDOSDisk(); // Frees the memory used in the dictionary and the volume bitmap
2022-07-12 01:09:34 -04:00
// Called from the constructor, it parses the volume and fills the hashmap with files
bool open(const Common::String filename);
2022-07-12 01:09:34 -04:00
// These are the Common::Archive related methods
bool hasFile(const Common::Path &path) const override;
int listMembers(Common::ArchiveMemberList &list) const override;
const Common::ArchiveMemberPtr getMember(const Common::Path &path) const override;
Common::SeekableReadStream *createReadStreamForMember(const Common::Path &path) const override;
private:
byte _loader1[kBlockSize]; // There's not much reason for these to be needed, but I included them just in case
byte _loader2[kBlockSize];
Common::String _name; // Name of volume
Common::File _disk; // The volume file itself
int _volBlocks; // Total blocks in volume
byte *_volBitmap; // This can determine if the volume is corrupt as it contains a bit for every block, where 0 = unused, 1 = used
Common::HashMap<Common::String, Common::SharedPtr<ProDOSFile>> _files; // Hashmap of files in the volume, where key=Path, Value=ProDOSFile
2022-07-12 01:09:34 -04:00
struct Date {
uint8 _day;
uint8 _month;
uint8 _year;
2022-07-12 01:09:34 -04:00
};
struct Time {
uint8 _hour;
uint8 _minute;
2022-07-12 01:09:34 -04:00
};
struct VolHeader {
uint8 _type; // Not really important for a volume header, as this will always be F
uint8 _nameLen;
char _name[16];
byte _reserved[8]; // Extra space reserved for possible future uses, not important
Date _date;
Time _time;
uint8 _ver;
uint8 _minVer; // Should pretty much always be 0 as far as I know
uint8 _access; // If this ends up useful, there should be an enum for the access values
uint8 _entryLen; // Always 27 in ProDOS 1.0
uint8 _entriesPerBlock; // Always 0D in ProDOS 1.0
2022-07-12 01:09:34 -04:00
uint16 _fileCount; // Number of files across all data blocks in this directory
uint16 _bitmapPtr; // Block pointer to the keyblock of the bitmap for the entire volume
uint16 _volBlocks; // Blocks in entire volume
};
struct DirHeader {
uint8 _type;
uint8 _nameLen;
char _name[16];
byte _reserved[8];
Date _date;
Time _time;
uint8 _ver;
uint8 _minVer;
uint8 _access;
uint8 _entryLen;
uint8 _entriesPerBlock;
2022-07-12 01:09:34 -04:00
uint16 _fileCount;
uint16 _parentBlockPtr; // These values allow ProDOS to navigate back out of a directory, but they aren't really needed by the class to navigate
uint8 _parentEntryIndex; // Index in the current directory
uint8 _parentEntryLen; // This is always 27 in ProDOS 1.0
2022-07-12 01:09:34 -04:00
};
struct FileEntry {
uint8 _type; // 0 = inactive, 1-3 = file, 4 = pascal area, 14 = subdirectory, 15 = volume directory
uint8 _nameLen;
char _name[16];
uint8 _ext; // File extension, uses the enum FileExt
2022-07-12 01:09:34 -04:00
uint16 _blockPtr; // Block pointer to data for seedling, index block for sapling, or master block for tree
uint16 _totalBlocks; // Really important to remember this is the total *including* the index block
uint32 _eof; // This is a long (3 bytes, read low to high) value representing the total readable data in a file (unless it's a sparse file, be careful!)
Date _date;
Time _time;
uint8 _ver;
uint8 _minVer;
uint8 _access;
2022-07-12 01:09:34 -04:00
uint16 _varUse;
Date _modDate;
Time _modTime;
2022-07-12 01:09:34 -04:00
uint16 _dirHeadPtr; // Pointer to the key block of the directory that contains this file entry
};
void getDate(Date *d, uint16 date); // Decompresses the date into a struct
void getTime(Time *t, uint16 time); // Decompresses the time into a struct
void getHeader(DirHeader *h); // Adds the main header values to the struct
void getDirectoryHeader(DirHeader *h); // Uses getHeader and then fills in the values for the parent directory
void getVolumeHeader(VolHeader *dir); // Uses getHeader and then fills in the volume related information (there is no parent directory to this one)
void getFileEntry(FileEntry *f); // Adds all of the file entry information to the struct
void searchDirectory(DirHeader *h, uint16 p, uint16 n, Common::String path); // Recursively searches all files within a directory, by calling itself for subdirectories
void getVolumeBitmap(VolHeader *h); // Puts together the volume bitmap
};
} // Namespace Common
#endif