2012-03-31 20:53:14 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Text;
|
|
|
|
|
|
|
|
|
|
namespace BizHawk.Emulation.Consoles.Atari._2600
|
|
|
|
|
{
|
2012-04-02 16:30:55 +00:00
|
|
|
|
/*
|
|
|
|
|
E7 (M-Network)
|
|
|
|
|
-----
|
|
|
|
|
|
|
|
|
|
M-network wanted something of their own too, so they came up with what they called
|
|
|
|
|
"Big Game" (this was printed on the prototype ASICs on the prototype carts). It
|
|
|
|
|
can handle up to 16K of ROM and 2K of RAM.
|
|
|
|
|
|
|
|
|
|
1000-17FF is selectable
|
|
|
|
|
1800-19FF is RAM
|
|
|
|
|
1A00-1FFF is fixed to the last 1.5K of ROM
|
|
|
|
|
|
|
|
|
|
Accessing 1FE0 through 1FE6 selects bank 0 through bank 6 of the ROM into 1000-17FF.
|
|
|
|
|
Accessing 1FE7 enables 1K of the 2K RAM, instead.
|
|
|
|
|
|
|
|
|
|
When the RAM is enabled, this 1K appears at 1000-17FF. 1000-13FF is the write port, 1400-17FF
|
|
|
|
|
is the read port.
|
|
|
|
|
|
|
|
|
|
1800-19FF also holds RAM. 1800-18FF is the write port, 1900-19FF is the read port.
|
|
|
|
|
Only 256 bytes of RAM is accessable at time, but there are four different 256 byte
|
|
|
|
|
banks making a total of 1K accessable here.
|
|
|
|
|
|
|
|
|
|
Accessing 1FE8 through 1FEB select which 256 byte bank shows up.
|
|
|
|
|
*/
|
|
|
|
|
|
2012-03-31 20:53:14 +00:00
|
|
|
|
class mE7 : MapperBase
|
|
|
|
|
{
|
2012-10-19 00:13:39 +00:00
|
|
|
|
int rombank_1k = 0;
|
2012-04-02 16:30:55 +00:00
|
|
|
|
int rambank1_toggle = 0;
|
|
|
|
|
ByteBuffer rambank0 = new ByteBuffer(1024);
|
|
|
|
|
ByteBuffer rambank1 = new ByteBuffer(1024);
|
|
|
|
|
bool EnableRam0 = false;
|
|
|
|
|
|
|
|
|
|
public override byte ReadMemory(ushort addr)
|
|
|
|
|
{
|
2012-10-19 00:13:39 +00:00
|
|
|
|
if (addr < 0x1000)
|
|
|
|
|
{
|
|
|
|
|
return base.ReadMemory(addr);
|
|
|
|
|
}
|
|
|
|
|
|
2012-04-02 16:30:55 +00:00
|
|
|
|
Address(addr);
|
2012-10-19 00:13:39 +00:00
|
|
|
|
if (addr < 0x1800)
|
2012-04-02 16:30:55 +00:00
|
|
|
|
{
|
|
|
|
|
if (EnableRam0)
|
|
|
|
|
{
|
2012-10-19 00:13:39 +00:00
|
|
|
|
if (addr < 0x1400) //Reading from the write port
|
|
|
|
|
{
|
|
|
|
|
return rambank0[addr & 0x3FF] = 0xFF; //Reading from 1k write port triggers an unwanted write
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return rambank0[addr & 0x3FF];
|
|
|
|
|
}
|
2012-04-02 16:30:55 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2012-10-19 00:13:39 +00:00
|
|
|
|
return core.rom[(rombank_1k * 0x800) + (addr & 0x7FF)];
|
2012-04-02 16:30:55 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2012-10-19 00:13:39 +00:00
|
|
|
|
else if (addr < 0x1900) //Ram 1 Read port
|
2012-10-15 14:29:13 +00:00
|
|
|
|
{
|
2012-10-19 00:13:39 +00:00
|
|
|
|
return rambank1[(rambank1_toggle * 0x100) + (addr & 0xFF)];
|
2012-10-15 14:29:13 +00:00
|
|
|
|
}
|
2012-10-19 00:13:39 +00:00
|
|
|
|
else if (addr < 0x1A00) //Ram 1 Write port
|
2012-10-15 14:29:13 +00:00
|
|
|
|
{
|
2012-10-19 00:13:39 +00:00
|
|
|
|
return rambank1[(rambank1_toggle * 0x100) + (addr & 0xFF)] = 0xFF; //Reading from the 256b write port @1800 riggers an unwanted write
|
2012-10-15 14:29:13 +00:00
|
|
|
|
}
|
|
|
|
|
else if (addr < 0x2000)
|
|
|
|
|
{
|
|
|
|
|
addr -= 0x1A00;
|
|
|
|
|
addr &= 0x5FF;
|
2012-10-19 00:13:39 +00:00
|
|
|
|
int offset = core.rom.Length - 0x0600;
|
|
|
|
|
return core.rom[offset + addr]; //Fixed to last 1.5K
|
2012-10-15 14:29:13 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return base.ReadMemory(addr);
|
|
|
|
|
}
|
2012-04-02 16:30:55 +00:00
|
|
|
|
}
|
|
|
|
|
public override void WriteMemory(ushort addr, byte value)
|
|
|
|
|
{
|
|
|
|
|
Address(addr);
|
2012-10-15 14:29:13 +00:00
|
|
|
|
if (addr < 0x1000)
|
|
|
|
|
{
|
|
|
|
|
base.WriteMemory(addr, value);
|
|
|
|
|
}
|
|
|
|
|
else if (addr < 0x1400)
|
|
|
|
|
{
|
|
|
|
|
rambank0[addr & 0x3FF] = value;
|
|
|
|
|
}
|
|
|
|
|
else if (addr >= 0x1800 && addr < 0x2000)
|
|
|
|
|
{
|
2012-10-19 00:13:39 +00:00
|
|
|
|
rambank1[(addr & 0xFF) + (rambank1_toggle * 0x100)] = value;
|
2012-10-15 14:29:13 +00:00
|
|
|
|
}
|
2012-04-02 16:30:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public override void SyncState(Serializer ser)
|
|
|
|
|
{
|
|
|
|
|
base.SyncState(ser);
|
2012-10-19 00:13:39 +00:00
|
|
|
|
ser.Sync("toggle", ref rombank_1k);
|
2012-04-02 16:30:55 +00:00
|
|
|
|
ser.Sync("rambank0", ref rambank0);
|
|
|
|
|
ser.Sync("rambank1", ref rambank1);
|
|
|
|
|
ser.Sync("EnableRam0", ref EnableRam0);
|
|
|
|
|
ser.Sync("rambank1_toggle", ref rambank1_toggle);
|
|
|
|
|
}
|
2012-03-31 20:53:14 +00:00
|
|
|
|
|
2012-04-02 16:30:55 +00:00
|
|
|
|
void Address(ushort addr)
|
|
|
|
|
{
|
|
|
|
|
switch (addr)
|
|
|
|
|
{
|
|
|
|
|
case 0x1FE0:
|
2012-10-19 00:13:39 +00:00
|
|
|
|
rombank_1k = 0;
|
2012-04-02 16:30:55 +00:00
|
|
|
|
break;
|
|
|
|
|
case 0x1FE1:
|
2012-10-19 00:13:39 +00:00
|
|
|
|
rombank_1k = 1;
|
2012-04-02 16:30:55 +00:00
|
|
|
|
break;
|
|
|
|
|
case 0x1FE2:
|
2012-10-19 00:13:39 +00:00
|
|
|
|
rombank_1k = 2;
|
2012-04-02 16:30:55 +00:00
|
|
|
|
break;
|
|
|
|
|
case 0x1FE3:
|
2012-10-19 00:13:39 +00:00
|
|
|
|
rombank_1k = 3;
|
2012-04-02 16:30:55 +00:00
|
|
|
|
break;
|
|
|
|
|
case 0x1FE4:
|
2012-10-19 00:13:39 +00:00
|
|
|
|
rombank_1k = 4;
|
2012-04-02 16:30:55 +00:00
|
|
|
|
break;
|
|
|
|
|
case 0x1FE5:
|
2012-10-19 00:13:39 +00:00
|
|
|
|
rombank_1k = 5;
|
2012-04-02 16:30:55 +00:00
|
|
|
|
break;
|
|
|
|
|
case 0x1FE6:
|
2012-10-19 00:13:39 +00:00
|
|
|
|
rombank_1k = 6;
|
2012-04-02 16:30:55 +00:00
|
|
|
|
break;
|
|
|
|
|
case 0x1FE7:
|
|
|
|
|
EnableRam0 = true;
|
|
|
|
|
break;
|
|
|
|
|
case 0x1FE8:
|
|
|
|
|
rambank1_toggle = 0;
|
|
|
|
|
break;
|
|
|
|
|
case 0x1FE9:
|
|
|
|
|
rambank1_toggle = 1;
|
|
|
|
|
break;
|
|
|
|
|
case 0x1FEA:
|
|
|
|
|
rambank1_toggle = 2;
|
|
|
|
|
break;
|
|
|
|
|
case 0x1FEB:
|
|
|
|
|
rambank1_toggle = 3;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
2012-03-31 20:53:14 +00:00
|
|
|
|
}
|
|
|
|
|
}
|