mirror of
https://github.com/JorjBauer/aiie.git
synced 2025-01-13 22:32:00 +00:00
woz uses built-in file management system
This commit is contained in:
parent
03b0a141bc
commit
687a096702
@ -47,7 +47,7 @@ bool DiskII::Serialize(int8_t fd)
|
|||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// FIXME: all the new variables are missing
|
// FIXME: all the new variables are missing ***
|
||||||
|
|
||||||
g_filemanager->writeByte(fd, DISKIIMAGIC);
|
g_filemanager->writeByte(fd, DISKIIMAGIC);
|
||||||
|
|
||||||
@ -84,7 +84,7 @@ bool DiskII::Serialize(int8_t fd)
|
|||||||
bool DiskII::Deserialize(int8_t fd)
|
bool DiskII::Deserialize(int8_t fd)
|
||||||
{
|
{
|
||||||
return false;
|
return false;
|
||||||
// FIXME: all the new variables are missing
|
// FIXME: all the new variables are missing ***
|
||||||
|
|
||||||
if (g_filemanager->readByte(fd) != DISKIIMAGIC) {
|
if (g_filemanager->readByte(fd) != DISKIIMAGIC) {
|
||||||
return false;
|
return false;
|
||||||
@ -167,11 +167,11 @@ uint8_t DiskII::readSwitches(uint8_t s)
|
|||||||
|
|
||||||
case 0x08: // drive off
|
case 0x08: // drive off
|
||||||
indicatorIsOn[selectedDisk] = 99;
|
indicatorIsOn[selectedDisk] = 99;
|
||||||
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, false); // FIXME: delay a bit? Queue for later drawing?
|
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, false); // FIXME: delay a bit? Queue for later drawing? ***
|
||||||
break;
|
break;
|
||||||
case 0x09: // drive on
|
case 0x09: // drive on
|
||||||
indicatorIsOn[selectedDisk] = 100;
|
indicatorIsOn[selectedDisk] = 100;
|
||||||
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, true); // FIXME: delay a bit? Queue for later drawing?
|
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, true); // FIXME: delay a bit? Queue for later drawing? ***
|
||||||
|
|
||||||
// Start the given disk drive spinning
|
// Start the given disk drive spinning
|
||||||
lastDiskRead[selectedDisk] = g_cpu->cycles;
|
lastDiskRead[selectedDisk] = g_cpu->cycles;
|
||||||
@ -186,6 +186,10 @@ uint8_t DiskII::readSwitches(uint8_t s)
|
|||||||
|
|
||||||
case 0x0C: // shift one read or write byte
|
case 0x0C: // shift one read or write byte
|
||||||
readWriteLatch = readOrWriteByte();
|
readWriteLatch = readOrWriteByte();
|
||||||
|
/*
|
||||||
|
if (readWriteLatch & 0x80)
|
||||||
|
printf(" => Disk II reads 0x%.2X @ $%.4X\n", sequencer, g_cpu->pc);
|
||||||
|
*/
|
||||||
if (readWriteLatch & 0x80)
|
if (readWriteLatch & 0x80)
|
||||||
sequencer = 0;
|
sequencer = 0;
|
||||||
break;
|
break;
|
||||||
@ -210,9 +214,9 @@ uint8_t DiskII::readSwitches(uint8_t s)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: improve the spin-down here. We need a CPU cycle callback
|
// FIXME: improve the spin-down here. We need a CPU cycle callback
|
||||||
// for some period of time instead of this silly decrement counter.
|
// for some period of time instead of this silly decrement counter ***
|
||||||
if (!indicatorIsOn[selectedDisk]) {
|
if (!indicatorIsOn[selectedDisk]) {
|
||||||
printf("Unexpected read while disk isn't on?\n");
|
// printf("Unexpected read while disk isn't on?\n");
|
||||||
indicatorIsOn[selectedDisk] = 100;
|
indicatorIsOn[selectedDisk] = 100;
|
||||||
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, true); // FIXME: queue for later drawing?
|
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, true); // FIXME: queue for later drawing?
|
||||||
}
|
}
|
||||||
@ -222,7 +226,7 @@ uint8_t DiskII::readSwitches(uint8_t s)
|
|||||||
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, false); // FIXME: queue for later drawing?
|
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, false); // FIXME: queue for later drawing?
|
||||||
|
|
||||||
// Stop the given disk drive spinning
|
// Stop the given disk drive spinning
|
||||||
lastDiskRead[selectedDisk] = 0; // FIXME: magic value. We need a tristate for this.
|
lastDiskRead[selectedDisk] = 0; // FIXME: magic value. We need a tristate for this. ***
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -351,10 +355,8 @@ void DiskII::setPhase(uint8_t phase)
|
|||||||
|
|
||||||
if (curHalfTrack[selectedDisk] != prevHalfTrack) {
|
if (curHalfTrack[selectedDisk] != prevHalfTrack) {
|
||||||
// We're changing track - flush the old track back to disk
|
// We're changing track - flush the old track back to disk
|
||||||
printf("track change\n");
|
|
||||||
|
|
||||||
curWozTrack[selectedDisk] = disk[selectedDisk]->trackNumberForQuarterTrack(curHalfTrack[selectedDisk]*2);
|
curWozTrack[selectedDisk] = disk[selectedDisk]->trackNumberForQuarterTrack(curHalfTrack[selectedDisk]*2);
|
||||||
printf("New half: %d; track: %d\n", curHalfTrack[selectedDisk]*2, curWozTrack[selectedDisk]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -400,7 +402,6 @@ void DiskII::insertDisk(int8_t driveNum, const char *filename, bool drawIt)
|
|||||||
disk[driveNum]->readFile(filename); // FIXME error checking
|
disk[driveNum]->readFile(filename); // FIXME error checking
|
||||||
|
|
||||||
curWozTrack[driveNum] = disk[driveNum]->trackNumberForQuarterTrack(curHalfTrack[driveNum]*2);
|
curWozTrack[driveNum] = disk[driveNum]->trackNumberForQuarterTrack(curHalfTrack[driveNum]*2);
|
||||||
printf("Cur track: %d\n", curWozTrack[driveNum]);
|
|
||||||
|
|
||||||
if (drawIt)
|
if (drawIt)
|
||||||
g_ui->drawOnOffUIElement(UIeDisk1_state + driveNum, false);
|
g_ui->drawOnOffUIElement(UIeDisk1_state + driveNum, false);
|
||||||
@ -420,8 +421,6 @@ void DiskII::select(int8_t which)
|
|||||||
if (which != 0 && which != 1)
|
if (which != 0 && which != 1)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
printf("Select disk %d\n", which);
|
|
||||||
|
|
||||||
if (which != selectedDisk) {
|
if (which != selectedDisk) {
|
||||||
indicatorIsOn[selectedDisk] = 100; // spindown time (fixme)
|
indicatorIsOn[selectedDisk] = 100; // spindown time (fixme)
|
||||||
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, false); // FIXME: queue for later drawing?
|
g_ui->drawOnOffUIElement(UIeDisk1_activity + selectedDisk, false); // FIXME: queue for later drawing?
|
||||||
@ -433,14 +432,12 @@ void DiskII::select(int8_t which)
|
|||||||
// Update the current woz track for the given disk drive
|
// Update the current woz track for the given disk drive
|
||||||
curWozTrack[selectedDisk] =
|
curWozTrack[selectedDisk] =
|
||||||
disk[selectedDisk]->trackNumberForQuarterTrack(curHalfTrack[selectedDisk]*2);
|
disk[selectedDisk]->trackNumberForQuarterTrack(curHalfTrack[selectedDisk]*2);
|
||||||
printf("Cur Woz track is %d\n", curWozTrack[selectedDisk]);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t DiskII::readOrWriteByte()
|
uint8_t DiskII::readOrWriteByte()
|
||||||
{
|
{
|
||||||
if (!disk[selectedDisk]) {
|
if (!disk[selectedDisk]) {
|
||||||
printf("reading from uninserted disk\n");
|
//printf("reading from uninserted disk\n");
|
||||||
return 0xFF;
|
return 0xFF;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -35,22 +35,24 @@ const static uint8_t _trans[64] = {0x96, 0x97, 0x9a, 0x9b, 0x9d, 0x9e, 0x9f, 0xa
|
|||||||
0xed, 0xee, 0xef, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6,
|
0xed, 0xee, 0xef, 0xf2, 0xf3, 0xf4, 0xf5, 0xf6,
|
||||||
0xf7, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff};
|
0xf7, 0xf9, 0xfa, 0xfb, 0xfc, 0xfd, 0xfe, 0xff};
|
||||||
|
|
||||||
const static uint8_t _detrans[0x80] = {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
// This is the inverted DOS 3.3 RWTS Write Table (high bit
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
// stripped). Any "bad" value is stored as 0xFF.
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x04,
|
const static uint8_t _detrans[0x80] = {0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
0x00, 0x00, 0x08, 0x0C, 0x00, 0x10, 0x14, 0x18,
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x1C, 0x20,
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x00, 0x04,
|
||||||
0x00, 0x00, 0x00, 0x24, 0x28, 0x2C, 0x30, 0x34,
|
0xFF, 0xFF, 0x08, 0x0C, 0xFF, 0x10, 0x14, 0x18,
|
||||||
0x00, 0x00, 0x38, 0x3C, 0x40, 0x44, 0x48, 0x4C,
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0x1C, 0x20,
|
||||||
0x00, 0x50, 0x54, 0x58, 0x5C, 0x60, 0x64, 0x68,
|
0xFF, 0xFF, 0xFF, 0x24, 0x28, 0x2C, 0x30, 0x34,
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
|
0xFF, 0xFF, 0x38, 0x3C, 0x40, 0x44, 0x48, 0x4C,
|
||||||
0x00, 0x00, 0x00, 0x6C, 0x00, 0x70, 0x74, 0x78,
|
0xFF, 0x50, 0x54, 0x58, 0x5C, 0x60, 0x64, 0x68,
|
||||||
0x00, 0x00, 0x00, 0x7C, 0x00, 0x00, 0x80, 0x84,
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF,
|
||||||
0x00, 0x88, 0x8C, 0x90, 0x94, 0x98, 0x9C, 0xA0,
|
0xFF, 0xFF, 0xFF, 0x6C, 0xFF, 0x70, 0x74, 0x78,
|
||||||
0x00, 0x00, 0x00, 0x00, 0x00, 0xA4, 0xA8, 0xAC,
|
0xFF, 0xFF, 0xFF, 0x7C, 0xFF, 0xFF, 0x80, 0x84,
|
||||||
0x00, 0xB0, 0xB4, 0xB8, 0xBC, 0xC0, 0xC4, 0xC8,
|
0xFF, 0x88, 0x8C, 0x90, 0x94, 0x98, 0x9C, 0xA0,
|
||||||
0x00, 0x00, 0xCC, 0xD0, 0xD4, 0xD8, 0xDC, 0xE0,
|
0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xA4, 0xA8, 0xAC,
|
||||||
0x00, 0xE4, 0xE8, 0xEC, 0xF0, 0xF4, 0xF8, 0xFC };
|
0xFF, 0xB0, 0xB4, 0xB8, 0xBC, 0xC0, 0xC4, 0xC8,
|
||||||
|
0xFF, 0xFF, 0xCC, 0xD0, 0xD4, 0xD8, 0xDC, 0xE0,
|
||||||
|
0xFF, 0xE4, 0xE8, 0xEC, 0xF0, 0xF4, 0xF8, 0xFC };
|
||||||
|
|
||||||
// dos 3.3 to physical sector conversion
|
// dos 3.3 to physical sector conversion
|
||||||
const static uint8_t dephys[16] = {
|
const static uint8_t dephys[16] = {
|
||||||
@ -232,6 +234,8 @@ static bool _decodeData(const uint8_t trackBuffer[343], uint8_t output[256])
|
|||||||
for (int i=0; i<342; i++) {
|
for (int i=0; i<342; i++) {
|
||||||
uint8_t in = *(trackBuffer++) & 0x7F; // strip high bit
|
uint8_t in = *(trackBuffer++) & 0x7F; // strip high bit
|
||||||
workbuf[i] = _detrans[in];
|
workbuf[i] = _detrans[in];
|
||||||
|
if (workbuf[i] == 0xFF) // bad data is untranslatable
|
||||||
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// fixme: collapse this in to the previous loop
|
// fixme: collapse this in to the previous loop
|
||||||
@ -343,7 +347,7 @@ nibErr denibblizeTrack(const uint8_t input[NIBTRACKSIZE], uint8_t rawTrackBuffer
|
|||||||
nibData[j] = input[(i+j)%NIBTRACKSIZE];
|
nibData[j] = input[(i+j)%NIBTRACKSIZE];
|
||||||
}
|
}
|
||||||
if (!_decodeData(nibData, output)) {
|
if (!_decodeData(nibData, output)) {
|
||||||
continue;
|
return errorBadData;
|
||||||
}
|
}
|
||||||
i += 343;
|
i += 343;
|
||||||
|
|
||||||
|
369
apple/woz.cpp
369
apple/woz.cpp
@ -3,6 +3,7 @@
|
|||||||
#include "crc32.h"
|
#include "crc32.h"
|
||||||
#include "nibutil.h"
|
#include "nibutil.h"
|
||||||
#include "version.h"
|
#include "version.h"
|
||||||
|
#include "globals.h"
|
||||||
|
|
||||||
// Block number we start packing data bits after (Woz 2.0 images)
|
// Block number we start packing data bits after (Woz 2.0 images)
|
||||||
#define STARTBLOCK 3
|
#define STARTBLOCK 3
|
||||||
@ -13,16 +14,16 @@
|
|||||||
return false; \
|
return false; \
|
||||||
if (!write32(f, 0)) \
|
if (!write32(f, 0)) \
|
||||||
return false; \
|
return false; \
|
||||||
curpos = ftello(f); \
|
curpos = g_filemanager->getSeekPosition(f); \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define END_SECTION(f) { \
|
#define END_SECTION(f) { \
|
||||||
long endpos = ftello(f); \
|
uint32_t endpos = g_filemanager->getSeekPosition(f); \
|
||||||
fseeko(f, curpos-4, SEEK_SET); \
|
g_filemanager->setSeekPosition(f, curpos-4); \
|
||||||
uint32_t chunksize = endpos - curpos; \
|
uint32_t chunksize = endpos - curpos; \
|
||||||
if (!write32(f, chunksize)) \
|
if (!write32(f, chunksize)) \
|
||||||
return false; \
|
return false; \
|
||||||
fseeko(f, 0, SEEK_END); \
|
g_filemanager->seekToEnd(f); \
|
||||||
}
|
}
|
||||||
|
|
||||||
Woz::Woz()
|
Woz::Woz()
|
||||||
@ -95,49 +96,47 @@ uint8_t Woz::nextDiskByte(uint8_t track)
|
|||||||
return d;
|
return d;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool write8(FILE *f, uint8_t v)
|
static bool write8(uint8_t fh, uint8_t v)
|
||||||
{
|
{
|
||||||
if (fwrite(&v, 1, 1, f) != 1)
|
if (!g_filemanager->writeByte(fh, v))
|
||||||
return false;
|
return false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool write16(FILE *f, uint16_t v)
|
static bool write16(uint8_t fh, uint16_t v)
|
||||||
{
|
{
|
||||||
if (!write8(f, v & 0xFF))
|
if (!write8(fh, v & 0xFF))
|
||||||
return false;
|
return false;
|
||||||
v >>= 8;
|
v >>= 8;
|
||||||
if (!write8(f, v & 0xFF))
|
if (!write8(fh, v & 0xFF))
|
||||||
return false;
|
return false;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool write32(FILE *f, uint32_t v)
|
static bool write32(uint8_t fh, uint32_t v)
|
||||||
{
|
{
|
||||||
for (int i=0; i<4; i++) {
|
for (int i=0; i<4; i++) {
|
||||||
if (!write8(f, v&0xFF))
|
if (!write8(fh, v&0xFF))
|
||||||
return false;
|
return false;
|
||||||
v >>= 8;
|
v >>= 8;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool read8(FILE *f, uint8_t *toWhere)
|
static bool read8(uint8_t fd, uint8_t *toWhere)
|
||||||
{
|
{
|
||||||
uint8_t r;
|
// FIXME: no error checking
|
||||||
if (fread(&r, 1, 1, f) != 1)
|
*toWhere = g_filemanager->readByte(fd);
|
||||||
return false;
|
|
||||||
*toWhere = r;
|
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool read16(FILE *f, uint16_t *toWhere)
|
static bool read16(uint8_t fh, uint16_t *toWhere)
|
||||||
{
|
{
|
||||||
uint16_t ret = 0;
|
uint16_t ret = 0;
|
||||||
for (int i=0; i<2; i++) {
|
for (int i=0; i<2; i++) {
|
||||||
uint8_t r;
|
uint8_t r;
|
||||||
if (!read8(f, &r)) {
|
if (!read8(fh, &r)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
ret >>= 8;
|
ret >>= 8;
|
||||||
@ -149,12 +148,12 @@ static bool read16(FILE *f, uint16_t *toWhere)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static bool read32(FILE *f, uint32_t *toWhere)
|
static bool read32(uint8_t fh, uint32_t *toWhere)
|
||||||
{
|
{
|
||||||
uint32_t ret = 0;
|
uint32_t ret = 0;
|
||||||
for (int i=0; i<4; i++) {
|
for (int i=0; i<4; i++) {
|
||||||
uint8_t r;
|
uint8_t r;
|
||||||
if (!read8(f, &r)) {
|
if (!read8(fh, &r)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
ret >>= 8;
|
ret >>= 8;
|
||||||
@ -168,7 +167,7 @@ static bool read32(FILE *f, uint32_t *toWhere)
|
|||||||
|
|
||||||
bool Woz::writeFile(uint8_t version, const char *filename)
|
bool Woz::writeFile(uint8_t version, const char *filename)
|
||||||
{
|
{
|
||||||
FILE *f = NULL;
|
int8_t fh = -1; // filehandle (-1 == closed)
|
||||||
bool retval = false;
|
bool retval = false;
|
||||||
uint32_t tmp32; // scratch 32-bit value
|
uint32_t tmp32; // scratch 32-bit value
|
||||||
off_t crcPos, endPos;
|
off_t crcPos, endPos;
|
||||||
@ -178,13 +177,17 @@ bool Woz::writeFile(uint8_t version, const char *filename)
|
|||||||
|
|
||||||
|
|
||||||
if (version > 2 || !version) {
|
if (version > 2 || !version) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "ERROR: version must be 1 or 2\n");
|
fprintf(stderr, "ERROR: version must be 1 or 2\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
f = fopen(filename, "w+");
|
fh = g_filemanager->openFile(filename);
|
||||||
if (!f) {
|
if (fh==-1) {
|
||||||
perror("ERROR: Unable to open output file");
|
#ifndef TEENSYDUINO
|
||||||
|
printf("ERROR: Unable to open output file\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -194,83 +197,102 @@ bool Woz::writeFile(uint8_t version, const char *filename)
|
|||||||
} else {
|
} else {
|
||||||
tmp32 = 0x325A4F57;
|
tmp32 = 0x325A4F57;
|
||||||
}
|
}
|
||||||
if (!write32(f, tmp32)) {
|
if (!write32(fh, tmp32)) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "ERROR: failed to write\n");
|
fprintf(stderr, "ERROR: failed to write\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
tmp32 = 0x0A0D0AFF;
|
tmp32 = 0x0A0D0AFF;
|
||||||
if (!write32(f, tmp32)) {
|
if (!write32(fh, tmp32)) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "ERROR: failed to write\n");
|
fprintf(stderr, "ERROR: failed to write\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
// We'll come back and write the checksum later
|
// We'll come back and write the checksum later
|
||||||
crcPos = ftello(f);
|
crcPos = g_filemanager->getSeekPosition(fh);
|
||||||
tmp32 = 0;
|
tmp32 = 0;
|
||||||
if (!write32(f, tmp32)) {
|
if (!write32(fh, tmp32)) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "ERROR: failed to write\n");
|
fprintf(stderr, "ERROR: failed to write\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
PREP_SECTION(f, 0x4F464E49); // 'INFO'
|
PREP_SECTION(fh, 0x4F464E49); // 'INFO'
|
||||||
if (!writeInfoChunk(version, f)) {
|
if (!writeInfoChunk(version, fh)) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "ERROR: failed to write INFO chunk\n");
|
fprintf(stderr, "ERROR: failed to write INFO chunk\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
END_SECTION(f);
|
END_SECTION(fh);
|
||||||
|
|
||||||
PREP_SECTION(f, 0x50414D54); // 'TMAP'
|
PREP_SECTION(fh, 0x50414D54); // 'TMAP'
|
||||||
if (!writeTMAPChunk(version, f)) {
|
if (!writeTMAPChunk(version, fh)) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "ERROR: failed to write TMAP chunk\n");
|
fprintf(stderr, "ERROR: failed to write TMAP chunk\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
END_SECTION(f);
|
END_SECTION(fh);
|
||||||
|
|
||||||
PREP_SECTION(f, 0x534B5254); // 'TRKS'
|
PREP_SECTION(fh, 0x534B5254); // 'TRKS'
|
||||||
if (!writeTRKSChunk(version, f)) {
|
if (!writeTRKSChunk(version, fh)) {
|
||||||
fprintf(stderr, "ERROR: failed to write TRKS chunk\n");
|
fprintf(stderr, "ERROR: failed to write TRKS chunk\n");
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
END_SECTION(f);
|
END_SECTION(fh);
|
||||||
|
|
||||||
// Write the metadata if we have any
|
// Write the metadata if we have any
|
||||||
if (metaData) {
|
if (metaData) {
|
||||||
PREP_SECTION(f, 0x4154454D); // 'META'
|
PREP_SECTION(fh, 0x4154454D); // 'META'
|
||||||
if (fwrite(metaData, 1, strlen(metaData), f) != strlen(metaData)) {
|
for (int i=0; i<strlen(metaData); i++) {
|
||||||
|
if (!write8(fh, metaData[i])) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "ERROR: failed to write META chunk\n");
|
fprintf(stderr, "ERROR: failed to write META chunk\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
END_SECTION(f);
|
}
|
||||||
|
END_SECTION(fh);
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: missing the WRIT chunk, if it exists
|
// FIXME: missing the WRIT chunk, if it exists
|
||||||
|
|
||||||
// Fix up the checksum
|
// Fix up the checksum
|
||||||
endPos = ftello(f);
|
endPos = g_filemanager->getSeekPosition(fh);
|
||||||
crcDataSize = endPos-crcPos-4;
|
crcDataSize = endPos-crcPos-4;
|
||||||
crcData = (uint8_t *)malloc(crcDataSize);
|
crcData = (uint8_t *)malloc(crcDataSize);
|
||||||
if (!crcData) {
|
if (!crcData) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "ERROR: failed to malloc crc data chunk\n");
|
fprintf(stderr, "ERROR: failed to malloc crc data chunk\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read the data in for checksumming
|
// Read the data in for checksumming
|
||||||
if (fseeko(f, crcPos+4, SEEK_SET)) {
|
// FIXME: no error checking on seek
|
||||||
fprintf(stderr, "ERROR: failed to fseek to crcPos+4 (0x%llX)\n", crcPos+4);
|
g_filemanager->setSeekPosition(fh, crcPos+4);
|
||||||
|
|
||||||
|
for (int i=0; i<crcDataSize; i++) {
|
||||||
|
if (!read8(fh, &crcData[i])) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
|
fprintf(stderr, "ERROR: failed to read in data for checksum [read %d, wanted %d]\n", tmp32, crcDataSize);
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
tmp32 = fread(crcData, 1, crcDataSize, f);
|
|
||||||
if (tmp32 != crcDataSize) {
|
|
||||||
fprintf(stderr, "ERROR: failed to read in data for checksum [read %d, wanted %d]\n", tmp32, crcDataSize);
|
|
||||||
goto done;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
tmp32 = compute_crc_32(crcData, crcDataSize);
|
tmp32 = compute_crc_32(crcData, crcDataSize);
|
||||||
// Write it back out
|
// Write it back out
|
||||||
fseeko(f, crcPos, SEEK_SET);
|
g_filemanager->setSeekPosition(fh, crcPos);
|
||||||
if (!write32(f, tmp32)) {
|
if (!write32(fh, tmp32)) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "ERROR: failed to write CRC\n");
|
fprintf(stderr, "ERROR: failed to write CRC\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -279,8 +301,9 @@ bool Woz::writeFile(uint8_t version, const char *filename)
|
|||||||
done:
|
done:
|
||||||
if (crcData)
|
if (crcData)
|
||||||
free(crcData);
|
free(crcData);
|
||||||
if (f)
|
if (fh != -1) {
|
||||||
fclose(f);
|
g_filemanager->closeFile(fh);
|
||||||
|
}
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -317,9 +340,11 @@ bool Woz::readDskFile(const char *filename, uint8_t subtype)
|
|||||||
{
|
{
|
||||||
bool retval = false;
|
bool retval = false;
|
||||||
|
|
||||||
FILE *f = fopen(filename, "r");
|
int8_t fh = g_filemanager->openFile(filename);
|
||||||
if (!f) {
|
if (fh == -1) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
perror("Unable to open input file");
|
perror("Unable to open input file");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -328,15 +353,16 @@ bool Woz::readDskFile(const char *filename, uint8_t subtype)
|
|||||||
// Now read in the 35 tracks of data from the DSK file and convert them to NIB
|
// Now read in the 35 tracks of data from the DSK file and convert them to NIB
|
||||||
uint8_t sectorData[256*16];
|
uint8_t sectorData[256*16];
|
||||||
for (int track=0; track<35; track++) {
|
for (int track=0; track<35; track++) {
|
||||||
uint32_t bytesRead = fread(sectorData, 1, 256*16, f);
|
for (int i=0; i<256*16; i++) {
|
||||||
if (bytesRead != 256*16) {
|
// FIXME: no error checking
|
||||||
fprintf(stderr, "Failed to read DSK data; got %d bytes, wanted %d\n", bytesRead, 256);
|
sectorData[i] = g_filemanager->readByte(fh);
|
||||||
goto done;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
tracks[track].trackData = (uint8_t *)calloc(NIBTRACKSIZE, 1);
|
tracks[track].trackData = (uint8_t *)calloc(NIBTRACKSIZE, 1);
|
||||||
if (!tracks[track].trackData) {
|
if (!tracks[track].trackData) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
fprintf(stderr, "Failed to malloc track data\n");
|
fprintf(stderr, "Failed to malloc track data\n");
|
||||||
|
#endif
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
tracks[track].startingBlock = STARTBLOCK + 13*track;
|
tracks[track].startingBlock = STARTBLOCK + 13*track;
|
||||||
@ -348,16 +374,18 @@ bool Woz::readDskFile(const char *filename, uint8_t subtype)
|
|||||||
retval = true;
|
retval = true;
|
||||||
|
|
||||||
done:
|
done:
|
||||||
if (f)
|
if (fh != -1)
|
||||||
fclose(f);
|
g_filemanager->closeFile(fh);
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Woz::readNibFile(const char *filename)
|
bool Woz::readNibFile(const char *filename)
|
||||||
{
|
{
|
||||||
FILE *f = fopen(filename, "r");
|
int8_t fh = g_filemanager->openFile(filename);
|
||||||
if (!f) {
|
if (fh == -1) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
perror("Unable to open input file");
|
perror("Unable to open input file");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -365,16 +393,18 @@ bool Woz::readNibFile(const char *filename)
|
|||||||
|
|
||||||
// Now read in the 35 tracks of data from the nib file
|
// Now read in the 35 tracks of data from the nib file
|
||||||
nibSector nibData[16];
|
nibSector nibData[16];
|
||||||
|
uint8_t *nibDataPtr = (uint8_t *)nibData;
|
||||||
for (int track=0; track<35; track++) {
|
for (int track=0; track<35; track++) {
|
||||||
uint32_t bytesRead = fread(nibData, 1, NIBTRACKSIZE, f);
|
for (int i=0; i<NIBTRACKSIZE; i++) {
|
||||||
if (bytesRead != NIBTRACKSIZE) {
|
// FIXME: no error checking
|
||||||
printf("Failed to read NIB data; got %d bytes, wanted %d\n", bytesRead, NIBTRACKSIZE);
|
nibDataPtr[i] = g_filemanager->readByte(fh);
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
tracks[track].trackData = (uint8_t *)calloc(NIBTRACKSIZE, 1);
|
tracks[track].trackData = (uint8_t *)calloc(NIBTRACKSIZE, 1);
|
||||||
if (!tracks[track].trackData) {
|
if (!tracks[track].trackData) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Failed to malloc track data\n");
|
printf("Failed to malloc track data\n");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -383,43 +413,53 @@ bool Woz::readNibFile(const char *filename)
|
|||||||
tracks[track].blockCount = 13;
|
tracks[track].blockCount = 13;
|
||||||
tracks[track].bitCount = NIBTRACKSIZE*8;
|
tracks[track].bitCount = NIBTRACKSIZE*8;
|
||||||
}
|
}
|
||||||
fclose(f);
|
g_filemanager->closeFile(fh);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Woz::readWozFile(const char *filename)
|
bool Woz::readWozFile(const char *filename)
|
||||||
{
|
{
|
||||||
FILE *f = fopen(filename, "r");
|
int8_t fh = g_filemanager->openFile(filename);
|
||||||
if (!f) {
|
if (fh == -1) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
perror("Unable to open input file");
|
perror("Unable to open input file");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Header
|
// Header
|
||||||
uint32_t h;
|
uint32_t h;
|
||||||
read32(f, &h);
|
read32(fh, &h);
|
||||||
if (h == 0x325A4F57 || h == 0x315A4F57) {
|
if (h == 0x325A4F57 || h == 0x315A4F57) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("WOZ%c disk image\n", (h & 0xFF000000)>>24);
|
printf("WOZ%c disk image\n", (h & 0xFF000000)>>24);
|
||||||
|
#endif
|
||||||
} else {
|
} else {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Unknown disk image type; can't continue\n");
|
printf("Unknown disk image type; can't continue\n");
|
||||||
fclose(f);
|
#endif
|
||||||
|
g_filemanager->closeFile(fh);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint32_t tmp;
|
uint32_t tmp;
|
||||||
if (!read32(f, &tmp)) {
|
if (!read32(fh, &tmp)) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Read failure\n");
|
printf("Read failure\n");
|
||||||
fclose(f);
|
#endif
|
||||||
|
g_filemanager->closeFile(fh);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (tmp != 0x0A0D0AFF) {
|
if (tmp != 0x0A0D0AFF) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("WOZ header failure; exiting\n");
|
printf("WOZ header failure; exiting\n");
|
||||||
fclose(f);
|
#endif
|
||||||
|
g_filemanager->closeFile(fh);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
uint32_t crc32;
|
uint32_t crc32;
|
||||||
read32(f, &crc32);
|
read32(fh, &crc32);
|
||||||
// printf("Disk crc32 should be 0x%X\n", crc32);
|
// printf("Disk crc32 should be 0x%X\n", crc32);
|
||||||
// FIXME: check CRC. Remember that 0x00 means "don't check CRC"
|
// FIXME: check CRC. Remember that 0x00 means "don't check CRC"
|
||||||
|
|
||||||
@ -431,45 +471,47 @@ bool Woz::readWozFile(const char *filename)
|
|||||||
#define cTRKS 4
|
#define cTRKS 4
|
||||||
|
|
||||||
while (1) {
|
while (1) {
|
||||||
if (fseeko(f, fpos, SEEK_SET)) {
|
g_filemanager->setSeekPosition(fh, fpos); // FIXME: no error checking
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint32_t chunkType;
|
uint32_t chunkType;
|
||||||
if (!read32(f, &chunkType)) {
|
if (!read32(fh, &chunkType)) {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
uint32_t chunkDataSize;
|
uint32_t chunkDataSize;
|
||||||
read32(f, &chunkDataSize);
|
read32(fh, &chunkDataSize);
|
||||||
|
|
||||||
bool isOk;
|
bool isOk;
|
||||||
|
|
||||||
switch (chunkType) {
|
switch (chunkType) {
|
||||||
case 0x4F464E49: // 'INFO'
|
case 0x4F464E49: // 'INFO'
|
||||||
isOk = parseInfoChunk(f, chunkDataSize);
|
isOk = parseInfoChunk(fh, chunkDataSize);
|
||||||
haveData |= cINFO;
|
haveData |= cINFO;
|
||||||
break;
|
break;
|
||||||
case 0x50414D54: // 'TMAP'
|
case 0x50414D54: // 'TMAP'
|
||||||
isOk = parseTMAPChunk(f, chunkDataSize);
|
isOk = parseTMAPChunk(fh, chunkDataSize);
|
||||||
haveData |= cTMAP;
|
haveData |= cTMAP;
|
||||||
break;
|
break;
|
||||||
case 0x534B5254: // 'TRKS'
|
case 0x534B5254: // 'TRKS'
|
||||||
isOk = parseTRKSChunk(f, chunkDataSize);
|
isOk = parseTRKSChunk(fh, chunkDataSize);
|
||||||
haveData |= cTRKS;
|
haveData |= cTRKS;
|
||||||
break;
|
break;
|
||||||
case 0x4154454D: // 'META'
|
case 0x4154454D: // 'META'
|
||||||
isOk = parseMetaChunk(f, chunkDataSize);
|
isOk = parseMetaChunk(fh, chunkDataSize);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Unknown chunk type 0x%X\n", chunkType);
|
printf("Unknown chunk type 0x%X\n", chunkType);
|
||||||
fclose(f);
|
#endif
|
||||||
|
g_filemanager->closeFile(fh);
|
||||||
return false;
|
return false;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isOk) {
|
if (!isOk) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Chunk parsing [0x%X] failed; exiting\n", chunkType);
|
printf("Chunk parsing [0x%X] failed; exiting\n", chunkType);
|
||||||
fclose(f);
|
#endif
|
||||||
|
g_filemanager->closeFile(fh);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -477,20 +519,26 @@ bool Woz::readWozFile(const char *filename)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (haveData != 0x07) {
|
if (haveData != 0x07) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("ERROR: missing one or more critical sections\n");
|
printf("ERROR: missing one or more critical sections\n");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i=0; i<35; i++) {
|
for (int i=0; i<35; i++) {
|
||||||
if (!readQuarterTrackData(f, i*4)) {
|
if (!readQuarterTrackData(fh, i*4)) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Failed to read QTD for track %d\n", i);
|
printf("Failed to read QTD for track %d\n", i);
|
||||||
fclose(f);
|
#endif
|
||||||
|
g_filemanager->closeFile(fh);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fclose(f);
|
g_filemanager->closeFile(fh);
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("File read successful\n");
|
printf("File read successful\n");
|
||||||
|
#endif
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -500,7 +548,9 @@ bool Woz::readFile(const char *filename, uint8_t forceType)
|
|||||||
// Try to determine type from the file extension
|
// Try to determine type from the file extension
|
||||||
const char *p = strrchr(filename, '.');
|
const char *p = strrchr(filename, '.');
|
||||||
if (!p) {
|
if (!p) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Unable to determine file type of '%s'\n", filename);
|
printf("Unable to determine file type of '%s'\n", filename);
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
if (strcasecmp(p, ".woz") == 0) {
|
if (strcasecmp(p, ".woz") == 0) {
|
||||||
@ -513,37 +563,47 @@ bool Woz::readFile(const char *filename, uint8_t forceType)
|
|||||||
} else if (strcasecmp(p, ".nib") == 0) {
|
} else if (strcasecmp(p, ".nib") == 0) {
|
||||||
forceType = T_NIB;
|
forceType = T_NIB;
|
||||||
} else {
|
} else {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Unable to determine file type of '%s'\n", filename);
|
printf("Unable to determine file type of '%s'\n", filename);
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
switch (forceType) {
|
switch (forceType) {
|
||||||
case T_WOZ:
|
case T_WOZ:
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("reading woz file %s\n", filename);
|
printf("reading woz file %s\n", filename);
|
||||||
|
#endif
|
||||||
return readWozFile(filename);
|
return readWozFile(filename);
|
||||||
case T_DSK:
|
case T_DSK:
|
||||||
case T_PO:
|
case T_PO:
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("reading DSK file %s\n", filename);
|
printf("reading DSK file %s\n", filename);
|
||||||
|
#endif
|
||||||
return readDskFile(filename, forceType);
|
return readDskFile(filename, forceType);
|
||||||
case T_NIB:
|
case T_NIB:
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("reading NIB file %s\n", filename);
|
printf("reading NIB file %s\n", filename);
|
||||||
|
#endif
|
||||||
return readNibFile(filename);
|
return readNibFile(filename);
|
||||||
default:
|
default:
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Unknown disk type; unable to read\n");
|
printf("Unknown disk type; unable to read\n");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Woz::parseTRKSChunk(FILE *f, uint32_t chunkSize)
|
bool Woz::parseTRKSChunk(int8_t fh, uint32_t chunkSize)
|
||||||
{
|
{
|
||||||
if (di.version == 2) {
|
if (di.version == 2) {
|
||||||
for (int i=0; i<160; i++) {
|
for (int i=0; i<160; i++) {
|
||||||
if (!read16(f, &tracks[i].startingBlock))
|
if (!read16(fh, &tracks[i].startingBlock))
|
||||||
return false;
|
return false;
|
||||||
if (!read16(f, &tracks[i].blockCount))
|
if (!read16(fh, &tracks[i].blockCount))
|
||||||
return false;
|
return false;
|
||||||
if (!read32(f, &tracks[i].bitCount))
|
if (!read32(fh, &tracks[i].bitCount))
|
||||||
return false;
|
return false;
|
||||||
tracks[i].startingByte = 0; // v1-specific
|
tracks[i].startingByte = 0; // v1-specific
|
||||||
}
|
}
|
||||||
@ -557,9 +617,9 @@ bool Woz::parseTRKSChunk(FILE *f, uint32_t chunkSize)
|
|||||||
tracks[trackNumber].startingByte = trackNumber * 6656 + 256;
|
tracks[trackNumber].startingByte = trackNumber * 6656 + 256;
|
||||||
tracks[trackNumber].startingBlock = 0; // v2-specific
|
tracks[trackNumber].startingBlock = 0; // v2-specific
|
||||||
tracks[trackNumber].blockCount = 13;
|
tracks[trackNumber].blockCount = 13;
|
||||||
fseeko(f, (trackNumber * 6656 + 256) + 6648, SEEK_SET);
|
g_filemanager->setSeekPosition(fh, (trackNumber * 6656 + 256) + 6648); // FIXME: no error checking
|
||||||
uint16_t numBits;
|
uint16_t numBits;
|
||||||
if (!read16(f, &numBits)) {
|
if (!read16(fh, &numBits)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
tracks[trackNumber].bitCount = numBits;
|
tracks[trackNumber].bitCount = numBits;
|
||||||
@ -570,15 +630,17 @@ bool Woz::parseTRKSChunk(FILE *f, uint32_t chunkSize)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Woz::parseTMAPChunk(FILE *f, uint32_t chunkSize)
|
bool Woz::parseTMAPChunk(int8_t fh, uint32_t chunkSize)
|
||||||
{
|
{
|
||||||
if (chunkSize != 0xa0) {
|
if (chunkSize != 0xa0) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("TMAP chunk is the wrong size; aborting\n");
|
printf("TMAP chunk is the wrong size; aborting\n");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (int i=0; i<40*4; i++) {
|
for (int i=0; i<40*4; i++) {
|
||||||
if (!read8(f, (uint8_t *)&quarterTrackMap[i]))
|
if (!read8(fh, (uint8_t *)&quarterTrackMap[i]))
|
||||||
return false;
|
return false;
|
||||||
chunkSize--;
|
chunkSize--;
|
||||||
}
|
}
|
||||||
@ -587,54 +649,60 @@ bool Woz::parseTMAPChunk(FILE *f, uint32_t chunkSize)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// return true if successful
|
// return true if successful
|
||||||
bool Woz::parseInfoChunk(FILE *f, uint32_t chunkSize)
|
bool Woz::parseInfoChunk(int8_t fh, uint32_t chunkSize)
|
||||||
{
|
{
|
||||||
if (chunkSize != 60) {
|
if (chunkSize != 60) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("INFO chunk size is not 60; aborting\n");
|
printf("INFO chunk size is not 60; aborting\n");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!read8(f, &di.version))
|
if (!read8(fh, &di.version))
|
||||||
return false;
|
return false;
|
||||||
if (di.version > 2) {
|
if (di.version > 2) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Incorrect version header; aborting\n");
|
printf("Incorrect version header; aborting\n");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!read8(f, &di.diskType))
|
if (!read8(fh, &di.diskType))
|
||||||
return false;
|
return false;
|
||||||
if (di.diskType != 1) {
|
if (di.diskType != 1) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("Not a 5.25\" disk image; aborting\n");
|
printf("Not a 5.25\" disk image; aborting\n");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!read8(f, &di.writeProtected))
|
if (!read8(fh, &di.writeProtected))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (!read8(f, &di.synchronized))
|
if (!read8(fh, &di.synchronized))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (!read8(f, &di.cleaned))
|
if (!read8(fh, &di.cleaned))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
di.creator[32] = 0;
|
di.creator[32] = 0;
|
||||||
for (int i=0; i<32; i++) {
|
for (int i=0; i<32; i++) {
|
||||||
if (!read8(f, (uint8_t *)&di.creator[i]))
|
if (!read8(fh, (uint8_t *)&di.creator[i]))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (di.version >= 2) {
|
if (di.version >= 2) {
|
||||||
if (!read8(f, &di.diskSides))
|
if (!read8(fh, &di.diskSides))
|
||||||
return false;
|
return false;
|
||||||
if (!read8(f, &di.bootSectorFormat))
|
if (!read8(fh, &di.bootSectorFormat))
|
||||||
return false;
|
return false;
|
||||||
if (!read8(f, &di.optimalBitTiming))
|
if (!read8(fh, &di.optimalBitTiming))
|
||||||
return false;
|
return false;
|
||||||
if (!read16(f, &di.compatHardware))
|
if (!read16(fh, &di.compatHardware))
|
||||||
return false;
|
return false;
|
||||||
if (!read16(f, &di.requiredRam))
|
if (!read16(fh, &di.requiredRam))
|
||||||
return false;
|
return false;
|
||||||
if (!read16(f, &di.largestTrack))
|
if (!read16(fh, &di.largestTrack))
|
||||||
return false;
|
return false;
|
||||||
} else {
|
} else {
|
||||||
di.diskSides = 0;
|
di.diskSides = 0;
|
||||||
@ -649,21 +717,22 @@ bool Woz::parseInfoChunk(FILE *f, uint32_t chunkSize)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Woz::parseMetaChunk(FILE *f, uint32_t chunkSize)
|
bool Woz::parseMetaChunk(int8_t fh, uint32_t chunkSize)
|
||||||
{
|
{
|
||||||
metaData = (char *)calloc(chunkSize+1, 1);
|
metaData = (char *)calloc(chunkSize+1, 1);
|
||||||
if (!metaData)
|
if (!metaData)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if (fread(metaData, 1, chunkSize, f) != chunkSize)
|
for (int i=0; i<chunkSize; i++) {
|
||||||
return false;
|
metaData[i] = g_filemanager->readByte(fh); // FIXME: no error checking
|
||||||
|
}
|
||||||
|
|
||||||
metaData[chunkSize] = 0;
|
metaData[chunkSize] = 0;
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Woz::readQuarterTrackData(FILE *f, uint8_t quartertrack)
|
bool Woz::readQuarterTrackData(int8_t fh, uint8_t quartertrack)
|
||||||
{
|
{
|
||||||
uint8_t targetImageTrack = quarterTrackMap[quartertrack];
|
uint8_t targetImageTrack = quarterTrackMap[quartertrack];
|
||||||
if (targetImageTrack == 0xFF) {
|
if (targetImageTrack == 0xFF) {
|
||||||
@ -681,25 +750,20 @@ bool Woz::readQuarterTrackData(FILE *f, uint8_t quartertrack)
|
|||||||
if (di.version == 1) count = (tracks[targetImageTrack].bitCount / 8) + ((tracks[targetImageTrack].bitCount % 8) ? 1 : 0);
|
if (di.version == 1) count = (tracks[targetImageTrack].bitCount / 8) + ((tracks[targetImageTrack].bitCount % 8) ? 1 : 0);
|
||||||
tracks[targetImageTrack].trackData = (uint8_t *)calloc(count, 1);
|
tracks[targetImageTrack].trackData = (uint8_t *)calloc(count, 1);
|
||||||
if (!tracks[targetImageTrack].trackData) {
|
if (!tracks[targetImageTrack].trackData) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
perror("Failed to alloc buf to read track magnetic data");
|
perror("Failed to alloc buf to read track magnetic data");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (di.version == 1) {
|
if (di.version == 1) {
|
||||||
if (fseeko(f, tracks[targetImageTrack].startingByte, SEEK_SET)) {
|
g_filemanager->setSeekPosition(fh, tracks[targetImageTrack].startingByte); // FIXME: no error checking
|
||||||
perror("Failed to seek to start of block");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
if (fseeko(f, bitsStartBlock*512, SEEK_SET)) {
|
g_filemanager->setSeekPosition(fh, bitsStartBlock*512); // FIXME: no error checking
|
||||||
perror("Failed to seek to start of block");
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
}
|
for (int i=0; i<count; i++) {
|
||||||
uint32_t didRead = fread(tracks[targetImageTrack].trackData, 1, count, f);
|
// FIXME: no error checking
|
||||||
if (didRead != count) {
|
tracks[targetImageTrack].trackData[i] = g_filemanager->readByte(fh);
|
||||||
printf("Failed to read all track data for track [read %d, wanted %d]\n", didRead, count);
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@ -781,17 +845,17 @@ bool Woz::readSectorData(uint8_t track, uint8_t sector, nibSector *sectorData)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Woz::writeInfoChunk(uint8_t version, FILE *f)
|
bool Woz::writeInfoChunk(uint8_t version, int8_t fh)
|
||||||
{
|
{
|
||||||
if (!write8(f, version) ||
|
if (!write8(fh, version) ||
|
||||||
!write8(f, di.diskType) ||
|
!write8(fh, di.diskType) ||
|
||||||
!write8(f, di.writeProtected) ||
|
!write8(fh, di.writeProtected) ||
|
||||||
!write8(f, di.synchronized) ||
|
!write8(fh, di.synchronized) ||
|
||||||
!write8(f, di.cleaned))
|
!write8(fh, di.cleaned))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
for (int i=0; i<32; i++) {
|
for (int i=0; i<32; i++) {
|
||||||
if (!write8(f, di.creator[i]))
|
if (!write8(fh, di.creator[i]))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -800,37 +864,39 @@ bool Woz::writeInfoChunk(uint8_t version, FILE *f)
|
|||||||
if (di.diskSides == 0)
|
if (di.diskSides == 0)
|
||||||
di.diskSides = 1;
|
di.diskSides = 1;
|
||||||
|
|
||||||
if ( !write8(f, di.diskSides) ||
|
if ( !write8(fh, di.diskSides) ||
|
||||||
!write8(f, di.bootSectorFormat) ||
|
!write8(fh, di.bootSectorFormat) ||
|
||||||
!write8(f, di.optimalBitTiming) ||
|
!write8(fh, di.optimalBitTiming) ||
|
||||||
!write16(f, di.compatHardware) ||
|
!write16(fh, di.compatHardware) ||
|
||||||
!write16(f, di.requiredRam) ||
|
!write16(fh, di.requiredRam) ||
|
||||||
!write16(f, di.largestTrack))
|
!write16(fh, di.largestTrack))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Padding
|
// Padding
|
||||||
for (int i=0; i<((version==1)?23:14); i++) {
|
for (int i=0; i<((version==1)?23:14); i++) {
|
||||||
if (!write8(f, 0))
|
if (!write8(fh, 0))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Woz::writeTMAPChunk(uint8_t version, FILE *f)
|
bool Woz::writeTMAPChunk(uint8_t version, int8_t fh)
|
||||||
{
|
{
|
||||||
for (int i=0; i<40*4; i++) {
|
for (int i=0; i<40*4; i++) {
|
||||||
if (!write8(f, quarterTrackMap[i]))
|
if (!write8(fh, quarterTrackMap[i]))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Woz::writeTRKSChunk(uint8_t version, FILE *f)
|
bool Woz::writeTRKSChunk(uint8_t version, int8_t fh)
|
||||||
{
|
{
|
||||||
if (version == 1) {
|
if (version == 1) {
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
printf("V1 write is not implemented\n");
|
printf("V1 write is not implemented\n");
|
||||||
|
#endif
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -851,11 +917,11 @@ bool Woz::writeTRKSChunk(uint8_t version, FILE *f)
|
|||||||
tracks[i].bitCount = 0;
|
tracks[i].bitCount = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!write16(f, tracks[i].startingBlock))
|
if (!write16(fh, tracks[i].startingBlock))
|
||||||
return false;
|
return false;
|
||||||
if (!write16(f, tracks[i].blockCount))
|
if (!write16(fh, tracks[i].blockCount))
|
||||||
return false;
|
return false;
|
||||||
if (!write32(f, tracks[i].bitCount))
|
if (!write32(fh, tracks[i].bitCount))
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -863,13 +929,16 @@ bool Woz::writeTRKSChunk(uint8_t version, FILE *f)
|
|||||||
for (int i=0; i<160; i++) {
|
for (int i=0; i<160; i++) {
|
||||||
if (tracks[i].startingBlock &&
|
if (tracks[i].startingBlock &&
|
||||||
tracks[i].blockCount) {
|
tracks[i].blockCount) {
|
||||||
fseeko(f, tracks[i].startingBlock * 512, SEEK_SET);
|
g_filemanager->setSeekPosition(fh, tracks[i].startingBlock*512); // FIXME: no error checking
|
||||||
uint32_t writeSize = (tracks[i].bitCount / 8) + ((tracks[i].bitCount % 8) ? 1 : 0);
|
uint32_t writeSize = (tracks[i].bitCount / 8) + ((tracks[i].bitCount % 8) ? 1 : 0);
|
||||||
if (fwrite(tracks[i].trackData, 1, writeSize, f) != writeSize)
|
for (int j=0; j<writeSize; j++) {
|
||||||
|
if (!g_filemanager->writeByte(fh, tracks[i].trackData[j])) {
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
uint8_t c = 0;
|
uint8_t c = 0;
|
||||||
while (writeSize < tracks[i].blockCount * 512) {
|
while (writeSize < tracks[i].blockCount * 512) {
|
||||||
if (fwrite(&c, 1, 1, f) != 1)
|
if (!write8(fh, c))
|
||||||
return false;
|
return false;
|
||||||
writeSize++;
|
writeSize++;
|
||||||
}
|
}
|
||||||
@ -902,6 +971,7 @@ bool Woz::decodeWozTrackToDsk(uint8_t track, uint8_t subtype, uint8_t sectorData
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
void Woz::dumpInfo()
|
void Woz::dumpInfo()
|
||||||
{
|
{
|
||||||
printf("WOZ image version %d\n", di.version);
|
printf("WOZ image version %d\n", di.version);
|
||||||
@ -1074,6 +1144,7 @@ void Woz::dumpInfo()
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
bool Woz::isSynchronized()
|
bool Woz::isSynchronized()
|
||||||
{
|
{
|
||||||
|
18
apple/woz.h
18
apple/woz.h
@ -46,7 +46,9 @@ class Woz {
|
|||||||
bool decodeWozTrackToNib(uint8_t track, nibSector sectorData[16]);
|
bool decodeWozTrackToNib(uint8_t track, nibSector sectorData[16]);
|
||||||
bool decodeWozTrackToDsk(uint8_t track, uint8_t subtype, uint8_t sectorData[256*16]);
|
bool decodeWozTrackToDsk(uint8_t track, uint8_t subtype, uint8_t sectorData[256*16]);
|
||||||
|
|
||||||
|
#ifndef TEENSYDUINO
|
||||||
void dumpInfo();
|
void dumpInfo();
|
||||||
|
#endif
|
||||||
|
|
||||||
bool isSynchronized();
|
bool isSynchronized();
|
||||||
|
|
||||||
@ -59,16 +61,16 @@ class Woz {
|
|||||||
|
|
||||||
uint8_t fakeBit();
|
uint8_t fakeBit();
|
||||||
|
|
||||||
bool parseTRKSChunk(FILE *f, uint32_t chunkSize);
|
bool parseTRKSChunk(int8_t fh, uint32_t chunkSize);
|
||||||
bool parseTMAPChunk(FILE *f, uint32_t chunkSize);
|
bool parseTMAPChunk(int8_t fh, uint32_t chunkSize);
|
||||||
bool parseInfoChunk(FILE *f, uint32_t chunkSize);
|
bool parseInfoChunk(int8_t fh, uint32_t chunkSize);
|
||||||
bool parseMetaChunk(FILE *f, uint32_t chunkSize);
|
bool parseMetaChunk(int8_t fh, uint32_t chunkSize);
|
||||||
|
|
||||||
bool writeInfoChunk(uint8_t version, FILE *f);
|
bool writeInfoChunk(uint8_t version, int8_t fh);
|
||||||
bool writeTMAPChunk(uint8_t version, FILE *f);
|
bool writeTMAPChunk(uint8_t version, int8_t fh);
|
||||||
bool writeTRKSChunk(uint8_t version, FILE *f);
|
bool writeTRKSChunk(uint8_t version, int8_t fh);
|
||||||
|
|
||||||
bool readQuarterTrackData(FILE *f, uint8_t quartertrack);
|
bool readQuarterTrackData(int8_t fh, uint8_t quartertrack);
|
||||||
bool readSectorData(uint8_t track, uint8_t sector, nibSector *sectorData);
|
bool readSectorData(uint8_t track, uint8_t sector, nibSector *sectorData);
|
||||||
|
|
||||||
void _initInfo();
|
void _initInfo();
|
||||||
|
37
cpu.cpp
37
cpu.cpp
@ -9,6 +9,13 @@
|
|||||||
|
|
||||||
#include "globals.h"
|
#include "globals.h"
|
||||||
|
|
||||||
|
// define DEBUGSTEPS to show disassembly of each instruction as it's processed
|
||||||
|
//#define DEBUGSTEPS
|
||||||
|
#ifdef DEBUGSTEPS
|
||||||
|
#include "disassembler.h"
|
||||||
|
extern Disassembler dis;
|
||||||
|
#endif
|
||||||
|
|
||||||
// To see calls to unimplemented opcodes, define this:
|
// To see calls to unimplemented opcodes, define this:
|
||||||
//#define VERBOSE_CPU_ERRORS
|
//#define VERBOSE_CPU_ERRORS
|
||||||
|
|
||||||
@ -487,6 +494,36 @@ uint8_t Cpu::step()
|
|||||||
irq();
|
irq();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifdef DEBUGSTEPS
|
||||||
|
static uint8_t cmdbuf[10];
|
||||||
|
static char buf[50];
|
||||||
|
|
||||||
|
uint16_t loc=g_cpu->pc;
|
||||||
|
for (int idx=0; idx<sizeof(cmdbuf); idx++) {
|
||||||
|
cmdbuf[idx] = g_vm->getMMU()->read(loc+idx);
|
||||||
|
}
|
||||||
|
dis.instructionToMnemonic(loc, cmdbuf, buf, sizeof(buf));
|
||||||
|
while (strlen(buf) < 25) {
|
||||||
|
strcat(buf, " ");
|
||||||
|
}
|
||||||
|
printf("%s ;", buf);
|
||||||
|
|
||||||
|
uint8_t p = g_cpu->flags;
|
||||||
|
printf("BS/BT: %02x/%02x A: %02x X: %02x Y: %02x SP: %02x Flags: %c%cx%c%c%c%c%c\n",
|
||||||
|
g_vm->getMMU()->read(0x3D),
|
||||||
|
g_vm->getMMU()->read(0x41),
|
||||||
|
g_cpu->a, g_cpu->x, g_cpu->y, g_cpu->sp,
|
||||||
|
p & (1<<7) ? 'N':' ',
|
||||||
|
p & (1<<6) ? 'V':' ',
|
||||||
|
p & (1<<4) ? 'B':' ',
|
||||||
|
p & (1<<3) ? 'D':' ',
|
||||||
|
p & (1<<2) ? 'I':' ',
|
||||||
|
p & (1<<1) ? 'Z':' ',
|
||||||
|
p & (1<<0) ? 'C':' '
|
||||||
|
);
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
||||||
uint8_t m = readmem(pc++);
|
uint8_t m = readmem(pc++);
|
||||||
|
|
||||||
optype_t opcode = opcodes[m];
|
optype_t opcode = opcodes[m];
|
||||||
|
@ -112,6 +112,16 @@ class FileManager {
|
|||||||
|
|
||||||
virtual void getRootPath(char *toWhere, int8_t maxLen) = 0;
|
virtual void getRootPath(char *toWhere, int8_t maxLen) = 0;
|
||||||
|
|
||||||
|
virtual uint32_t getSeekPosition(int8_t fd) {
|
||||||
|
return fileSeekPositions[fd];
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual void setSeekPosition(int8_t fd, uint32_t pos) {
|
||||||
|
fileSeekPositions[fd] = pos;
|
||||||
|
};
|
||||||
|
|
||||||
|
virtual void seekToEnd(int8_t fd) = 0;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
unsigned long fileSeekPositions[MAXFILES];
|
unsigned long fileSeekPositions[MAXFILES];
|
||||||
char cachedNames[MAXFILES][MAXPATH];
|
char cachedNames[MAXFILES][MAXPATH];
|
||||||
|
@ -384,3 +384,14 @@ void NixFileManager::getRootPath(char *toWhere, int8_t maxLen)
|
|||||||
strcpy(toWhere, ROOTDIR);
|
strcpy(toWhere, ROOTDIR);
|
||||||
// strncpy(toWhere, ROOTDIR, maxLen);
|
// strncpy(toWhere, ROOTDIR, maxLen);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void NixFileManager::seekToEnd(int8_t fd)
|
||||||
|
{
|
||||||
|
// This could just be a stat call...
|
||||||
|
FILE *f = fopen(cachedNames[fd], "r");
|
||||||
|
if (f) {
|
||||||
|
fseeko(f, 0, SEEK_END);
|
||||||
|
fileSeekPositions[fd] = ftello(f);
|
||||||
|
fclose(f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -29,6 +29,8 @@ class NixFileManager : public FileManager {
|
|||||||
|
|
||||||
void getRootPath(char *toWhere, int8_t maxLen);
|
void getRootPath(char *toWhere, int8_t maxLen);
|
||||||
|
|
||||||
|
virtual void seekToEnd(int8_t fd);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
int8_t numCached;
|
int8_t numCached;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user