211 lines
6.3 KiB
C++
211 lines
6.3 KiB
C++
//
|
|
// Apple 2 floppy disk support
|
|
//
|
|
|
|
#ifndef __FLOPPY_H__
|
|
#define __FLOPPY_H__
|
|
|
|
// MAX_PATH isn't defined in stdlib.h on *nix, so we do it here...
|
|
#ifdef __GCCUNIX__
|
|
#include <limits.h>
|
|
#define MAX_PATH _POSIX_PATH_MAX
|
|
#else
|
|
#include <stdlib.h> // for MAX_PATH on MinGW/Darwin
|
|
// Kludge for Win64
|
|
#ifndef MAX_PATH
|
|
#define MAX_PATH _MAX_PATH
|
|
#endif
|
|
#endif
|
|
|
|
#include <stdint.h>
|
|
#include <stdio.h>
|
|
|
|
enum { DT_EMPTY = 0, DT_WOZ, DT_DOS33, DT_DOS33_HDR, DT_PRODOS, DT_NYBBLE,
|
|
DFT_UNKNOWN };
|
|
enum { DLS_OFF, DLS_READ, DLS_WRITE };
|
|
|
|
// N.B.: All 32/16-bit values are stored in little endian. Which means, to
|
|
// read/write them safely, we need to use translators as this code may or
|
|
// may not be compiled on an architecture that supports little endian
|
|
// natively.
|
|
|
|
struct WOZTrack
|
|
{
|
|
uint8_t bits[6646];
|
|
uint16_t byteCount;
|
|
uint16_t bitCount;
|
|
uint16_t splicePoint;
|
|
uint8_t spliceNibble;
|
|
uint8_t spliceBitCount;
|
|
uint16_t reserved;
|
|
};
|
|
|
|
struct WOZMetadata
|
|
{
|
|
uint8_t metaTag[4]; // "META"
|
|
uint32_t metaSize; // Size of the META chunk
|
|
uint8_t data[]; // Variable length array of metadata
|
|
};
|
|
|
|
struct WOZ
|
|
{
|
|
// Header
|
|
uint8_t magic[8]; // "WOZ1" $FF $0A $0D $0A
|
|
uint32_t crc32; // CRC32 of the remaining data in the file
|
|
|
|
// INFO chunk
|
|
uint8_t infoTag[4]; // "INFO"
|
|
uint32_t infoSize; // Always 60 bytes long
|
|
uint8_t infoVersion; // Currently 1
|
|
uint8_t diskType; // 1 = 5 1/4", 2 = 3 1/2"
|
|
uint8_t writeProtected; // 1 = write protected disk
|
|
uint8_t synchronized; // 1 = cross-track sync was used during imaging
|
|
uint8_t cleaned; // 1 = fake bits removed from image
|
|
uint8_t creator[32]; // Software that made this image, padded with 0x20
|
|
uint8_t pad1[23]; // Padding to 60 bytes
|
|
|
|
// TMAP chunk
|
|
uint8_t tmapTag[4]; // "TMAP"
|
|
uint32_t tmapSize; // Always 160 bytes long
|
|
uint8_t tmap[160]; // Track map, with empty tracks set to $FF
|
|
|
|
// TRKS chunk
|
|
uint8_t trksTag[4]; // "TRKS"
|
|
uint32_t trksSize; // Varies, depending on # of tracks imaged
|
|
WOZTrack track[]; // Variable length array for the track data proper
|
|
};
|
|
|
|
struct WOZTrack2
|
|
{
|
|
uint16_t startingBlock; // 512 byte block # where this track starts (relative to the start of the file)
|
|
uint16_t blockCount; // # of blocks in this track
|
|
uint32_t bitCount; // # of bits in this track
|
|
};
|
|
|
|
struct WOZ2
|
|
{
|
|
// Header
|
|
uint8_t magic[8]; // "WOZ2" $FF $0A $0D $0A
|
|
uint32_t crc32; // CRC32 of the remaining data in the file
|
|
|
|
// INFO chunk
|
|
uint8_t infoTag[4]; // "INFO"
|
|
uint32_t infoSize; // Always 60 bytes long
|
|
uint8_t infoVersion; // Currently 1
|
|
uint8_t diskType; // 1 = 5 1/4", 2 = 3 1/2"
|
|
uint8_t writeProtected; // 1 = write protected disk
|
|
uint8_t synchronized; // 1 = cross-track sync was used during imaging
|
|
uint8_t cleaned; // 1 = fake bits removed from image
|
|
uint8_t creator[32]; // Software that made this image, padded with 0x20
|
|
uint8_t diskSides; // 5 1/4" disks always have 1 side (v2 from here on)
|
|
uint8_t bootSectorFmt; // 5 1/4" only (0=unknown, 1=16 sector, 2=13 sector, 3=both)
|
|
uint8_t optimalBitTmg; // In ticks, standard for 5 1/4" is 32 (4 µs)
|
|
uint16_t compatibleHW; // Bitfield showing hardware compatibility (1=][, 2=][+, 4=//e (unenh), 8=//c, 16=//e (enh), 32=IIgs, 64=//c+, 128=///, 256=///+)
|
|
uint16_t requiredRAM; // Minimum size in K, 0=unknown
|
|
uint16_t largestTrack; // Number of 512 byte blocks used by largest track
|
|
uint8_t pad1[14]; // Padding to 60 bytes
|
|
|
|
// TMAP chunk
|
|
uint8_t tmapTag[4]; // "TMAP"
|
|
uint32_t tmapSize; // Always 160 bytes long
|
|
uint8_t tmap[160]; // Track map, with empty tracks set to $FF
|
|
|
|
// TRKS chunk
|
|
uint8_t trksTag[4]; // "TRKS"
|
|
uint32_t trksSize; // Varies, depending on # of tracks imaged
|
|
WOZTrack2 track[160]; // Actual track info (corresponding to TMAP data)
|
|
uint8_t data[]; // Variable length array for the track data proper
|
|
};
|
|
|
|
class FloppyDrive
|
|
{
|
|
public:
|
|
FloppyDrive();
|
|
~FloppyDrive();
|
|
|
|
bool LoadImage(const char * filename, uint8_t driveNum = 0);
|
|
bool SaveImage(uint8_t driveNum = 0);
|
|
bool SaveImageAs(const char * filename, uint8_t driveNum = 0);
|
|
void CreateBlankImage(uint8_t driveNum = 0);
|
|
void SwapImages(void);
|
|
const char * ImageName(uint8_t driveNum = 0);
|
|
void EjectImage(uint8_t driveNum = 0);
|
|
bool IsEmpty(uint8_t driveNum = 0);
|
|
bool IsWriteProtected(uint8_t driveNum = 0);
|
|
void SetWriteProtect(bool, uint8_t driveNum = 0);
|
|
int DriveLightStatus(uint8_t driveNum = 0);
|
|
void SaveState(FILE *);
|
|
void LoadState(FILE *);
|
|
void InitWOZ(uint8_t driveNum = 0);
|
|
bool CheckWOZ(const uint8_t * wozData, uint32_t wozSize, uint8_t driveNum = 0);
|
|
bool SaveWOZ(uint8_t driveNum);
|
|
|
|
private:
|
|
uint32_t ReadLong(FILE *);
|
|
void WriteLong(FILE *, uint32_t);
|
|
void WriteLongLE(FILE *, uint32_t);
|
|
void WriteWordLE(FILE *, uint16_t);
|
|
void WriteZeroes(FILE *, uint32_t);
|
|
|
|
// I/O functions ($C0Ex accesses)
|
|
|
|
public:
|
|
void ControlStepper(uint8_t addr);
|
|
void ControlMotor(uint8_t addr);
|
|
void DriveEnable(uint8_t addr);
|
|
void SetShiftLoadSwitch(uint8_t state);
|
|
void SetReadWriteSwitch(uint8_t state);
|
|
uint8_t DataRegister(void);
|
|
void DataRegister(uint8_t);
|
|
void RunSequencer(uint32_t);
|
|
|
|
protected:
|
|
void DetectImageType(const char * filename, uint8_t driveNum);
|
|
void WriteBits(uint8_t * dest, uint8_t * src, uint16_t bits, uint16_t * start);
|
|
void WOZifyImage(uint8_t driveNum);
|
|
|
|
private:
|
|
char imageName[2][MAX_PATH];
|
|
uint8_t * disk[2];
|
|
uint32_t diskSize[2];
|
|
uint8_t diskType[2];
|
|
bool imageDirty[2];
|
|
uint8_t motorOn;
|
|
uint8_t activeDrive;
|
|
uint8_t ioMode;
|
|
uint8_t dataRegister;
|
|
uint8_t phase[2];
|
|
uint8_t headPos[2];
|
|
bool ioHappened;
|
|
|
|
uint32_t currentPos[2];
|
|
WOZ * woz[2];
|
|
|
|
uint8_t cpuDataBus;
|
|
uint8_t slSwitch; // Shift/Load soft switch
|
|
uint8_t rwSwitch; // Read/Write soft switch
|
|
uint8_t readPulse; // Disk read head "pulse" signal
|
|
uint8_t pulseClock; // Disk read head bitstream "pulse clock"
|
|
uint8_t sequencerState;
|
|
uint32_t driveOffTimeout;
|
|
uint8_t zeroBitCount;
|
|
uint16_t trackLength[2];
|
|
|
|
// And here are some private class variables (to reduce function
|
|
// redundancy):
|
|
static uint8_t doSector[16];
|
|
static uint8_t poSector[16];
|
|
static uint8_t wozHeader[9];
|
|
static uint8_t wozHeader2[9];
|
|
static uint8_t standardTMAP[141];
|
|
static uint8_t sequencerROM[256];
|
|
static uint8_t bitMask[8];
|
|
static char nameBuf[MAX_PATH];
|
|
};
|
|
|
|
void InstallFloppy(uint8_t slot);
|
|
extern FloppyDrive floppyDrive[];
|
|
|
|
#endif // __FLOPPY_H__
|
|
|