2006-02-25 20:50:29 +00:00
|
|
|
/*
|
|
|
|
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
|
2010-01-03 18:43:08 +00:00
|
|
|
Copyright (C) 2006-2010, Tom Charlesworth, Michael Pohoreski
|
2006-02-25 20:50:29 +00:00
|
|
|
|
|
|
|
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: Disk Image
|
|
|
|
*
|
|
|
|
* Author: Various
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "StdAfx.h"
|
2014-08-14 17:48:38 +01:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
#include "DiskImage.h"
|
2020-11-11 21:15:27 +00:00
|
|
|
#include "Common.h"
|
2010-01-03 18:43:08 +00:00
|
|
|
#include "DiskImageHelper.h"
|
2010-02-14 21:11:26 +00:00
|
|
|
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
static CDiskImageHelper sg_DiskImageHelper;
|
2016-09-19 22:14:57 +01:00
|
|
|
static CHardDiskImageHelper sg_HardDiskImageHelper;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
// Pre: *pWriteProtected_ already set to file's r/w status - see DiskInsert()
|
2019-09-07 09:02:39 +01:00
|
|
|
ImageError_e ImageOpen( const std::string & pszImageFilename,
|
2016-10-25 21:09:48 +01:00
|
|
|
ImageInfo** ppImageInfo,
|
2015-02-13 22:40:53 +00:00
|
|
|
bool* pWriteProtected,
|
2010-01-03 18:43:08 +00:00
|
|
|
const bool bCreateIfNecessary,
|
2015-02-13 22:40:53 +00:00
|
|
|
std::string& strFilenameInZip,
|
2016-10-25 22:26:28 +01:00
|
|
|
const bool bExpectFloppy /*=true*/)
|
2009-02-17 22:53:52 +00:00
|
|
|
{
|
2019-09-07 18:51:07 +01:00
|
|
|
if (!(!pszImageFilename.empty() && ppImageInfo && pWriteProtected))
|
2010-01-03 18:43:08 +00:00
|
|
|
return eIMAGE_ERROR_BAD_POINTER;
|
2009-02-24 22:13:46 +00:00
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
// CREATE A RECORD FOR THE FILE
|
2019-11-11 14:09:29 +00:00
|
|
|
*ppImageInfo = new ImageInfo();
|
2009-02-17 22:53:52 +00:00
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
ImageInfo* pImageInfo = *ppImageInfo;
|
2015-02-13 22:40:53 +00:00
|
|
|
pImageInfo->bWriteProtected = *pWriteProtected;
|
2016-10-25 22:26:28 +01:00
|
|
|
if (bExpectFloppy) pImageInfo->pImageHelper = &sg_DiskImageHelper;
|
|
|
|
else pImageInfo->pImageHelper = &sg_HardDiskImageHelper;
|
2009-02-24 22:13:46 +00:00
|
|
|
|
2019-09-07 09:02:39 +01:00
|
|
|
ImageError_e Err = pImageInfo->pImageHelper->Open(pszImageFilename.c_str(), pImageInfo, bCreateIfNecessary, strFilenameInZip);
|
2010-01-03 18:43:08 +00:00
|
|
|
if (Err != eIMAGE_ERROR_NONE)
|
2009-02-24 22:13:46 +00:00
|
|
|
{
|
2020-02-09 21:23:15 +00:00
|
|
|
ImageClose(*ppImageInfo);
|
2016-10-25 21:09:48 +01:00
|
|
|
*ppImageInfo = NULL;
|
2015-02-13 22:40:53 +00:00
|
|
|
return Err;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (pImageInfo->pImageType && pImageInfo->pImageType->GetType() == eImageHDV)
|
|
|
|
{
|
|
|
|
if (bExpectFloppy)
|
2024-08-17 16:14:09 +01:00
|
|
|
{
|
|
|
|
ImageClose(*ppImageInfo);
|
|
|
|
*ppImageInfo = NULL;
|
2015-02-13 22:40:53 +00:00
|
|
|
Err = eIMAGE_ERROR_UNSUPPORTED_HDV;
|
2024-08-17 16:14:09 +01:00
|
|
|
}
|
2010-01-03 18:43:08 +00:00
|
|
|
return Err;
|
2009-02-17 22:53:52 +00:00
|
|
|
}
|
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
// THE FILE MATCHES A KNOWN FORMAT
|
2009-02-17 22:53:52 +00:00
|
|
|
|
2016-10-25 22:26:28 +01:00
|
|
|
_ASSERT(bExpectFloppy);
|
2024-08-17 16:14:09 +01:00
|
|
|
if (!bExpectFloppy || !pImageInfo->uNumTracks)
|
|
|
|
{
|
|
|
|
ImageClose(*ppImageInfo);
|
|
|
|
*ppImageInfo = NULL;
|
2023-06-17 20:41:48 +01:00
|
|
|
return eIMAGE_ERROR_UNSUPPORTED;
|
2024-08-17 16:14:09 +01:00
|
|
|
}
|
2009-02-17 22:53:52 +00:00
|
|
|
|
2015-02-13 22:40:53 +00:00
|
|
|
*pWriteProtected = pImageInfo->bWriteProtected;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
return eIMAGE_ERROR_NONE;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2020-02-09 21:23:15 +00:00
|
|
|
void ImageClose(ImageInfo* const pImageInfo)
|
2010-01-03 18:43:08 +00:00
|
|
|
{
|
2020-02-09 21:23:15 +00:00
|
|
|
pImageInfo->pImageHelper->Close(pImageInfo);
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2019-11-11 14:09:29 +00:00
|
|
|
delete pImageInfo;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
BOOL ImageBoot(ImageInfo* const pImageInfo)
|
2009-02-17 22:53:52 +00:00
|
|
|
{
|
2010-01-03 18:43:08 +00:00
|
|
|
BOOL result = 0;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
if (pImageInfo->pImageType->AllowBoot())
|
|
|
|
result = pImageInfo->pImageType->Boot(pImageInfo);
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
if (result)
|
2016-10-25 21:09:48 +01:00
|
|
|
pImageInfo->bWriteProtected = 1;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
return result;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
void ImageReadTrack( ImageInfo* const pImageInfo,
|
2019-07-05 23:01:19 +01:00
|
|
|
float phase, // phase [0..79] +/- 0.5
|
2010-01-03 18:43:08 +00:00
|
|
|
LPBYTE pTrackImageBuffer,
|
2019-04-07 14:22:05 +01:00
|
|
|
int* pNibbles,
|
2019-07-05 23:01:19 +01:00
|
|
|
UINT* pBitCount,
|
2019-04-11 22:34:40 +01:00
|
|
|
bool enhanceDisk)
|
2006-03-09 21:40:16 +00:00
|
|
|
{
|
2019-07-05 23:01:19 +01:00
|
|
|
_ASSERT(phase >= 0);
|
|
|
|
if (phase < 0)
|
|
|
|
phase = 0;
|
2006-03-09 21:40:16 +00:00
|
|
|
|
2019-07-05 23:01:19 +01:00
|
|
|
const UINT track = pImageInfo->pImageType->PhaseToTrack(phase);
|
|
|
|
|
2020-02-09 21:23:15 +00:00
|
|
|
if (pImageInfo->pImageType->AllowRW())
|
2006-03-09 21:40:16 +00:00
|
|
|
{
|
2019-07-05 23:01:19 +01:00
|
|
|
pImageInfo->pImageType->Read(pImageInfo, phase, pTrackImageBuffer, pNibbles, pBitCount, enhanceDisk);
|
2006-03-09 21:40:16 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-12-31 12:07:45 +00:00
|
|
|
*pNibbles = (int) ImageGetMaxNibblesPerTrack(pImageInfo);
|
|
|
|
for (int i = 0; i < *pNibbles; i++)
|
|
|
|
pTrackImageBuffer[i] = (BYTE)(rand() & 0xFF);
|
2006-03-09 21:40:16 +00:00
|
|
|
}
|
2010-01-03 18:43:08 +00:00
|
|
|
}
|
2006-03-09 21:40:16 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
//===========================================================================
|
2006-03-09 21:40:16 +00:00
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
void ImageWriteTrack( ImageInfo* const pImageInfo,
|
2019-07-05 23:01:19 +01:00
|
|
|
float phase, // phase [0..79] +/- 0.5
|
|
|
|
LPBYTE pTrackImageBuffer,
|
2010-01-03 18:43:08 +00:00
|
|
|
const int nNibbles)
|
|
|
|
{
|
2019-07-05 23:01:19 +01:00
|
|
|
_ASSERT(phase >= 0);
|
|
|
|
if (phase < 0)
|
|
|
|
phase = 0;
|
|
|
|
|
|
|
|
const UINT track = pImageInfo->pImageType->PhaseToTrack(phase);
|
2006-03-09 21:40:16 +00:00
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
if (pImageInfo->pImageType->AllowRW() && !pImageInfo->bWriteProtected)
|
2010-01-03 18:43:08 +00:00
|
|
|
{
|
2019-07-05 23:01:19 +01:00
|
|
|
pImageInfo->pImageType->Write(pImageInfo, phase, pTrackImageBuffer, nNibbles);
|
2020-02-09 21:23:15 +00:00
|
|
|
|
|
|
|
eImageType imageType = pImageInfo->pImageType->GetType();
|
|
|
|
if (imageType == eImageWOZ1 || imageType == eImageWOZ2)
|
|
|
|
{
|
|
|
|
DWORD dummy;
|
|
|
|
bool res = sg_DiskImageHelper.WOZUpdateInfo(pImageInfo, dummy);
|
|
|
|
_ASSERT(res);
|
|
|
|
}
|
2010-01-03 18:43:08 +00:00
|
|
|
}
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
bool ImageReadBlock( ImageInfo* const pImageInfo,
|
2015-02-13 22:40:53 +00:00
|
|
|
UINT nBlock,
|
|
|
|
LPBYTE pBlockBuffer)
|
|
|
|
{
|
|
|
|
bool bRes = false;
|
2016-10-25 21:09:48 +01:00
|
|
|
if (pImageInfo->pImageType->AllowRW())
|
|
|
|
bRes = pImageInfo->pImageType->Read(pImageInfo, nBlock, pBlockBuffer);
|
2015-02-13 22:40:53 +00:00
|
|
|
|
|
|
|
return bRes;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
bool ImageWriteBlock( ImageInfo* const pImageInfo,
|
2015-02-13 22:40:53 +00:00
|
|
|
UINT nBlock,
|
|
|
|
LPBYTE pBlockBuffer)
|
|
|
|
{
|
|
|
|
bool bRes = false;
|
2016-10-25 21:09:48 +01:00
|
|
|
if (pImageInfo->pImageType->AllowRW() && !pImageInfo->bWriteProtected)
|
|
|
|
bRes = pImageInfo->pImageType->Write(pImageInfo, nBlock, pBlockBuffer);
|
2015-02-13 22:40:53 +00:00
|
|
|
|
|
|
|
return bRes;
|
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2019-07-05 23:01:19 +01:00
|
|
|
UINT ImageGetNumTracks(ImageInfo* const pImageInfo)
|
2010-01-03 18:43:08 +00:00
|
|
|
{
|
2016-10-25 21:09:48 +01:00
|
|
|
return pImageInfo ? pImageInfo->uNumTracks : 0;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
2009-05-29 21:39:13 +00:00
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
bool ImageIsMultiFileZip(ImageInfo* const pImageInfo)
|
2009-05-29 21:39:13 +00:00
|
|
|
{
|
2020-02-09 21:23:15 +00:00
|
|
|
return pImageInfo ? (pImageInfo->uNumValidImagesInZip > 1) : false;
|
2009-05-29 21:39:13 +00:00
|
|
|
}
|
2015-02-13 22:40:53 +00:00
|
|
|
|
2019-09-07 09:02:39 +01:00
|
|
|
const std::string & ImageGetPathname(ImageInfo* const pImageInfo)
|
2015-02-13 22:40:53 +00:00
|
|
|
{
|
2019-09-07 09:02:39 +01:00
|
|
|
static const std::string szEmpty;
|
2016-10-25 21:09:48 +01:00
|
|
|
return pImageInfo ? pImageInfo->szFilename : szEmpty;
|
2015-02-13 22:40:53 +00:00
|
|
|
}
|
|
|
|
|
2016-10-25 21:09:48 +01:00
|
|
|
UINT ImageGetImageSize(ImageInfo* const pImageInfo)
|
2015-02-13 22:40:53 +00:00
|
|
|
{
|
2016-10-25 21:09:48 +01:00
|
|
|
return pImageInfo ? pImageInfo->uImageSize : 0;
|
2015-02-13 22:40:53 +00:00
|
|
|
}
|
|
|
|
|
2019-07-05 23:01:19 +01:00
|
|
|
bool ImageIsWOZ(ImageInfo* const pImageInfo)
|
|
|
|
{
|
|
|
|
return pImageInfo ? (pImageInfo->pImageType->GetType() == eImageWOZ1 || pImageInfo->pImageType->GetType() == eImageWOZ2) : false;
|
|
|
|
}
|
|
|
|
|
|
|
|
BYTE ImageGetOptimalBitTiming(ImageInfo* const pImageInfo)
|
|
|
|
{
|
|
|
|
return pImageInfo ? pImageInfo->optimalBitTiming : 32;
|
|
|
|
}
|
|
|
|
|
2020-02-22 11:38:25 +00:00
|
|
|
bool ImageIsBootSectorFormatSector13(ImageInfo* const pImageInfo)
|
|
|
|
{
|
|
|
|
return pImageInfo ? pImageInfo->bootSectorFormat == CWOZHelper::bootSector13 : false;
|
|
|
|
}
|
|
|
|
|
2019-07-05 23:01:19 +01:00
|
|
|
UINT ImagePhaseToTrack(ImageInfo* const pImageInfo, const float phase, const bool limit/*=true*/)
|
|
|
|
{
|
|
|
|
if (!pImageInfo)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
UINT track = pImageInfo->pImageType->PhaseToTrack(phase);
|
|
|
|
|
|
|
|
if (limit)
|
|
|
|
{
|
|
|
|
const UINT numTracksInImage = ImageGetNumTracks(pImageInfo);
|
|
|
|
track = (numTracksInImage == 0) ? 0
|
|
|
|
: MIN(numTracksInImage - 1, track);
|
|
|
|
}
|
|
|
|
|
|
|
|
return track;
|
|
|
|
}
|
|
|
|
|
2019-12-31 12:07:45 +00:00
|
|
|
UINT ImageGetMaxNibblesPerTrack(ImageInfo* const pImageInfo)
|
|
|
|
{
|
|
|
|
return pImageInfo ? pImageInfo->maxNibblesPerTrack : NIBBLES_PER_TRACK;
|
|
|
|
}
|
|
|
|
|
2019-09-07 09:02:39 +01:00
|
|
|
void GetImageTitle(LPCTSTR pPathname, std::string & pImageName, std::string & pFullName)
|
2015-02-13 22:40:53 +00:00
|
|
|
{
|
|
|
|
TCHAR imagetitle[ MAX_DISK_FULL_NAME+1 ];
|
|
|
|
LPCTSTR startpos = pPathname;
|
|
|
|
|
|
|
|
// imagetitle = <FILENAME.EXT>
|
2021-05-19 21:10:22 +01:00
|
|
|
if (_tcsrchr(startpos, TEXT(PATH_SEPARATOR)))
|
|
|
|
startpos = _tcsrchr(startpos, TEXT(PATH_SEPARATOR))+1;
|
2015-02-13 22:40:53 +00:00
|
|
|
|
|
|
|
_tcsncpy(imagetitle, startpos, MAX_DISK_FULL_NAME);
|
|
|
|
imagetitle[MAX_DISK_FULL_NAME] = 0;
|
|
|
|
|
|
|
|
// if imagetitle contains a lowercase char, then found=1 (why?)
|
|
|
|
BOOL found = 0;
|
|
|
|
int loop = 0;
|
|
|
|
while (imagetitle[loop] && !found)
|
|
|
|
{
|
|
|
|
if (IsCharLower(imagetitle[loop]))
|
|
|
|
found = 1;
|
|
|
|
else
|
|
|
|
loop++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((!found) && (loop > 2))
|
|
|
|
CharLowerBuff(imagetitle+1, _tcslen(imagetitle+1));
|
|
|
|
|
|
|
|
// pFullName = <FILENAME.EXT>
|
2019-09-07 09:02:39 +01:00
|
|
|
pFullName = imagetitle;
|
2015-02-13 22:40:53 +00:00
|
|
|
|
|
|
|
if (imagetitle[0])
|
|
|
|
{
|
|
|
|
LPTSTR dot = imagetitle;
|
|
|
|
if (_tcsrchr(dot, TEXT('.')))
|
|
|
|
dot = _tcsrchr(dot, TEXT('.'));
|
|
|
|
if (dot > imagetitle)
|
|
|
|
*dot = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
// pImageName = <FILENAME> (ie. no extension)
|
2019-09-07 09:02:39 +01:00
|
|
|
pImageName = imagetitle;
|
2015-02-13 22:40:53 +00:00
|
|
|
}
|