mirror of https://github.com/stella-emu/stella.git
'3E' bankswitch scheme now always reports 256+32 banks, even if there
really aren't that many ROM banks. This is done so that the RAM banks always start at number 256. Added latest DPC+ updates from Spiceware. git-svn-id: svn://svn.code.sf.net/p/stella/code/trunk@1983 8b62c5a3-ac7e-4cc8-8f21-d9a121418aba
This commit is contained in:
parent
fe0e1c757e
commit
8c3ea00bb2
|
@ -232,9 +232,12 @@ int Cartridge3E::bank()
|
|||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
int Cartridge3E::bankCount()
|
||||
{
|
||||
// In addition to the number of 2K banks in ROM, there are 32 more 1K
|
||||
// banks for RAM (doubled to 2K because of a read and write port)
|
||||
return (mySize >> 11) + 32;
|
||||
// Because the RAM banks always start at 256 and above, we require the
|
||||
// number of ROM banks to be 256
|
||||
// If the RAM banks were simply appended to the number of actual
|
||||
// ROM banks, bank numbers would be ambiguous (ie, would bank 128 be
|
||||
// the last bank of ROM, or one of the banks of RAM?)
|
||||
return 256 + 32;
|
||||
}
|
||||
|
||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
|
|
|
@ -69,13 +69,11 @@ void Cartridge4A50::install(System& system)
|
|||
|
||||
// Map all of the accesses to call peek and poke
|
||||
System::PageAccess access;
|
||||
for(uInt32 i = 0x1000; i < 0x2000; i += (1 << shift))
|
||||
{
|
||||
access.directPeekBase = 0;
|
||||
access.directPokeBase = 0;
|
||||
access.device = this;
|
||||
for(uInt32 i = 0x1000; i < 0x2000; i += (1 << shift))
|
||||
mySystem->setPageAccess(i >> shift, access);
|
||||
}
|
||||
|
||||
// Mirror all access in TIA and RIOT; by doing so we're taking responsibility
|
||||
// for that address space in peek and poke below.
|
||||
|
|
|
@ -92,14 +92,13 @@ void CartridgeAR::install(System& system)
|
|||
// Make sure the system we're being installed in has a page size that'll work
|
||||
assert((0x1000 & mask) == 0);
|
||||
|
||||
// Map all of the accesses to call peek and poke
|
||||
System::PageAccess access;
|
||||
for(uInt32 i = 0x1000; i < 0x2000; i += (1 << shift))
|
||||
{
|
||||
access.directPeekBase = 0;
|
||||
access.directPokeBase = 0;
|
||||
access.device = this;
|
||||
for(uInt32 i = 0x1000; i < 0x2000; i += (1 << shift))
|
||||
mySystem->setPageAccess(i >> shift, access);
|
||||
}
|
||||
|
||||
bankConfiguration(0);
|
||||
}
|
||||
|
|
|
@ -18,13 +18,14 @@
|
|||
|
||||
#include <cassert>
|
||||
#include <cstring>
|
||||
#include <iostream>
|
||||
|
||||
#include "System.hxx"
|
||||
#include "CartDPC.hxx"
|
||||
|
||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
CartridgeDPC::CartridgeDPC(const uInt8* image, uInt32 size)
|
||||
: mySystemCycles(0),
|
||||
myFractionalClocks(0.0)
|
||||
{
|
||||
// Make a copy of the entire image as-is, for use by getImage()
|
||||
// (this wastes 12K of RAM, should be controlled by a #ifdef)
|
||||
|
@ -46,10 +47,6 @@ CartridgeDPC::CartridgeDPC(const uInt8* image, uInt32 size)
|
|||
// Initialize the DPC's random number generator register (must be non-zero)
|
||||
myRandomNumber = 1;
|
||||
|
||||
// Initialize the system cycles counter & fractional clock values
|
||||
mySystemCycles = 0;
|
||||
myFractionalClocks = 0.0;
|
||||
|
||||
// Remember startup bank
|
||||
myStartBank = 1;
|
||||
}
|
||||
|
@ -190,6 +187,11 @@ 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[(myCurrentBank << 12) + 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.
|
||||
|
|
|
@ -26,6 +26,9 @@
|
|||
|
||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
CartridgeDPCPlus::CartridgeDPCPlus(const uInt8* image, uInt32 size)
|
||||
: myFastFetch(false),
|
||||
mySystemCycles(0),
|
||||
myFractionalClocks(0.0)
|
||||
{
|
||||
// Make a copy of the entire image as-is, for use by getImage()
|
||||
// (this wastes 29K of RAM, should be controlled by a #ifdef)
|
||||
|
@ -50,10 +53,6 @@ CartridgeDPCPlus::CartridgeDPCPlus(const uInt8* image, uInt32 size)
|
|||
// Initialize the DPC's random number generator register (must be non-zero)
|
||||
myRandomNumber = 0x2B435044; // "DPC+"
|
||||
|
||||
// Initialize the system cycles counter & fractional clock values
|
||||
mySystemCycles = 0;
|
||||
myFractionalClocks = 0.0;
|
||||
|
||||
// Remember startup bank
|
||||
myStartBank = 5;
|
||||
}
|
||||
|
@ -94,24 +93,13 @@ void CartridgeDPCPlus::install(System& system)
|
|||
// Make sure the system we're being installed in has a page size that'll work
|
||||
assert(((0x1080 & mask) == 0) && ((0x1100 & mask) == 0));
|
||||
|
||||
// Set the page accessing methods for the hot spots
|
||||
// Map all of the accesses to call peek and poke
|
||||
System::PageAccess access;
|
||||
for(uInt32 i = (0x1FF6 & ~mask); i < 0x2000; i += (1 << shift))
|
||||
{
|
||||
access.directPeekBase = 0;
|
||||
access.directPokeBase = 0;
|
||||
access.device = this;
|
||||
for(uInt32 i = 0x1000; i < 0x2000; i += (1 << shift))
|
||||
mySystem->setPageAccess(i >> shift, access);
|
||||
}
|
||||
|
||||
// Set the page accessing method for the DPC reading & writing pages
|
||||
for(uInt32 j = 0x1000; j < 0x1080; j += (1 << shift))
|
||||
{
|
||||
access.directPeekBase = 0;
|
||||
access.directPokeBase = 0;
|
||||
access.device = this;
|
||||
mySystem->setPageAccess(j >> shift, access);
|
||||
}
|
||||
|
||||
// Install pages for the startup bank
|
||||
bank(myStartBank);
|
||||
|
@ -121,8 +109,7 @@ void CartridgeDPCPlus::install(System& system)
|
|||
inline void CartridgeDPCPlus::clockRandomNumberGenerator()
|
||||
{
|
||||
// Update random number generator (32-bit LFSR)
|
||||
myRandomNumber = ((myRandomNumber & 1) ? 0xa260012b: 0x00) ^
|
||||
((myRandomNumber >> 1) & 0x7FFFFFFF);
|
||||
myRandomNumber = ((myRandomNumber & 1) ? 0xa260012b: 0x00) ^ ((myRandomNumber >> 1) & 0x7FFFFFFF);
|
||||
}
|
||||
|
||||
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
||||
|
@ -165,7 +152,22 @@ uInt8 CartridgeDPCPlus::peek(uInt16 address)
|
|||
{
|
||||
address &= 0x0FFF;
|
||||
|
||||
if(address < 0x0040)
|
||||
uInt8 peekvalue = myProgramImage[(myCurrentBank << 12) + address];
|
||||
|
||||
// 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 peekvalue;
|
||||
|
||||
if(myFastFetch && myLDAimmediate)
|
||||
{
|
||||
peekvalue = myProgramImage[(myCurrentBank << 12) + address];
|
||||
if(peekvalue < 0x0028)
|
||||
address = peekvalue;
|
||||
}
|
||||
myLDAimmediate = false;
|
||||
|
||||
if(address < 0x0028)
|
||||
{
|
||||
uInt8 result = 0;
|
||||
|
||||
|
@ -174,11 +176,11 @@ uInt8 CartridgeDPCPlus::peek(uInt16 address)
|
|||
uInt32 function = (address >> 3) & 0x07;
|
||||
|
||||
// Update flag register for selected data fetcher
|
||||
if(((myCounters[index] & 0x00ff00) >> 8) == ((myTops[index]+1) & 0xff))
|
||||
if((myCounters[index] & 0x00ff) == ((myTops[index]+1) & 0xff))
|
||||
{
|
||||
myFlags[index] = 0xff;
|
||||
}
|
||||
else if(((myCounters[index] & 0x00ff00) >> 8) == myBottoms[index])
|
||||
else if((myCounters[index] & 0x00ff) == myBottoms[index])
|
||||
{
|
||||
myFlags[index] = 0x00;
|
||||
}
|
||||
|
@ -211,10 +213,7 @@ uInt8 CartridgeDPCPlus::peek(uInt16 address)
|
|||
result = (myRandomNumber>>24) & 0xFF;
|
||||
break;
|
||||
|
||||
case 0x05:
|
||||
case 0x06:
|
||||
case 0x07:
|
||||
// No, it's a music read
|
||||
case 0x05: // No, it's a music read
|
||||
{
|
||||
static const uInt8 musicAmplitudes[8] = {
|
||||
0x00, 0x04, 0x05, 0x09, 0x06, 0x0a, 0x0b, 0x0f
|
||||
|
@ -238,7 +237,11 @@ uInt8 CartridgeDPCPlus::peek(uInt16 address)
|
|||
}
|
||||
|
||||
result = musicAmplitudes[i];
|
||||
break;
|
||||
}
|
||||
|
||||
case 0x06: // reserved
|
||||
case 0x07: // reserved
|
||||
break;
|
||||
}
|
||||
break;
|
||||
|
@ -247,31 +250,51 @@ uInt8 CartridgeDPCPlus::peek(uInt16 address)
|
|||
// DFx display data read
|
||||
case 0x01:
|
||||
{
|
||||
result = myDisplayImage[myCounters[index] >> 8];
|
||||
myCounters[index] = (myCounters[index] + 0x100) & 0x0fffff;
|
||||
result = myDisplayImage[myCounters[index]];
|
||||
myCounters[index] = (myCounters[index] + 0x1) & 0x0fff;
|
||||
break;
|
||||
}
|
||||
|
||||
// DFx display data read AND'd w/flag
|
||||
case 0x02:
|
||||
{
|
||||
result = myDisplayImage[myCounters[index] >> 8] & myFlags[index];
|
||||
myCounters[index] = (myCounters[index] + 0x100) & 0x0fffff;
|
||||
result = myDisplayImage[myCounters[index]] & myFlags[index];
|
||||
myCounters[index] = (myCounters[index] + 0x1) & 0x0fff;
|
||||
break;
|
||||
}
|
||||
|
||||
// DFx display data read w/fractional increment
|
||||
case 0x03:
|
||||
{
|
||||
result = myDisplayImage[myCounters[index] >> 8];
|
||||
myCounters[index] = (myCounters[index] + myFractionalIncrements[index]) & 0x0fffff;
|
||||
result = myDisplayImage[myFractionalCounters[index] >> 8];
|
||||
myFractionalCounters[index] = (myFractionalCounters[index] + myFractionalIncrements[index]) & 0x0fffff;
|
||||
break;
|
||||
}
|
||||
|
||||
case 0x04:
|
||||
{
|
||||
switch (index)
|
||||
{
|
||||
case 0x00:
|
||||
case 0x01:
|
||||
case 0x02:
|
||||
case 0x03:
|
||||
{
|
||||
result = myFlags[index];
|
||||
break;
|
||||
}
|
||||
case 0x04: // reserved
|
||||
case 0x05: // reserved
|
||||
case 0x06: // reserved
|
||||
case 0x07: // reserved
|
||||
break;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
default:
|
||||
{
|
||||
result = 0;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -315,7 +338,12 @@ uInt8 CartridgeDPCPlus::peek(uInt16 address)
|
|||
default:
|
||||
break;
|
||||
}
|
||||
return myProgramImage[(myCurrentBank << 12) + address];
|
||||
|
||||
peekvalue = myProgramImage[(myCurrentBank << 12) + address];
|
||||
if(myFastFetch)
|
||||
myLDAimmediate = (peekvalue == 0xA9);
|
||||
|
||||
return peekvalue;
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -324,23 +352,37 @@ bool CartridgeDPCPlus::poke(uInt16 address, uInt8 value)
|
|||
{
|
||||
address &= 0x0FFF;
|
||||
|
||||
if((address >= 0x0038) && (address < 0x0080))
|
||||
if((address >= 0x0028) && (address < 0x0080))
|
||||
{
|
||||
// Get the index of the data fetcher that's being accessed
|
||||
uInt32 index = address & 0x07;
|
||||
uInt32 function = ((address - 0x38) >> 3) & 0x0f;
|
||||
uInt32 function = ((address - 0x28) >> 3) & 0x0f;
|
||||
|
||||
switch(function)
|
||||
{
|
||||
//DFxFrac counter low
|
||||
case 0x00:
|
||||
{
|
||||
myFractionalCounters[index] = (myFractionalCounters[index] & 0x0F0000) | ((uInt16)value << 8);
|
||||
break;
|
||||
}
|
||||
|
||||
// DFxFrac counter high
|
||||
case 0x01:
|
||||
{
|
||||
myFractionalCounters[index] = (((uInt16)value & 0x0F) << 16) | (myFractionalCounters[index] & 0x00ffff);
|
||||
break;
|
||||
}
|
||||
|
||||
case 0x02:
|
||||
{
|
||||
myFractionalIncrements[index] = value;
|
||||
myCounters[index] = myCounters[index] & 0x0FFF00;
|
||||
myFractionalCounters[index] = myFractionalCounters[index] & 0x0FFF00;
|
||||
break;
|
||||
}
|
||||
|
||||
// DFx top count
|
||||
case 0x01:
|
||||
case 0x03:
|
||||
{
|
||||
myTops[index] = value;
|
||||
myFlags[index] = 0x00;
|
||||
|
@ -348,53 +390,64 @@ bool CartridgeDPCPlus::poke(uInt16 address, uInt8 value)
|
|||
}
|
||||
|
||||
// DFx bottom count
|
||||
case 0x02:
|
||||
case 0x04:
|
||||
{
|
||||
myBottoms[index] = value;
|
||||
break;
|
||||
}
|
||||
|
||||
// DFx counter low
|
||||
case 0x03:
|
||||
case 0x05:
|
||||
{
|
||||
myCounters[index] = (myCounters[index] & 0x0F0000) | ((uInt16)value << 8);
|
||||
myCounters[index] = (myCounters[index] & 0x0F00) | value ;
|
||||
break;
|
||||
}
|
||||
|
||||
// DFx counter high
|
||||
case 0x04:
|
||||
// Control registers
|
||||
case 0x06:
|
||||
{
|
||||
myCounters[index] = (((uInt16)value & 0x0F) << 16) | (myCounters[index] & 0x00ffff);
|
||||
|
||||
// Execute special code for music mode data fetchers
|
||||
if(index >= 5)
|
||||
switch (index)
|
||||
{
|
||||
// FastFetch control
|
||||
case 0x00:
|
||||
{
|
||||
myFastFetch = (value == 0);
|
||||
break;
|
||||
}
|
||||
case 0x01: // reserved
|
||||
case 0x02: // reserved
|
||||
case 0x03: // reserved
|
||||
case 0x04: // reserved
|
||||
break;
|
||||
case 0x05:
|
||||
case 0x06:
|
||||
case 0x07:
|
||||
{
|
||||
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;
|
||||
}
|
||||
break;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
// DF Push
|
||||
case 0x05:
|
||||
case 0x07:
|
||||
{
|
||||
myCounters[index] = (myCounters[index] - 0x100) & 0x0fffff;
|
||||
myDisplayImage[myCounters[index] >> 8] = value;
|
||||
myCounters[index] = (myCounters[index] - 0x1) & 0x0fff;
|
||||
myDisplayImage[myCounters[index]] = value;
|
||||
break;
|
||||
}
|
||||
|
||||
// DFx counter hi, same as counter high, but w/out music mode
|
||||
case 0x06:
|
||||
case 0x08:
|
||||
{
|
||||
myCounters[index] = (((uInt16)value & 0x0F) << 16) | (myCounters[index] & 0x00ffff);
|
||||
myCounters[index] = (((uInt16)value & 0x0F) << 8) | (myCounters[index] & 0x00ff);
|
||||
break;
|
||||
}
|
||||
|
||||
// Random Number Generator Reset
|
||||
case 0x07:
|
||||
case 0x09:
|
||||
{
|
||||
switch (index)
|
||||
{
|
||||
|
@ -431,7 +484,6 @@ bool CartridgeDPCPlus::poke(uInt16 address, uInt8 value)
|
|||
(myFrequencyImage[(value<<2)+1]<<8) +
|
||||
(myFrequencyImage[(value<<2)+2]<<16) +
|
||||
(myFrequencyImage[(value<<2)+3]<<24);
|
||||
|
||||
break;
|
||||
}
|
||||
default:
|
||||
|
@ -441,17 +493,19 @@ bool CartridgeDPCPlus::poke(uInt16 address, uInt8 value)
|
|||
}
|
||||
|
||||
// DF Write
|
||||
case 0x08:
|
||||
case 0x0a:
|
||||
{
|
||||
myDisplayImage[myCounters[index] >> 8] = value;
|
||||
myCounters[index] = (myCounters[index] + 0x100) & 0x0fffff;
|
||||
myDisplayImage[myCounters[index]] = value;
|
||||
myCounters[index] = (myCounters[index] + 0x1) & 0x0fff;
|
||||
break;
|
||||
}
|
||||
|
||||
default:
|
||||
{
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// Switch banks if necessary
|
||||
|
@ -501,22 +555,7 @@ void CartridgeDPCPlus::bank(uInt16 bank)
|
|||
|
||||
// Remember what bank we're in
|
||||
myCurrentBank = bank;
|
||||
uInt16 offset = myCurrentBank << 12;
|
||||
uInt16 shift = mySystem->pageShift();
|
||||
uInt16 mask = mySystem->pageMask();
|
||||
|
||||
// Setup the page access methods for the current bank
|
||||
System::PageAccess access;
|
||||
access.device = this;
|
||||
access.directPokeBase = 0;
|
||||
|
||||
// Map Program ROM image into the system
|
||||
for(uInt32 address = 0x1080; address < (0x1FF8U & ~mask);
|
||||
address += (1 << shift))
|
||||
{
|
||||
access.directPeekBase = &myProgramImage[offset + (address & 0x0FFF)];
|
||||
mySystem->setPageAccess(address >> shift, access);
|
||||
}
|
||||
myBankChanged = true;
|
||||
}
|
||||
|
||||
|
@ -583,6 +622,11 @@ bool CartridgeDPCPlus::save(Serializer& out) const
|
|||
for(i = 0; i < 8; ++i)
|
||||
out.putInt(myCounters[i]);
|
||||
|
||||
// The counter registers for the fractional data fetchers
|
||||
out.putInt(8);
|
||||
for(i = 0; i < 8; ++i)
|
||||
out.putInt(myFractionalCounters[i]);
|
||||
|
||||
// The fractional registers for the data fetchers
|
||||
out.putInt(8);
|
||||
for(i = 0; i < 8; ++i)
|
||||
|
@ -593,6 +637,10 @@ bool CartridgeDPCPlus::save(Serializer& out) const
|
|||
for(i = 0; i < 8; ++i)
|
||||
out.putByte((char)myFlags[i]);
|
||||
|
||||
// The Fast Fetcher Enabled flag
|
||||
out.putBool(myFastFetch);
|
||||
out.putBool(myLDAimmediate);
|
||||
|
||||
// The music mode flags for the data fetchers
|
||||
out.putInt(3);
|
||||
for(i = 0; i < 3; ++i)
|
||||
|
@ -651,7 +699,12 @@ bool CartridgeDPCPlus::load(Serializer& in)
|
|||
// The counter registers for the data fetchers
|
||||
limit = (uInt32) in.getInt();
|
||||
for(i = 0; i < limit; ++i)
|
||||
myCounters[i] = (uInt32) in.getInt();
|
||||
myCounters[i] = (uInt16) in.getInt();
|
||||
|
||||
// The counter registers for the data fetchers
|
||||
limit = (uInt32) in.getInt();
|
||||
for(i = 0; i < limit; ++i)
|
||||
myFractionalCounters[i] = (uInt32) in.getInt();
|
||||
|
||||
// The fractional registers for the data fetchers
|
||||
limit = (uInt32) in.getInt();
|
||||
|
@ -663,6 +716,10 @@ bool CartridgeDPCPlus::load(Serializer& in)
|
|||
for(i = 0; i < limit; ++i)
|
||||
myFlags[i] = (uInt8) in.getByte();
|
||||
|
||||
// The Fast Fetcher Enabled flag
|
||||
myFastFetch = in.getBool();
|
||||
myLDAimmediate = in.getBool();
|
||||
|
||||
// The music mode flags for the data fetchers
|
||||
limit = (uInt32) in.getInt();
|
||||
for(i = 0; i < limit; ++i)
|
||||
|
|
|
@ -182,7 +182,10 @@ class CartridgeDPCPlus : public Cartridge
|
|||
uInt8 myBottoms[8];
|
||||
|
||||
// The counter registers for the data fetchers
|
||||
uInt32 myCounters[8];
|
||||
uInt16 myCounters[8];
|
||||
|
||||
// The counter registers for the fractional data fetchers
|
||||
uInt32 myFractionalCounters[8];
|
||||
|
||||
// The fractional increments for the data fetchers
|
||||
uInt8 myFractionalIncrements[8];
|
||||
|
@ -190,6 +193,10 @@ class CartridgeDPCPlus : public Cartridge
|
|||
// The flag registers for the data fetchers
|
||||
uInt8 myFlags[8];
|
||||
|
||||
// The Fast Fetcher Enabled flag
|
||||
bool myFastFetch;
|
||||
bool myLDAimmediate;
|
||||
|
||||
// The music mode DF5, DF6, & DF7 enabled flags
|
||||
bool myMusicMode[3];
|
||||
|
||||
|
|
Loading…
Reference in New Issue