mirror of https://github.com/stella-emu/stella.git
541 lines
14 KiB
C++
541 lines
14 KiB
C++
//============================================================================
|
|
//
|
|
// SSSS tt lll lll
|
|
// SS SS tt ll ll
|
|
// SS tttttt eeee ll ll aaaa
|
|
// SSSS tt ee ee ll ll aa
|
|
// SS tt eeeeee ll ll aaaaa -- "An Atari 2600 VCS Emulator"
|
|
// SS SS tt ee ll ll aa aa
|
|
// SSSS ttt eeeee llll llll aaaaa
|
|
//
|
|
// Copyright (c) 1995-2017 by Bradford W. Mott, Stephen Anthony
|
|
// and the Stella Team
|
|
//
|
|
// See the file "License.txt" for information on usage and redistribution of
|
|
// this file, and for a DISCLAIMER OF ALL WARRANTIES.
|
|
//============================================================================
|
|
|
|
#include "System.hxx"
|
|
#include "CartDPC.hxx"
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
CartridgeDPC::CartridgeDPC(const BytePtr& image, uInt32 size,
|
|
const Settings& settings)
|
|
: Cartridge(settings),
|
|
mySize(size),
|
|
mySystemCycles(0),
|
|
myFractionalClocks(0.0),
|
|
myBankOffset(0)
|
|
{
|
|
// Make a copy of the entire image
|
|
memcpy(myImage, image.get(), std::min(size, 8192u + 2048u + 256u));
|
|
createCodeAccessBase(8192);
|
|
|
|
// Pointer to the program ROM (8K @ 0 byte offset)
|
|
myProgramImage = myImage;
|
|
|
|
// Pointer to the display ROM (2K @ 8K offset)
|
|
myDisplayImage = myProgramImage + 8192;
|
|
|
|
// Initialize the DPC data fetcher registers
|
|
for(int i = 0; i < 8; ++i)
|
|
myTops[i] = myBottoms[i] = myCounters[i] = myFlags[i] = 0;
|
|
|
|
// None of the data fetchers are in music mode
|
|
myMusicMode[0] = myMusicMode[1] = myMusicMode[2] = false;
|
|
|
|
// Initialize the DPC's random number generator register (must be non-zero)
|
|
myRandomNumber = 1;
|
|
|
|
// Remember startup bank
|
|
myStartBank = 1;
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
void CartridgeDPC::reset()
|
|
{
|
|
// Update cycles to the current system cycles
|
|
mySystemCycles = mySystem->cycles();
|
|
myFractionalClocks = 0.0;
|
|
|
|
// Upon reset we switch to the startup bank
|
|
bank(myStartBank);
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
void CartridgeDPC::systemCyclesReset()
|
|
{
|
|
// Adjust the cycle counter so that it reflects the new value
|
|
mySystemCycles -= mySystem->cycles();
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
void CartridgeDPC::install(System& system)
|
|
{
|
|
mySystem = &system;
|
|
|
|
// Set the page accessing method for the DPC reading & writing pages
|
|
System::PageAccess access(this, System::PA_READWRITE);
|
|
for(uInt32 j = 0x1000; j < 0x1080; j += (1 << System::PAGE_SHIFT))
|
|
mySystem->setPageAccess(j >> System::PAGE_SHIFT, access);
|
|
|
|
// Install pages for the startup bank
|
|
bank(myStartBank);
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
inline void CartridgeDPC::clockRandomNumberGenerator()
|
|
{
|
|
// Table for computing the input bit of the random number generator's
|
|
// shift register (it's the NOT of the EOR of four bits)
|
|
static constexpr uInt8 f[16] = {
|
|
1, 0, 0, 1, 0, 1, 1, 0, 0, 1, 1, 0, 1, 0, 0, 1
|
|
};
|
|
|
|
// Using bits 7, 5, 4, & 3 of the shift register compute the input
|
|
// bit for the shift register
|
|
uInt8 bit = f[((myRandomNumber >> 3) & 0x07) |
|
|
((myRandomNumber & 0x80) ? 0x08 : 0x00)];
|
|
|
|
// Update the shift register
|
|
myRandomNumber = (myRandomNumber << 1) | bit;
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
inline void CartridgeDPC::updateMusicModeDataFetchers()
|
|
{
|
|
// Calculate the number of cycles since the last update
|
|
Int32 cycles = mySystem->cycles() - mySystemCycles;
|
|
mySystemCycles = mySystem->cycles();
|
|
|
|
// Calculate the number of DPC OSC clocks since the last update
|
|
double clocks = ((20000.0 * cycles) / 1193191.66666667) + myFractionalClocks;
|
|
Int32 wholeClocks = Int32(clocks);
|
|
myFractionalClocks = clocks - double(wholeClocks);
|
|
|
|
if(wholeClocks <= 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
// Let's update counters and flags of the music mode data fetchers
|
|
for(int x = 5; x <= 7; ++x)
|
|
{
|
|
// Update only if the data fetcher is in music mode
|
|
if(myMusicMode[x - 5])
|
|
{
|
|
Int32 top = myTops[x] + 1;
|
|
Int32 newLow = Int32(myCounters[x] & 0x00ff);
|
|
|
|
if(myTops[x] != 0)
|
|
{
|
|
newLow -= (wholeClocks % top);
|
|
if(newLow < 0)
|
|
{
|
|
newLow += top;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
newLow = 0;
|
|
}
|
|
|
|
// Update flag register for this data fetcher
|
|
if(newLow <= myBottoms[x])
|
|
{
|
|
myFlags[x] = 0x00;
|
|
}
|
|
else if(newLow <= myTops[x])
|
|
{
|
|
myFlags[x] = 0xff;
|
|
}
|
|
|
|
myCounters[x] = (myCounters[x] & 0x0700) | uInt16(newLow);
|
|
}
|
|
}
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
uInt8 CartridgeDPC::peek(uInt16 address)
|
|
{
|
|
address &= 0x0FFF;
|
|
|
|
// In debugger/bank-locked mode, we ignore all hotspots and in general
|
|
// anything that can change the internal state of the cart
|
|
if(bankLocked())
|
|
return myProgramImage[myBankOffset + address];
|
|
|
|
// Clock the random number generator. This should be done for every
|
|
// cartridge access, however, we're only doing it for the DPC and
|
|
// hot-spot accesses to save time.
|
|
clockRandomNumberGenerator();
|
|
|
|
if(address < 0x0040)
|
|
{
|
|
uInt8 result = 0;
|
|
|
|
// Get the index of the data fetcher that's being accessed
|
|
uInt32 index = address & 0x07;
|
|
uInt32 function = (address >> 3) & 0x07;
|
|
|
|
// Update flag register for selected data fetcher
|
|
if((myCounters[index] & 0x00ff) == myTops[index])
|
|
{
|
|
myFlags[index] = 0xff;
|
|
}
|
|
else if((myCounters[index] & 0x00ff) == myBottoms[index])
|
|
{
|
|
myFlags[index] = 0x00;
|
|
}
|
|
|
|
switch(function)
|
|
{
|
|
case 0x00:
|
|
{
|
|
// Is this a random number read
|
|
if(index < 4)
|
|
{
|
|
result = myRandomNumber;
|
|
}
|
|
// No, it's a music read
|
|
else
|
|
{
|
|
static constexpr uInt8 musicAmplitudes[8] = {
|
|
0x00, 0x04, 0x05, 0x09, 0x06, 0x0a, 0x0b, 0x0f
|
|
};
|
|
|
|
// Update the music data fetchers (counter & flag)
|
|
updateMusicModeDataFetchers();
|
|
|
|
uInt8 i = 0;
|
|
if(myMusicMode[0] && myFlags[5])
|
|
{
|
|
i |= 0x01;
|
|
}
|
|
if(myMusicMode[1] && myFlags[6])
|
|
{
|
|
i |= 0x02;
|
|
}
|
|
if(myMusicMode[2] && myFlags[7])
|
|
{
|
|
i |= 0x04;
|
|
}
|
|
|
|
result = musicAmplitudes[i];
|
|
}
|
|
break;
|
|
}
|
|
|
|
// DFx display data read
|
|
case 0x01:
|
|
{
|
|
result = myDisplayImage[2047 - myCounters[index]];
|
|
break;
|
|
}
|
|
|
|
// DFx display data read AND'd w/flag
|
|
case 0x02:
|
|
{
|
|
result = myDisplayImage[2047 - myCounters[index]] & myFlags[index];
|
|
break;
|
|
}
|
|
|
|
// DFx flag
|
|
case 0x07:
|
|
{
|
|
result = myFlags[index];
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
result = 0;
|
|
}
|
|
}
|
|
|
|
// Clock the selected data fetcher's counter if needed
|
|
if((index < 5) || ((index >= 5) && (!myMusicMode[index - 5])))
|
|
{
|
|
myCounters[index] = (myCounters[index] - 1) & 0x07ff;
|
|
}
|
|
|
|
return result;
|
|
}
|
|
else
|
|
{
|
|
// Switch banks if necessary
|
|
switch(address)
|
|
{
|
|
case 0x0FF8:
|
|
// Set the current bank to the lower 4k bank
|
|
bank(0);
|
|
break;
|
|
|
|
case 0x0FF9:
|
|
// Set the current bank to the upper 4k bank
|
|
bank(1);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
return myProgramImage[myBankOffset + address];
|
|
}
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
bool CartridgeDPC::poke(uInt16 address, uInt8 value)
|
|
{
|
|
address &= 0x0FFF;
|
|
|
|
// Clock the random number generator. This should be done for every
|
|
// cartridge access, however, we're only doing it for the DPC and
|
|
// hot-spot accesses to save time.
|
|
clockRandomNumberGenerator();
|
|
|
|
if((address >= 0x0040) && (address < 0x0080))
|
|
{
|
|
// Get the index of the data fetcher that's being accessed
|
|
uInt32 index = address & 0x07;
|
|
uInt32 function = (address >> 3) & 0x07;
|
|
|
|
switch(function)
|
|
{
|
|
// DFx top count
|
|
case 0x00:
|
|
{
|
|
myTops[index] = value;
|
|
myFlags[index] = 0x00;
|
|
break;
|
|
}
|
|
|
|
// DFx bottom count
|
|
case 0x01:
|
|
{
|
|
myBottoms[index] = value;
|
|
break;
|
|
}
|
|
|
|
// DFx counter low
|
|
case 0x02:
|
|
{
|
|
if((index >= 5) && myMusicMode[index - 5])
|
|
{
|
|
// Data fetcher is in music mode so its low counter value
|
|
// should be loaded from the top register not the poked value
|
|
myCounters[index] = (myCounters[index] & 0x0700) |
|
|
uInt16(myTops[index]);
|
|
}
|
|
else
|
|
{
|
|
// Data fetcher is either not a music mode data fetcher or it
|
|
// isn't in music mode so it's low counter value should be loaded
|
|
// with the poked value
|
|
myCounters[index] = (myCounters[index] & 0x0700) | uInt16(value);
|
|
}
|
|
break;
|
|
}
|
|
|
|
// DFx counter high
|
|
case 0x03:
|
|
{
|
|
myCounters[index] = ((uInt16(value) & 0x07) << 8) |
|
|
(myCounters[index] & 0x00ff);
|
|
|
|
// Execute special code for music mode data fetchers
|
|
if(index >= 5)
|
|
{
|
|
myMusicMode[index - 5] = (value & 0x10);
|
|
|
|
// NOTE: We are not handling the clock source input for
|
|
// the music mode data fetchers. We're going to assume
|
|
// they always use the OSC input.
|
|
}
|
|
break;
|
|
}
|
|
|
|
// Random Number Generator Reset
|
|
case 0x06:
|
|
{
|
|
myRandomNumber = 1;
|
|
break;
|
|
}
|
|
|
|
default:
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Switch banks if necessary
|
|
switch(address)
|
|
{
|
|
case 0x0FF8:
|
|
// Set the current bank to the lower 4k bank
|
|
bank(0);
|
|
break;
|
|
|
|
case 0x0FF9:
|
|
// Set the current bank to the upper 4k bank
|
|
bank(1);
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
bool CartridgeDPC::bank(uInt16 bank)
|
|
{
|
|
if(bankLocked()) return false;
|
|
|
|
// Remember what bank we're in
|
|
myBankOffset = bank << 12;
|
|
|
|
System::PageAccess access(this, System::PA_READ);
|
|
|
|
// Set the page accessing methods for the hot spots
|
|
for(uInt32 i = (0x1FF8 & ~System::PAGE_MASK); i < 0x2000;
|
|
i += (1 << System::PAGE_SHIFT))
|
|
{
|
|
access.codeAccessBase = &myCodeAccessBase[myBankOffset + (i & 0x0FFF)];
|
|
mySystem->setPageAccess(i >> System::PAGE_SHIFT, access);
|
|
}
|
|
|
|
// Setup the page access methods for the current bank
|
|
for(uInt32 address = 0x1080; address < (0x1FF8U & ~System::PAGE_MASK);
|
|
address += (1 << System::PAGE_SHIFT))
|
|
{
|
|
access.directPeekBase = &myProgramImage[myBankOffset + (address & 0x0FFF)];
|
|
access.codeAccessBase = &myCodeAccessBase[myBankOffset + (address & 0x0FFF)];
|
|
mySystem->setPageAccess(address >> System::PAGE_SHIFT, access);
|
|
}
|
|
return myBankChanged = true;
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
uInt16 CartridgeDPC::getBank() const
|
|
{
|
|
return myBankOffset >> 12;
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
uInt16 CartridgeDPC::bankCount() const
|
|
{
|
|
return 2;
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
bool CartridgeDPC::patch(uInt16 address, uInt8 value)
|
|
{
|
|
address &= 0x0FFF;
|
|
|
|
// For now, we ignore attempts to patch the DPC address space
|
|
if(address >= 0x0080)
|
|
{
|
|
myProgramImage[myBankOffset + (address & 0x0FFF)] = value;
|
|
return myBankChanged = true;
|
|
}
|
|
else
|
|
return false;
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
const uInt8* CartridgeDPC::getImage(uInt32& size) const
|
|
{
|
|
size = mySize;
|
|
return myImage;
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
bool CartridgeDPC::save(Serializer& out) const
|
|
{
|
|
try
|
|
{
|
|
out.putString(name());
|
|
|
|
// Indicates which bank is currently active
|
|
out.putShort(myBankOffset);
|
|
|
|
// The top registers for the data fetchers
|
|
out.putByteArray(myTops, 8);
|
|
|
|
// The bottom registers for the data fetchers
|
|
out.putByteArray(myBottoms, 8);
|
|
|
|
// The counter registers for the data fetchers
|
|
out.putShortArray(myCounters, 8);
|
|
|
|
// The flag registers for the data fetchers
|
|
out.putByteArray(myFlags, 8);
|
|
|
|
// The music mode flags for the data fetchers
|
|
for(int i = 0; i < 3; ++i)
|
|
out.putBool(myMusicMode[i]);
|
|
|
|
// The random number generator register
|
|
out.putByte(myRandomNumber);
|
|
|
|
out.putInt(mySystemCycles);
|
|
out.putInt(uInt32(myFractionalClocks * 100000000.0));
|
|
}
|
|
catch(...)
|
|
{
|
|
cerr << "ERROR: CartridgeDPC::save" << endl;
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
bool CartridgeDPC::load(Serializer& in)
|
|
{
|
|
try
|
|
{
|
|
if(in.getString() != name())
|
|
return false;
|
|
|
|
// Indicates which bank is currently active
|
|
myBankOffset = in.getShort();
|
|
|
|
// The top registers for the data fetchers
|
|
in.getByteArray(myTops, 8);
|
|
|
|
// The bottom registers for the data fetchers
|
|
in.getByteArray(myBottoms, 8);
|
|
|
|
// The counter registers for the data fetchers
|
|
in.getShortArray(myCounters, 8);
|
|
|
|
// The flag registers for the data fetchers
|
|
in.getByteArray(myFlags, 8);
|
|
|
|
// The music mode flags for the data fetchers
|
|
for(int i = 0; i < 3; ++i)
|
|
myMusicMode[i] = in.getBool();
|
|
|
|
// The random number generator register
|
|
myRandomNumber = in.getByte();
|
|
|
|
// Get system cycles and fractional clocks
|
|
mySystemCycles = Int32(in.getInt());
|
|
myFractionalClocks = double(in.getInt()) / 100000000.0;
|
|
}
|
|
catch(...)
|
|
{
|
|
cerr << "ERROR: CartridgeDPC::load" << endl;
|
|
return false;
|
|
}
|
|
|
|
// Now, go to the current bank
|
|
bank(myBankOffset >> 12);
|
|
|
|
return true;
|
|
}
|