AppleWin/source/Harddisk.cpp

1427 lines
45 KiB
C++

/*
AppleWin : An Apple //e emulator for Windows
Copyright (C) 1994-1996, Michael O'Brien
Copyright (C) 1999-2001, Oliver Schmidt
Copyright (C) 2002-2005, Tom Charlesworth
Copyright (C) 2006-2015, Tom Charlesworth, Michael Pohoreski
AppleWin 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 2 of the License, or
(at your option) any later version.
AppleWin 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 AppleWin; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
/* Description: Hard drive emulation
*
* Author: Copyright (c) 2005, Robert Hoem
*/
#include "StdAfx.h"
#include "Harddisk.h"
#include "Core.h"
#include "Interface.h"
#include "CPU.h"
#include "DiskImage.h" // ImageError_e, Disk_Status_e
#include "Memory.h"
#include "Registry.h"
#include "SaveState.h"
#include "YamlHelper.h"
#include "Debugger/Debug.h"
#include "../resource/resource.h"
/*
Memory map ProDOS BLK device (IO addr + s*$10):
. "hddrvr" v1 and v2 firmware
C080 (r) EXECUTE AND RETURN STATUS
C081 (r) STATUS (or ERROR): b7=busy, b0=error
C082 (r/w) COMMAND
C083 (r/w) UNIT NUMBER
C084 (r/w) LOW BYTE OF MEMORY BUFFER
C085 (r/w) HIGH BYTE OF MEMORY BUFFER
C086 (r/w) LOW BYTE OF BLOCK NUMBER
C087 (r/w) HIGH BYTE OF BLOCK NUMBER
C088 (r) NEXT BYTE (legacy read-only port - still supported)
C089 (r) LOW BYTE OF DISK IMAGE SIZE IN BLOCKS
C08A (r) HIGH BYTE OF DISK IMAGE SIZE IN BLOCKS
Firmware notes:
. ROR ABS16,X and ROL ABS16,X - only used for $C081+s*$10 STATUS register:
6502: double read (old data), write (old data), write (new data). The writes are harmless as writes to STATUS are ignored.
65C02: double read (old data), write (new data). The write is harmless as writes to STATUS are ignored.
. STA ABS16,X does a false-read. This is harmless for writable I/O registers, since the false-read has no side effect.
---
Memory map SmartPort device (IO addr + s*$10):
. "hdc-smartport" firmware
. I/O basically compatible with older "hddrvr" firmware
C080 (r) EXECUTE AND RETURN STATUS; subsequent reads just return STATUS (need to write COMMAND again for EXECUTE)
C081 (r) STATUS : b7=busy, b0=error
C082 (w) COMMAND : BLK = $00 status, $01 read, $02 write. SP = $80 status, $81 read, $82 write,
C083 (w) UNIT NUMBER : BLK = DSSS0000 if SSS != n from CnXX, add 2 to D (4 drives support). SP = $00,$01.....
C084 (w) LOW BYTE OF MEMORY BUFFER
C085 (w) HIGH BYTE OF MEMORY BUFFER
C086 (w) STATUS CODE : write SP status code $00(device status), $03(device info block)
C086 (w) LOW BYTE OF BLOCK NUMBER : BLK = 16 bit value. SP = 24 bit value
C087 (w) MIDDLE BYTE OF BLOCK NUMBER
C088 (w) HIGH BYTE OF BLOCK NUMBER (SP only)
; C088 (r) NEXT BYTE (legacy read-only port - still supported)
C089 (r) LOW BYTE OF DISK IMAGE SIZE IN BLOCKS
C08A (r) HIGH BYTE OF DISK IMAGE SIZE IN BLOCKS
C089 (w) a 6-deep FIFO to write: command, unitNum, memPtr(2), blockNum(2); for BLK driver
C08A (w) a 7-deep FIFO to write: command, unitNum, memPtr(2), blockNum(3); for SP: first byte gets OR'd with $80 (ie. to indicate it's an SP command)
*/
/*
Hard drive emulation in AppleWin - for the HDDRVR v1 & v2 firmware
Concept
To emulate a 32MiB hard drive connected to an Apple IIe via AppleWin.
Designed to work with Autoboot ROM and ProDOS.
Overview
1. Hard drive image file
The hard drive image file (.HDV) will be formatted into blocks of 512
bytes, in a linear fashion. The internal formatting and meaning of each
block to be decided by the Apple's operating system (ProDOS). To create
an empty .HDV file, just create a 0 byte file.
Use the -harddisknumblocks n command line option to set the disk size
returned by ProDOS status calls.
2. Emulation code
There are 4 commands ProDOS will send to a block device.
Listed below are each command and how it's handled:
1. STATUS
In the emulation's case, returns only a DEVICE OK (0), DEVICE I/O ERROR ($27) or DEVICE NOT CONNECTED ($28)
DEVICE NOT CONNECTED only returned if no HDV file is selected.
2. READ
Loads requested block into a 512 byte buffer by attempting to seek to
location in HDV file.
If seek fails, returns a DEVICE I/O ERROR. Resets m_buf_ptr used by legacy HD_NEXTBYTE
Copies requested block from a 512 byte buffer to the Apple's memory.
Sets STATUS.busy=1 until the DMA operation completes.
Returns a DEVICE OK if read was successful, or a DEVICE I/O ERROR otherwise.
3. WRITE
Copies requested block from the Apple's memory to a 512 byte buffer
then attempts to seek to requested block.
If the seek fails (usually because the seek is beyond the EOF for the
HDV file), the emulation will attempt to "grow" the HDV file to accommodate.
Once the file can accommodate, or if the seek did not fail, the buffer is
written to the HDV file. NOTE: A2PC will grow *AND* shrink the HDV file.
Sets STATUS.busy=1 until the DMA operation completes.
I didn't see the point in shrinking the file as this behaviour would require
patching prodos (to detect DELETE FILE calls).
4. FORMAT
This is used for low level formatting of the device. (GH#88)
(Also in the case of a tape or SCSI drive, perhaps.)
3. Bugs
The only thing I've noticed is that Copy II+ 7.1 seems to crash or stall
occasionally when trying to calculate how many free blocks are available
when running a catalog. This might be due to the great number of blocks
available. Also, DDD pro will not optimise the disk correctly (it's
doing a disk defragment of some sort, and when it requests a block outside
the range of the image file, it starts getting I/O errors), so don't
bother. Any program that preforms a read before write to an "unwritten"
block (a block that should be located beyond the EOF of the .HDV, which is
valid for writing but not for reading until written to) will fail with I/O
errors (although these are few and far between).
I'm sure there are programs out there that may try to use the I/O ports in
ways they weren't designed (like telling Ultima 5 that you have a Phasor
sound card in slot 7 is a generally bad idea) will cause problems.
*/
HarddiskInterfaceCard::HarddiskInterfaceCard(UINT slot) :
Card(CT_GenericHDD, slot), m_userNumBlocks(0), m_isFirmwareV1or2(false), m_useHdcFirmwareV1(false), m_useHdcFirmwareV2(false), m_useHdcFirmwareMode(HdcDefault)
{
if (m_slot != SLOT5 && m_slot != SLOT7) // fixme
ThrowErrorInvalidSlot();
m_unitNum = (HARDDISK_1 << 7) | (m_slot << 4); // b7=unit, b6:4=slot
// The HDD interface has a single Command register for both drives:
// . ProDOS will write to Command before switching drives
m_command = 0;
m_fifoIdx = 0;
// SmartPort Status cmd's Status code
m_statusCode = 0;
// SmartPort Controller is always loaded
m_smartPortController.m_imageloaded = true;
// Interface busy doing DMA for r/w when current cycle is earlier than this cycle
m_notBusyCycle = 0;
m_saveDiskImage = true; // Save the DiskImage name to Registry
m_saveStateFirmwareV1 = false;
m_saveStateFirmwareV2 = false;
m_saveStateFirmwareValid = false;
memset(m_saveStateFirmware, 0, sizeof(m_saveStateFirmware));
}
HarddiskInterfaceCard::~HarddiskInterfaceCard(void)
{
for (UINT i = 0; i < NUM_HARDDISKS; i++)
CleanupDriveInternal(i);
}
void HarddiskInterfaceCard::Reset(const bool powerCycle)
{
for (UINT i = 0; i < NUM_HARDDISKS; i++)
m_hardDiskDrive[i].m_error = 0;
m_fifoIdx = 0;
}
//===========================================================================
void HarddiskInterfaceCard::InitializeIO(LPBYTE pCxRomPeripheral)
{
const DWORD HARDDISK_FW_SIZE = APPLE_SLOT_SIZE;
WORD id = IDR_HDC_SMARTPORT_FW; // If not enhanced //e, then modify the firmware later
// Use any cmd line override
if (m_useHdcFirmwareV1 || m_saveStateFirmwareV1) id = IDR_HDDRVR_FW;
else if (m_useHdcFirmwareV2 || m_saveStateFirmwareV2) id = IDR_HDDRVR_V2_FW;
m_saveStateFirmwareV1 = false;
m_saveStateFirmwareV2 = false;
bool allowFirmwareMods = false;
BYTE* pData = NULL;
if (!m_saveStateFirmwareValid)
{
pData = GetFrame().GetResource(id, "FIRMWARE", HARDDISK_FW_SIZE);
if (pData == NULL)
return;
if (id == IDR_HDDRVR_FW || id == IDR_HDDRVR_V2_FW)
m_isFirmwareV1or2 = true;
allowFirmwareMods = true;
}
else
{
m_saveStateFirmwareValid = false;
pData = m_saveStateFirmware;
}
BYTE* pFirmwareBase = pCxRomPeripheral + m_slot * APPLE_SLOT_SIZE;
memcpy(pFirmwareBase, pData, HARDDISK_FW_SIZE);
if (allowFirmwareMods)
{
if (m_useHdcFirmwareMode == HdcBlockMode2Devices || m_useHdcFirmwareMode == HdcBlockMode4Devices)
{
pFirmwareBase[0x07] = 0x3C; // Block mode (not SmartPort)
const BYTE numDrives = (m_useHdcFirmwareMode == HdcBlockMode2Devices) ? 1 : 3;
pFirmwareBase[0xFE] = (pFirmwareBase[0xFE] & 0xCF) | (numDrives << 4); // 2 or 4 drives
}
else if (!IsEnhancedIIE())
{
if (m_useHdcFirmwareMode != HdcSmartPort)
pFirmwareBase[0x07] = 0x3C; // Block mode (not SmartPort)
}
}
RegisterIoHandler(m_slot, IORead, IOWrite, NULL, NULL, this, NULL);
}
//===========================================================================
void HarddiskInterfaceCard::CleanupDriveInternal(const int iDrive)
{
if (m_hardDiskDrive[iDrive].m_imagehandle)
{
ImageClose(m_hardDiskDrive[iDrive].m_imagehandle);
m_hardDiskDrive[iDrive].m_imagehandle = NULL;
}
m_hardDiskDrive[iDrive].m_imageloaded = false;
m_hardDiskDrive[iDrive].m_imagename.clear();
m_hardDiskDrive[iDrive].m_fullname.clear();
m_hardDiskDrive[iDrive].m_strFilenameInZip.clear();
}
void HarddiskInterfaceCard::CleanupDrive(const int iDrive)
{
CleanupDriveInternal(iDrive);
SaveLastDiskImage(iDrive);
}
//===========================================================================
void HarddiskInterfaceCard::NotifyInvalidImage(const std::string & szImageFilename)
{
// TC: TO DO - see Disk2InterfaceCard::NotifyInvalidImage()
std::string strText = StrFormat("Unable to open the file %s.",
szImageFilename.c_str());
GetFrame().FrameMessageBox(strText.c_str(),
g_pAppTitle.c_str(),
MB_ICONEXCLAMATION | MB_SETFOREGROUND);
}
//===========================================================================
void HarddiskInterfaceCard::LoadLastDiskImage(const int drive)
{
_ASSERT(drive >= HARDDISK_1 && drive < NUM_HARDDISKS);
const std::string regKey = std::string(REGVALUE_LAST_HARDDISK_) + (char)('1' + drive);
char pathname[MAX_PATH];
std::string regSection = RegGetConfigSlotSection(m_slot);
if (RegLoadString(regSection.c_str(), regKey.c_str(), TRUE, pathname, MAX_PATH, TEXT("")) && (pathname[0] != 0))
{
m_saveDiskImage = false;
bool res = Insert(drive, pathname);
m_saveDiskImage = true;
if (!res)
{
NotifyInvalidImage(pathname);
CleanupDrive(drive);
}
}
}
//===========================================================================
void HarddiskInterfaceCard::SaveLastDiskImage(const int drive)
{
_ASSERT(drive >= HARDDISK_1 && drive < NUM_HARDDISKS);
if (!m_saveDiskImage)
return;
std::string regSection = RegGetConfigSlotSection(m_slot);
RegSaveValue(regSection.c_str(), REGVALUE_CARD_TYPE, TRUE, CT_GenericHDD);
const std::string regKey = std::string(REGVALUE_LAST_HARDDISK_) + (char)('1' + drive);
const std::string& pathName = HarddiskGetFullPathName(drive);
RegSaveString(regSection.c_str(), regKey.c_str(), TRUE, pathName);
//
// For now, only update 'HDV Starting Directory' for slot7 & drive1
// . otherwise you'll get inconsistent results if you set drive1, then drive2 (and the images were in different folders)
if (m_slot != SLOT7 || drive != HARDDISK_1)
return;
const size_t slash = pathName.find_last_of(PATH_SEPARATOR);
if (slash != std::string::npos)
{
const std::string dirName = pathName.substr(0, slash + 1);
RegSaveString(REG_PREFS, REGVALUE_PREF_HDV_START_DIR, 1, dirName);
}
}
//===========================================================================
const std::string& HarddiskInterfaceCard::GetFullName(const int iDrive)
{
return m_hardDiskDrive[iDrive].m_fullname;
}
const std::string& HarddiskInterfaceCard::HarddiskGetFullPathName(const int iDrive)
{
return ImageGetPathname(m_hardDiskDrive[iDrive].m_imagehandle);
}
const std::string& HarddiskInterfaceCard::DiskGetBaseName(const int iDrive)
{
return m_hardDiskDrive[iDrive].m_imagename;
}
void HarddiskInterfaceCard::GetFilenameAndPathForSaveState(std::string& filename, std::string& path)
{
filename = "";
path = "";
for (UINT i = 0; i < NUM_HARDDISKS; i++)
{
if (!m_hardDiskDrive[i].m_imageloaded)
continue;
filename = DiskGetBaseName(i);
std::string pathname = HarddiskGetFullPathName(i);
int idx = pathname.find_last_of(PATH_SEPARATOR);
if (idx >= 0 && idx+1 < (int)pathname.length()) // path exists?
{
path = pathname.substr(0, idx+1);
return;
}
_ASSERT(0);
break;
}
}
//===========================================================================
void HarddiskInterfaceCard::Destroy(void)
{
for (UINT i = 0; i < NUM_HARDDISKS; i++)
{
m_saveDiskImage = false;
CleanupDrive(i);
}
m_saveDiskImage = true;
}
//===========================================================================
// Pre: pathname likely to include path (but can also just be filename)
bool HarddiskInterfaceCard::Insert(const int iDrive, const std::string& pathname)
{
if (pathname.empty())
return false;
if (m_hardDiskDrive[iDrive].m_imageloaded)
Unplug(iDrive);
const DWORD dwAttributes = GetFileAttributes(pathname.c_str());
if (dwAttributes == INVALID_FILE_ATTRIBUTES)
m_hardDiskDrive[iDrive].m_bWriteProtected = false; // File doesn't exist - so ImageOpen() below will fail
else
m_hardDiskDrive[iDrive].m_bWriteProtected = (dwAttributes & FILE_ATTRIBUTE_READONLY) ? true : false;
// Check if image is being used by the other HDD, and unplug it in order to be swapped
{
const std::string & pszOtherPathname = HarddiskGetFullPathName(!iDrive);
char szCurrentPathname[MAX_PATH];
DWORD uNameLen = GetFullPathName(pathname.c_str(), MAX_PATH, szCurrentPathname, NULL);
if (uNameLen == 0 || uNameLen >= MAX_PATH)
strcpy_s(szCurrentPathname, MAX_PATH, pathname.c_str());
if (!strcmp(pszOtherPathname.c_str(), szCurrentPathname))
{
Unplug(!iDrive);
GetFrame().FrameRefreshStatus(DRAW_LEDS | DRAW_DISK_STATUS);
}
}
const bool bCreateIfNecessary = false; // NB. Don't allow creation of HDV files
const bool bExpectFloppy = false;
const bool bIsHarddisk = true;
ImageError_e Error = ImageOpen(pathname,
&m_hardDiskDrive[iDrive].m_imagehandle,
&m_hardDiskDrive[iDrive].m_bWriteProtected,
bCreateIfNecessary,
m_hardDiskDrive[iDrive].m_strFilenameInZip, // TODO: Use this
bExpectFloppy);
m_hardDiskDrive[iDrive].m_imageloaded = (Error == eIMAGE_ERROR_NONE);
m_hardDiskDrive[iDrive].m_status_next = DISK_STATUS_OFF;
m_hardDiskDrive[iDrive].m_status_prev = DISK_STATUS_OFF;
if (Error == eIMAGE_ERROR_NONE)
{
GetImageTitle(pathname.c_str(), m_hardDiskDrive[iDrive].m_imagename, m_hardDiskDrive[iDrive].m_fullname);
Snapshot_UpdatePath();
}
SaveLastDiskImage(iDrive);
return m_hardDiskDrive[iDrive].m_imageloaded;
}
//-----------------------------------------------------------------------------
bool HarddiskInterfaceCard::SelectImage(const int drive, LPCSTR pszFilename)
{
TCHAR directory[MAX_PATH];
TCHAR filename[MAX_PATH];
StringCbCopy(filename, MAX_PATH, pszFilename);
RegLoadString(TEXT(REG_PREFS), TEXT(REGVALUE_PREF_HDV_START_DIR), 1, directory, MAX_PATH, TEXT(""));
std::string title = StrFormat("Select HDV Image For HDD %d", drive + 1);
OPENFILENAME ofn;
memset(&ofn, 0, sizeof(OPENFILENAME));
ofn.lStructSize = sizeof(OPENFILENAME);
ofn.hwndOwner = GetFrame().g_hFrameWindow;
ofn.hInstance = GetFrame().g_hInstance;
ofn.lpstrFilter = TEXT("Hard Disk Images (*.hdv,*.po,*.2mg,*.2img,*.gz,*.zip)\0*.hdv;*.po;*.2mg;*.2img;*.gz;*.zip\0")
TEXT("All Files\0*.*\0");
ofn.lpstrFile = filename;
ofn.nMaxFile = MAX_PATH;
ofn.lpstrInitialDir = directory;
ofn.Flags = OFN_FILEMUSTEXIST | OFN_HIDEREADONLY; // Don't allow creation & hide the read-only checkbox
ofn.lpstrTitle = title.c_str();
bool bRes = false;
if (GetOpenFileName(&ofn))
{
std::string openFilename = filename;
if ((!ofn.nFileExtension) || !filename[ofn.nFileExtension])
openFilename += TEXT(".hdv");
if (Insert(drive, openFilename))
{
bRes = true;
}
else
{
NotifyInvalidImage(openFilename);
}
}
return bRes;
}
bool HarddiskInterfaceCard::Select(const int iDrive)
{
return SelectImage(iDrive, TEXT(""));
}
//===========================================================================
void HarddiskInterfaceCard::Unplug(const int iDrive)
{
if (m_hardDiskDrive[iDrive].m_imageloaded)
{
CleanupDrive(iDrive);
Snapshot_UpdatePath();
}
}
//===========================================================================
#if 0 // Enable HDD command logging
#define LOG_DISK(format, ...) LOG(format, __VA_ARGS__)
#define DEBUG_SKIP_BUSY_STATUS 1
#else
#define LOG_DISK(...)
#define DEBUG_SKIP_BUSY_STATUS 0
#endif
// ProDOS BLK & SmartPort commands
//
const UINT BLK_Cmd_Status = 0x00;
const UINT BLK_Cmd_Read = 0x01;
const UINT BLK_Cmd_Write = 0x02;
const UINT BLK_Cmd_Format = 0x03;
//
const UINT SP_Cmd_base = 0x80;
const UINT SP_Cmd_extended = 0x40;
const UINT SP_Cmd_status = SP_Cmd_base + 0x00;
const UINT SP_Cmd_status_STATUS = 0x00;
const UINT SP_Cmd_status_GETDCB = 0x01;
const UINT SP_Cmd_status_GETNL = 0x02;
const UINT SP_Cmd_status_GETDIB = 0x03;
const UINT SP_Cmd_readblock = SP_Cmd_base + 0x01;
const UINT SP_Cmd_writeblock = SP_Cmd_base + 0x02;
const UINT SP_Cmd_format = SP_Cmd_base + 0x03;
const UINT SP_Cmd_control = SP_Cmd_base + 0x04;
const UINT SP_Cmd_init = SP_Cmd_base + 0x05;
const UINT SP_Cmd_open = SP_Cmd_base + 0x06;
const UINT SP_Cmd_close = SP_Cmd_base + 0x07;
const UINT SP_Cmd_read = SP_Cmd_base + 0x08;
const UINT SP_Cmd_write = SP_Cmd_base + 0x09;
const UINT SP_Cmd_busyStatus = SP_Cmd_base + 0x3F; // AppleWin vendor-specific command
#define DEVICE_OK 0x00
#define BUSERR 0x06
#define BADCTL 0x21
#define DEVICE_IO_ERROR 0x27
#define DEVICE_NOT_CONNECTED 0x28 // No device detected/connected
#define NOWRITE 0x2B // Disk write protected
#define ERRORCODE_MASK 0x3F // limit to just 6 bits (as 'error' byte uses b0 & b7 for other status)
#define STATUS_OK 0x00
#define STATUS_ERROR 0x01
#define STATUS_BUSY 0x80
#define SET_STATUS_ERROR(err) (((err)<<1)|STATUS_ERROR)
BYTE __stdcall HarddiskInterfaceCard::IORead(WORD pc, WORD addr, BYTE bWrite, BYTE d, ULONG nExecutedCycles)
{
const UINT slot = ((addr & 0xff) >> 4) - 8;
HarddiskInterfaceCard* pCard = (HarddiskInterfaceCard*)MemGetSlotParameters(slot);
HardDiskDrive* pHDD = NULL;
const BYTE addrIdx = addr & 0xF;
if (addrIdx != 0x2 && addrIdx != 0x3) // GetUnit() depends on m_command & m_unitNum
{
pHDD = pCard->GetUnit();
if (pHDD == NULL)
return SET_STATUS_ERROR(DEVICE_NOT_CONNECTED);
}
CpuCalcCycles(nExecutedCycles);
BYTE r = STATUS_OK;
switch (addrIdx)
{
case 0x0: // EXECUTE & RETURN STATUS
r = pCard->CmdExecute(pHDD);
pCard->m_command = (pCard->m_command & SP_Cmd_base) ? SP_Cmd_busyStatus : BLK_Cmd_Status; // Subsequent reads from IO addr 0x0 just executes 'Status' cmd
_ASSERT(pCard->m_fifoIdx == 0);
pCard->m_fifoIdx = 0;
break;
case 0x1: // STATUS
r = pCard->CmdStatus(pHDD);
break;
case 0x2:
r = pCard->m_command;
break;
case 0x3:
r = pCard->m_unitNum;
break;
case 0x4:
r = (BYTE)(pHDD->m_memblock & 0x00FF);
break;
case 0x5:
r = (BYTE)((pHDD->m_memblock & 0xFF00) >> 8);
break;
case 0x6:
r = (BYTE)(pHDD->m_diskblock & 0x00FF);
break;
case 0x7:
r = (BYTE)((pHDD->m_diskblock & 0xFF00) >> 8);
break;
case 0x8: // Legacy: continue to support this I/O port for old HDC firmware
r = pHDD->m_buf[pHDD->m_buf_ptr];
if (pHDD->m_buf_ptr < sizeof(pHDD->m_buf)-1)
pHDD->m_buf_ptr++;
break;
case 0x9:
if (pHDD->m_imageloaded)
r = (BYTE)(pCard->GetImageSizeInBlocks(pHDD->m_imagehandle, true) & 0x00ff);
else
r = 0;
break;
case 0xa:
if (pHDD->m_imageloaded)
r = (BYTE)((pCard->GetImageSizeInBlocks(pHDD->m_imagehandle, true) & 0xff00) >> 8);
else
r = 0;
break;
default:
LOG_DISK("slot-%d, Bad IORead(), io reg=$%1X, PC=$%04X\n", pCard->m_slot, addrIdx, pc);
pHDD->m_status_next = DISK_STATUS_OFF;
r = IO_Null(pc, addr, bWrite, d, nExecutedCycles);
}
if (pHDD)
pCard->UpdateLightStatus(pHDD);
return r;
}
BYTE HarddiskInterfaceCard::CmdExecute(HardDiskDrive* pHDD)
{
LOG_DISK("slot-%d, HDD-%d(%02X): Cmd=%02X ", m_slot, (m_command & SP_Cmd_base) ? m_unitNum : GetProDOSBlockDeviceUnit(), m_unitNum, m_command);
if (!pHDD->m_imageloaded && m_command != BLK_Cmd_Status && m_command != SP_Cmd_status)
{
pHDD->m_status_next = DISK_STATUS_OFF;
pHDD->m_error = DEVICE_NOT_CONNECTED; // GH#452
return CmdStatus(pHDD);
}
if ((m_command == SP_Cmd_readblock || m_command == SP_Cmd_writeblock) && pHDD->m_diskblock > kHarddiskMaxNumBlocks)
{
pHDD->m_status_next = DISK_STATUS_OFF;
pHDD->m_error = DEVICE_IO_ERROR;
return CmdStatus(pHDD);
}
const UINT CYCLES_FOR_DMA_RW_BLOCK = HD_BLOCK_SIZE;
const UINT CYCLES_FOR_FORMATTING_1_BLOCK = 100; // Arbitrary
const UINT PAGE_SIZE = 256;
pHDD->m_error = DEVICE_OK;
switch (m_command)
{
case BLK_Cmd_Status:
if (ImageGetImageSize(pHDD->m_imagehandle) == 0)
pHDD->m_error = DEVICE_IO_ERROR;
LOG_DISK("ST-BLK: %02X\n", pHDD->m_error);
break;
case SP_Cmd_busyStatus:
LOG_DISK("ST-BSY: %02X\n", pHDD->m_error);
break;
case SP_Cmd_status:
pHDD->m_error = SmartPortCmdStatus(pHDD);
LOG_DISK("ST: %02X (statusCode: %02X)\n", pHDD->m_error, m_statusCode);
break;
case BLK_Cmd_Read:
case SP_Cmd_readblock:
LOG_DISK("RD: %08X (to addr: %04X)\n", pHDD->m_diskblock, pHDD->m_memblock);
pHDD->m_status_next = DISK_STATUS_READ;
if ((pHDD->m_diskblock * HD_BLOCK_SIZE) < ImageGetImageSize(pHDD->m_imagehandle))
{
bool breakpointHit = false;
bool bRes = ImageReadBlock(pHDD->m_imagehandle, pHDD->m_diskblock, pHDD->m_buf);
if (bRes)
{
pHDD->m_buf_ptr = 0;
// Apple II's MMU could be setup so that read & write memory is different,
// so can't use 'mem' (like we can for HDD block writes)
WORD dstAddr = pHDD->m_memblock;
UINT remaining = HD_BLOCK_SIZE;
BYTE* pSrc = pHDD->m_buf;
while (remaining)
{
memdirty[dstAddr >> 8] = 0xFF;
LPBYTE page = memwrite[dstAddr >> 8];
if (!page) // I/O space or ROM
{
if (g_nAppMode == MODE_STEPPING)
DebuggerBreakOnDmaToOrFromIoMemory(dstAddr, true); // GH#1007
//else // Show MessageBox?
bRes = false;
break;
}
// handle both page-aligned & non-page aligned destinations
UINT size = PAGE_SIZE - (dstAddr & 0xff);
if (size > remaining) size = remaining; // clip the last memcpy for the unaligned case
if (g_nAppMode == MODE_STEPPING)
breakpointHit = DebuggerCheckMemBreakpoints(dstAddr, size, true); // GH#1103
memcpy(page + (dstAddr & 0xff), pSrc, size);
pSrc += size;
dstAddr = (dstAddr + size) & (MEMORY_LENGTH - 1); // wraps at 64KiB boundary
remaining -= size;
}
}
if (bRes)
{
pHDD->m_error = DEVICE_OK;
if (!breakpointHit)
m_notBusyCycle = g_nCumulativeCycles + (UINT64)CYCLES_FOR_DMA_RW_BLOCK;
#if DEBUG_SKIP_BUSY_STATUS
m_notBusyCycle = 0;
#endif
}
else
{
pHDD->m_error = DEVICE_IO_ERROR;
}
}
else
{
pHDD->m_error = DEVICE_IO_ERROR;
}
break;
case BLK_Cmd_Write:
case SP_Cmd_writeblock:
{
LOG_DISK("WR: %08X (from addr: %04X) %s\n", pHDD->m_diskblock, pHDD->m_memblock, pHDD->m_bWriteProtected ? "write-protected" : "");
if (pHDD->m_bWriteProtected)
{
pHDD->m_status_next = DISK_STATUS_PROT;
pHDD->m_error = NOWRITE;
}
else
{
pHDD->m_status_next = DISK_STATUS_WRITE;
bool bRes = true;
const bool bAppendBlocks = (pHDD->m_diskblock * HD_BLOCK_SIZE) >= ImageGetImageSize(pHDD->m_imagehandle);
bool breakpointHit = false;
if (bAppendBlocks)
{
memset(pHDD->m_buf, 0, HD_BLOCK_SIZE);
// Inefficient (especially for gzip/zip files!)
UINT uBlock = ImageGetImageSize(pHDD->m_imagehandle) / HD_BLOCK_SIZE;
while (uBlock < pHDD->m_diskblock)
{
bRes = ImageWriteBlock(pHDD->m_imagehandle, uBlock++, pHDD->m_buf);
_ASSERT(bRes);
if (!bRes)
break;
}
}
// Trap and error on any accesses that overlap with I/O memory (GH#1007)
if ((pHDD->m_memblock < APPLE_IO_BEGIN && ((pHDD->m_memblock + HD_BLOCK_SIZE - 1) >= APPLE_IO_BEGIN)) // 1) Starts before I/O, but ends in I/O memory
|| ((pHDD->m_memblock >> 12) == (APPLE_IO_BEGIN >> 12))) // 2) Starts in I/O memory
{
WORD dstAddr = ((pHDD->m_memblock >> 12) == (APPLE_IO_BEGIN >> 12)) ? pHDD->m_memblock : APPLE_IO_BEGIN;
if (g_nAppMode == MODE_STEPPING)
DebuggerBreakOnDmaToOrFromIoMemory(dstAddr, false);
//else // Show MessageBox?
bRes = false;
}
else
{
// NB. Do the writes in units of PAGE_SIZE so that DMA breakpoints are consistent with reads
WORD srcAddr = pHDD->m_memblock;
UINT remaining = HD_BLOCK_SIZE;
BYTE* pDst = pHDD->m_buf;
while (remaining)
{
UINT size = PAGE_SIZE - (srcAddr & 0xff);
if (size > remaining) size = remaining; // clip the last memcpy for the unaligned case
if (g_nAppMode == MODE_STEPPING)
breakpointHit = DebuggerCheckMemBreakpoints(srcAddr, size, false);
memcpy(pDst, mem + srcAddr, size);
pDst += size;
srcAddr = (srcAddr + size) & (MEMORY_LENGTH - 1); // wraps at 64KiB boundary
remaining -= size;
}
}
if (bRes)
bRes = ImageWriteBlock(pHDD->m_imagehandle, pHDD->m_diskblock, pHDD->m_buf);
if (bRes)
{
pHDD->m_error = DEVICE_OK;
if (!breakpointHit)
m_notBusyCycle = g_nCumulativeCycles + (UINT64)CYCLES_FOR_DMA_RW_BLOCK;
#if DEBUG_SKIP_BUSY_STATUS
m_notBusyCycle = 0;
#endif
}
else
{
pHDD->m_error = DEVICE_IO_ERROR;
}
} // if (pHDD->m_bWriteProtected)
}
break;
case BLK_Cmd_Format:
case SP_Cmd_format:
LOG_DISK("FORMAT: write-protected=%d\n", pHDD->m_bWriteProtected);
if (pHDD->m_bWriteProtected)
{
pHDD->m_error = NOWRITE;
}
else
{
const UINT numBlocks = GetImageSizeInBlocks(pHDD->m_imagehandle);
memset(pHDD->m_buf, 0, HD_BLOCK_SIZE);
bool res = false;
m_notBusyCycle = g_nCumulativeCycles;
for (UINT block = 0; block < numBlocks; block++)
{
// Inefficient (especially for gzip/zip files!)
res = ImageWriteBlock(pHDD->m_imagehandle, block, pHDD->m_buf);
_ASSERT(res);
if (!res)
break;
m_notBusyCycle += (UINT64)CYCLES_FOR_FORMATTING_1_BLOCK;
#if DEBUG_SKIP_BUSY_STATUS
m_notBusyCycle = 0;
#endif
}
pHDD->m_error = res ? DEVICE_OK : DEVICE_IO_ERROR;
}
break;
default:
_ASSERT(0);
pHDD->m_error = DEVICE_IO_ERROR;
break;
}
return CmdStatus(pHDD);
}
BYTE HarddiskInterfaceCard::CmdStatus(HardDiskDrive* pHDD)
{
BYTE r = 0;
if (pHDD->m_error)
r = STATUS_ERROR; // Firmware requires that b0=1 for an error
if (g_nCumulativeCycles <= m_notBusyCycle)
r |= STATUS_BUSY; // Firmware requires that b7=1 for busy (eg. busy doing r/w DMA operation)
else
pHDD->m_status_next = DISK_STATUS_OFF; // TODO: FIXME: ??? YELLOW ??? WARNING
// Firmware requires that error code is [b6..1]
_ASSERT(pHDD->m_error <= ERRORCODE_MASK);
r |= (pHDD->m_error & ERRORCODE_MASK) << 1;
return r;
}
//-----------------------------------------------------------------------------
BYTE __stdcall HarddiskInterfaceCard::IOWrite(WORD pc, WORD addr, BYTE bWrite, BYTE d, ULONG nExecutedCycles)
{
const UINT slot = ((addr & 0xff) >> 4) - 8;
HarddiskInterfaceCard* pCard = (HarddiskInterfaceCard*)MemGetSlotParameters(slot);
HardDiskDrive* pHDD = NULL;
BYTE addrIdx = addr & 0xF;
if (addrIdx != 0x2 && addrIdx != 0x3 // GetUnit() depends on m_command & m_unitNum
&& !((addrIdx == 0x9 || addrIdx == 0xA) && pCard->m_fifoIdx < 2))
{
pHDD = pCard->GetUnit();
if (pHDD == NULL)
{
// Ensure that fifoIdx returns to 0, even if unitNum is out of range
// EG. ProDOS 8 v2.5.0: Bitsy Bye: TAB through S7 volumes
const UINT fifoSize = (addrIdx == 0x9) ? 6 : 7;
pCard->m_fifoIdx = (pCard->m_fifoIdx + 1) % fifoSize;
return SET_STATUS_ERROR(DEVICE_NOT_CONNECTED);
}
}
if (addrIdx == 0x9 || addrIdx == 0xA) // BLK or SP cmd FIFO
{
if (addrIdx == 0xA && pCard->m_fifoIdx == 0)
d |= SP_Cmd_base;
const UINT fifoSize = (addrIdx == 0x9) ? 6 : 7;
addrIdx = 0x2 + pCard->m_fifoIdx;
pCard->m_fifoIdx = (pCard->m_fifoIdx + 1) % fifoSize;
}
switch (addrIdx)
{
case 0x0: // r/o: status
case 0x1: // r/o: execute
// Writing to these read-only registers is a no-op.
// NB. Don't change m_status_next, as UpdateLightStatus() has a huge performance cost!
// Some HDC's firmware has a busy-wait loop doing "rol hd_status,x"
// - this RMW opcode does an IORead() then an IOWrite(), and the loop iterates ~100 times!
break;
case 0x2:
pCard->m_command = d;
break;
case 0x3:
// b7 = drive#
// b6..4 = slot#
// b3..0 = ?
pCard->m_unitNum = d;
pCard->FixupUnitNum();
break;
case 0x4:
pHDD->m_memblock = (pHDD->m_memblock & 0xFF00) | d;
break;
case 0x5:
pHDD->m_memblock = (pHDD->m_memblock & 0x00FF) | (d << 8);
break;
case 0x6:
if (pCard->m_command != SP_Cmd_status)
pHDD->m_diskblock = (pHDD->m_diskblock & 0xFFFF00) | d;
else
pCard->m_statusCode = d;
break;
case 0x7:
pHDD->m_diskblock = (pHDD->m_diskblock & 0xFF00FF) | (d << 8);
break;
case 0x8:
if (pCard->m_command & SP_Cmd_base)
pHDD->m_diskblock = (pHDD->m_diskblock & 0x00FFFF) | (d << 16);
break;
default:
pHDD->m_status_next = DISK_STATUS_OFF;
}
if (pHDD)
{
if ((pCard->m_command & SP_Cmd_base) == 0)
pHDD->m_diskblock &= 0x00FFFF; // BLK cmds are only 16-bit
pCard->UpdateLightStatus(pHDD);
}
return 0;
}
//===========================================================================
void HarddiskInterfaceCard::FixupUnitNum(void)
{
if (!m_isFirmwareV1or2)
return;
// Older firmwares can write unitNum with 0x00
if ((m_unitNum >> 4) != m_slot)
m_unitNum = (m_unitNum & 0x8F) | (m_slot << 4);
}
BYTE HarddiskInterfaceCard::GetNumConnectedDevices(void)
{
// Scan backwards to find the index of the last attached HDD
int numDevices = NUM_HARDDISKS - 1;
for (; numDevices >= 0; numDevices--)
{
if (m_hardDiskDrive[numDevices].m_imageloaded)
break;
}
return numDevices + 1;
}
BYTE HarddiskInterfaceCard::GetProDOSBlockDeviceUnit(void)
{
const BYTE slotFromUnitNum = (m_unitNum >> 4) & 7;
const BYTE offset = (slotFromUnitNum == m_slot) ? 0 : 2;
return offset + (m_unitNum >> 7); // bit7 = drive select
}
HardDiskDrive* HarddiskInterfaceCard::GetUnit(void)
{
const bool isSmartPortCmd = m_command & SP_Cmd_base;
if (!isSmartPortCmd)
return &m_hardDiskDrive[GetProDOSBlockDeviceUnit()];
if (m_unitNum > kMaxSmartPortUnits)
return NULL;
if (m_unitNum > GetNumConnectedDevices())
return NULL;
if (m_unitNum == 0)
return &m_smartPortController;
return &m_hardDiskDrive[m_unitNum - 1];
}
void HarddiskInterfaceCard::SetIdString(WORD addr, const char* str)
{
BYTE& idStrLen = mem[addr]; // ID string length
idStrLen = 0;
WORD idStrAddr = addr + 1;
for (UINT i = 0; i < 16; i++)
mem[idStrAddr + i] = ' '; // ID string padded with ASCII spaces
if (str == NULL)
return;
while (*str && idStrLen < 16)
{
idStrLen++;
mem[idStrAddr++] = *str++;
}
}
BYTE HarddiskInterfaceCard::SmartPortCmdStatus(HardDiskDrive* pHDD)
{
// Make Firmware version: eg. 1.30.18.0 => 130.18
UINT fwVerMajorCheck = g_AppleWinVersion[0] * 100 + g_AppleWinVersion[1];
_ASSERT(fwVerMajorCheck < 256);
if (fwVerMajorCheck >= 256) fwVerMajorCheck = 255;
UINT fwVerMinorCheck = g_AppleWinVersion[2];
_ASSERT(fwVerMinorCheck < 256);
if (fwVerMinorCheck >= 256) fwVerMinorCheck = 255;
const BYTE fwVerMajor = fwVerMajorCheck;
const BYTE fwVerMinor = fwVerMinorCheck;
//
WORD statusListAddr = pHDD->m_memblock;
BYTE r = DEVICE_OK;
if (m_unitNum == 0) // Unit-0: SmartPort Controller
{
BYTE numDevices = GetNumConnectedDevices();
switch (m_statusCode)
{
case SP_Cmd_status_STATUS:
case SP_Cmd_status_GETDIB:
{
// SmartPort driver status (8 bytes)
mem[statusListAddr++] = numDevices;
for (UINT i = 0; i < 7; i++)
mem[statusListAddr++] = 0; // reserved
if (m_statusCode == SP_Cmd_status_STATUS)
break;
// Device Information Block (DIB)
std::string idStr = "AppleWin SP";
SetIdString(statusListAddr, idStr.c_str());
statusListAddr += 17;
mem[statusListAddr++] = 0x00; // device type (0x00: Apple II memory expansion card)
mem[statusListAddr++] = 0x00; // device subtype (0x00: Apple II memory expansion card)
mem[statusListAddr++] = fwVerMajor; // f/w version (major)
mem[statusListAddr++] = fwVerMinor; // f/w version (minor)
break;
}
case SP_Cmd_status_GETDCB:
case SP_Cmd_status_GETNL:
default:
pHDD->m_error = 1;
r = BADCTL;
break;
}
}
else // Unit > 0: SmartPort Devices
{
switch (m_statusCode)
{
case SP_Cmd_status_STATUS:
case SP_Cmd_status_GETDIB:
{
// Device status (4 bytes)
const bool isImageLoaded = m_hardDiskDrive[m_unitNum - 1].m_imageloaded;
// general status:
// . b7=block device, b6=write allowed, b5=read allowed, b4=device online or disk in drive,
// . b3=format allowed, b2=media write protected (block devices only), b1=device currently interrupting (//c only), b0=device currently open (char device only)
BYTE generalStatus = isImageLoaded ? 0xF8 : 0xE8; // Loaded: b#11111000: bwrlf--- / Not loaded: b#11101000: bwr-f---
if (pHDD->m_bWriteProtected) generalStatus |= (1 << 2);
mem[statusListAddr++] = generalStatus;
const UINT imageSizeInBlocks = isImageLoaded ? GetImageSizeInBlocks(pHDD->m_imagehandle) : 0;
mem[statusListAddr++] = imageSizeInBlocks & 0xff; // num blocks (lo)
mem[statusListAddr++] = (imageSizeInBlocks >> 8) & 0xff; // num blocks (med)
mem[statusListAddr++] = (imageSizeInBlocks >> 16) & 0xff; // num blocks (hi)
if (m_statusCode == SP_Cmd_status_STATUS)
break;
// Device Information Block (DIB)
std::string idStr = "AppleWin SP D#"; // + "01".."99" (device number in decimal)
idStr += (char)('0' + m_unitNum / 10);
idStr += (char)('0' + m_unitNum % 10);
SetIdString(statusListAddr, idStr.c_str());
statusListAddr += 17;
mem[statusListAddr++] = 0x02; // device type (0x02: Hard disk)
mem[statusListAddr++] = 0x20; // device subtype (0x20: Hard disk)
mem[statusListAddr++] = fwVerMajor; // f/w version (major)
mem[statusListAddr++] = fwVerMinor; // f/w version (minor)
break;
}
case SP_Cmd_status_GETDCB:
case SP_Cmd_status_GETNL:
default:
pHDD->m_error = 1;
r = BADCTL;
break;
}
}
return r;
}
//===========================================================================
UINT HarddiskInterfaceCard::GetImageSizeInBlocks(ImageInfo* const pImageInfo, const bool is16bit/*=false*/)
{
if (m_userNumBlocks != 0)
return m_userNumBlocks;
UINT numberOfBlocks = (pImageInfo ? pImageInfo->uImageSize : 0) / HD_BLOCK_SIZE;
if (numberOfBlocks > kHarddiskMaxNumBlocks)
numberOfBlocks = kHarddiskMaxNumBlocks;
if (is16bit && numberOfBlocks > 0xffff)
numberOfBlocks = 0xffff;
return numberOfBlocks;
}
//===========================================================================
void HarddiskInterfaceCard::UpdateLightStatus(HardDiskDrive* pHDD)
{
if (pHDD->m_status_prev != pHDD->m_status_next) // Update LEDs if state changes
{
pHDD->m_status_prev = pHDD->m_status_next;
GetFrame().FrameRefreshStatus(DRAW_LEDS | DRAW_DISK_STATUS);
}
}
void HarddiskInterfaceCard::GetLightStatus(Disk_Status_e *pDisk1Status)
{
const BYTE unit = (m_command & SP_Cmd_base) ? m_unitNum : GetProDOSBlockDeviceUnit();
HardDiskDrive* pHDD = &m_hardDiskDrive[unit];
*pDisk1Status = pHDD->m_status_prev;
}
//===========================================================================
bool HarddiskInterfaceCard::ImageSwap(void)
{
std::swap(m_hardDiskDrive[HARDDISK_1], m_hardDiskDrive[HARDDISK_2]);
SaveLastDiskImage(HARDDISK_1);
SaveLastDiskImage(HARDDISK_2);
GetFrame().FrameRefreshStatus(DRAW_LEDS);
return true;
}
//===========================================================================
// Unit version history:
// 2: Updated $C7nn firmware to fix GH#319
// 3: Updated $Csnn firmware to fix GH#996 (now slot-independent code)
// Added: Not Busy Cycle
// 4: Updated $Csnn firmware to fix GH#1264
// 5: Added: SP Status Code, FIFO Index & 256-byte firmware
// Units are 1-based (up to v4 they were 0-based)
// 6: Added: absolute path
static const UINT kUNIT_VERSION = 6;
#define SS_YAML_VALUE_CARD_HDD "Generic HDD"
#define SS_YAML_KEY_CURRENT_UNIT "Current Unit"
#define SS_YAML_KEY_COMMAND "Command"
#define SS_YAML_KEY_HDDUNIT "Unit"
#define SS_YAML_KEY_FILENAME "Filename"
#define SS_YAML_KEY_ABSOLUTE_PATH "Absolute Path"
#define SS_YAML_KEY_ERROR "Error"
#define SS_YAML_KEY_MEMBLOCK "MemBlock"
#define SS_YAML_KEY_DISKBLOCK "DiskBlock"
#define SS_YAML_KEY_IMAGELOADED "ImageLoaded"
#define SS_YAML_KEY_STATUS_NEXT "Status Next"
#define SS_YAML_KEY_STATUS_PREV "Status Prev"
#define SS_YAML_KEY_BUF_PTR "Buffer Offset"
#define SS_YAML_KEY_BUF "Buffer"
#define SS_YAML_KEY_NOT_BUSY_CYCLE "Not Busy Cycle"
#define SS_YAML_KEY_SP_STATUS_CODE "SP Status Code"
#define SS_YAML_KEY_FIFO_INDEX "FIFO Index"
#define SS_YAML_KEY_FIRMWARE "Firmware"
const std::string& HarddiskInterfaceCard::GetSnapshotCardName(void)
{
static const std::string name(SS_YAML_VALUE_CARD_HDD);
return name;
}
void HarddiskInterfaceCard::SaveSnapshotHDDUnit(YamlSaveHelper& yamlSaveHelper, const UINT unit)
{
const UINT baseUnitNum = 1; // Unit0 is the SP Controller, so SP units start from 1
YamlSaveHelper::Label label(yamlSaveHelper, "%s%d:\n", SS_YAML_KEY_HDDUNIT, baseUnitNum + unit);
yamlSaveHelper.SaveString(SS_YAML_KEY_FILENAME, m_hardDiskDrive[unit].m_fullname);
yamlSaveHelper.SaveString(SS_YAML_KEY_ABSOLUTE_PATH, ImageGetPathname(m_hardDiskDrive[unit].m_imagehandle));
yamlSaveHelper.SaveHexUint8(SS_YAML_KEY_ERROR, m_hardDiskDrive[unit].m_error);
yamlSaveHelper.SaveHexUint16(SS_YAML_KEY_MEMBLOCK, m_hardDiskDrive[unit].m_memblock);
yamlSaveHelper.SaveHexUint32(SS_YAML_KEY_DISKBLOCK, m_hardDiskDrive[unit].m_diskblock);
yamlSaveHelper.SaveBool(SS_YAML_KEY_IMAGELOADED, m_hardDiskDrive[unit].m_imageloaded);
yamlSaveHelper.SaveUint(SS_YAML_KEY_STATUS_NEXT, m_hardDiskDrive[unit].m_status_next);
yamlSaveHelper.SaveUint(SS_YAML_KEY_STATUS_PREV, m_hardDiskDrive[unit].m_status_prev);
yamlSaveHelper.SaveHexUint16(SS_YAML_KEY_BUF_PTR, m_hardDiskDrive[unit].m_buf_ptr);
// New label
{
YamlSaveHelper::Label buffer(yamlSaveHelper, "%s:\n", SS_YAML_KEY_BUF);
yamlSaveHelper.SaveMemory(m_hardDiskDrive[unit].m_buf, HD_BLOCK_SIZE);
}
}
void HarddiskInterfaceCard::SaveSnapshot(YamlSaveHelper& yamlSaveHelper)
{
YamlSaveHelper::Slot slot(yamlSaveHelper, GetSnapshotCardName(), m_slot, kUNIT_VERSION);
YamlSaveHelper::Label state(yamlSaveHelper, "%s:\n", SS_YAML_KEY_STATE);
yamlSaveHelper.Save("%s: %d # b7=unit for ProDOS BLK device\n", SS_YAML_KEY_CURRENT_UNIT, m_unitNum);
yamlSaveHelper.SaveHexUint8(SS_YAML_KEY_COMMAND, m_command);
yamlSaveHelper.SaveHexUint64(SS_YAML_KEY_NOT_BUSY_CYCLE, m_notBusyCycle);
yamlSaveHelper.SaveHexUint8(SS_YAML_KEY_SP_STATUS_CODE, m_statusCode);
yamlSaveHelper.SaveHexUint8(SS_YAML_KEY_FIFO_INDEX, m_fifoIdx);
// New label
{
YamlSaveHelper::Label buffer(yamlSaveHelper, "%s:\n", SS_YAML_KEY_FIRMWARE);
yamlSaveHelper.SaveMemory(mem + APPLE_IO_BEGIN + m_slot * APPLE_SLOT_SIZE, APPLE_SLOT_SIZE);
}
for (UINT i = 0; i < NUM_HARDDISKS; i++)
{
if (m_hardDiskDrive[i].m_imageloaded)
SaveSnapshotHDDUnit(yamlSaveHelper, i);
}
}
bool HarddiskInterfaceCard::LoadSnapshotHDDUnit(YamlLoadHelper& yamlLoadHelper, const UINT unit, const UINT version)
{
const UINT baseUnitNum = (version >= 5) ? 1 : 0;
std::string hddUnitName = std::string(SS_YAML_KEY_HDDUNIT) + (char)('0' + baseUnitNum + unit);
if (!yamlLoadHelper.GetSubMap(hddUnitName))
return false; // No HDD plugged in for this unit#
m_hardDiskDrive[unit].m_fullname.clear();
m_hardDiskDrive[unit].m_imagename.clear();
m_hardDiskDrive[unit].m_imageloaded = false; // Default to false (until image is successfully loaded below)
m_hardDiskDrive[unit].m_status_next = DISK_STATUS_OFF;
m_hardDiskDrive[unit].m_status_prev = DISK_STATUS_OFF;
const std::string simpleFilename = yamlLoadHelper.LoadString(SS_YAML_KEY_FILENAME);
const std::string absolutePath = version >= 6 ? yamlLoadHelper.LoadString(SS_YAML_KEY_ABSOLUTE_PATH) : "";
m_hardDiskDrive[unit].m_error = yamlLoadHelper.LoadUint(SS_YAML_KEY_ERROR);
m_hardDiskDrive[unit].m_memblock = yamlLoadHelper.LoadUint(SS_YAML_KEY_MEMBLOCK);
m_hardDiskDrive[unit].m_diskblock = yamlLoadHelper.LoadUint(SS_YAML_KEY_DISKBLOCK);
yamlLoadHelper.LoadBool(SS_YAML_KEY_IMAGELOADED); // Consume
Disk_Status_e diskStatusNext = (Disk_Status_e) yamlLoadHelper.LoadUint(SS_YAML_KEY_STATUS_NEXT);
Disk_Status_e diskStatusPrev = (Disk_Status_e) yamlLoadHelper.LoadUint(SS_YAML_KEY_STATUS_PREV);
m_hardDiskDrive[unit].m_buf_ptr = yamlLoadHelper.LoadUint(SS_YAML_KEY_BUF_PTR);
if (m_hardDiskDrive[unit].m_buf_ptr >= sizeof(m_hardDiskDrive[unit].m_buf)) // pre-v3 save-states would leave m_buf_ptr==0x200 after reading a block
m_hardDiskDrive[unit].m_buf_ptr = sizeof(m_hardDiskDrive[unit].m_buf) - 1;
if (!yamlLoadHelper.GetSubMap(SS_YAML_KEY_BUF))
throw std::runtime_error(hddUnitName + ": Missing: " + SS_YAML_KEY_BUF);
yamlLoadHelper.LoadMemory(m_hardDiskDrive[unit].m_buf, HD_BLOCK_SIZE);
yamlLoadHelper.PopMap();
yamlLoadHelper.PopMap();
//
bool userSelectedImageFolder = false;
std::string filename = simpleFilename;
if (!filename.empty())
{
DWORD dwAttributes = GetFileAttributes(filename.c_str());
if (dwAttributes == INVALID_FILE_ATTRIBUTES && !absolutePath.empty())
{
// try the absolute path if present
filename = absolutePath;
dwAttributes = GetFileAttributes(filename.c_str());
}
if (dwAttributes == INVALID_FILE_ATTRIBUTES)
{
// ignore absolute name when opening the file dialog
filename = simpleFilename;
// Get user to browse for file
userSelectedImageFolder = SelectImage(unit, filename.c_str());
dwAttributes = GetFileAttributes(filename.c_str());
}
bool bImageError = (dwAttributes == INVALID_FILE_ATTRIBUTES);
if (!bImageError)
{
if (!Insert(unit, filename.c_str()))
bImageError = true;
// Insert() sets up:
// . m_imagename
// . m_fullname
// . m_imageloaded
// . hd_status_next = DISK_STATUS_OFF
// . hd_status_prev = DISK_STATUS_OFF
m_hardDiskDrive[unit].m_status_next = diskStatusNext;
m_hardDiskDrive[unit].m_status_prev = diskStatusPrev;
}
}
return userSelectedImageFolder;
}
bool HarddiskInterfaceCard::LoadSnapshot(YamlLoadHelper& yamlLoadHelper, UINT version)
{
if (version < 1 || version > kUNIT_VERSION)
ThrowErrorInvalidVersion(version);
if (version <= 2 && (regs.pc >> 8) == (0xC0|m_slot))
throw std::runtime_error("HDC card: 6502 is running old HDD firmware");
m_unitNum = yamlLoadHelper.LoadUint(SS_YAML_KEY_CURRENT_UNIT);
if (version < 5)
{
m_isFirmwareV1or2 = true;
FixupUnitNum();
}
m_command = yamlLoadHelper.LoadUint(SS_YAML_KEY_COMMAND);
if (version >= 3)
m_notBusyCycle = yamlLoadHelper.LoadUint64(SS_YAML_KEY_NOT_BUSY_CYCLE);
m_saveStateFirmwareV1 = m_saveStateFirmwareV2 = false;
if (version < 4)
m_saveStateFirmwareV1 = true;
else if (version == 4)
m_saveStateFirmwareV2 = true;
if (version >= 5)
{
m_statusCode = yamlLoadHelper.LoadUint(SS_YAML_KEY_SP_STATUS_CODE);
m_fifoIdx = yamlLoadHelper.LoadUint(SS_YAML_KEY_FIFO_INDEX);
//
if (!yamlLoadHelper.GetSubMap(SS_YAML_KEY_FIRMWARE))
throw std::runtime_error(std::string("HDC") + ": Missing: " + SS_YAML_KEY_FIRMWARE);
yamlLoadHelper.LoadMemory(m_saveStateFirmware, APPLE_SLOT_SIZE);
yamlLoadHelper.PopMap();
m_saveStateFirmwareValid = true;
// NB. A command line option can be used to ignore the HDC's firmware:
// . Used by AppleWin-Test (regression suite) so that an older save-state file can be used to test newer HDC firmware (eg. GH#1207).
if (Snapshot_GetIgnoreHdcFirmware())
m_saveStateFirmwareValid = false;
}
// Unplug all HDDs first in case eg. HDD-2 is to be plugged in as HDD-1
for (UINT i = 0; i < NUM_HARDDISKS; i++)
{
Unplug(i);
m_hardDiskDrive[i].clear();
}
bool userSelectedImageFolder = false;
for (UINT i = 0; i < NUM_HARDDISKS; i++)
userSelectedImageFolder |= LoadSnapshotHDDUnit(yamlLoadHelper, i, version);
if (!userSelectedImageFolder)
RegSaveString(TEXT(REG_PREFS), TEXT(REGVALUE_PREF_HDV_START_DIR), 1, Snapshot_GetPath());
GetFrame().FrameRefreshStatus(DRAW_LEDS | DRAW_DISK_STATUS);
return true;
}