nes-support tengen board (mapper64)
This commit is contained in:
parent
e5f1c142f8
commit
b7863fb7d1
|
@ -139,6 +139,7 @@
|
|||
</Compile>
|
||||
<Compile Include="Consoles\Nintendo\NES\Boards\TAITO_74_161_161_32.cs" />
|
||||
<Compile Include="Consoles\Nintendo\NES\Boards\Taito_TC0190FMC.cs" />
|
||||
<Compile Include="Consoles\Nintendo\NES\Boards\TENGEN-800032.cs" />
|
||||
<Compile Include="Consoles\Nintendo\NES\Boards\UxROM.cs">
|
||||
<SubType>Code</SubType>
|
||||
</Compile>
|
||||
|
|
|
@ -51,7 +51,7 @@ Open bus and bus conflict emulation is not considered complete or thorough in an
|
|||
060 Multicart Junk
|
||||
061 Multicart Junk
|
||||
062 Multicart Junk
|
||||
064 Tengen Needed
|
||||
064 Tengen Good
|
||||
065 Misc (J) Nothing
|
||||
066 GxROM Complete
|
||||
067 Misc (J) Nothing
|
||||
|
|
|
@ -0,0 +1,267 @@
|
|||
using System;
|
||||
using System.IO;
|
||||
using System.Diagnostics;
|
||||
|
||||
namespace BizHawk.Emulation.Consoles.Nintendo
|
||||
{
|
||||
//AKA mapper 64
|
||||
public class TENGEN_800032 : NES.NESBoardBase
|
||||
{
|
||||
//configuration
|
||||
int prg_bank_mask_8k;
|
||||
int chr_bank_mask_1k;
|
||||
|
||||
//regenerable state
|
||||
IntBuffer prg_banks_8k = new IntBuffer(4);
|
||||
IntBuffer chr_banks_1k = new IntBuffer(8);
|
||||
//state
|
||||
IntBuffer regs = new IntBuffer(16);
|
||||
int address;
|
||||
bool chr_1k, chr_mode, prg_mode;
|
||||
//irq
|
||||
int irq_countdown;
|
||||
int a12_old;
|
||||
int irq_reload, irq_counter;
|
||||
bool irq_pending, irq_enable;
|
||||
bool irq_mode;
|
||||
bool irq_reload_pending;
|
||||
|
||||
|
||||
public override void Dispose()
|
||||
{
|
||||
base.Dispose();
|
||||
prg_banks_8k.Dispose();
|
||||
chr_banks_1k.Dispose();
|
||||
|
||||
}
|
||||
|
||||
public override void SyncState(Serializer ser)
|
||||
{
|
||||
base.SyncState(ser);
|
||||
ser.Sync("regs", ref regs);
|
||||
ser.Sync("address", ref address);
|
||||
ser.Sync("chr_1k", ref chr_1k);
|
||||
ser.Sync("chr_mode", ref chr_mode);
|
||||
ser.Sync("prg_mode", ref prg_mode);
|
||||
ser.Sync("irq_countdown", ref irq_countdown);
|
||||
ser.Sync("a12_old", ref a12_old);
|
||||
ser.Sync("irq_reload", ref irq_reload);
|
||||
ser.Sync("irq_counter", ref irq_counter);
|
||||
ser.Sync("irq_pending", ref irq_pending);
|
||||
ser.Sync("irq_enable", ref irq_enable);
|
||||
ser.Sync("irq_mode", ref irq_mode);
|
||||
ser.Sync("irq_reload_pending", ref irq_reload_pending);
|
||||
|
||||
if (ser.IsReader)
|
||||
Sync();
|
||||
}
|
||||
|
||||
public override bool Configure(NES.EDetectionOrigin origin)
|
||||
{
|
||||
switch (Cart.board_type)
|
||||
{
|
||||
case "TENGEN-800032":
|
||||
AssertPrg(64,128); AssertChr(64,128); AssertVram(0); AssertWram(00);
|
||||
break;
|
||||
}
|
||||
|
||||
prg_bank_mask_8k = Cart.prg_size / 8 - 1;
|
||||
chr_bank_mask_1k = Cart.chr_size / 1 - 1;
|
||||
|
||||
SetMirrorType(EMirrorType.Vertical);
|
||||
|
||||
Sync();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
void Sync()
|
||||
{
|
||||
if (prg_mode)
|
||||
{
|
||||
prg_banks_8k[0] = regs[0xF] & prg_bank_mask_8k;
|
||||
prg_banks_8k[1] = regs[0x6] & prg_bank_mask_8k;
|
||||
prg_banks_8k[2] = regs[0x7] & prg_bank_mask_8k;
|
||||
prg_banks_8k[3] = 0xFF & prg_bank_mask_8k;
|
||||
}
|
||||
else
|
||||
{
|
||||
prg_banks_8k[0] = regs[0x6] & prg_bank_mask_8k;
|
||||
prg_banks_8k[1] = regs[0x7] & prg_bank_mask_8k;
|
||||
prg_banks_8k[2] = regs[0xF] & prg_bank_mask_8k;
|
||||
prg_banks_8k[3] = 0xFF & prg_bank_mask_8k;
|
||||
}
|
||||
|
||||
if (chr_mode)
|
||||
{
|
||||
chr_banks_1k[0] = regs[0x2] & chr_bank_mask_1k;
|
||||
chr_banks_1k[1] = regs[0x3] & chr_bank_mask_1k;
|
||||
chr_banks_1k[2] = regs[0x4] & chr_bank_mask_1k;
|
||||
chr_banks_1k[3] = regs[0x5] & chr_bank_mask_1k;
|
||||
if (chr_1k)
|
||||
{
|
||||
chr_banks_1k[4] = regs[0x0] & chr_bank_mask_1k;
|
||||
chr_banks_1k[5] = regs[0x8] & chr_bank_mask_1k;
|
||||
chr_banks_1k[6] = regs[0x1] & chr_bank_mask_1k;
|
||||
chr_banks_1k[7] = regs[0x9] & chr_bank_mask_1k;
|
||||
}
|
||||
else
|
||||
{
|
||||
chr_banks_1k[4] = ((regs[0x0] & 0xFE) + 0) & chr_bank_mask_1k;
|
||||
chr_banks_1k[5] = ((regs[0x0] & 0xFE) + 1) & chr_bank_mask_1k;
|
||||
chr_banks_1k[6] = ((regs[0x1] & 0xFE) + 0) & chr_bank_mask_1k;
|
||||
chr_banks_1k[7] = ((regs[0x1] & 0xFE) + 1) & chr_bank_mask_1k;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
chr_banks_1k[4] = regs[0x2] & chr_bank_mask_1k;
|
||||
chr_banks_1k[5] = regs[0x3] & chr_bank_mask_1k;
|
||||
chr_banks_1k[6] = regs[0x4] & chr_bank_mask_1k;
|
||||
chr_banks_1k[7] = regs[0x5] & chr_bank_mask_1k;
|
||||
if (chr_1k)
|
||||
{
|
||||
chr_banks_1k[0] = regs[0x0] & chr_bank_mask_1k;
|
||||
chr_banks_1k[1] = regs[0x8] & chr_bank_mask_1k;
|
||||
chr_banks_1k[2] = regs[0x1] & chr_bank_mask_1k;
|
||||
chr_banks_1k[3] = regs[0x9] & chr_bank_mask_1k;
|
||||
}
|
||||
else
|
||||
{
|
||||
chr_banks_1k[0] = ((regs[0x0] & 0xFE) + 0) & chr_bank_mask_1k;
|
||||
chr_banks_1k[1] = ((regs[0x0] & 0xFE) + 1) & chr_bank_mask_1k;
|
||||
chr_banks_1k[2] = ((regs[0x1] & 0xFE) + 0) & chr_bank_mask_1k;
|
||||
chr_banks_1k[3] = ((regs[0x1] & 0xFE) + 1) & chr_bank_mask_1k;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public override void WritePRG(int addr, byte value)
|
||||
{
|
||||
//Console.WriteLine("mapping {0:X4} = {1:X2}", addr, value);
|
||||
addr &= 0xE001;
|
||||
switch (addr)
|
||||
{
|
||||
case 0x0000:
|
||||
address = value & 0xF;
|
||||
chr_1k = value.Bit(5);
|
||||
prg_mode = value.Bit(6);
|
||||
chr_mode = value.Bit(7);
|
||||
Sync();
|
||||
break;
|
||||
|
||||
case 0x0001: //data port
|
||||
regs[address] = value;
|
||||
Sync();
|
||||
break;
|
||||
|
||||
case 0x2000:
|
||||
if (value.Bit(0)) SetMirrorType(EMirrorType.Horizontal);
|
||||
else SetMirrorType(EMirrorType.Vertical);
|
||||
break;
|
||||
|
||||
case 0x4000:
|
||||
irq_reload = value;
|
||||
break;
|
||||
|
||||
case 0x4001:
|
||||
irq_mode = value.Bit(0);
|
||||
if(irq_mode) irq_countdown = 12;
|
||||
irq_reload_pending = true;
|
||||
break;
|
||||
|
||||
case 0x6000:
|
||||
irq_enable = false;
|
||||
irq_pending = false;
|
||||
SyncIRQ();
|
||||
break;
|
||||
case 0x6001:
|
||||
irq_enable = true;
|
||||
SyncIRQ();
|
||||
break;
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public override byte ReadPRG(int addr)
|
||||
{
|
||||
int bank_8k = addr >> 13;
|
||||
int ofs = addr & ((1 << 13) - 1);
|
||||
bank_8k = prg_banks_8k[bank_8k];
|
||||
addr = (bank_8k << 13) | ofs;
|
||||
return ROM[addr];
|
||||
}
|
||||
|
||||
|
||||
public override byte ReadPPU(int addr)
|
||||
{
|
||||
if (addr < 0x2000)
|
||||
{
|
||||
int bank_1k = addr >> 10;
|
||||
int ofs = addr & ((1 << 10) - 1);
|
||||
bank_1k = chr_banks_1k[bank_1k];
|
||||
addr = (bank_1k << 10) | ofs;
|
||||
return VROM[addr];
|
||||
}
|
||||
else
|
||||
return base.ReadPPU(addr);
|
||||
}
|
||||
|
||||
void SyncIRQ()
|
||||
{
|
||||
NES.irq_cart = irq_pending;
|
||||
}
|
||||
|
||||
void ClockIRQ()
|
||||
{
|
||||
if (irq_reload_pending)
|
||||
{
|
||||
irq_counter = irq_reload + 1;
|
||||
irq_reload_pending = false;
|
||||
}
|
||||
else if (irq_counter == 0)
|
||||
{
|
||||
irq_counter = irq_reload;
|
||||
}
|
||||
else
|
||||
{
|
||||
irq_counter--;
|
||||
if (irq_counter == 0 && irq_enable)
|
||||
{
|
||||
irq_pending = true;
|
||||
SyncIRQ();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public override void ClockPPU()
|
||||
{
|
||||
if (irq_countdown > 0)
|
||||
{
|
||||
irq_countdown--;
|
||||
if (irq_countdown == 0)
|
||||
{
|
||||
ClockIRQ();
|
||||
if (irq_mode)
|
||||
irq_countdown = 12;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public override void AddressPPU(int addr)
|
||||
{
|
||||
int a12 = (addr >> 12) & 1;
|
||||
bool rising_edge = (a12 == 1 && a12_old == 0);
|
||||
if (rising_edge)
|
||||
{
|
||||
//this number is totally guessed to make klax work
|
||||
irq_countdown = 25;
|
||||
}
|
||||
|
||||
a12_old = a12;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue