2016-03-23 11:11:37 +01:00
|
|
|
/* 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.
|
|
|
|
*
|
2021-12-26 18:47:58 +01:00
|
|
|
* 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.
|
2016-03-23 11:11:37 +01:00
|
|
|
*
|
|
|
|
* 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
|
2021-12-26 18:47:58 +01:00
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2016-03-23 11:11:37 +01:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "common/ptr.h"
|
|
|
|
#include "common/file.h"
|
2016-03-24 19:01:01 +01:00
|
|
|
#include "common/debug.h"
|
2016-03-23 11:11:37 +01:00
|
|
|
|
|
|
|
#ifndef ADL_DISK_H
|
|
|
|
#define ADL_DISK_H
|
|
|
|
|
|
|
|
namespace Common {
|
|
|
|
class SeekableReadStream;
|
|
|
|
class String;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace Adl {
|
|
|
|
|
2018-02-01 22:20:52 +01:00
|
|
|
// Used for disk image detection
|
|
|
|
int32 computeMD5(const Common::FSNode &node, Common::String &md5, uint32 md5Bytes);
|
|
|
|
|
2016-03-23 11:11:37 +01:00
|
|
|
class DataBlock {
|
|
|
|
public:
|
|
|
|
virtual ~DataBlock() { }
|
|
|
|
|
|
|
|
virtual Common::SeekableReadStream *createReadStream() const = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
typedef Common::SharedPtr<DataBlock> DataBlockPtr;
|
2016-03-26 01:01:10 +01:00
|
|
|
typedef Common::ScopedPtr<Common::SeekableReadStream> StreamPtr;
|
2016-03-23 11:11:37 +01:00
|
|
|
|
2016-03-24 09:50:52 +01:00
|
|
|
class Files {
|
|
|
|
public:
|
|
|
|
virtual ~Files() { }
|
|
|
|
|
2016-03-24 19:01:01 +01:00
|
|
|
virtual const DataBlockPtr getDataBlock(const Common::String &filename, uint offset = 0) const = 0;
|
|
|
|
virtual Common::SeekableReadStream *createReadStream(const Common::String &filename, uint offset = 0) const = 0;
|
2021-07-28 00:53:38 +02:00
|
|
|
virtual bool exists(const Common::String &filename) const = 0;
|
2016-03-24 09:50:52 +01:00
|
|
|
|
2016-03-26 01:01:10 +01:00
|
|
|
protected:
|
|
|
|
class DataBlock : public Adl::DataBlock {
|
|
|
|
public:
|
|
|
|
DataBlock(const Files *files, const Common::String &filename, uint offset) :
|
|
|
|
_files(files),
|
|
|
|
_filename(filename),
|
|
|
|
_offset(offset) { }
|
|
|
|
|
2017-08-12 12:02:26 +02:00
|
|
|
Common::SeekableReadStream *createReadStream() const override {
|
2016-03-26 01:01:10 +01:00
|
|
|
return _files->createReadStream(_filename, _offset);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
const Common::String _filename;
|
|
|
|
uint _offset;
|
|
|
|
const Files *_files;
|
|
|
|
};
|
2016-03-24 09:50:52 +01:00
|
|
|
};
|
|
|
|
|
2016-03-23 11:11:37 +01:00
|
|
|
class DiskImage {
|
|
|
|
public:
|
2016-03-26 01:01:10 +01:00
|
|
|
DiskImage() :
|
2016-08-19 15:45:44 +02:00
|
|
|
_stream(nullptr),
|
2016-08-27 01:47:43 +02:00
|
|
|
_tracks(0),
|
|
|
|
_sectorsPerTrack(0),
|
|
|
|
_bytesPerSector(0),
|
2017-08-12 12:02:26 +02:00
|
|
|
_sectorLimit(0),
|
|
|
|
_firstSector(0) { }
|
2016-03-26 01:01:10 +01:00
|
|
|
|
2016-08-19 15:45:44 +02:00
|
|
|
~DiskImage() {
|
|
|
|
delete _stream;
|
2016-03-26 01:01:10 +01:00
|
|
|
}
|
2016-03-23 11:11:37 +01:00
|
|
|
|
2016-08-19 15:45:44 +02:00
|
|
|
bool open(const Common::String &filename);
|
|
|
|
const DataBlockPtr getDataBlock(uint track, uint sector, uint offset = 0, uint size = 0) const;
|
2016-08-27 01:47:43 +02:00
|
|
|
Common::SeekableReadStream *createReadStream(uint track, uint sector, uint offset = 0, uint size = 0, uint sectorsUsed = 0) const;
|
|
|
|
void setSectorLimit(uint sectorLimit) { _sectorLimit = sectorLimit; } // Maximum number of sectors to read per track before stepping
|
2017-06-10 17:07:17 +02:00
|
|
|
uint getBytesPerSector() const { return _bytesPerSector; }
|
|
|
|
uint getSectorsPerTrack() const { return _sectorsPerTrack; }
|
2018-01-27 14:06:02 +01:00
|
|
|
uint getTracks() const { return _tracks; }
|
2016-03-23 11:11:37 +01:00
|
|
|
|
|
|
|
protected:
|
2016-03-26 01:01:10 +01:00
|
|
|
class DataBlock : public Adl::DataBlock {
|
|
|
|
public:
|
2016-08-27 01:47:43 +02:00
|
|
|
DataBlock(const DiskImage *disk, uint track, uint sector, uint offset, uint size, uint sectorLimit) :
|
2016-03-26 01:01:10 +01:00
|
|
|
_track(track),
|
|
|
|
_sector(sector),
|
|
|
|
_offset(offset),
|
|
|
|
_size(size),
|
2016-08-27 01:47:43 +02:00
|
|
|
_sectorLimit(sectorLimit),
|
2016-03-26 01:01:10 +01:00
|
|
|
_disk(disk) { }
|
|
|
|
|
2017-08-12 12:02:26 +02:00
|
|
|
Common::SeekableReadStream *createReadStream() const override {
|
2016-08-27 01:47:43 +02:00
|
|
|
return _disk->createReadStream(_track, _sector, _offset, _size, _sectorLimit);
|
2016-03-26 01:01:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
uint _track, _sector, _offset, _size;
|
2016-08-27 01:47:43 +02:00
|
|
|
uint _sectorLimit;
|
2016-03-26 01:01:10 +01:00
|
|
|
const DiskImage *_disk;
|
|
|
|
};
|
|
|
|
|
2016-08-19 15:45:44 +02:00
|
|
|
Common::SeekableReadStream *_stream;
|
2017-08-12 12:02:26 +02:00
|
|
|
uint _tracks, _sectorsPerTrack, _bytesPerSector, _firstSector;
|
2016-08-27 01:47:43 +02:00
|
|
|
uint _sectorLimit;
|
2016-03-23 11:11:37 +01:00
|
|
|
};
|
|
|
|
|
2016-03-26 01:01:10 +01:00
|
|
|
// Data in plain files
|
|
|
|
class Files_Plain : public Files {
|
2016-03-23 11:11:37 +01:00
|
|
|
public:
|
2017-08-12 12:02:26 +02:00
|
|
|
const DataBlockPtr getDataBlock(const Common::String &filename, uint offset = 0) const override;
|
|
|
|
Common::SeekableReadStream *createReadStream(const Common::String &filename, uint offset = 0) const override;
|
2021-07-28 00:53:38 +02:00
|
|
|
bool exists(const Common::String &filename) const override { return Common::File::exists(filename); }
|
2016-03-23 11:11:37 +01:00
|
|
|
};
|
|
|
|
|
2016-03-26 01:01:10 +01:00
|
|
|
// Data in files contained in Apple DOS 3.3 disk image
|
2018-02-03 11:42:03 +01:00
|
|
|
class Files_AppleDOS : public Files {
|
2016-03-24 19:01:01 +01:00
|
|
|
public:
|
2018-02-03 11:42:03 +01:00
|
|
|
Files_AppleDOS();
|
2020-02-09 12:05:27 +01:00
|
|
|
~Files_AppleDOS() override;
|
2016-03-24 19:01:01 +01:00
|
|
|
|
2021-06-29 12:54:35 +02:00
|
|
|
bool open(const Common::String &filename);
|
2017-08-12 12:02:26 +02:00
|
|
|
const DataBlockPtr getDataBlock(const Common::String &filename, uint offset = 0) const override;
|
|
|
|
Common::SeekableReadStream *createReadStream(const Common::String &filename, uint offset = 0) const override;
|
2021-07-28 00:53:38 +02:00
|
|
|
bool exists(const Common::String &filename) const override { return _toc.contains(filename); }
|
2016-03-24 19:01:01 +01:00
|
|
|
|
|
|
|
private:
|
|
|
|
enum FileType {
|
|
|
|
kFileTypeText = 0,
|
|
|
|
kFileTypeAppleSoft = 2,
|
2016-10-09 14:59:58 +02:00
|
|
|
kFileTypeBinary = 4
|
2016-03-24 19:01:01 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
enum {
|
|
|
|
kSectorSize = 256,
|
|
|
|
kFilenameLen = 30
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TrackSector {
|
|
|
|
byte track;
|
|
|
|
byte sector;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct TOCEntry {
|
|
|
|
byte type;
|
|
|
|
uint16 totalSectors;
|
|
|
|
Common::Array<TrackSector> sectors;
|
|
|
|
};
|
|
|
|
|
2021-06-29 12:54:35 +02:00
|
|
|
void readVTOC();
|
2016-03-24 19:01:01 +01:00
|
|
|
void readSectorList(TrackSector start, Common::Array<TrackSector> &list);
|
|
|
|
Common::SeekableReadStream *createReadStreamText(const TOCEntry &entry) const;
|
|
|
|
Common::SeekableReadStream *createReadStreamBinary(const TOCEntry &entry) const;
|
|
|
|
|
2016-03-26 01:01:10 +01:00
|
|
|
DiskImage *_disk;
|
2016-03-24 19:01:01 +01:00
|
|
|
Common::HashMap<Common::String, TOCEntry> _toc;
|
|
|
|
};
|
|
|
|
|
2017-08-12 12:02:26 +02:00
|
|
|
// On the Apple II, sector headers contain a disk volume number. This number
|
|
|
|
// is used by ADL multi-disk games. The PC port has the disk volume number
|
|
|
|
// as the first data byte of every sector that contains game data. We need
|
|
|
|
// to skip this number as we read in the data. Additionally, the data is now
|
|
|
|
// prefixed with an uint16 containing the data size.
|
|
|
|
class DataBlock_PC : public DataBlock {
|
|
|
|
public:
|
|
|
|
DataBlock_PC(DiskImage *disk, byte track, byte sector, uint16 offset = 0) :
|
|
|
|
_disk(disk),
|
|
|
|
_track(track),
|
2021-05-04 11:45:03 +03:00
|
|
|
_sector(sector),
|
2017-08-12 12:02:26 +02:00
|
|
|
_offset(offset) { }
|
|
|
|
|
2020-02-09 12:05:27 +01:00
|
|
|
~DataBlock_PC() override { }
|
2017-08-12 12:02:26 +02:00
|
|
|
|
|
|
|
Common::SeekableReadStream *createReadStream() const override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
void read(Common::SeekableReadStream &stream, byte *const dataPtr, const uint32 size) const;
|
|
|
|
|
|
|
|
DiskImage *_disk;
|
|
|
|
byte _track, _sector;
|
|
|
|
uint16 _offset;
|
|
|
|
};
|
|
|
|
|
2016-03-23 11:11:37 +01:00
|
|
|
} // End of namespace Adl
|
|
|
|
|
|
|
|
#endif
|