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"
|
2010-01-03 18:43:08 +00:00
|
|
|
#include "DiskImage.h"
|
|
|
|
#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;
|
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()
|
|
|
|
ImageError_e ImageOpen( LPCTSTR pszImageFilename,
|
|
|
|
HIMAGE* hDiskImage_,
|
|
|
|
bool* pWriteProtected_,
|
|
|
|
const bool bCreateIfNecessary,
|
|
|
|
std::string& strFilenameInZip)
|
2009-02-17 22:53:52 +00:00
|
|
|
{
|
2010-01-03 18:43:08 +00:00
|
|
|
if (! (pszImageFilename && hDiskImage_ && pWriteProtected_ && sg_DiskImageHelper.GetWorkBuffer()))
|
|
|
|
return eIMAGE_ERROR_BAD_POINTER;
|
2009-02-24 22:13:46 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
// CREATE A RECORD FOR THE FILE, AND RETURN AN IMAGE HANDLE
|
|
|
|
*hDiskImage_ = (HIMAGE) VirtualAlloc(NULL, sizeof(ImageInfo), MEM_COMMIT, PAGE_READWRITE);
|
|
|
|
if (*hDiskImage_ == NULL)
|
|
|
|
return eIMAGE_ERROR_BAD_POINTER;
|
2009-02-17 22:53:52 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
ZeroMemory(*hDiskImage_, sizeof(ImageInfo));
|
|
|
|
ImageInfo* pImageInfo = (ImageInfo*) *hDiskImage_;
|
|
|
|
pImageInfo->bWriteProtected = *pWriteProtected_;
|
2009-02-24 22:13:46 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
ImageError_e Err = sg_DiskImageHelper.Open(pszImageFilename, pImageInfo, bCreateIfNecessary, strFilenameInZip);
|
2009-02-24 22:13:46 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
if (pImageInfo->pImageType != NULL && Err == eIMAGE_ERROR_NONE && pImageInfo->pImageType->GetType() == eImageHDV)
|
|
|
|
Err = eIMAGE_ERROR_UNSUPPORTED_HDV;
|
|
|
|
|
|
|
|
if (Err != eIMAGE_ERROR_NONE)
|
2009-02-24 22:13:46 +00:00
|
|
|
{
|
2010-01-03 18:43:08 +00:00
|
|
|
ImageClose(*hDiskImage_, true);
|
|
|
|
*hDiskImage_ = (HIMAGE)0;
|
|
|
|
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
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
pImageInfo->uNumTracks = sg_DiskImageHelper.GetNumTracksInImage(pImageInfo->pImageType);
|
2009-02-17 22:53:52 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
for (UINT uTrack = 0; uTrack < pImageInfo->uNumTracks; uTrack++)
|
|
|
|
pImageInfo->ValidTrack[uTrack] = (pImageInfo->uImageSize > 0) ? 1 : 0;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +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
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
void ImageClose(const HIMAGE hDiskImage, const bool bOpenError /*=false*/)
|
|
|
|
{
|
|
|
|
ImageInfo* ptr = (ImageInfo*) hDiskImage;
|
|
|
|
bool bDeleteFile = false;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
if (!bOpenError)
|
|
|
|
{
|
|
|
|
for (UINT uTrack = 0; uTrack < ptr->uNumTracks; uTrack++)
|
|
|
|
{
|
|
|
|
if (!ptr->ValidTrack[uTrack])
|
|
|
|
{
|
2010-02-14 21:11:26 +00:00
|
|
|
// TODO: Comment using info from this URL:
|
|
|
|
// http://groups.google.de/group/comp.emulators.apple2/msg/7a1b9317e7905152
|
2010-01-03 18:43:08 +00:00
|
|
|
bDeleteFile = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
sg_DiskImageHelper.Close(ptr, bDeleteFile);
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
VirtualFree(ptr, 0, MEM_RELEASE);
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
BOOL ImageBoot(const HIMAGE hDiskImage)
|
2009-02-17 22:53:52 +00:00
|
|
|
{
|
2010-01-03 18:43:08 +00:00
|
|
|
ImageInfo* ptr = (ImageInfo*) hDiskImage;
|
|
|
|
BOOL result = 0;
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
if (ptr->pImageType->AllowBoot())
|
|
|
|
result = ptr->pImageType->Boot(ptr);
|
2006-02-25 20:50:29 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
if (result)
|
|
|
|
ptr->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
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
void ImageDestroy(void)
|
|
|
|
{
|
|
|
|
VirtualFree(sg_DiskImageHelper.GetWorkBuffer(), 0, MEM_RELEASE);
|
|
|
|
sg_DiskImageHelper.SetWorkBuffer(NULL);
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
void ImageInitialize(void)
|
|
|
|
{
|
|
|
|
LPBYTE pBuffer = (LPBYTE) VirtualAlloc(NULL, TRACK_DENIBBLIZED_SIZE*2, MEM_COMMIT, PAGE_READWRITE);
|
|
|
|
sg_DiskImageHelper.SetWorkBuffer(pBuffer);
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
void ImageReadTrack( const HIMAGE hDiskImage,
|
|
|
|
const int nTrack,
|
|
|
|
const int nQuarterTrack,
|
|
|
|
LPBYTE pTrackImageBuffer,
|
|
|
|
int* pNibbles)
|
2006-03-09 21:40:16 +00:00
|
|
|
{
|
2010-01-03 18:43:08 +00:00
|
|
|
_ASSERT(nTrack >= 0);
|
|
|
|
if (nTrack < 0)
|
|
|
|
return;
|
2006-03-09 21:40:16 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
ImageInfo* ptr = (ImageInfo*) hDiskImage;
|
|
|
|
if (ptr->pImageType->AllowRW() && ptr->ValidTrack[nTrack])
|
2006-03-09 21:40:16 +00:00
|
|
|
{
|
2010-01-03 18:43:08 +00:00
|
|
|
ptr->pImageType->Read(ptr, nTrack, nQuarterTrack, pTrackImageBuffer, pNibbles);
|
2006-03-09 21:40:16 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-01-03 18:43:08 +00:00
|
|
|
for (*pNibbles = 0; *pNibbles < NIBBLES_PER_TRACK; (*pNibbles)++)
|
|
|
|
pTrackImageBuffer[*pNibbles] = (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
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
void ImageWriteTrack( const HIMAGE hDiskImage,
|
|
|
|
const int nTrack,
|
|
|
|
const int nQuarterTrack,
|
|
|
|
LPBYTE pTrackImage,
|
|
|
|
const int nNibbles)
|
|
|
|
{
|
|
|
|
_ASSERT(nTrack >= 0);
|
|
|
|
if (nTrack < 0)
|
|
|
|
return;
|
2006-03-09 21:40:16 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
ImageInfo* ptr = (ImageInfo*) hDiskImage;
|
|
|
|
if (ptr->pImageType->AllowRW() && !ptr->bWriteProtected)
|
|
|
|
{
|
|
|
|
ptr->pImageType->Write(ptr, nTrack, nQuarterTrack, pTrackImage, nNibbles);
|
|
|
|
ptr->ValidTrack[nTrack] = 1;
|
|
|
|
}
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//===========================================================================
|
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
int ImageGetNumTracks(const HIMAGE hDiskImage)
|
|
|
|
{
|
|
|
|
ImageInfo* ptr = (ImageInfo*) hDiskImage;
|
|
|
|
return ptr ? ptr->uNumTracks : 0;
|
2006-02-25 20:50:29 +00:00
|
|
|
}
|
2009-05-29 21:39:13 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
bool ImageIsWriteProtected(const HIMAGE hDiskImage)
|
|
|
|
{
|
|
|
|
ImageInfo* ptr = (ImageInfo*) hDiskImage;
|
|
|
|
return ptr ? ptr->bWriteProtected : true;
|
|
|
|
}
|
2009-05-29 21:39:13 +00:00
|
|
|
|
2010-01-03 18:43:08 +00:00
|
|
|
bool ImageIsMultiFileZip(const HIMAGE hDiskImage)
|
2009-05-29 21:39:13 +00:00
|
|
|
{
|
2010-01-03 18:43:08 +00:00
|
|
|
ImageInfo* ptr = (ImageInfo*) hDiskImage;
|
|
|
|
return ptr ? (ptr->uNumEntriesInZip > 1) : false;
|
2009-05-29 21:39:13 +00:00
|
|
|
}
|