commit
6a0fe162a1
Binary file not shown.
|
@ -231,7 +231,7 @@ B19256C6716147A9744F5BD528F14450 Magic Knight Rayearth 2 - Making of Magic Knig
|
|||
846D48D0F4024C8094117599D0E1EEF1 Magic Knight Rayearth (J) GG SRAM=8192 Japan
|
||||
E496FF2196C372F4D6111538950D25CA Magical Puzzle Popils (W) (En,Ja) GG Puzzle SRAM=8192 World
|
||||
3AF0C6DDF5F00A493E1F159FCEDC0933 Magical Taruruuto-kun (J) GG Japan
|
||||
B0C35BC53AB7C184D34E5624F69AAD24 The Majors Pro Baseball (U) GG Sports;Baseball SRAM=128;GGLink USA
|
||||
B0C35BC53AB7C184D34E5624F69AAD24 The Majors Pro Baseball (U) GG Sports;Baseball SRAM=128;GGLink;EEPROM USA
|
||||
A15C5219F766D516D1B8D9A09B9A2BB4 Mappy (J) GG Japan
|
||||
B83F36FD113A8F75F1A29652ACB641FC Marble Madness (UE) GG Arcade USA;Europe
|
||||
BA846684A66E90372C3C234955EE28BC Marko's Magic Football (E) (En,Fr,De,Es) GG Europe
|
||||
|
@ -461,9 +461,9 @@ A23E89266DDAD3C856E7401D04A49C6C Woody Pop (W) (Rev 1) GG World
|
|||
13F72ACFEA47587F9AA9F655BF98653C World Class Leader Board Golf (UE) GG Sports;Golf USA;Europe
|
||||
D95D381C6AFFB8345EE5457655E393D1 World Cup USA 94 (UE) (En,Fr,De,Es,It,Nl,Pt,Sv) GG Sports;Soccer USA;Europe
|
||||
D8939B64458FAF174CDC1241F777CB59 World Derby (J) GG GGLink Japan
|
||||
E7EABBFC7A1F1339C4720249AEA92A32 World Series Baseball '95 (U) GG Sports;Baseball SRAM=128;GGLink USA
|
||||
59359FC38865CFF00C90D6EB148DDC2F World Series Baseball (U) GG Sports;Baseball SRAM=128;GGLink USA
|
||||
05CAC33029F0CAAC27774504C1AA8597 World Series Baseball (U) (Rev 1) GG Sports;Baseball SRAM=128;GGLink USA
|
||||
E7EABBFC7A1F1339C4720249AEA92A32 World Series Baseball '95 (U) GG Sports;Baseball SRAM=128;GGLink;EEPROM USA
|
||||
59359FC38865CFF00C90D6EB148DDC2F World Series Baseball (U) GG Sports;Baseball SRAM=128;GGLink;EEPROM USA
|
||||
05CAC33029F0CAAC27774504C1AA8597 World Series Baseball (U) (Rev 1) GG Sports;Baseball SRAM=128;GGLink;EEPROM USA
|
||||
D810E851AD60ED5BA50B6246C2CE12F2 WWF Raw (UE) GG Sports;Wrestling USA;Europe
|
||||
571AC03B80E3075C699CD583BF8651FD X-Men - Gamemaster's Legacy (UE) GG Marvel USA;Europe
|
||||
CA15F2BA2507EBD836C42D9D10231EB1 X-Men - Mojo World (UE) GG Marvel USA;Europe
|
||||
|
|
|
@ -106,7 +106,7 @@ F2535DF9BDC3A84221303FA62D61AD6E Back to the Future Part II (E) SMS Europe
|
|||
3D24A52E98E6C85D7C059386451CE749 Back to the Future Part III (E) SMS PAL Europe
|
||||
8A94DED3D95AA46DAE8800B92E66D3EE Baku Baku (B) SMS Brazil
|
||||
7A5D3B9963E316CB7F73BBDC2A7311C6 Bank Panic (E) SMS Europe
|
||||
26DF4404950CB8DA47235833C0C101C6 Bart vs. the Space Mutants (E) SMS Europe
|
||||
26DF4404950CB8DA47235833C0C101C6 Bart vs. the Space Mutants (E) SMS PAL Europe
|
||||
CBA2EC2940619956359801618E7AAB17 Bart vs. the World (E) SMS Europe
|
||||
0069B1BD9C5B6B88ACE6324D7E61539F Basketball Nightmare (E) SMS Sports;Basketball Europe
|
||||
215876A62D3CA48D28E98CD8A2C70A15 Batman Returns (E) SMS Europe
|
||||
|
|
|
@ -470,7 +470,7 @@ namespace BizHawk.Client.Common
|
|||
System = "PSX"
|
||||
};
|
||||
}
|
||||
else if (ext == ".iso" || ext == ".cue" || ext == ".ccd")
|
||||
else if (ext == ".iso" || ext == ".cue" || ext == ".ccd" || ext == ".mds")
|
||||
{
|
||||
if (file.IsArchive)
|
||||
{
|
||||
|
@ -494,7 +494,7 @@ namespace BizHawk.Client.Common
|
|||
throw new InvalidOperationException("\r\n" + discMountJob.OUT_Log);
|
||||
}
|
||||
|
||||
var disc = discMountJob.OUT_Disc;
|
||||
var disc = discMountJob.OUT_Disc;
|
||||
|
||||
// -----------
|
||||
// TODO - use more sophisticated IDer
|
||||
|
|
|
@ -284,29 +284,61 @@ namespace BizHawk.Client.Common
|
|||
case WatchSize.Byte:
|
||||
return (byte?)_val;
|
||||
case WatchSize.Word:
|
||||
if (addr == _watch.Address)
|
||||
if (_watch.BigEndian)
|
||||
{
|
||||
if (addr == _watch.Address)
|
||||
{
|
||||
return (byte)(_val & 0xFF);
|
||||
}
|
||||
return (byte)(_val >> 8);
|
||||
}
|
||||
|
||||
return (byte)(_val & 0xFF);
|
||||
case WatchSize.DWord:
|
||||
if (addr == _watch.Address)
|
||||
else
|
||||
{
|
||||
if (addr == _watch.Address)
|
||||
{
|
||||
return (byte)(_val >> 8);
|
||||
}
|
||||
return (byte)(_val & 0xFF);
|
||||
}
|
||||
case WatchSize.DWord:
|
||||
if (_watch.BigEndian)
|
||||
{
|
||||
if (addr == _watch.Address)
|
||||
{
|
||||
return (byte)((_val >> 24) & 0xFF);
|
||||
}
|
||||
|
||||
if (addr == _watch.Address + 1)
|
||||
{
|
||||
return (byte)((_val >> 16) & 0xFF);
|
||||
}
|
||||
|
||||
if (addr == _watch.Address + 2)
|
||||
{
|
||||
return (byte)((_val >> 8) & 0xFF);
|
||||
}
|
||||
|
||||
return (byte)(_val & 0xFF);
|
||||
}
|
||||
else
|
||||
{
|
||||
if (addr == _watch.Address)
|
||||
{
|
||||
return (byte)(_val & 0xFF);
|
||||
}
|
||||
|
||||
if (addr == _watch.Address + 1)
|
||||
{
|
||||
return (byte)((_val >> 8) & 0xFF);
|
||||
}
|
||||
|
||||
if (addr == _watch.Address + 2)
|
||||
{
|
||||
return (byte)((_val >> 16) & 0xFF);
|
||||
}
|
||||
|
||||
return (byte)((_val >> 24) & 0xFF);
|
||||
}
|
||||
|
||||
if (addr == _watch.Address + 1)
|
||||
{
|
||||
return (byte)((_val >> 16) & 0xFF);
|
||||
}
|
||||
|
||||
if (addr == _watch.Address + 2)
|
||||
{
|
||||
return (byte)((_val >> 8) & 0xFF);
|
||||
}
|
||||
|
||||
return (byte)(_val & 0xFF);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -111,7 +111,7 @@ namespace BizHawk.Client.DiscoHawk
|
|||
foreach (string str in files)
|
||||
{
|
||||
string ext = Path.GetExtension(str).ToUpper();
|
||||
if(!ext.In(new string[]{".CUE",".ISO",".CCD"}))
|
||||
if(!ext.In(new string[]{".CUE",".ISO",".CCD", ".MDS"}))
|
||||
{
|
||||
return new List<string>();
|
||||
}
|
||||
|
|
|
@ -51,11 +51,11 @@ namespace BizHawk.Client.EmuHawk
|
|||
return new[]
|
||||
{
|
||||
".NES", ".FDS", ".UNF", ".SMS", ".GG", ".SG", ".GB", ".GBC", ".GBA", ".PCE", ".SGX", ".BIN", ".SMD", ".GEN", ".MD", ".SMC", ".SFC", ".A26", ".A78", ".LNX", ".COL", ".ROM", ".M3U", ".CUE", ".CCD", ".SGB", ".Z64", ".V64", ".N64", ".WS", ".WSC", ".XML", ".DSK", ".DO", ".PO", ".PSF", ".MINIPSF", ".NSF",
|
||||
".EXE", ".PRG", ".D64", "*G64", ".CRT", ".TAP", ".32X"
|
||||
".EXE", ".PRG", ".D64", "*G64", ".CRT", ".TAP", ".32X", ".MDS"
|
||||
};
|
||||
}
|
||||
|
||||
return new[] { ".NES", ".FDS", ".UNF", ".SMS", ".GG", ".SG", ".GB", ".GBC", ".GBA", ".PCE", ".SGX", ".BIN", ".SMD", ".GEN", ".MD", ".SMC", ".SFC", ".A26", ".A78", ".LNX", ".COL", ".ROM", ".M3U", ".CUE", ".CCD", ".SGB", ".Z64", ".V64", ".N64", ".WS", ".WSC", ".XML", ".DSK", ".DO", ".PO", ".PSF", ".MINIPSF", ".NSF", ".32X" };
|
||||
return new[] { ".NES", ".FDS", ".UNF", ".SMS", ".GG", ".SG", ".GB", ".GBC", ".GBA", ".PCE", ".SGX", ".BIN", ".SMD", ".GEN", ".MD", ".SMC", ".SFC", ".A26", ".A78", ".LNX", ".COL", ".ROM", ".M3U", ".CUE", ".CCD", ".SGB", ".Z64", ".V64", ".N64", ".WS", ".WSC", ".XML", ".DSK", ".DO", ".PO", ".PSF", ".MINIPSF", ".NSF", ".32X", ".MDS" };
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -2077,13 +2077,13 @@ namespace BizHawk.Client.EmuHawk
|
|||
if (VersionInfo.DeveloperBuild)
|
||||
{
|
||||
return FormatFilter(
|
||||
"Rom Files", "*.nes;*.fds;*.unf;*.sms;*.gg;*.sg;*.pce;*.sgx;*.bin;*.smd;*.rom;*.a26;*.a78;*.lnx;*.m3u;*.cue;*.ccd;*.exe;*.gb;*.gbc;*.gba;*.gen;*.md;*.32x;*.col;*.int;*.smc;*.sfc;*.prg;*.d64;*.g64;*.crt;*.tap;*.sgb;*.xml;*.z64;*.v64;*.n64;*.ws;*.wsc;*.dsk;*.do;*.po;*.vb;*.ngp;*.ngc;*.psf;*.minipsf;*.nsf;%ARCH%",
|
||||
"Rom Files", "*.nes;*.fds;*.unf;*.sms;*.gg;*.sg;*.pce;*.sgx;*.bin;*.smd;*.rom;*.a26;*.a78;*.lnx;*.m3u;*.cue;*.ccd;*.mds;*.exe;*.gb;*.gbc;*.gba;*.gen;*.md;*.32x;*.col;*.int;*.smc;*.sfc;*.prg;*.d64;*.g64;*.crt;*.tap;*.sgb;*.xml;*.z64;*.v64;*.n64;*.ws;*.wsc;*.dsk;*.do;*.po;*.vb;*.ngp;*.ngc;*.psf;*.minipsf;*.nsf;%ARCH%",
|
||||
"Music Files", "*.psf;*.minipsf;*.sid;*.nsf",
|
||||
"Disc Images", "*.cue;*.ccd;*.m3u",
|
||||
"Disc Images", "*.cue;*.ccd;*.mds;*.m3u",
|
||||
"NES", "*.nes;*.fds;*.unf;*.nsf;%ARCH%",
|
||||
"Super NES", "*.smc;*.sfc;*.xml;%ARCH%",
|
||||
"Master System", "*.sms;*.gg;*.sg;%ARCH%",
|
||||
"PC Engine", "*.pce;*.sgx;*.cue;*.ccd;%ARCH%",
|
||||
"PC Engine", "*.pce;*.sgx;*.cue;*.ccd;*.mds;%ARCH%",
|
||||
"TI-83", "*.rom;%ARCH%",
|
||||
"Archive Files", "%ARCH%",
|
||||
"Savestate", "*.state",
|
||||
|
@ -2095,7 +2095,7 @@ namespace BizHawk.Client.EmuHawk
|
|||
"Gameboy Advance", "*.gba;%ARCH%",
|
||||
"Colecovision", "*.col;%ARCH%",
|
||||
"Intellivision", "*.int;*.bin;*.rom;%ARCH%",
|
||||
"PlayStation", "*.cue;*.ccd;*.m3u",
|
||||
"PlayStation", "*.cue;*.ccd;*.mds;*.m3u",
|
||||
"PSX Executables (experimental)", "*.exe",
|
||||
"PSF Playstation Sound File", "*.psf;*.minipsf",
|
||||
"Commodore 64", "*.prg; *.d64, *.g64; *.crt; *.tap;%ARCH%",
|
||||
|
@ -2109,17 +2109,17 @@ namespace BizHawk.Client.EmuHawk
|
|||
}
|
||||
|
||||
return FormatFilter(
|
||||
"Rom Files", "*.nes;*.fds;*.unf;*.sms;*.gg;*.sg;*.gb;*.gbc;*.gba;*.pce;*.sgx;*.bin;*.smd;*.gen;*.md;*.32x;*.smc;*.sfc;*.a26;*.a78;*.lnx;*.col;*.int;*.rom;*.m3u;*.cue;*.ccd;*.sgb;*.z64;*.v64;*.n64;*.ws;*.wsc;*.xml;*.dsk;*.do;*.po;*.psf;*.ngp;*.ngc;*.prg;*.d64;*.g64;*.minipsf;*.nsf;%ARCH%",
|
||||
"Disc Images", "*.cue;*.ccd;*.m3u",
|
||||
"Rom Files", "*.nes;*.fds;*.unf;*.sms;*.gg;*.sg;*.gb;*.gbc;*.gba;*.pce;*.sgx;*.bin;*.smd;*.gen;*.md;*.32x;*.smc;*.sfc;*.a26;*.a78;*.lnx;*.col;*.int;*.rom;*.m3u;*.cue;*.ccd;*.mds;*.sgb;*.z64;*.v64;*.n64;*.ws;*.wsc;*.xml;*.dsk;*.do;*.po;*.psf;*.ngp;*.ngc;*.prg;*.d64;*.g64;*.minipsf;*.nsf;%ARCH%",
|
||||
"Disc Images", "*.cue;*.ccd;*.mds;*.m3u",
|
||||
"NES", "*.nes;*.fds;*.unf;*.nsf;%ARCH%",
|
||||
"Super NES", "*.smc;*.sfc;*.xml;%ARCH%",
|
||||
"PlayStation", "*.cue;*.ccd;*.m3u",
|
||||
"PlayStation", "*.cue;*.ccd;*.mds;*.m3u",
|
||||
"PSF Playstation Sound File", "*.psf;*.minipsf",
|
||||
"Nintendo 64", "*.z64;*.v64;*.n64",
|
||||
"Gameboy", "*.gb;*.gbc;*.sgb;%ARCH%",
|
||||
"Gameboy Advance", "*.gba;%ARCH%",
|
||||
"Master System", "*.sms;*.gg;*.sg;%ARCH%",
|
||||
"PC Engine", "*.pce;*.sgx;*.cue;*.ccd;%ARCH%",
|
||||
"PC Engine", "*.pce;*.sgx;*.cue;*.ccd;*.mds;%ARCH%",
|
||||
"Atari 2600", "*.a26;%ARCH%",
|
||||
"Atari 7800", "*.a78;%ARCH%",
|
||||
"Atari Lynx", "*.lnx;%ARCH%",
|
||||
|
|
|
@ -526,7 +526,16 @@ namespace BizHawk.Client.EmuHawk
|
|||
{
|
||||
if (_addr + j + DataSize <= _domain.Size)
|
||||
{
|
||||
rowStr.AppendFormat(_digitFormatString, MakeValue(_addr + j));
|
||||
int t_val = 0;
|
||||
int t_next = 0;
|
||||
|
||||
for (int k = 0; k < DataSize; k++)
|
||||
{
|
||||
t_next = MakeValue(1, _addr + j + k);
|
||||
t_val += (t_next << ((DataSize - k - 1) * 8));
|
||||
}
|
||||
|
||||
rowStr.AppendFormat(_digitFormatString, t_val);
|
||||
}
|
||||
else
|
||||
{
|
||||
|
@ -569,7 +578,7 @@ namespace BizHawk.Client.EmuHawk
|
|||
{
|
||||
if (Global.CheatList.IsActive(_domain, address))
|
||||
{
|
||||
return Global.CheatList.GetCheatValue(_domain, address, (WatchSize)DataSize ).Value;
|
||||
return Global.CheatList.GetCheatValue(_domain, address, (WatchSize)dataSize ).Value;
|
||||
}
|
||||
|
||||
switch (dataSize)
|
||||
|
|
|
@ -1071,6 +1071,8 @@
|
|||
<Compile Include="Consoles\Sega\Saturn\LibSaturnus.cs" />
|
||||
<Compile Include="Consoles\Sega\Saturn\Saturnus.cs" />
|
||||
<Compile Include="Consoles\Sega\Saturn\SaturnusControllerDeck.cs" />
|
||||
<Compile Include="Consoles\Sega\SMS\EEPROM.93c46.cs" />
|
||||
<Compile Include="Consoles\Sega\SMS\MemoryMap.EEPROM.cs" />
|
||||
<Compile Include="Consoles\Sega\SMS\SMS.cs" />
|
||||
<Compile Include="Consoles\Sega\SMS\SMS.ICodeDataLogger.cs">
|
||||
<DependentUpon>SMS.cs</DependentUpon>
|
||||
|
@ -1186,12 +1188,14 @@
|
|||
<Compile Include="CPUs\Z80-GB\Registers.cs" />
|
||||
<Compile Include="CPUs\Z80-GB\Tables.cs" />
|
||||
<Compile Include="CPUs\Z80-GB\Z80.cs" />
|
||||
<Compile Include="CPUs\Z80\Disassembler.cs" />
|
||||
<Compile Include="CPUs\Z80\Execute.cs" />
|
||||
<Compile Include="CPUs\Z80\Interrupts.cs" />
|
||||
<Compile Include="CPUs\Z80\Registers.cs" />
|
||||
<Compile Include="CPUs\Z80\Tables.cs" />
|
||||
<Compile Include="CPUs\Z80\Z80A.cs" />
|
||||
<Compile Include="CPUs\Z80A\NewDisassembler.cs" />
|
||||
<Compile Include="CPUs\Z80A\Execute.cs" />
|
||||
<Compile Include="CPUs\Z80A\Interrupts.cs" />
|
||||
<Compile Include="CPUs\Z80A\Registers.cs" />
|
||||
<Compile Include="CPUs\Z80A\Operations.cs" />
|
||||
<Compile Include="CPUs\Z80A\Tables_Direct.cs" />
|
||||
<Compile Include="CPUs\Z80A\Tables_Indirect.cs" />
|
||||
<Compile Include="CPUs\Z80A\Z80A.cs" />
|
||||
<Compile Include="Properties\Resources.Designer.cs">
|
||||
<AutoGen>True</AutoGen>
|
||||
<DesignTime>True</DesignTime>
|
||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -1,55 +0,0 @@
|
|||
using System;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80
|
||||
{
|
||||
public partial class Z80A
|
||||
{
|
||||
private bool iff1;
|
||||
public bool IFF1 { get { return iff1; } set { iff1 = value; } }
|
||||
|
||||
private bool iff2;
|
||||
public bool IFF2 { get { return iff2; } set { iff2 = value; } }
|
||||
|
||||
private bool interrupt;
|
||||
public bool Interrupt { get { return interrupt; } set { interrupt = value; } }
|
||||
|
||||
private bool nonMaskableInterrupt;
|
||||
public bool NonMaskableInterrupt
|
||||
{
|
||||
get { return nonMaskableInterrupt; }
|
||||
set { if (value && !nonMaskableInterrupt) NonMaskableInterruptPending = true; nonMaskableInterrupt = value; }
|
||||
}
|
||||
|
||||
private bool nonMaskableInterruptPending;
|
||||
public bool NonMaskableInterruptPending { get { return nonMaskableInterruptPending; } set { nonMaskableInterruptPending = value; } }
|
||||
|
||||
private int interruptMode;
|
||||
public int InterruptMode
|
||||
{
|
||||
get { return interruptMode; }
|
||||
set { if (value < 0 || value > 2) throw new ArgumentOutOfRangeException(); interruptMode = value; }
|
||||
}
|
||||
|
||||
private bool halted;
|
||||
public bool Halted { get { return halted; } set { halted = value; } }
|
||||
|
||||
public Action IRQCallback = delegate() { };
|
||||
public Action NMICallback = delegate() { };
|
||||
|
||||
private void ResetInterrupts()
|
||||
{
|
||||
IFF1 = false;
|
||||
IFF2 = false;
|
||||
Interrupt = false;
|
||||
NonMaskableInterrupt = false;
|
||||
NonMaskableInterruptPending = false;
|
||||
InterruptMode = 1;
|
||||
Halted = false;
|
||||
}
|
||||
|
||||
private void Halt()
|
||||
{
|
||||
Halted = true;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,270 +0,0 @@
|
|||
using System.Runtime.InteropServices;
|
||||
using System;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80
|
||||
{
|
||||
public partial class Z80A
|
||||
{
|
||||
[StructLayout(LayoutKind.Explicit)]
|
||||
[Serializable()]
|
||||
public struct RegisterPair
|
||||
{
|
||||
[FieldOffset(0)]
|
||||
public ushort Word;
|
||||
|
||||
[FieldOffset(0)]
|
||||
public byte Low;
|
||||
|
||||
[FieldOffset(1)]
|
||||
public byte High;
|
||||
|
||||
public RegisterPair(ushort value)
|
||||
{
|
||||
Word = value;
|
||||
Low = (byte)(Word);
|
||||
High = (byte)(Word >> 8);
|
||||
}
|
||||
|
||||
public static implicit operator ushort(RegisterPair rp)
|
||||
{
|
||||
return rp.Word;
|
||||
}
|
||||
|
||||
public static implicit operator RegisterPair(ushort value)
|
||||
{
|
||||
return new RegisterPair(value);
|
||||
}
|
||||
}
|
||||
|
||||
private bool RegFlagC
|
||||
{
|
||||
get { return (RegAF.Low & 0x01) != 0; }
|
||||
set { RegAF.Low = (byte)((RegAF.Low & ~0x01) | (value ? 0x01 : 0x00)); }
|
||||
}
|
||||
|
||||
private bool RegFlagN
|
||||
{
|
||||
get { return (RegAF.Low & 0x02) != 0; }
|
||||
set { RegAF.Low = (byte)((RegAF.Low & ~0x02) | (value ? 0x02 : 0x00)); }
|
||||
}
|
||||
|
||||
private bool RegFlagP
|
||||
{
|
||||
get { return (RegAF.Low & 0x04) != 0; }
|
||||
set { RegAF.Low = (byte)((RegAF.Low & ~0x04) | (value ? 0x04 : 0x00)); }
|
||||
}
|
||||
|
||||
private bool RegFlag3
|
||||
{
|
||||
get { return (RegAF.Low & 0x08) != 0; }
|
||||
set { RegAF.Low = (byte)((RegAF.Low & ~0x08) | (value ? 0x08 : 0x00)); }
|
||||
}
|
||||
|
||||
private bool RegFlagH
|
||||
{
|
||||
get { return (RegAF.Low & 0x10) != 0; }
|
||||
set { RegAF.Low = (byte)((RegAF.Low & ~0x10) | (value ? 0x10 : 0x00)); }
|
||||
}
|
||||
|
||||
private bool RegFlag5
|
||||
{
|
||||
get { return (RegAF.Low & 0x20) != 0; }
|
||||
set { RegAF.Low = (byte)((RegAF.Low & ~0x20) | (value ? 0x20 : 0x00)); }
|
||||
}
|
||||
|
||||
private bool RegFlagZ
|
||||
{
|
||||
get { return (RegAF.Low & 0x40) != 0; }
|
||||
set { RegAF.Low = (byte)((RegAF.Low & ~0x40) | (value ? 0x40 : 0x00)); }
|
||||
}
|
||||
|
||||
private bool RegFlagS
|
||||
{
|
||||
get { return (RegAF.Low & 0x80) != 0; }
|
||||
set { RegAF.Low = (byte)((RegAF.Low & ~0x80) | (value ? 0x80 : 0x00)); }
|
||||
}
|
||||
|
||||
private RegisterPair RegAF;
|
||||
private RegisterPair RegBC;
|
||||
private RegisterPair RegDE;
|
||||
private RegisterPair RegHL;
|
||||
private RegisterPair RegWZ;
|
||||
|
||||
private RegisterPair RegAltAF; // Shadow for A and F
|
||||
private RegisterPair RegAltBC; // Shadow for B and C
|
||||
private RegisterPair RegAltDE; // Shadow for D and E
|
||||
private RegisterPair RegAltHL; // Shadow for H and L
|
||||
// NOTE: There is no AltWZ register (despite it being shown on various block diagrams)
|
||||
|
||||
private byte RegI; // I (interrupt vector)
|
||||
private byte RegR; // R (memory refresh)
|
||||
|
||||
private RegisterPair RegIX; // IX (index register x)
|
||||
private RegisterPair RegIY; // IY (index register y)
|
||||
|
||||
private RegisterPair RegSP; // SP (stack pointer)
|
||||
private RegisterPair RegPC; // PC (program counter)
|
||||
|
||||
private void ResetRegisters()
|
||||
{
|
||||
// Clear main registers
|
||||
RegAF = 0; RegBC = 0; RegDE = 0; RegHL = 0; RegWZ = 0;
|
||||
// Clear alternate registers
|
||||
RegAltAF = 0; RegAltBC = 0; RegAltDE = 0; RegAltHL = 0;
|
||||
// Clear special purpose registers
|
||||
RegI = 0; RegR = 0;
|
||||
RegIX.Word = 0; RegIY.Word = 0;
|
||||
RegSP.Word = 0; RegPC.Word = 0;
|
||||
}
|
||||
|
||||
public byte RegisterA
|
||||
{
|
||||
get { return RegAF.High; }
|
||||
set { RegAF.High = value; }
|
||||
}
|
||||
|
||||
public byte RegisterF
|
||||
{
|
||||
get { return RegAF.Low; }
|
||||
set { RegAF.Low = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterAF
|
||||
{
|
||||
get { return RegAF.Word; }
|
||||
set { RegAF.Word = value; }
|
||||
}
|
||||
|
||||
public byte RegisterB
|
||||
{
|
||||
get { return RegBC.High; }
|
||||
set { RegBC.High = value; }
|
||||
}
|
||||
|
||||
public byte RegisterC
|
||||
{
|
||||
get { return RegBC.Low; }
|
||||
set { RegBC.Low = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterBC
|
||||
{
|
||||
get { return RegBC.Word; }
|
||||
set { RegBC.Word = value; }
|
||||
}
|
||||
|
||||
public byte RegisterD
|
||||
{
|
||||
get { return RegDE.High; }
|
||||
set { RegDE.High = value; }
|
||||
}
|
||||
|
||||
public byte RegisterE
|
||||
{
|
||||
get { return RegDE.Low; }
|
||||
set { RegDE.Low = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterDE
|
||||
{
|
||||
get { return RegDE.Word; }
|
||||
set { RegDE.Word = value; }
|
||||
}
|
||||
|
||||
public byte RegisterH
|
||||
{
|
||||
get { return RegHL.High; }
|
||||
set { RegHL.High = value; }
|
||||
}
|
||||
|
||||
public byte RegisterL
|
||||
{
|
||||
get { return RegHL.Low; }
|
||||
set { RegHL.Low = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterHL
|
||||
{
|
||||
get { return RegHL.Word; }
|
||||
set { RegHL.Word = value; }
|
||||
}
|
||||
|
||||
public byte RegisterW
|
||||
{
|
||||
get { return RegWZ.High; }
|
||||
set { RegWZ.High = value; }
|
||||
}
|
||||
|
||||
public byte RegisterZ
|
||||
{
|
||||
get { return RegWZ.Low; }
|
||||
set { RegWZ.Low = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterWZ
|
||||
{
|
||||
get { return RegWZ.Word; }
|
||||
set { RegWZ.Word = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterPC
|
||||
{
|
||||
get { return RegPC.Word; }
|
||||
set { RegPC.Word = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterSP
|
||||
{
|
||||
get { return RegSP.Word; }
|
||||
set { RegSP.Word = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterIX
|
||||
{
|
||||
get { return RegIX.Word; }
|
||||
set { RegIX.Word = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterIY
|
||||
{
|
||||
get { return RegIY.Word; }
|
||||
set { RegIY.Word = value; }
|
||||
}
|
||||
|
||||
public byte RegisterI
|
||||
{
|
||||
get { return RegI; }
|
||||
set { RegI = value; }
|
||||
}
|
||||
|
||||
public byte RegisterR
|
||||
{
|
||||
get { return RegR; }
|
||||
set { RegR = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterShadowAF
|
||||
{
|
||||
get { return RegAltAF.Word; }
|
||||
set { RegAltAF.Word = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterShadowBC
|
||||
{
|
||||
get { return RegAltBC.Word; }
|
||||
set { RegAltBC.Word = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterShadowDE
|
||||
{
|
||||
get { return RegAltDE.Word; }
|
||||
set { RegAltDE.Word = value; }
|
||||
}
|
||||
|
||||
public ushort RegisterShadowHL
|
||||
{
|
||||
get { return RegAltHL.Word; }
|
||||
set { RegAltHL.Word = value; }
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,331 +0,0 @@
|
|||
namespace BizHawk.Emulation.Cores.Components.Z80
|
||||
{
|
||||
public partial class Z80A
|
||||
{
|
||||
private void InitialiseTables()
|
||||
{
|
||||
InitTableInc();
|
||||
InitTableDec();
|
||||
InitTableParity();
|
||||
InitTableALU();
|
||||
InitTableRotShift();
|
||||
InitTableHalfBorrow();
|
||||
InitTableHalfCarry();
|
||||
InitTableNeg();
|
||||
InitTableDaa();
|
||||
}
|
||||
|
||||
private byte[] TableInc;
|
||||
private void InitTableInc()
|
||||
{
|
||||
TableInc = new byte[256];
|
||||
for (int i = 0; i < 256; ++i)
|
||||
TableInc[i] = FlagByte(false, false, i == 0x80, UndocumentedX(i), (i & 0xF) == 0x0, UndocumentedY(i), i == 0, i > 127);
|
||||
}
|
||||
|
||||
private byte[] TableDec;
|
||||
private void InitTableDec()
|
||||
{
|
||||
TableDec = new byte[256];
|
||||
for (int i = 0; i < 256; ++i)
|
||||
TableDec[i] = FlagByte(false, true, i == 0x7F, UndocumentedX(i), (i & 0xF) == 0xF, UndocumentedY(i), i == 0, i > 127);
|
||||
}
|
||||
|
||||
private bool[] TableParity;
|
||||
private void InitTableParity()
|
||||
{
|
||||
TableParity = new bool[256];
|
||||
for (int i = 0; i < 256; ++i)
|
||||
{
|
||||
int Bits = 0;
|
||||
for (int j = 0; j < 8; ++j)
|
||||
{
|
||||
Bits += (i >> j) & 1;
|
||||
}
|
||||
TableParity[i] = (Bits & 1) == 0;
|
||||
}
|
||||
}
|
||||
|
||||
private ushort[, , ,] TableALU;
|
||||
private void InitTableALU()
|
||||
{
|
||||
TableALU = new ushort[8, 256, 256, 2]; // Class, OP1, OP2, Carry
|
||||
|
||||
for (int i = 0; i < 8; ++i)
|
||||
{
|
||||
for (int op1 = 0; op1 < 256; ++op1)
|
||||
{
|
||||
for (int op2 = 0; op2 < 256; ++op2)
|
||||
{
|
||||
for (int c = 0; c < 2; ++c)
|
||||
{
|
||||
|
||||
int ac = (i == 1 || i == 3) ? c : 0;
|
||||
|
||||
bool S = false;
|
||||
bool Z = false;
|
||||
bool C = false;
|
||||
bool H = false;
|
||||
bool N = false;
|
||||
bool P = false;
|
||||
|
||||
byte result_b = 0;
|
||||
int result_si = 0;
|
||||
int result_ui = 0;
|
||||
|
||||
// Fetch result
|
||||
switch (i)
|
||||
{
|
||||
case 0:
|
||||
case 1:
|
||||
result_si = (sbyte)op1 + (sbyte)op2 + ac;
|
||||
result_ui = op1 + op2 + ac;
|
||||
break;
|
||||
case 2:
|
||||
case 3:
|
||||
case 7:
|
||||
result_si = (sbyte)op1 - (sbyte)op2 - ac;
|
||||
result_ui = op1 - op2 - ac;
|
||||
break;
|
||||
case 4:
|
||||
result_si = op1 & op2;
|
||||
break;
|
||||
case 5:
|
||||
result_si = op1 ^ op2;
|
||||
break;
|
||||
case 6:
|
||||
result_si = op1 | op2;
|
||||
break;
|
||||
}
|
||||
|
||||
result_b = (byte)result_si;
|
||||
|
||||
// Parity/Carry
|
||||
|
||||
switch (i)
|
||||
{
|
||||
case 0:
|
||||
case 1:
|
||||
case 2:
|
||||
case 3:
|
||||
case 7:
|
||||
P = result_si < -128 || result_si > 127;
|
||||
C = result_ui < 0 || result_ui > 255;
|
||||
break;
|
||||
case 4:
|
||||
case 5:
|
||||
case 6:
|
||||
P = TableParity[result_b];
|
||||
C = false;
|
||||
break;
|
||||
}
|
||||
|
||||
// Subtraction
|
||||
N = i == 2 || i == 3 || i == 7;
|
||||
|
||||
// Half carry
|
||||
switch (i)
|
||||
{
|
||||
case 0:
|
||||
case 1:
|
||||
H = ((op1 & 0xF) + (op2 & 0xF) + (ac & 0xF)) > 0xF;
|
||||
break;
|
||||
case 2:
|
||||
case 3:
|
||||
case 7:
|
||||
H = ((op1 & 0xF) - (op2 & 0xF) - (ac & 0xF)) < 0x0;
|
||||
break;
|
||||
case 4:
|
||||
H = true;
|
||||
break;
|
||||
case 5:
|
||||
case 6:
|
||||
H = false;
|
||||
break;
|
||||
}
|
||||
|
||||
// Undocumented
|
||||
byte UndocumentedFlags = (byte)(result_b & 0x28);
|
||||
if (i == 7) UndocumentedFlags = (byte)(op2 & 0x28);
|
||||
|
||||
S = result_b > 127;
|
||||
Z = result_b == 0;
|
||||
|
||||
if (i == 7) result_b = (byte)op1;
|
||||
|
||||
TableALU[i, op1, op2, c] = (ushort)(
|
||||
result_b * 256 +
|
||||
((C ? 0x01 : 0) + (N ? 0x02 : 0) + (P ? 0x04 : 0) + (H ? 0x10 : 0) + (Z ? 0x40 : 0) + (S ? 0x80 : 0)) +
|
||||
(UndocumentedFlags));
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private bool[,] TableHalfBorrow;
|
||||
private void InitTableHalfBorrow()
|
||||
{
|
||||
TableHalfBorrow = new bool[256, 256];
|
||||
for (int i = 0; i < 256; i++)
|
||||
{
|
||||
for (int j = 0; j < 256; j++)
|
||||
{
|
||||
TableHalfBorrow[i, j] = ((i & 0xF) - (j & 0xF)) < 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private bool[,] TableHalfCarry;
|
||||
private void InitTableHalfCarry()
|
||||
{
|
||||
TableHalfCarry = new bool[256, 256];
|
||||
for (int i = 0; i < 256; i++)
|
||||
{
|
||||
for (int j = 0; j < 256; j++)
|
||||
{
|
||||
TableHalfCarry[i, j] = ((i & 0xF) + (j & 0xF)) > 0xF;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private ushort[, ,] TableRotShift;
|
||||
private void InitTableRotShift()
|
||||
{
|
||||
TableRotShift = new ushort[2, 8, 65536]; // All, operation, AF
|
||||
for (int all = 0; all < 2; all++)
|
||||
{
|
||||
for (int y = 0; y < 8; ++y)
|
||||
{
|
||||
for (int af = 0; af < 65536; af++)
|
||||
{
|
||||
byte Old = (byte)(af >> 8);
|
||||
bool OldCarry = (af & 0x01) != 0;
|
||||
|
||||
ushort newAf = (ushort)(af & ~(0x13)); // Clear HALF-CARRY, SUBTRACT and CARRY flags
|
||||
|
||||
byte New = Old;
|
||||
if ((y & 1) == 0)
|
||||
{
|
||||
if ((Old & 0x80) != 0) ++newAf;
|
||||
|
||||
New <<= 1;
|
||||
|
||||
if ((y & 0x04) == 0)
|
||||
{
|
||||
if (((y & 0x02) == 0) ? ((newAf & 0x01) != 0) : OldCarry) New |= 0x01;
|
||||
}
|
||||
else
|
||||
{
|
||||
if ((y & 0x02) != 0) New |= 0x01;
|
||||
}
|
||||
|
||||
}
|
||||
else
|
||||
{
|
||||
|
||||
if ((Old & 0x01) != 0) ++newAf;
|
||||
|
||||
New >>= 1;
|
||||
|
||||
if ((y & 0x04) == 0)
|
||||
{
|
||||
if (((y & 0x02) == 0) ? ((newAf & 0x01) != 0) : OldCarry) New |= 0x80;
|
||||
}
|
||||
else
|
||||
{
|
||||
if ((y & 0x02) == 0) New |= (byte)(Old & 0x80);
|
||||
}
|
||||
}
|
||||
|
||||
newAf &= 0xFF;
|
||||
newAf |= (ushort)(New * 256);
|
||||
|
||||
if (all == 1)
|
||||
{
|
||||
newAf &= unchecked((ushort)~0xC4); // Clear S, Z, P
|
||||
if (New > 127) newAf |= 0x80;
|
||||
if (New == 0) newAf |= 0x40;
|
||||
if (TableParity[New]) newAf |= 0x04;
|
||||
}
|
||||
|
||||
TableRotShift[all, y, af] = (ushort)((newAf & ~0x28) | ((newAf >> 8) & 0x28));
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private ushort[] TableNeg;
|
||||
private void InitTableNeg()
|
||||
{
|
||||
TableNeg = new ushort[65536];
|
||||
for (int af = 0; af < 65536; af++)
|
||||
{
|
||||
ushort raf = 0;
|
||||
byte b = (byte)(af >> 8);
|
||||
byte a = (byte)-b;
|
||||
raf |= (ushort)(a * 256);
|
||||
raf |= FlagByte(b != 0x00, true, b == 0x80, UndocumentedX(a), TableHalfCarry[a, b], UndocumentedY(a), a == 0, a > 127);
|
||||
TableNeg[af] = raf;
|
||||
}
|
||||
}
|
||||
|
||||
private ushort[] TableDaa;
|
||||
private void InitTableDaa()
|
||||
{
|
||||
TableDaa = new ushort[65536];
|
||||
for (int af = 0; af < 65536; ++af)
|
||||
{
|
||||
byte a = (byte)(af >> 8);
|
||||
byte tmp = a;
|
||||
|
||||
if (IsN(af))
|
||||
{
|
||||
if (IsH(af) || ((a & 0x0F) > 0x09)) tmp -= 0x06;
|
||||
if (IsC(af) || a > 0x99) tmp -= 0x60;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (IsH(af) || ((a & 0x0F) > 0x09)) tmp += 0x06;
|
||||
if (IsC(af) || a > 0x99) tmp += 0x60;
|
||||
}
|
||||
|
||||
TableDaa[af] = (ushort)((tmp * 256) + FlagByte(IsC(af) || a > 0x99, IsN(af), TableParity[tmp], UndocumentedX(tmp), ((a ^ tmp) & 0x10) != 0, UndocumentedY(tmp), tmp == 0, tmp > 127));
|
||||
}
|
||||
}
|
||||
|
||||
private byte FlagByte(bool C, bool N, bool P, bool X, bool H, bool Y, bool Z, bool S)
|
||||
{
|
||||
return (byte)(
|
||||
(C ? 0x01 : 0) +
|
||||
(N ? 0x02 : 0) +
|
||||
(P ? 0x04 : 0) +
|
||||
(X ? 0x08 : 0) +
|
||||
(H ? 0x10 : 0) +
|
||||
(Y ? 0x20 : 0) +
|
||||
(Z ? 0x40 : 0) +
|
||||
(S ? 0x80 : 0)
|
||||
);
|
||||
}
|
||||
|
||||
private bool UndocumentedX(int value)
|
||||
{
|
||||
return (value & 0x08) != 0;
|
||||
}
|
||||
|
||||
private bool UndocumentedY(int value)
|
||||
{
|
||||
return (value & 0x20) != 0;
|
||||
}
|
||||
|
||||
private bool IsC(int value) { return (value & 0x01) != 0; }
|
||||
private bool IsN(int value) { return (value & 0x02) != 0; }
|
||||
private bool IsP(int value) { return (value & 0x04) != 0; }
|
||||
private bool IsX(int value) { return (value & 0x08) != 0; }
|
||||
private bool IsH(int value) { return (value & 0x10) != 0; }
|
||||
private bool IsY(int value) { return (value & 0x20) != 0; }
|
||||
private bool IsZ(int value) { return (value & 0x40) != 0; }
|
||||
private bool IsS(int value) { return (value & 0x80) != 0; }
|
||||
}
|
||||
}
|
|
@ -1,142 +0,0 @@
|
|||
using System;
|
||||
using System.Globalization;
|
||||
using System.IO;
|
||||
|
||||
using BizHawk.Common;
|
||||
using BizHawk.Emulation.Common;
|
||||
|
||||
// This Z80 emulator is a modified version of Ben Ryves 'Brazil' emulator.
|
||||
// It is MIT licensed.
|
||||
|
||||
// for WZ register details, see: http://www.grimware.org/lib/exe/fetch.php/documentations/devices/z80/z80.memptr.eng.txt
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80
|
||||
{
|
||||
public sealed partial class Z80A
|
||||
{
|
||||
public Z80A()
|
||||
{
|
||||
InitialiseTables();
|
||||
Reset();
|
||||
}
|
||||
|
||||
public void Reset()
|
||||
{
|
||||
ResetRegisters();
|
||||
ResetInterrupts();
|
||||
PendingCycles = 0;
|
||||
ExpectedExecutedCycles = 0;
|
||||
TotalExecutedCycles = 0;
|
||||
}
|
||||
|
||||
public void SoftReset()
|
||||
{
|
||||
ResetRegisters();
|
||||
ResetInterrupts();
|
||||
}
|
||||
|
||||
// Memory Access
|
||||
|
||||
public Func<ushort, bool, byte> FetchMemory;
|
||||
public Func<ushort, byte> ReadMemory;
|
||||
public Action<ushort, byte> WriteMemory;
|
||||
|
||||
public byte ReadMemoryWrapper(ushort addr)
|
||||
{
|
||||
if (MemoryCallbacks != null)
|
||||
{
|
||||
MemoryCallbacks.CallReads(addr);
|
||||
}
|
||||
|
||||
return ReadMemory(addr);
|
||||
}
|
||||
|
||||
public byte FetchFirstMemoryWrapper(ushort addr)
|
||||
{
|
||||
if (MemoryCallbacks != null)
|
||||
{
|
||||
MemoryCallbacks.CallReads(addr);
|
||||
}
|
||||
|
||||
if (FetchMemory != null)
|
||||
{
|
||||
return FetchMemory(addr, true);
|
||||
}
|
||||
|
||||
return ReadMemory(addr);
|
||||
}
|
||||
|
||||
public byte FetchMemoryWrapper(ushort addr)
|
||||
{
|
||||
if (MemoryCallbacks != null)
|
||||
{
|
||||
MemoryCallbacks.CallReads(addr);
|
||||
}
|
||||
|
||||
if (FetchMemory != null)
|
||||
{
|
||||
return FetchMemory(addr, false);
|
||||
}
|
||||
|
||||
return ReadMemory(addr);
|
||||
}
|
||||
|
||||
public void WriteMemoryWrapper(ushort addr, byte value)
|
||||
{
|
||||
if (MemoryCallbacks != null)
|
||||
{
|
||||
MemoryCallbacks.CallWrites(addr);
|
||||
}
|
||||
|
||||
WriteMemory(addr, value);
|
||||
}
|
||||
|
||||
public IMemoryCallbackSystem MemoryCallbacks { get; set; }
|
||||
|
||||
// Utility function, not used by core
|
||||
public ushort ReadWord(ushort addr)
|
||||
{
|
||||
ushort value = ReadMemory(addr++);
|
||||
value |= (ushort)(ReadMemory(addr) << 8);
|
||||
return value;
|
||||
}
|
||||
|
||||
// Hardware I/O Port Access
|
||||
|
||||
public Func<ushort, byte> ReadHardware;
|
||||
public Action<ushort, byte> WriteHardware;
|
||||
|
||||
// State Save/Load
|
||||
|
||||
public void SyncState(Serializer ser)
|
||||
{
|
||||
ser.BeginSection("Z80");
|
||||
ser.Sync("AF", ref RegAF.Word);
|
||||
ser.Sync("BC", ref RegBC.Word);
|
||||
ser.Sync("DE", ref RegDE.Word);
|
||||
ser.Sync("HL", ref RegHL.Word);
|
||||
ser.Sync("WZ", ref RegWZ.Word);
|
||||
ser.Sync("ShadowAF", ref RegAltAF.Word);
|
||||
ser.Sync("ShadowBC", ref RegAltBC.Word);
|
||||
ser.Sync("ShadowDE", ref RegAltDE.Word);
|
||||
ser.Sync("ShadowHL", ref RegAltHL.Word);
|
||||
ser.Sync("I", ref RegI);
|
||||
ser.Sync("R", ref RegR);
|
||||
ser.Sync("IX", ref RegIX.Word);
|
||||
ser.Sync("IY", ref RegIY.Word);
|
||||
ser.Sync("SP", ref RegSP.Word);
|
||||
ser.Sync("PC", ref RegPC.Word);
|
||||
ser.Sync("IRQ", ref interrupt);
|
||||
ser.Sync("NMI", ref nonMaskableInterrupt);
|
||||
ser.Sync("NMIPending", ref nonMaskableInterruptPending);
|
||||
ser.Sync("IM", ref interruptMode);
|
||||
ser.Sync("IFF1", ref iff1);
|
||||
ser.Sync("IFF2", ref iff2);
|
||||
ser.Sync("Halted", ref halted);
|
||||
ser.Sync("ExecutedCycles", ref totalExecutedCycles);
|
||||
ser.Sync("PendingCycles", ref pendingCycles);
|
||||
ser.Sync("EI_pending", ref EI_pending);
|
||||
ser.EndSection();
|
||||
}
|
||||
}
|
||||
}
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,123 @@
|
|||
using System;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80A
|
||||
{
|
||||
public partial class Z80A
|
||||
{
|
||||
private bool iff1;
|
||||
public bool IFF1 { get { return iff1; } set { iff1 = value; } }
|
||||
|
||||
private bool iff2;
|
||||
public bool IFF2 { get { return iff2; } set { iff2 = value; } }
|
||||
|
||||
private bool nonMaskableInterrupt;
|
||||
public bool NonMaskableInterrupt
|
||||
{
|
||||
get { return nonMaskableInterrupt; }
|
||||
set { if (value && !nonMaskableInterrupt) NonMaskableInterruptPending = true; nonMaskableInterrupt = value; }
|
||||
}
|
||||
|
||||
private bool nonMaskableInterruptPending;
|
||||
public bool NonMaskableInterruptPending { get { return nonMaskableInterruptPending; } set { nonMaskableInterruptPending = value; } }
|
||||
|
||||
private int interruptMode;
|
||||
public int InterruptMode
|
||||
{
|
||||
get { return interruptMode; }
|
||||
set { if (value < 0 || value > 2) throw new ArgumentOutOfRangeException(); interruptMode = value; }
|
||||
}
|
||||
|
||||
public Action IRQCallback = delegate () { };
|
||||
public Action NMICallback = delegate () { };
|
||||
|
||||
private void NMI_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
DEC16, SPl, SPh,
|
||||
WR, SPl, SPh, PCh,
|
||||
IDLE,
|
||||
DEC16, SPl, SPh,
|
||||
WR, SPl, SPh, PCl,
|
||||
IDLE,
|
||||
ASGN, PCl, 0x66,
|
||||
ASGN, PCh, 0,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
// Mode 0 interrupts only take effect if a CALL or RST is on the data bus
|
||||
// Otherwise operation just continues as normal
|
||||
// For now assume a NOP is on the data bus, in which case no stack operations occur
|
||||
|
||||
//NOTE: TODO: When a CALL is present on the data bus, adjust WZ accordingly
|
||||
private void INTERRUPT_0(ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
// Just jump to $0038
|
||||
private void INTERRUPT_1()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{DEC16, SPl, SPh,
|
||||
IDLE,
|
||||
WR, SPl, SPh, PCh,
|
||||
IDLE,
|
||||
DEC16, SPl, SPh,
|
||||
IDLE,
|
||||
WR, SPl, SPh, PCl,
|
||||
IDLE,
|
||||
ASGN, PCl, 0x38,
|
||||
IDLE,
|
||||
ASGN, PCh, 0,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
// Interrupt mode 2 uses the I vector combined with a byte on the data bus
|
||||
// Again for now we assume only a 0 on the data bus and jump to (0xI00)
|
||||
private void INTERRUPT_2(ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
DEC16, SPl, SPh,
|
||||
WR, SPl, SPh, PCh,
|
||||
IDLE,
|
||||
DEC16, SPl, SPh,
|
||||
WR, SPl, SPh, PCl,
|
||||
IDLE,
|
||||
ASGN, PCl, 0,
|
||||
TR, PCh, I,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, W, PCl, PCh,
|
||||
IDLE,
|
||||
TR16, PCl, PCh, Z, W,
|
||||
OP };
|
||||
}
|
||||
|
||||
private static ushort[] INT_vectors = new ushort[] {0x40, 0x48, 0x50, 0x58, 0x60};
|
||||
|
||||
private void ResetInterrupts()
|
||||
{
|
||||
IFF1 = false;
|
||||
IFF2 = false;
|
||||
NonMaskableInterrupt = false;
|
||||
NonMaskableInterruptPending = false;
|
||||
FlagI = false;
|
||||
InterruptMode = 1;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -1,114 +1,27 @@
|
|||
//http://www.zophar.net/fileuploads/2/10819kouzv/z80undoc.html
|
||||
|
||||
//TODO: ex. (IX+00h) could be turned into (IX)
|
||||
|
||||
//usage:
|
||||
//VgMuseum.Z80.Disassembler disasm = new Disassembler();
|
||||
//ushort pc = RegPC.Word;
|
||||
//string str = disasm.Disassemble(() => ReadMemory(pc++));
|
||||
//Console.WriteLine(str);
|
||||
|
||||
//please note that however much youre tempted to, timings can't be put in a table here because they depend on how the instruction executes at runtime
|
||||
|
||||
using System;
|
||||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
||||
using BizHawk.Emulation.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80A
|
||||
{
|
||||
public class Disassembler : IDisassemblable
|
||||
public sealed partial class Z80A : IDisassemblable
|
||||
{
|
||||
readonly static sbyte[,] opcodeSizes = new sbyte[7, 256];
|
||||
|
||||
public static void GenerateOpcodeSizes()
|
||||
{
|
||||
Disassembler disasm = new Disassembler();
|
||||
|
||||
for (int i = 0; i < 256; i++)
|
||||
{
|
||||
int pc = 0;
|
||||
byte[] opcode = { (byte)i, 0, 0, 0 };
|
||||
disasm.Disassemble(() => opcode[pc++]);
|
||||
opcodeSizes[0, i] = (sbyte)pc;
|
||||
}
|
||||
|
||||
opcodeSizes[0, 0xCB] = -1;
|
||||
opcodeSizes[0, 0xED] = -2;
|
||||
opcodeSizes[0, 0xDD] = -3;
|
||||
opcodeSizes[0, 0xFD] = -4;
|
||||
|
||||
for (int i = 0; i < 256; i++)
|
||||
{
|
||||
int pc = 0;
|
||||
byte[] opcode = { 0xCB, (byte)i, 0, 0, 0 };
|
||||
disasm.Disassemble(() => opcode[pc++]);
|
||||
opcodeSizes[1, i] = (sbyte)pc;
|
||||
}
|
||||
|
||||
for (int i = 0; i < 256; i++)
|
||||
{
|
||||
int pc = 0;
|
||||
byte[] opcode = { 0xED, (byte)i, 0, 0, 0 };
|
||||
disasm.Disassemble(() => opcode[pc++]);
|
||||
opcodeSizes[2, i] = (sbyte)pc;
|
||||
}
|
||||
|
||||
for (int i = 0; i < 256; i++)
|
||||
{
|
||||
int pc = 0;
|
||||
byte[] opcode = { 0xDD, (byte)i, 0, 0, 0 };
|
||||
disasm.Disassemble(() => opcode[pc++]);
|
||||
opcodeSizes[3, i] = (sbyte)pc;
|
||||
}
|
||||
|
||||
opcodeSizes[3, 0xCB] = -5;
|
||||
opcodeSizes[3, 0xED] = -2;
|
||||
|
||||
for (int i = 0; i < 256; i++)
|
||||
{
|
||||
int pc = 0;
|
||||
byte[] opcode = { 0xFD, (byte)i, 0, 0, 0 };
|
||||
disasm.Disassemble(() => opcode[pc++]);
|
||||
opcodeSizes[4, i] = (sbyte)pc;
|
||||
}
|
||||
|
||||
opcodeSizes[3, 0xCB] = -6;
|
||||
opcodeSizes[3, 0xED] = -2;
|
||||
|
||||
|
||||
for (int i = 0; i < 256; i++)
|
||||
{
|
||||
int pc = 0;
|
||||
byte[] opcode = { 0xDD, 0xCB, (byte)i, 0, 0, 0 };
|
||||
disasm.Disassemble(() => opcode[pc++]);
|
||||
opcodeSizes[5, i] = (sbyte)pc;
|
||||
}
|
||||
|
||||
for (int i = 0; i < 256; i++)
|
||||
{
|
||||
int pc = 0;
|
||||
byte[] opcode = { 0xFD, 0xCB, (byte)i, 0, 0, 0 };
|
||||
disasm.Disassemble(() => opcode[pc++]);
|
||||
opcodeSizes[6, i] = (sbyte)pc;
|
||||
}
|
||||
}
|
||||
|
||||
static string Result(string format, Func<byte> read)
|
||||
static string Result(string format, Func<ushort, byte> read, ref ushort addr)
|
||||
{
|
||||
//d immediately succeeds the opcode
|
||||
//n immediate succeeds the opcode and the displacement (if present)
|
||||
//nn immediately succeeds the opcode and the displacement (if present)
|
||||
if (format.IndexOf("nn") != -1)
|
||||
{
|
||||
byte B = read();
|
||||
byte C = read();
|
||||
byte B = read(addr++);
|
||||
byte C = read(addr++);
|
||||
format = format.Replace("nn", string.Format("{0:X4}h", B + C * 256));
|
||||
}
|
||||
|
||||
if (format.IndexOf("n") != -1)
|
||||
{
|
||||
byte B = read();
|
||||
byte B = read(addr++);
|
||||
format = format.Replace("n", string.Format("{0:X2}h", B));
|
||||
}
|
||||
|
||||
|
@ -116,7 +29,7 @@ namespace BizHawk.Emulation.Cores.Components.Z80
|
|||
|
||||
if (format.IndexOf("d") != -1)
|
||||
{
|
||||
byte B = read();
|
||||
byte B = read(addr++);
|
||||
bool neg = ((B & 0x80) != 0);
|
||||
char sign = neg ? '-' : '+';
|
||||
int val = neg ? 256 - B : B;
|
||||
|
@ -480,46 +393,49 @@ namespace BizHawk.Emulation.Cores.Components.Z80
|
|||
"NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", "NOP", //0x100
|
||||
};
|
||||
|
||||
string DisassembleInternal(Func<byte> read)
|
||||
public string Disassemble(ushort addr, Func<ushort, byte> read, out ushort size)
|
||||
{
|
||||
byte A = read();
|
||||
ushort start_addr = addr;
|
||||
ushort extra_inc = 0;
|
||||
byte A = read(addr++);
|
||||
string format;
|
||||
switch (A)
|
||||
{
|
||||
case 0xCB:
|
||||
A = read();
|
||||
A = read(addr++);
|
||||
format = mnemonicsCB[A];
|
||||
break;
|
||||
case 0xDD:
|
||||
A = read();
|
||||
A = read(addr++);
|
||||
switch (A)
|
||||
{
|
||||
case 0xCB: format = mnemonicsDDCB[A]; break;
|
||||
case 0xCB: format = mnemonicsDDCB[A]; extra_inc = 1; break;
|
||||
case 0xED: format = mnemonicsED[A]; break;
|
||||
default: format = mnemonicsDD[A]; break;
|
||||
}
|
||||
break;
|
||||
case 0xED:
|
||||
A = read();
|
||||
A = read(addr++);
|
||||
format = mnemonicsED[A];
|
||||
break;
|
||||
case 0xFD:
|
||||
A = read();
|
||||
A = read(addr++);
|
||||
switch (A)
|
||||
{
|
||||
case 0xCB: format = mnemonicsFDCB[A]; break;
|
||||
case 0xCB: format = mnemonicsFDCB[A]; extra_inc = 1; break;
|
||||
case 0xED: format = mnemonicsED[A]; break;
|
||||
default: format = mnemonicsFD[A]; break;
|
||||
}
|
||||
break;
|
||||
default: format = mnemonics[A]; break;
|
||||
}
|
||||
return format;
|
||||
}
|
||||
|
||||
string temp = Result(format, read, ref addr);
|
||||
|
||||
public string Disassemble(Func<byte> read)
|
||||
{
|
||||
return Result(DisassembleInternal(read), read);
|
||||
addr += extra_inc;
|
||||
|
||||
size = (ushort)(addr - start_addr);
|
||||
return temp;
|
||||
}
|
||||
|
||||
#region IDisassemblable
|
||||
|
@ -543,7 +459,8 @@ namespace BizHawk.Emulation.Cores.Components.Z80
|
|||
public string Disassemble(MemoryDomain m, uint addr, out int length)
|
||||
{
|
||||
int loc = (int)addr;
|
||||
string ret = Disassemble(() => m.PeekByte(loc++));
|
||||
ushort unused = 0;
|
||||
string ret = Disassemble((ushort) addr, a => m.PeekByte(a), out unused);
|
||||
length = loc - (int)addr;
|
||||
return ret;
|
||||
}
|
|
@ -0,0 +1,742 @@
|
|||
using BizHawk.Common.NumberExtensions;
|
||||
using System;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80A
|
||||
{
|
||||
public partial class Z80A
|
||||
{
|
||||
public void Read_Func(ushort dest, ushort src_l, ushort src_h)
|
||||
{
|
||||
Regs[dest] = ReadMemory((ushort)(Regs[src_l] | (Regs[src_h]) << 8));
|
||||
}
|
||||
|
||||
public void I_Read_Func(ushort dest, ushort src_l, ushort src_h, ushort inc)
|
||||
{
|
||||
Regs[dest] = ReadMemory((ushort)((Regs[src_l] | (Regs[src_h] << 8)) + inc));
|
||||
}
|
||||
|
||||
public void Write_Func(ushort dest_l, ushort dest_h, ushort src)
|
||||
{
|
||||
WriteMemory((ushort)(Regs[dest_l] | (Regs[dest_h] << 8)), (byte)Regs[src]);
|
||||
}
|
||||
|
||||
public void I_Write_Func(ushort dest_l, ushort dest_h, ushort inc, ushort src)
|
||||
{
|
||||
WriteMemory((ushort)((Regs[dest_l] | (Regs[dest_h] << 8)) + inc), (byte)Regs[src]);
|
||||
}
|
||||
|
||||
public void OUT_Func(ushort dest, ushort src)
|
||||
{
|
||||
WriteHardware(Regs[dest], (byte)(Regs[src]));
|
||||
}
|
||||
|
||||
public void IN_Func(ushort dest, ushort src)
|
||||
{
|
||||
Regs[dest] = ReadHardware(Regs[src]);
|
||||
}
|
||||
|
||||
public void TR_Func(ushort dest, ushort src)
|
||||
{
|
||||
Regs[dest] = Regs[src];
|
||||
}
|
||||
|
||||
public void TR16_Func(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
Regs[dest_l] = Regs[src_l];
|
||||
Regs[dest_h] = Regs[src_h];
|
||||
}
|
||||
|
||||
public void ADD16_Func(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
int Reg16_d = Regs[dest_l] | (Regs[dest_h] << 8);
|
||||
int Reg16_s = Regs[src_l] | (Regs[src_h] << 8);
|
||||
int temp = Reg16_d + Reg16_s;
|
||||
|
||||
FlagC = temp.Bit(16);
|
||||
FlagH = ((Reg16_d & 0xFFF) + (Reg16_s & 0xFFF)) > 0xFFF;
|
||||
FlagN = false;
|
||||
Flag3 = (temp & 0x0800) != 0;
|
||||
Flag5 = (temp & 0x2000) != 0;
|
||||
|
||||
Regs[dest_l] = (ushort)(temp & 0xFF);
|
||||
Regs[dest_h] = (ushort)((temp & 0xFF00) >> 8);
|
||||
}
|
||||
|
||||
public void ADD8_Func(ushort dest, ushort src)
|
||||
{
|
||||
int Reg16_d = Regs[dest];
|
||||
Reg16_d += Regs[src];
|
||||
|
||||
FlagC = Reg16_d.Bit(8);
|
||||
FlagZ = (Reg16_d & 0xFF) == 0;
|
||||
|
||||
ushort ans = (ushort)(Reg16_d & 0xFF);
|
||||
|
||||
// redo for half carry flag
|
||||
Reg16_d = Regs[dest] & 0xF;
|
||||
Reg16_d += (Regs[src] & 0xF);
|
||||
|
||||
FlagH = Reg16_d.Bit(4);
|
||||
FlagN = false;
|
||||
Flag3 = (ans & 0x08) != 0;
|
||||
Flag5 = (ans & 0x20) != 0;
|
||||
FlagP = (Regs[dest].Bit(7) == Regs[src].Bit(7)) && (Regs[dest].Bit(7) != ans.Bit(7));
|
||||
FlagS = ans > 127;
|
||||
|
||||
Regs[dest] = ans;
|
||||
}
|
||||
|
||||
public void SUB8_Func(ushort dest, ushort src)
|
||||
{
|
||||
int Reg16_d = Regs[dest];
|
||||
Reg16_d -= Regs[src];
|
||||
|
||||
FlagC = Reg16_d.Bit(8);
|
||||
FlagZ = (Reg16_d & 0xFF) == 0;
|
||||
|
||||
ushort ans = (ushort)(Reg16_d & 0xFF);
|
||||
|
||||
// redo for half carry flag
|
||||
Reg16_d = Regs[dest] & 0xF;
|
||||
Reg16_d -= (Regs[src] & 0xF);
|
||||
|
||||
FlagH = Reg16_d.Bit(4);
|
||||
FlagN = true;
|
||||
Flag3 = (ans & 0x08) != 0;
|
||||
Flag5 = (ans & 0x20) != 0;
|
||||
FlagP = (Regs[dest].Bit(7) != Regs[src].Bit(7)) && (Regs[dest].Bit(7) != ans.Bit(7));
|
||||
FlagS = ans > 127;
|
||||
|
||||
Regs[dest] = ans;
|
||||
}
|
||||
|
||||
public void BIT_Func(ushort bit, ushort src)
|
||||
{
|
||||
FlagZ = !Regs[src].Bit(bit);
|
||||
FlagP = FlagZ; // special case
|
||||
FlagH = true;
|
||||
FlagN = false;
|
||||
FlagS = ((bit == 7) && Regs[src].Bit(bit));
|
||||
Flag5 = Regs[src].Bit(5);
|
||||
Flag3 = Regs[src].Bit(3);
|
||||
}
|
||||
|
||||
// When doing I* + n bit tests, flags 3 and 5 come from I* + n
|
||||
// This cooresponds to the high byte of WZ
|
||||
// This is the same for the (HL) bit tests, except that WZ were not assigned to before the test occurs
|
||||
public void I_BIT_Func(ushort bit, ushort src)
|
||||
{
|
||||
FlagZ = !Regs[src].Bit(bit);
|
||||
FlagP = FlagZ; // special case
|
||||
FlagH = true;
|
||||
FlagN = false;
|
||||
FlagS = ((bit == 7) && Regs[src].Bit(bit));
|
||||
Flag5 = Regs[W].Bit(5);
|
||||
Flag3 = Regs[W].Bit(3);
|
||||
}
|
||||
|
||||
public void SET_Func(ushort bit, ushort src)
|
||||
{
|
||||
Regs[src] |= (ushort)(1 << bit);
|
||||
}
|
||||
|
||||
public void RES_Func(ushort bit, ushort src)
|
||||
{
|
||||
Regs[src] &= (ushort)(0xFF - (1 << bit));
|
||||
}
|
||||
|
||||
public void ASGN_Func(ushort src, ushort val)
|
||||
{
|
||||
Regs[src] = val;
|
||||
}
|
||||
|
||||
public void SLL_Func(ushort src)
|
||||
{
|
||||
FlagC = Regs[src].Bit(7);
|
||||
|
||||
Regs[src] = (ushort)(((Regs[src] << 1) & 0xFF) | 0x1);
|
||||
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagZ = Regs[src] == 0;
|
||||
FlagP = TableParity[Regs[src]];
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
}
|
||||
|
||||
public void SLA_Func(ushort src)
|
||||
{
|
||||
FlagC = Regs[src].Bit(7);
|
||||
|
||||
Regs[src] = (ushort)((Regs[src] << 1) & 0xFF);
|
||||
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagZ = Regs[src] == 0;
|
||||
FlagP = TableParity[Regs[src]];
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
}
|
||||
|
||||
public void SRA_Func(ushort src)
|
||||
{
|
||||
FlagC = Regs[src].Bit(0);
|
||||
|
||||
ushort temp = (ushort)(Regs[src] & 0x80); // MSB doesn't change in this operation
|
||||
|
||||
Regs[src] = (ushort)((Regs[src] >> 1) | temp);
|
||||
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagZ = Regs[src] == 0;
|
||||
FlagP = TableParity[Regs[src]];
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
}
|
||||
|
||||
public void SRL_Func(ushort src)
|
||||
{
|
||||
FlagC = Regs[src].Bit(0) ? true : false;
|
||||
|
||||
Regs[src] = (ushort)(Regs[src] >> 1);
|
||||
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagZ = Regs[src] == 0;
|
||||
FlagP = TableParity[Regs[src]];
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
}
|
||||
|
||||
public void CPL_Func(ushort src)
|
||||
{
|
||||
Regs[src] = (ushort)((~Regs[src]) & 0xFF);
|
||||
|
||||
FlagH = true;
|
||||
FlagN = true;
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
}
|
||||
|
||||
public void CCF_Func(ushort src)
|
||||
{
|
||||
FlagH = FlagC;
|
||||
FlagC = !FlagC;
|
||||
FlagN = false;
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
}
|
||||
|
||||
public void SCF_Func(ushort src)
|
||||
{
|
||||
FlagC = true;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
}
|
||||
|
||||
public void AND8_Func(ushort dest, ushort src)
|
||||
{
|
||||
Regs[dest] = (ushort)(Regs[dest] & Regs[src]);
|
||||
|
||||
FlagZ = Regs[dest] == 0;
|
||||
FlagC = false;
|
||||
FlagH = true;
|
||||
FlagN = false;
|
||||
Flag3 = (Regs[dest] & 0x08) != 0;
|
||||
Flag5 = (Regs[dest] & 0x20) != 0;
|
||||
FlagS = Regs[dest] > 127;
|
||||
FlagP = TableParity[Regs[dest]];
|
||||
}
|
||||
|
||||
public void OR8_Func(ushort dest, ushort src)
|
||||
{
|
||||
Regs[dest] = (ushort)(Regs[dest] | Regs[src]);
|
||||
|
||||
FlagZ = Regs[dest] == 0;
|
||||
FlagC = false;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
Flag3 = (Regs[dest] & 0x08) != 0;
|
||||
Flag5 = (Regs[dest] & 0x20) != 0;
|
||||
FlagS = Regs[dest] > 127;
|
||||
FlagP = TableParity[Regs[dest]];
|
||||
}
|
||||
|
||||
public void XOR8_Func(ushort dest, ushort src)
|
||||
{
|
||||
Regs[dest] = (ushort)((Regs[dest] ^ Regs[src]));
|
||||
|
||||
FlagZ = Regs[dest] == 0;
|
||||
FlagC = false;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
Flag3 = (Regs[dest] & 0x08) != 0;
|
||||
Flag5 = (Regs[dest] & 0x20) != 0;
|
||||
FlagS = Regs[dest] > 127;
|
||||
FlagP = TableParity[Regs[dest]];
|
||||
}
|
||||
|
||||
public void CP8_Func(ushort dest, ushort src)
|
||||
{
|
||||
int Reg16_d = Regs[dest];
|
||||
Reg16_d -= Regs[src];
|
||||
|
||||
FlagC = Reg16_d.Bit(8);
|
||||
FlagZ = (Reg16_d & 0xFF) == 0;
|
||||
|
||||
ushort ans = (ushort)(Reg16_d & 0xFF);
|
||||
|
||||
// redo for half carry flag
|
||||
Reg16_d = Regs[dest] & 0xF;
|
||||
Reg16_d -= (Regs[src] & 0xF);
|
||||
|
||||
FlagH = Reg16_d.Bit(4);
|
||||
FlagN = true;
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
FlagP = (Regs[dest].Bit(7) != Regs[src].Bit(7)) && (Regs[dest].Bit(7) != ans.Bit(7));
|
||||
FlagS = ans > 127;
|
||||
}
|
||||
|
||||
public void RRC_Func(ushort src)
|
||||
{
|
||||
bool imm = src == Aim;
|
||||
if (imm) { src = A; }
|
||||
|
||||
FlagC = Regs[src].Bit(0);
|
||||
|
||||
Regs[src] = (ushort)((FlagC ? 0x80 : 0) | (Regs[src] >> 1));
|
||||
|
||||
if (!imm)
|
||||
{
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagZ = Regs[src] == 0;
|
||||
FlagP = TableParity[Regs[src]];
|
||||
}
|
||||
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
}
|
||||
|
||||
public void RR_Func(ushort src)
|
||||
{
|
||||
bool imm = src == Aim;
|
||||
if (imm) { src = A; }
|
||||
|
||||
ushort c = (ushort)(FlagC ? 0x80 : 0);
|
||||
|
||||
FlagC = Regs[src].Bit(0);
|
||||
|
||||
Regs[src] = (ushort)(c | (Regs[src] >> 1));
|
||||
|
||||
if (!imm)
|
||||
{
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagZ = Regs[src] == 0;
|
||||
FlagP = TableParity[Regs[src]];
|
||||
}
|
||||
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
}
|
||||
|
||||
public void RLC_Func(ushort src)
|
||||
{
|
||||
bool imm = src == Aim;
|
||||
if (imm) { src = A; }
|
||||
|
||||
ushort c = (ushort)(Regs[src].Bit(7) ? 1 : 0);
|
||||
FlagC = Regs[src].Bit(7);
|
||||
|
||||
Regs[src] = (ushort)(((Regs[src] << 1) & 0xFF) | c);
|
||||
|
||||
if (!imm)
|
||||
{
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagZ = Regs[src] == 0;
|
||||
FlagP = TableParity[Regs[src]];
|
||||
}
|
||||
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
}
|
||||
|
||||
public void RL_Func(ushort src)
|
||||
{
|
||||
bool imm = src == Aim;
|
||||
if (imm) { src = A; }
|
||||
|
||||
ushort c = (ushort)(FlagC ? 1 : 0);
|
||||
FlagC = Regs[src].Bit(7);
|
||||
|
||||
Regs[src] = (ushort)(((Regs[src] << 1) & 0xFF) | c);
|
||||
|
||||
if (!imm)
|
||||
{
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagZ = Regs[src] == 0;
|
||||
FlagP = TableParity[Regs[src]];
|
||||
}
|
||||
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
}
|
||||
|
||||
public void INC8_Func(ushort src)
|
||||
{
|
||||
int Reg16_d = Regs[src];
|
||||
Reg16_d += 1;
|
||||
|
||||
FlagZ = (Reg16_d & 0xFF) == 0;
|
||||
|
||||
ushort ans = (ushort)(Reg16_d & 0xFF);
|
||||
|
||||
// redo for half carry flag
|
||||
Reg16_d = Regs[src] & 0xF;
|
||||
Reg16_d += 1;
|
||||
|
||||
FlagH = Reg16_d.Bit(4);
|
||||
FlagN = false;
|
||||
|
||||
Regs[src] = ans;
|
||||
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagP = Regs[src] == 0x80;
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
}
|
||||
|
||||
public void DEC8_Func(ushort src)
|
||||
{
|
||||
int Reg16_d = Regs[src];
|
||||
Reg16_d -= 1;
|
||||
|
||||
FlagZ = (Reg16_d & 0xFF) == 0;
|
||||
|
||||
ushort ans = (ushort)(Reg16_d & 0xFF);
|
||||
|
||||
// redo for half carry flag
|
||||
Reg16_d = Regs[src] & 0xF;
|
||||
Reg16_d -= 1;
|
||||
|
||||
FlagH = Reg16_d.Bit(4);
|
||||
FlagN = true;
|
||||
|
||||
Regs[src] = ans;
|
||||
|
||||
FlagS = Regs[src].Bit(7);
|
||||
FlagP = Regs[src] == 0x7F;
|
||||
Flag3 = (Regs[src] & 0x08) != 0;
|
||||
Flag5 = (Regs[src] & 0x20) != 0;
|
||||
}
|
||||
|
||||
public void INC16_Func(ushort src_l, ushort src_h)
|
||||
{
|
||||
int Reg16_d = Regs[src_l] | (Regs[src_h] << 8);
|
||||
|
||||
Reg16_d += 1;
|
||||
|
||||
Regs[src_l] = (ushort)(Reg16_d & 0xFF);
|
||||
Regs[src_h] = (ushort)((Reg16_d & 0xFF00) >> 8);
|
||||
}
|
||||
|
||||
public void DEC16_Func(ushort src_l, ushort src_h)
|
||||
{
|
||||
int Reg16_d = Regs[src_l] | (Regs[src_h] << 8);
|
||||
|
||||
Reg16_d -= 1;
|
||||
|
||||
Regs[src_l] = (ushort)(Reg16_d & 0xFF);
|
||||
Regs[src_h] = (ushort)((Reg16_d & 0xFF00) >> 8);
|
||||
}
|
||||
|
||||
public void ADC8_Func(ushort dest, ushort src)
|
||||
{
|
||||
int Reg16_d = Regs[dest];
|
||||
int c = FlagC ? 1 : 0;
|
||||
|
||||
Reg16_d += (Regs[src] + c);
|
||||
|
||||
FlagC = Reg16_d.Bit(8);
|
||||
FlagZ = (Reg16_d & 0xFF) == 0;
|
||||
|
||||
ushort ans = (ushort)(Reg16_d & 0xFF);
|
||||
|
||||
// redo for half carry flag
|
||||
Reg16_d = Regs[dest] & 0xF;
|
||||
Reg16_d += ((Regs[src] & 0xF) + c);
|
||||
|
||||
FlagH = Reg16_d.Bit(4);
|
||||
FlagN = false;
|
||||
Flag3 = (ans & 0x08) != 0;
|
||||
Flag5 = (ans & 0x20) != 0;
|
||||
FlagP = (Regs[dest].Bit(7) == Regs[src].Bit(7)) && (Regs[dest].Bit(7) != ans.Bit(7));
|
||||
FlagS = ans > 127;
|
||||
|
||||
Regs[dest] = ans;
|
||||
}
|
||||
|
||||
public void SBC8_Func(ushort dest, ushort src)
|
||||
{
|
||||
int Reg16_d = Regs[dest];
|
||||
int c = FlagC ? 1 : 0;
|
||||
|
||||
Reg16_d -= (Regs[src] + c);
|
||||
|
||||
FlagC = Reg16_d.Bit(8);
|
||||
FlagZ = (Reg16_d & 0xFF) == 0;
|
||||
|
||||
ushort ans = (ushort)(Reg16_d & 0xFF);
|
||||
|
||||
// redo for half carry flag
|
||||
Reg16_d = Regs[dest] & 0xF;
|
||||
Reg16_d -= ((Regs[src] & 0xF) + c);
|
||||
|
||||
FlagH = Reg16_d.Bit(4);
|
||||
FlagN = true;
|
||||
Flag3 = (ans & 0x08) != 0;
|
||||
Flag5 = (ans & 0x20) != 0;
|
||||
FlagP = (Regs[dest].Bit(7) != Regs[src].Bit(7)) && (Regs[dest].Bit(7) != ans.Bit(7));
|
||||
FlagS = ans > 127;
|
||||
|
||||
Regs[dest] = ans;
|
||||
}
|
||||
|
||||
public void DA_Func(ushort src)
|
||||
{
|
||||
byte a = (byte)Regs[src];
|
||||
byte temp = a;
|
||||
|
||||
if (FlagN)
|
||||
{
|
||||
if (FlagH || ((a & 0x0F) > 0x09)) { temp -= 0x06; }
|
||||
if (FlagC || a > 0x99) { temp -= 0x60; }
|
||||
}
|
||||
else
|
||||
{
|
||||
if (FlagH || ((a & 0x0F) > 0x09)) { temp += 0x06; }
|
||||
if (FlagC || a > 0x99) { temp += 0x60; }
|
||||
}
|
||||
|
||||
temp &= 0xFF;
|
||||
|
||||
FlagC = FlagC || a > 0x99;
|
||||
FlagZ = temp == 0;
|
||||
FlagH = ((a ^ temp) & 0x10) != 0;
|
||||
FlagP = TableParity[temp];
|
||||
FlagS = temp > 127;
|
||||
Flag3 = (temp & 0x08) != 0;
|
||||
Flag5 = (temp & 0x20) != 0;
|
||||
|
||||
Regs[src] = temp;
|
||||
}
|
||||
|
||||
// used for signed operations
|
||||
public void ADDS_Func(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
int Reg16_d = Regs[dest_l];
|
||||
int Reg16_s = Regs[src_l];
|
||||
|
||||
Reg16_d += Reg16_s;
|
||||
|
||||
ushort temp = 0;
|
||||
|
||||
// since this is signed addition, calculate the high byte carry appropriately
|
||||
// note that flags are unaffected by this operation
|
||||
if (Reg16_s.Bit(7))
|
||||
{
|
||||
if (((Reg16_d & 0xFF) >= Regs[dest_l]))
|
||||
{
|
||||
temp = 0xFF;
|
||||
}
|
||||
else
|
||||
{
|
||||
temp = 0;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
temp = (ushort)(Reg16_d.Bit(8) ? 1 : 0);
|
||||
}
|
||||
|
||||
ushort ans_l = (ushort)(Reg16_d & 0xFF);
|
||||
|
||||
Regs[dest_l] = ans_l;
|
||||
Regs[dest_h] += temp;
|
||||
Regs[dest_h] &= 0xFF;
|
||||
|
||||
}
|
||||
|
||||
public void EXCH_16_Func(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
ushort temp = Regs[dest_l];
|
||||
Regs[dest_l] = Regs[src_l];
|
||||
Regs[src_l] = temp;
|
||||
|
||||
temp = Regs[dest_h];
|
||||
Regs[dest_h] = Regs[src_h];
|
||||
Regs[src_h] = temp;
|
||||
}
|
||||
|
||||
public void SBC_16_Func(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
int Reg16_d = Regs[dest_l] | (Regs[dest_h] << 8);
|
||||
int Reg16_s = Regs[src_l] | (Regs[src_h] << 8);
|
||||
int c = FlagC ? 1 : 0;
|
||||
|
||||
int ans = Reg16_d - Reg16_s - c;
|
||||
|
||||
FlagN = true;
|
||||
FlagC = ans.Bit(16);
|
||||
FlagP = (Reg16_d.Bit(15) != Reg16_s.Bit(15)) && (Reg16_d.Bit(15) != ans.Bit(15));
|
||||
FlagS = (ushort)(ans & 0xFFFF) > 32767;
|
||||
FlagZ = (ans & 0xFFFF) == 0;
|
||||
Flag3 = (ans & 0x0800) != 0;
|
||||
Flag5 = (ans & 0x2000) != 0;
|
||||
|
||||
// redo for half carry flag
|
||||
Reg16_d &= 0xFFF;
|
||||
Reg16_d -= ((Reg16_s & 0xFFF) + c);
|
||||
|
||||
FlagH = Reg16_d.Bit(12);
|
||||
|
||||
Regs[dest_l] = (ushort)(ans & 0xFF);
|
||||
Regs[dest_h] = (ushort)((ans >> 8) & 0xFF);
|
||||
}
|
||||
|
||||
public void ADC_16_Func(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
int Reg16_d = Regs[dest_l] | (Regs[dest_h] << 8);
|
||||
int Reg16_s = Regs[src_l] | (Regs[src_h] << 8);
|
||||
|
||||
int ans = Reg16_d + Reg16_s + (FlagC ? 1 : 0);
|
||||
|
||||
FlagH = ((Reg16_d & 0xFFF) + (Reg16_s & 0xFFF) + (FlagC ? 1 : 0)) > 0xFFF;
|
||||
FlagN = false;
|
||||
FlagC = ans.Bit(16);
|
||||
FlagP = (Reg16_d.Bit(15) == Reg16_s.Bit(15)) && (Reg16_d.Bit(15) != ans.Bit(15));
|
||||
FlagS = (ans & 0xFFFF) > 32767;
|
||||
FlagZ = (ans & 0xFFFF) == 0;
|
||||
Flag3 = (ans & 0x0800) != 0;
|
||||
Flag5 = (ans & 0x2000) != 0;
|
||||
|
||||
Regs[dest_l] = (ushort)(ans & 0xFF);
|
||||
Regs[dest_h] = (ushort)((ans >> 8) & 0xFF);
|
||||
}
|
||||
|
||||
public void NEG_8_Func(ushort src)
|
||||
{
|
||||
int Reg16_d = 0;
|
||||
Reg16_d -= Regs[src];
|
||||
|
||||
FlagC = Regs[src] != 0x0;
|
||||
FlagZ = (Reg16_d & 0xFF) == 0;
|
||||
FlagP = Regs[src] == 0x80;
|
||||
FlagS = (Reg16_d & 0xFF) > 127;
|
||||
|
||||
ushort ans = (ushort)(Reg16_d & 0xFF);
|
||||
// redo for half carry flag
|
||||
Reg16_d = 0;
|
||||
Reg16_d -= (Regs[src] & 0xF);
|
||||
FlagH = Reg16_d.Bit(4);
|
||||
Regs[src] = ans;
|
||||
FlagN = true;
|
||||
Flag3 = (ans & 0x08) != 0;
|
||||
Flag5 = (ans & 0x20) != 0;
|
||||
}
|
||||
|
||||
public void RRD_Func(ushort dest, ushort src)
|
||||
{
|
||||
ushort temp1 = Regs[src];
|
||||
ushort temp2 = Regs[dest];
|
||||
Regs[dest] = (ushort)(((temp1 & 0x0F) << 4) + ((temp2 & 0xF0) >> 4));
|
||||
Regs[src] = (ushort)((temp1 & 0xF0) + (temp2 & 0x0F));
|
||||
|
||||
temp1 = Regs[src];
|
||||
FlagS = temp1 > 127;
|
||||
FlagZ = temp1 == 0;
|
||||
FlagH = false;
|
||||
FlagP = TableParity[temp1];
|
||||
FlagN = false;
|
||||
Flag3 = (temp1 & 0x08) != 0;
|
||||
Flag5 = (temp1 & 0x20) != 0;
|
||||
}
|
||||
|
||||
public void RLD_Func(ushort dest, ushort src)
|
||||
{
|
||||
ushort temp1 = Regs[src];
|
||||
ushort temp2 = Regs[dest];
|
||||
Regs[dest] = (ushort)((temp1 & 0x0F) + ((temp2 & 0x0F) << 4));
|
||||
Regs[src] = (ushort)((temp1 & 0xF0) + ((temp2 & 0xF0) >> 4));
|
||||
|
||||
temp1 = Regs[src];
|
||||
FlagS = temp1 > 127;
|
||||
FlagZ = temp1 == 0;
|
||||
FlagH = false;
|
||||
FlagP = TableParity[temp1];
|
||||
FlagN = false;
|
||||
Flag3 = (temp1 & 0x08) != 0;
|
||||
Flag5 = (temp1 & 0x20) != 0;
|
||||
}
|
||||
|
||||
// sets flags for LD/R
|
||||
public void SET_FL_LD_Func()
|
||||
{
|
||||
FlagP = (Regs[C] | (Regs[B] << 8)) != 0;
|
||||
FlagH = false;
|
||||
FlagN = false;
|
||||
Flag5 = ((Regs[ALU] + Regs[A]) & 0x02) != 0;
|
||||
Flag3 = ((Regs[ALU] + Regs[A]) & 0x08) != 0;
|
||||
}
|
||||
|
||||
// set flags for CP/R
|
||||
public void SET_FL_CP_Func()
|
||||
{
|
||||
int Reg8_d = Regs[A];
|
||||
int Reg8_s = Regs[ALU];
|
||||
|
||||
// get half carry flag
|
||||
byte temp = (byte)((Reg8_d & 0xF) - (Reg8_s & 0xF));
|
||||
FlagH = temp.Bit(4);
|
||||
|
||||
temp = (byte)(Reg8_d - Reg8_s);
|
||||
FlagN = true;
|
||||
FlagZ = temp == 0;
|
||||
FlagS = temp > 127;
|
||||
FlagP = (Regs[C] | (Regs[B] << 8)) != 0;
|
||||
|
||||
temp = (byte)(Reg8_d - Reg8_s - (FlagH ? 1 : 0));
|
||||
Flag5 = (temp & 0x02) != 0;
|
||||
Flag3 = (temp & 0x08) != 0;
|
||||
}
|
||||
|
||||
// set flags for LD A, I/R
|
||||
public void SET_FL_IR_Func(ushort dest)
|
||||
{
|
||||
if (dest == A)
|
||||
{
|
||||
FlagN = false;
|
||||
FlagH = false;
|
||||
FlagZ = Regs[A] == 0;
|
||||
FlagS = Regs[A] > 127;
|
||||
FlagP = iff2;
|
||||
Flag5 = (Regs[A] & 0x02) != 0;
|
||||
Flag3 = (Regs[A] & 0x08) != 0;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,8 @@
|
|||
TODO:
|
||||
|
||||
Mode 0 and 2 interrupts
|
||||
Check T-cycle level memory access timing
|
||||
Check R register
|
||||
new tests for WZ Registers
|
||||
Memory refresh - IR is pushed onto the address bus at instruction start, does anything need this?
|
||||
Data Bus - For mode zero and 2 interrupts, need a system that uses it to test
|
|
@ -0,0 +1,132 @@
|
|||
using System.Runtime.InteropServices;
|
||||
using System;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80A
|
||||
{
|
||||
public partial class Z80A
|
||||
{
|
||||
// registers
|
||||
// note these are not constants. When shadows are used, they will be changed accordingly
|
||||
public ushort PCl = 0;
|
||||
public ushort PCh = 1;
|
||||
public ushort SPl = 2;
|
||||
public ushort SPh = 3;
|
||||
public ushort A = 4;
|
||||
public ushort F = 5;
|
||||
public ushort B = 6;
|
||||
public ushort C = 7;
|
||||
public ushort D = 8;
|
||||
public ushort E = 9;
|
||||
public ushort H = 10;
|
||||
public ushort L = 11;
|
||||
public ushort W = 12;
|
||||
public ushort Z = 13;
|
||||
public ushort Aim = 14; // use this indicator for RLCA etc., since the Z flag is reset on those
|
||||
public ushort Ixl = 15;
|
||||
public ushort Ixh = 16;
|
||||
public ushort Iyl = 17;
|
||||
public ushort Iyh = 18;
|
||||
public ushort Int = 19;
|
||||
public ushort R = 20;
|
||||
public ushort I = 21;
|
||||
public ushort ZERO = 22; // it is convenient to have a register that is always zero, to reuse instructions
|
||||
public ushort ALU = 23; // This will be temporary arthimatic storage
|
||||
// shadow registers
|
||||
public ushort A_s = 24;
|
||||
public ushort F_s = 25;
|
||||
public ushort B_s = 26;
|
||||
public ushort C_s = 27;
|
||||
public ushort D_s = 28;
|
||||
public ushort E_s = 29;
|
||||
public ushort H_s = 30;
|
||||
public ushort L_s = 31;
|
||||
|
||||
public ushort[] Regs = new ushort[36];
|
||||
|
||||
public bool FlagI;
|
||||
|
||||
public bool FlagC
|
||||
{
|
||||
get { return (Regs[5] & 0x01) != 0; }
|
||||
set { Regs[5] = (ushort)((Regs[5] & ~0x01) | (value ? 0x01 : 0x00)); }
|
||||
}
|
||||
|
||||
public bool FlagN
|
||||
{
|
||||
get { return (Regs[5] & 0x02) != 0; }
|
||||
set { Regs[5] = (ushort)((Regs[5] & ~0x02) | (value ? 0x02 : 0x00)); }
|
||||
}
|
||||
|
||||
public bool FlagP
|
||||
{
|
||||
get { return (Regs[5] & 0x04) != 0; }
|
||||
set { Regs[5] = (ushort)((Regs[5] & ~0x04) | (value ? 0x04 : 0x00)); }
|
||||
}
|
||||
|
||||
public bool Flag3
|
||||
{
|
||||
get { return (Regs[5] & 0x08) != 0; }
|
||||
set { Regs[5] = (ushort)((Regs[5] & ~0x08) | (value ? 0x08 : 0x00)); }
|
||||
}
|
||||
|
||||
public bool FlagH
|
||||
{
|
||||
get { return (Regs[5] & 0x10) != 0; }
|
||||
set { Regs[5] = (ushort)((Regs[5] & ~0x10) | (value ? 0x10 : 0x00)); }
|
||||
}
|
||||
|
||||
public bool Flag5
|
||||
{
|
||||
get { return (Regs[5] & 0x20) != 0; }
|
||||
set { Regs[5] = (ushort)((Regs[5] & ~0x20) | (value ? 0x20 : 0x00)); }
|
||||
}
|
||||
|
||||
public bool FlagZ
|
||||
{
|
||||
get { return (Regs[5] & 0x40) != 0; }
|
||||
set { Regs[5] = (ushort)((Regs[5] & ~0x40) | (value ? 0x40 : 0x00)); }
|
||||
}
|
||||
|
||||
public bool FlagS
|
||||
{
|
||||
get { return (Regs[5] & 0x80) != 0; }
|
||||
set { Regs[5] = (ushort)((Regs[5] & ~0x80) | (value ? 0x80 : 0x00)); }
|
||||
}
|
||||
|
||||
public ushort RegPC
|
||||
{
|
||||
get { return (ushort)(Regs[0] | (Regs[1] << 8)); }
|
||||
set
|
||||
{
|
||||
Regs[0] = (ushort)(value & 0xFF);
|
||||
Regs[1] = (ushort)((value >> 8) & 0xFF);
|
||||
}
|
||||
}
|
||||
|
||||
private void ResetRegisters()
|
||||
{
|
||||
for (int i=0; i < 36; i++)
|
||||
{
|
||||
Regs[i] = 0;
|
||||
}
|
||||
}
|
||||
|
||||
private bool[] TableParity;
|
||||
private void InitTableParity()
|
||||
{
|
||||
TableParity = new bool[256];
|
||||
for (int i = 0; i < 256; ++i)
|
||||
{
|
||||
int Bits = 0;
|
||||
for (int j = 0; j < 8; ++j)
|
||||
{
|
||||
Bits += (i >> j) & 1;
|
||||
}
|
||||
TableParity[i] = (Bits & 1) == 0;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
}
|
||||
}
|
|
@ -0,0 +1,572 @@
|
|||
using System;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80A
|
||||
{
|
||||
public partial class Z80A
|
||||
{
|
||||
// this contains the vectors of instrcution operations
|
||||
// NOTE: This list is NOT confirmed accurate for each individual cycle
|
||||
|
||||
private void NOP_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
// NOTE: In a real Z80, this operation just flips a switch to choose between 2 registers
|
||||
// but it's simpler to emulate just by exchanging the register with it's shadow
|
||||
private void EXCH_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{EXCH,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void EXX_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{EXX,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
// this exchanges 2 16 bit registers
|
||||
private void EXCH_16_(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{EXCH_16, dest_l, dest_h, src_l, src_h,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void INC_16(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
INC16, src_l, src_h,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
|
||||
private void DEC_16(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
DEC16, src_l, src_h,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
// this is done in two steps technically, but the flags don't work out using existing funcitons
|
||||
// so let's use a different function since it's an internal operation anyway
|
||||
private void ADD_16(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
TR16, Z, W, dest_l, dest_h,
|
||||
INC16, Z, W,
|
||||
IDLE,
|
||||
IDLE,
|
||||
ADD16, dest_l, dest_h, src_l, src_h,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void REG_OP(ushort operation, ushort dest, ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{operation, dest, src,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
// Operations using the I and R registers take one T-cycle longer
|
||||
private void REG_OP_IR(ushort operation, ushort dest, ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{operation, dest, src,
|
||||
IDLE,
|
||||
IDLE,
|
||||
SET_FL_IR, dest,
|
||||
OP };
|
||||
}
|
||||
|
||||
// note: do not use DEC here since no flags are affected by this operation
|
||||
private void DJNZ_()
|
||||
{
|
||||
if ((Regs[B] - 1) != 0)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
ASGN, B, (ushort)((Regs[B] - 1) & 0xFF),
|
||||
IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
ASGN, W, 0,
|
||||
IDLE,
|
||||
ADDS, PCl, PCh, Z, W,
|
||||
TR16, Z, W, PCl, PCh,
|
||||
OP };
|
||||
}
|
||||
else
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
ASGN, B, (ushort)((Regs[B] - 1) & 0xFF),
|
||||
IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
}
|
||||
|
||||
private void HALT_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
HALT };
|
||||
}
|
||||
|
||||
private void JR_COND(bool cond)
|
||||
{
|
||||
if (cond)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
IDLE,
|
||||
ASGN, W, 0,
|
||||
IDLE,
|
||||
ADDS, PCl, PCh, Z, W,
|
||||
TR16, Z, W, PCl, PCh,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
else
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
OP };
|
||||
}
|
||||
}
|
||||
|
||||
private void JP_COND(bool cond)
|
||||
{
|
||||
if (cond)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
RD, W, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
TR16, PCl, PCh, Z, W,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
else
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, W, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
}
|
||||
|
||||
private void RET_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, SPl, SPh,
|
||||
INC16, SPl, SPh,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, W, SPl, SPh,
|
||||
INC16, SPl, SPh,
|
||||
TR16, PCl, PCh, Z, W,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void RETI_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, SPl, SPh,
|
||||
INC16, SPl, SPh,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, W, SPl, SPh,
|
||||
INC16, SPl, SPh,
|
||||
TR16, PCl, PCh, Z, W,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void RETN_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, SPl, SPh,
|
||||
INC16, SPl, SPh,
|
||||
IDLE,
|
||||
RD, W, SPl, SPh,
|
||||
INC16, SPl, SPh,
|
||||
EI_RETN,
|
||||
TR16, PCl, PCh, Z, W,
|
||||
OP };
|
||||
}
|
||||
|
||||
|
||||
private void RET_COND(bool cond)
|
||||
{
|
||||
if (cond)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, SPl, SPh,
|
||||
INC16, SPl, SPh,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, W, SPl, SPh,
|
||||
INC16, SPl, SPh,
|
||||
IDLE,
|
||||
TR16, PCl, PCh, Z, W,
|
||||
OP };
|
||||
}
|
||||
else
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
}
|
||||
|
||||
private void CALL_COND(bool cond)
|
||||
{
|
||||
if (cond)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, W, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
DEC16, SPl, SPh,
|
||||
IDLE,
|
||||
WR, SPl, SPh, PCh,
|
||||
DEC16, SPl, SPh,
|
||||
WR, SPl, SPh, PCl,
|
||||
IDLE,
|
||||
TR, PCl, Z,
|
||||
TR, PCh, W,
|
||||
OP };
|
||||
}
|
||||
else
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, W, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
OP };
|
||||
}
|
||||
}
|
||||
|
||||
private void INT_OP(ushort operation, ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{operation, src,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void BIT_OP(ushort operation, ushort bit, ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{operation, bit, src,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void PUSH_(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
DEC16, SPl, SPh,
|
||||
IDLE,
|
||||
WR, SPl, SPh, src_h,
|
||||
IDLE,
|
||||
DEC16, SPl, SPh,
|
||||
IDLE,
|
||||
WR, SPl, SPh, src_l,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
|
||||
private void POP_(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
RD, src_l, SPl, SPh,
|
||||
IDLE,
|
||||
INC16, SPl, SPh,
|
||||
IDLE,
|
||||
RD, src_h, SPl, SPh,
|
||||
IDLE,
|
||||
INC16, SPl, SPh,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void RST_(ushort n)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
DEC16, SPl, SPh,
|
||||
WR, SPl, SPh, PCh,
|
||||
DEC16, SPl, SPh,
|
||||
WR, SPl, SPh, PCl,
|
||||
IDLE,
|
||||
ASGN, Z, n,
|
||||
ASGN, W, 0,
|
||||
TR16, PCl, PCh, Z, W,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void PREFIX_(ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
PREFIX, src};
|
||||
}
|
||||
|
||||
private void PREFETCH_(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{TR16, Z, W, src_l, src_h,
|
||||
ADDS, Z, W, ALU, ZERO,
|
||||
IDLE,
|
||||
PREFIX, IXYprefetch };
|
||||
}
|
||||
|
||||
private void DI_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{DI,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void EI_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{EI,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void JP_16(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{TR, PCl, src_l,
|
||||
IDLE,
|
||||
TR, PCh, src_h,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_SP_16(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
TR, SPl, src_l,
|
||||
TR, SPh, src_h,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void OUT_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
TR, W, A,
|
||||
OUT, ALU, A,
|
||||
TR, Z, ALU,
|
||||
INC16, Z, ALU,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP};
|
||||
}
|
||||
|
||||
private void OUT_REG_(ushort dest, ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
OUT, dest, src,
|
||||
IDLE,
|
||||
TR16, Z, W, C, B,
|
||||
INC16, Z, W,
|
||||
IDLE,
|
||||
OP};
|
||||
}
|
||||
|
||||
private void IN_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
TR, W, A,
|
||||
IN, A, ALU,
|
||||
TR, Z, ALU,
|
||||
INC16, Z, W,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP};
|
||||
}
|
||||
|
||||
private void IN_REG_(ushort dest, ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IN, dest, src,
|
||||
IDLE,
|
||||
TR16, Z, W, C, B,
|
||||
INC16, Z, W,
|
||||
IDLE,
|
||||
OP};
|
||||
}
|
||||
|
||||
private void REG_OP_16_(ushort op, ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
TR16, Z, W, dest_l, dest_h,
|
||||
INC16, Z, W,
|
||||
IDLE,
|
||||
IDLE,
|
||||
op, dest_l, dest_h, src_l, src_h,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP};
|
||||
}
|
||||
|
||||
private void INT_MODE_(ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
INT_MODE, src,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void RRD_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
TR16, Z, W, L, H,
|
||||
IDLE,
|
||||
RD, ALU, Z, W,
|
||||
IDLE,
|
||||
RRD, ALU, A,
|
||||
IDLE,
|
||||
WR, Z, W, ALU,
|
||||
IDLE,
|
||||
INC16, Z, W,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void RLD_()
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
TR16, Z, W, L, H,
|
||||
IDLE,
|
||||
RD, ALU, Z, W,
|
||||
IDLE,
|
||||
RLD, ALU, A,
|
||||
IDLE,
|
||||
WR, Z, W, ALU,
|
||||
IDLE,
|
||||
INC16, Z, W,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,478 @@
|
|||
namespace BizHawk.Emulation.Cores.Components.Z80A
|
||||
{
|
||||
public partial class Z80A
|
||||
{
|
||||
private void INT_OP_IND(ushort operation, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, ALU, src_l, src_h,
|
||||
IDLE,
|
||||
operation, ALU,
|
||||
IDLE,
|
||||
WR, src_l, src_h, ALU,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void BIT_OP_IND(ushort operation, ushort bit, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, ALU, src_l, src_h,
|
||||
IDLE,
|
||||
operation, bit, ALU,
|
||||
IDLE,
|
||||
WR, src_l, src_h, ALU,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
// Note that this operation uses I_BIT, same as indexed BIT.
|
||||
// This is where the strange behaviour in Flag bits 3 and 5 come from.
|
||||
// normally WZ contain I* + n when doing I_BIT ops, but here we use that code path
|
||||
// even though WZ is not assigned to, letting it's value from other operations show through
|
||||
private void BIT_TE_IND(ushort operation, ushort bit, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, ALU, src_l, src_h,
|
||||
IDLE,
|
||||
I_BIT, bit, ALU,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void REG_OP_IND_INC(ushort operation, ushort dest, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, ALU, src_l, src_h,
|
||||
IDLE,
|
||||
operation, dest, ALU,
|
||||
INC16, src_l, src_h,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void REG_OP_IND(ushort operation, ushort dest, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
TR16, Z, W, src_l, src_h,
|
||||
RD, ALU, Z, W,
|
||||
INC16, Z, W,
|
||||
operation, dest, ALU,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_16_IND_nn(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, W, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
WR, Z, W, src_l,
|
||||
IDLE,
|
||||
INC16, Z, W,
|
||||
IDLE,
|
||||
WR, Z, W, src_h,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_IND_16_nn(ushort dest_l, ushort dest_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, W, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, dest_l, Z, W,
|
||||
IDLE,
|
||||
INC16, Z, W,
|
||||
IDLE,
|
||||
RD, dest_h, Z, W,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_8_IND_nn(ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, W, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
WR, Z, W, src,
|
||||
INC16, Z, W,
|
||||
TR, W, A,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_IND_8_nn(ushort dest)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
RD, Z, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, W, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
RD, dest, Z, W,
|
||||
IDLE,
|
||||
INC16, Z, W,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_8_IND(ushort dest_l, ushort dest_h, ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
TR16, Z, W, dest_l, dest_h,
|
||||
WR, Z, W, src,
|
||||
INC16, Z, W,
|
||||
TR, W, A,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_8_IND_IND(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, ALU, src_l, src_h,
|
||||
IDLE,
|
||||
INC16, src_l, src_h,
|
||||
IDLE,
|
||||
WR, dest_l, dest_h, ALU,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_IND_8_INC(ushort dest, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, dest, src_l, src_h,
|
||||
IDLE,
|
||||
INC16, src_l, src_h,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_IND_8_DEC(ushort dest, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, dest, src_l, src_h,
|
||||
IDLE,
|
||||
DEC16, src_l, src_h,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_IND_16(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, dest_l, src_l, src_h,
|
||||
IDLE,
|
||||
INC16, src_l, src_h,
|
||||
RD, dest_h, src_l, src_h,
|
||||
IDLE,
|
||||
INC16, src_l, src_h,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void INC_8_IND(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, ALU, src_l, src_h,
|
||||
IDLE,
|
||||
INC8, ALU,
|
||||
IDLE,
|
||||
WR, src_l, src_h, ALU,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void DEC_8_IND(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, ALU, src_l, src_h,
|
||||
IDLE,
|
||||
DEC8, ALU,
|
||||
IDLE,
|
||||
WR, src_l, src_h, ALU,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
// NOTE: WZ implied for the wollowing 3 functions
|
||||
private void I_INT_OP(ushort operation, ushort dest)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, ALU, Z, W,
|
||||
IDLE,
|
||||
operation, ALU,
|
||||
IDLE,
|
||||
WR, Z, W, ALU,
|
||||
IDLE,
|
||||
TR, dest, ALU,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void I_BIT_OP(ushort operation, ushort bit, ushort dest)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, ALU, Z, W,
|
||||
IDLE,
|
||||
operation, bit, ALU,
|
||||
IDLE,
|
||||
WR, Z, W, ALU,
|
||||
IDLE,
|
||||
TR, dest, ALU,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void I_BIT_TE(ushort bit)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
RD, ALU, Z, W,
|
||||
IDLE,
|
||||
I_BIT, bit, ALU,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void I_OP_n(ushort operation, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
TR16, Z, W, src_l, src_h,
|
||||
IDLE,
|
||||
ADDS, Z, W, ALU, ZERO,
|
||||
IDLE,
|
||||
RD, ALU, Z, W,
|
||||
IDLE,
|
||||
IDLE,
|
||||
operation, ALU,
|
||||
IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
WR, Z, W, ALU,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void I_OP_n_n(ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
TR16, Z, W, src_l, src_h,
|
||||
IDLE,
|
||||
ADDS, Z, W, ALU, ZERO,
|
||||
IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
WR, Z, W, ALU,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void I_REG_OP_IND_n(ushort operation, ushort dest, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
TR16, Z, W, src_l, src_h,
|
||||
IDLE,
|
||||
ADDS, Z, W, ALU, ZERO,
|
||||
IDLE,
|
||||
RD, ALU, Z, W,
|
||||
IDLE,
|
||||
operation, dest, ALU,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void I_LD_8_IND_n(ushort dest_l, ushort dest_h, ushort src)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
RD, ALU, PCl, PCh,
|
||||
IDLE,
|
||||
INC16, PCl, PCh,
|
||||
IDLE,
|
||||
TR16, Z, W, dest_l, dest_h,
|
||||
IDLE,
|
||||
ADDS, Z, W, ALU, ZERO,
|
||||
IDLE,
|
||||
WR, Z, W, src,
|
||||
IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
|
||||
private void LD_OP_R(ushort operation, ushort repeat_instr)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{RD, ALU, L, H,
|
||||
IDLE,
|
||||
WR, E, D, ALU,
|
||||
IDLE,
|
||||
operation, L, H,
|
||||
IDLE,
|
||||
operation, E, D,
|
||||
IDLE,
|
||||
DEC16, C, B,
|
||||
SET_FL_LD,
|
||||
IDLE,
|
||||
OP_R, 0, operation, repeat_instr };
|
||||
}
|
||||
|
||||
private void CP_OP_R(ushort operation, ushort repeat_instr)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, ALU, L, H,
|
||||
operation, L, H,
|
||||
IDLE,
|
||||
IDLE,
|
||||
DEC16, C, B,
|
||||
SET_FL_CP,
|
||||
IDLE,
|
||||
operation, Z, W,
|
||||
IDLE,
|
||||
OP_R, 1, operation, repeat_instr };
|
||||
}
|
||||
|
||||
private void IN_OP_R(ushort operation, ushort repeat_instr)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IN, ALU, C,
|
||||
IDLE,
|
||||
WR, L, H, ALU,
|
||||
IDLE,
|
||||
operation, L, H,
|
||||
IDLE,
|
||||
TR16, Z, W, C, B,
|
||||
operation, Z, W,
|
||||
IDLE,
|
||||
DEC8, B,
|
||||
IDLE,
|
||||
OP_R, 2, operation, repeat_instr };
|
||||
}
|
||||
|
||||
private void OUT_OP_R(ushort operation, ushort repeat_instr)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{RD, ALU, L, H,
|
||||
IDLE,
|
||||
OUT, C, ALU,
|
||||
IDLE,
|
||||
IDLE,
|
||||
operation, L, H,
|
||||
DEC8, B,
|
||||
IDLE,
|
||||
TR16, Z, W, C, B,
|
||||
operation, Z, W,
|
||||
IDLE,
|
||||
OP_R, 3, operation, repeat_instr };
|
||||
}
|
||||
|
||||
// this is an indirect change of a a 16 bit register with memory
|
||||
private void EXCH_16_IND_(ushort dest_l, ushort dest_h, ushort src_l, ushort src_h)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
RD, Z, dest_l, dest_h,
|
||||
IDLE,
|
||||
IDLE,
|
||||
I_RD, W, dest_l, dest_h, 1,
|
||||
IDLE,
|
||||
IDLE,
|
||||
WR, dest_l, dest_h, src_l,
|
||||
IDLE,
|
||||
IDLE,
|
||||
I_WR, dest_l, dest_h, 1, src_h,
|
||||
IDLE,
|
||||
IDLE,
|
||||
TR16, src_l, src_h, Z, W,
|
||||
IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
OP };
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,687 @@
|
|||
using System;
|
||||
using System.Globalization;
|
||||
using System.IO;
|
||||
|
||||
using BizHawk.Common;
|
||||
using BizHawk.Emulation.Common;
|
||||
using BizHawk.Common.NumberExtensions;
|
||||
|
||||
// Z80A CPU
|
||||
namespace BizHawk.Emulation.Cores.Components.Z80A
|
||||
{
|
||||
public sealed partial class Z80A
|
||||
{
|
||||
// operations that can take place in an instruction
|
||||
public const ushort IDLE = 0;
|
||||
public const ushort OP = 1;
|
||||
public const ushort OP_R = 2; // used for repeating operations
|
||||
public const ushort HALT = 3;
|
||||
public const ushort RD = 4;
|
||||
public const ushort WR = 5;
|
||||
public const ushort I_RD = 6;
|
||||
public const ushort I_WR = 7;
|
||||
public const ushort TR = 8;
|
||||
public const ushort TR16 = 9;
|
||||
public const ushort ADD16 = 10;
|
||||
public const ushort ADD8 = 11;
|
||||
public const ushort SUB8 = 12;
|
||||
public const ushort ADC8 = 13;
|
||||
public const ushort SBC8 = 14;
|
||||
public const ushort SBC16 = 15;
|
||||
public const ushort ADC16 = 16;
|
||||
public const ushort INC16 = 17;
|
||||
public const ushort INC8 = 18;
|
||||
public const ushort DEC16 = 19;
|
||||
public const ushort DEC8 = 20;
|
||||
public const ushort RLC = 21;
|
||||
public const ushort RL = 22;
|
||||
public const ushort RRC = 23;
|
||||
public const ushort RR = 24;
|
||||
public const ushort CPL = 25;
|
||||
public const ushort DA = 26;
|
||||
public const ushort SCF = 27;
|
||||
public const ushort CCF = 28;
|
||||
public const ushort AND8 = 29;
|
||||
public const ushort XOR8 = 30;
|
||||
public const ushort OR8 = 31;
|
||||
public const ushort CP8 = 32;
|
||||
public const ushort SLA = 33;
|
||||
public const ushort SRA = 34;
|
||||
public const ushort SRL = 35;
|
||||
public const ushort SLL = 36;
|
||||
public const ushort BIT = 37;
|
||||
public const ushort RES = 38;
|
||||
public const ushort SET = 39;
|
||||
public const ushort EI = 40;
|
||||
public const ushort DI = 41;
|
||||
public const ushort EXCH = 42;
|
||||
public const ushort EXX = 43;
|
||||
public const ushort EXCH_16 = 44;
|
||||
public const ushort PREFIX = 45;
|
||||
public const ushort PREFETCH = 46;
|
||||
public const ushort ASGN = 47;
|
||||
public const ushort ADDS = 48; // signed 16 bit operation used in 2 instructions
|
||||
public const ushort INT_MODE = 49;
|
||||
public const ushort EI_RETN = 50;
|
||||
public const ushort EI_RETI = 51; // reti has no delay in interrupt enable
|
||||
public const ushort OUT = 52;
|
||||
public const ushort IN = 53;
|
||||
public const ushort NEG = 54;
|
||||
public const ushort RRD = 55;
|
||||
public const ushort RLD = 56;
|
||||
public const ushort SET_FL_LD = 57;
|
||||
public const ushort SET_FL_CP = 58;
|
||||
public const ushort SET_FL_IR = 59;
|
||||
public const ushort I_BIT = 60;
|
||||
public const ushort HL_BIT = 61;
|
||||
|
||||
public byte temp_R;
|
||||
|
||||
public Z80A()
|
||||
{
|
||||
Reset();
|
||||
InitTableParity();
|
||||
}
|
||||
|
||||
public void Reset()
|
||||
{
|
||||
ResetRegisters();
|
||||
ResetInterrupts();
|
||||
TotalExecutedCycles = 0;
|
||||
cur_instr = new ushort[] { OP };
|
||||
NO_prefix = true;
|
||||
}
|
||||
|
||||
public IMemoryCallbackSystem MemoryCallbacks { get; set; }
|
||||
|
||||
// Memory Access
|
||||
public Func<ushort, bool, byte> FetchMemory;
|
||||
public Func<ushort, byte> ReadMemory;
|
||||
public Action<ushort, byte> WriteMemory;
|
||||
public Func<ushort, byte> PeekMemory;
|
||||
public Func<ushort, byte> DummyReadMemory;
|
||||
|
||||
// Hardware I/O Port Access
|
||||
public Func<ushort, byte> ReadHardware;
|
||||
public Action<ushort, byte> WriteHardware;
|
||||
|
||||
//this only calls when the first byte of an instruction is fetched.
|
||||
public Action<ushort> OnExecFetch;
|
||||
|
||||
public void UnregisterMemoryMapper()
|
||||
{
|
||||
ReadMemory = null;
|
||||
WriteMemory = null;
|
||||
PeekMemory = null;
|
||||
DummyReadMemory = null;
|
||||
ReadHardware = null;
|
||||
WriteHardware = null;
|
||||
}
|
||||
|
||||
public void SetCallbacks
|
||||
(
|
||||
Func<ushort, byte> ReadMemory,
|
||||
Func<ushort, byte> DummyReadMemory,
|
||||
Func<ushort, byte> PeekMemory,
|
||||
Action<ushort, byte> WriteMemory,
|
||||
Func<ushort, byte> ReadHardware,
|
||||
Action<ushort, byte> WriteHardware
|
||||
)
|
||||
{
|
||||
this.ReadMemory = ReadMemory;
|
||||
this.DummyReadMemory = DummyReadMemory;
|
||||
this.PeekMemory = PeekMemory;
|
||||
this.WriteMemory = WriteMemory;
|
||||
this.ReadHardware = ReadHardware;
|
||||
this.WriteHardware = WriteHardware;
|
||||
}
|
||||
|
||||
// Execute instructions
|
||||
public void ExecuteOne()
|
||||
{
|
||||
if (Regs[A] > 255) { Console.WriteLine(RegPC); }
|
||||
switch (cur_instr[instr_pntr++])
|
||||
{
|
||||
case IDLE:
|
||||
// do nothing
|
||||
break;
|
||||
case OP:
|
||||
// Read the opcode of the next instruction
|
||||
if (EI_pending > 0)
|
||||
{
|
||||
EI_pending--;
|
||||
if (EI_pending == 0) { IFF1 = IFF2 = true; }
|
||||
}
|
||||
|
||||
// Process interrupt requests.
|
||||
if (nonMaskableInterruptPending)
|
||||
{
|
||||
nonMaskableInterruptPending = false;
|
||||
|
||||
if (TraceCallback != null)
|
||||
{
|
||||
TraceCallback(new TraceInfo{Disassembly = "====NMI====", RegisterInfo = ""});
|
||||
}
|
||||
|
||||
iff2 = iff1;
|
||||
iff1 = false;
|
||||
NMI_();
|
||||
NMICallback();
|
||||
}
|
||||
else if (iff1 && FlagI)
|
||||
{
|
||||
iff1 = iff2 = false;
|
||||
EI_pending = 0;
|
||||
|
||||
if (TraceCallback != null)
|
||||
{
|
||||
TraceCallback(new TraceInfo{Disassembly = "====IRQ====", RegisterInfo = ""});
|
||||
}
|
||||
|
||||
switch (interruptMode)
|
||||
{
|
||||
case 0:
|
||||
// Requires something to be pushed onto the data bus
|
||||
// we'll assume it's a zero for now
|
||||
INTERRUPT_0(0);
|
||||
break;
|
||||
case 1:
|
||||
INTERRUPT_1();
|
||||
break;
|
||||
case 2:
|
||||
// Low byte of interrupt vector comes from data bus
|
||||
// We'll assume it's zero for now
|
||||
INTERRUPT_2(0);
|
||||
break;
|
||||
}
|
||||
IRQCallback();
|
||||
}
|
||||
else
|
||||
{
|
||||
if (OnExecFetch != null) OnExecFetch(RegPC);
|
||||
if (TraceCallback != null) TraceCallback(State());
|
||||
FetchInstruction(ReadMemory(RegPC++));
|
||||
}
|
||||
instr_pntr = 0;
|
||||
|
||||
temp_R = (byte)(Regs[R] & 0x7F);
|
||||
temp_R++;
|
||||
temp_R &= 0x7F;
|
||||
Regs[R] = (byte)((Regs[R] & 0x80) | temp_R);
|
||||
break;
|
||||
case OP_R:
|
||||
// determine if we repeat based on what operation we are doing
|
||||
// single execution versions also come here, but never repeat
|
||||
ushort temp1 = cur_instr[instr_pntr++];
|
||||
ushort temp2 = cur_instr[instr_pntr++];
|
||||
ushort temp3 = cur_instr[instr_pntr++];
|
||||
|
||||
bool repeat = false;
|
||||
int Reg16_d = Regs[C] | (Regs[B] << 8);
|
||||
switch (temp1)
|
||||
{
|
||||
case 0:
|
||||
repeat = Reg16_d != 0;
|
||||
break;
|
||||
case 1:
|
||||
repeat = (Reg16_d != 0) && !FlagZ;
|
||||
break;
|
||||
case 2:
|
||||
repeat = Regs[B] != 0;
|
||||
break;
|
||||
case 3:
|
||||
repeat = Regs[B] != 0;
|
||||
break;
|
||||
}
|
||||
|
||||
// if we repeat, we do a 5 cycle refresh which decrements PC by 2
|
||||
// if we don't repeat, continue on as a normal opcode fetch
|
||||
if (repeat && temp3 > 0)
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
DEC16, PCl, PCh,
|
||||
IDLE,
|
||||
DEC16, PCl, PCh,
|
||||
OP };
|
||||
|
||||
// adjust WZ register accordingly
|
||||
switch (temp1)
|
||||
{
|
||||
case 0:
|
||||
// TEST: PC before or after the instruction?
|
||||
Regs[Z] = Regs[PCl];
|
||||
Regs[W] = Regs[PCh];
|
||||
INC16_Func(Z, W);
|
||||
break;
|
||||
case 1:
|
||||
// TEST: PC before or after the instruction?
|
||||
Regs[Z] = Regs[PCl];
|
||||
Regs[W] = Regs[PCh];
|
||||
INC16_Func(Z, W);
|
||||
break;
|
||||
case 2:
|
||||
// Nothing
|
||||
break;
|
||||
case 3:
|
||||
// Nothing
|
||||
break;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
// Interrupts can occur at this point, so process them accordingly
|
||||
// Read the opcode of the next instruction
|
||||
if (EI_pending > 0)
|
||||
{
|
||||
EI_pending--;
|
||||
if (EI_pending == 0) { IFF1 = IFF2 = true; }
|
||||
}
|
||||
|
||||
// Process interrupt requests.
|
||||
if (nonMaskableInterruptPending)
|
||||
{
|
||||
nonMaskableInterruptPending = false;
|
||||
|
||||
if (TraceCallback != null)
|
||||
{
|
||||
TraceCallback(new TraceInfo{Disassembly = "====NMI====", RegisterInfo = ""});
|
||||
}
|
||||
|
||||
iff2 = iff1;
|
||||
iff1 = false;
|
||||
NMI_();
|
||||
NMICallback();
|
||||
}
|
||||
else if (iff1 && FlagI)
|
||||
{
|
||||
iff1 = iff2 = false;
|
||||
EI_pending = 0;
|
||||
|
||||
if (TraceCallback != null)
|
||||
{
|
||||
TraceCallback(new TraceInfo{Disassembly = "====IRQ====", RegisterInfo = ""});
|
||||
}
|
||||
|
||||
switch (interruptMode)
|
||||
{
|
||||
case 0:
|
||||
// Requires something to be pushed onto the data bus
|
||||
// we'll assume it's a zero for now
|
||||
INTERRUPT_0(0);
|
||||
break;
|
||||
case 1:
|
||||
INTERRUPT_1();
|
||||
break;
|
||||
case 2:
|
||||
// Low byte of interrupt vector comes from data bus
|
||||
// We'll assume it's zero for now
|
||||
INTERRUPT_2(0);
|
||||
break;
|
||||
}
|
||||
IRQCallback();
|
||||
}
|
||||
else
|
||||
{
|
||||
if (OnExecFetch != null) OnExecFetch(RegPC);
|
||||
if (TraceCallback != null) TraceCallback(State());
|
||||
FetchInstruction(ReadMemory(RegPC++));
|
||||
}
|
||||
|
||||
temp_R = (byte)(Regs[R] & 0x7F);
|
||||
temp_R++;
|
||||
temp_R &= 0x7F;
|
||||
Regs[R] = (byte)((Regs[R] & 0x80) | temp_R);
|
||||
}
|
||||
instr_pntr = 0;
|
||||
break;
|
||||
|
||||
case HALT:
|
||||
halted = true;
|
||||
if (EI_pending > 0)
|
||||
{
|
||||
EI_pending--;
|
||||
if (EI_pending == 0) { IFF1 = IFF2 = true; }
|
||||
}
|
||||
|
||||
// Process interrupt requests.
|
||||
if (nonMaskableInterruptPending)
|
||||
{
|
||||
nonMaskableInterruptPending = false;
|
||||
|
||||
if (TraceCallback != null)
|
||||
{
|
||||
TraceCallback(new TraceInfo{Disassembly = "====NMI====", RegisterInfo = ""});
|
||||
}
|
||||
|
||||
iff2 = iff1;
|
||||
iff1 = false;
|
||||
NMI_();
|
||||
NMICallback();
|
||||
halted = false;
|
||||
}
|
||||
else if (iff1 && FlagI)
|
||||
{
|
||||
iff1 = iff2 = false;
|
||||
EI_pending = 0;
|
||||
|
||||
if (TraceCallback != null)
|
||||
{
|
||||
TraceCallback(new TraceInfo{Disassembly = "====IRQ====", RegisterInfo = ""});
|
||||
}
|
||||
|
||||
switch (interruptMode)
|
||||
{
|
||||
case 0:
|
||||
// Requires something to be pushed onto the data bus
|
||||
// we'll assume it's a zero for now
|
||||
INTERRUPT_0(0);
|
||||
break;
|
||||
case 1:
|
||||
INTERRUPT_1();
|
||||
break;
|
||||
case 2:
|
||||
// Low byte of interrupt vector comes from data bus
|
||||
// We'll assume it's zero for now
|
||||
INTERRUPT_2(0);
|
||||
break;
|
||||
}
|
||||
IRQCallback();
|
||||
halted = false;
|
||||
}
|
||||
else
|
||||
{
|
||||
cur_instr = new ushort[]
|
||||
{IDLE,
|
||||
IDLE,
|
||||
IDLE,
|
||||
HALT };
|
||||
}
|
||||
temp_R = (byte)(Regs[R] & 0x7F);
|
||||
temp_R++;
|
||||
temp_R &= 0x7F;
|
||||
Regs[R] = (byte)((Regs[R] & 0x80) | temp_R);
|
||||
|
||||
instr_pntr = 0;
|
||||
break;
|
||||
case RD:
|
||||
Read_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case WR:
|
||||
Write_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case I_RD:
|
||||
I_Read_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case I_WR:
|
||||
I_Write_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case TR:
|
||||
TR_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case TR16:
|
||||
TR16_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case ADD16:
|
||||
ADD16_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case ADD8:
|
||||
ADD8_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SUB8:
|
||||
SUB8_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case ADC8:
|
||||
ADC8_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case ADC16:
|
||||
ADC_16_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SBC8:
|
||||
SBC8_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SBC16:
|
||||
SBC_16_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case INC16:
|
||||
INC16_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case INC8:
|
||||
INC8_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case DEC16:
|
||||
DEC16_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case DEC8:
|
||||
DEC8_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case RLC:
|
||||
RLC_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case RL:
|
||||
RL_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case RRC:
|
||||
RRC_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case RR:
|
||||
RR_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case CPL:
|
||||
CPL_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case DA:
|
||||
DA_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SCF:
|
||||
SCF_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case CCF:
|
||||
CCF_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case AND8:
|
||||
AND8_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case XOR8:
|
||||
XOR8_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case OR8:
|
||||
OR8_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case CP8:
|
||||
CP8_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SLA:
|
||||
SLA_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SRA:
|
||||
SRA_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SRL:
|
||||
SRL_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SLL:
|
||||
SLL_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case BIT:
|
||||
BIT_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case I_BIT:
|
||||
I_BIT_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case RES:
|
||||
RES_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SET:
|
||||
SET_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case EI:
|
||||
EI_pending = 2;
|
||||
break;
|
||||
case DI:
|
||||
IFF1 = IFF2 = false;
|
||||
break;
|
||||
case EXCH:
|
||||
EXCH_16_Func(F_s, A_s, F, A);
|
||||
break;
|
||||
case EXX:
|
||||
EXCH_16_Func(C_s, B_s, C, B);
|
||||
EXCH_16_Func(E_s, D_s, E, D);
|
||||
EXCH_16_Func(L_s, H_s, L, H);
|
||||
break;
|
||||
case EXCH_16:
|
||||
EXCH_16_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case PREFIX:
|
||||
ushort prefix_src = cur_instr[instr_pntr++];
|
||||
NO_prefix = false;
|
||||
if (prefix_src == CBpre) { CB_prefix = true; }
|
||||
if (prefix_src == EXTDpre) { EXTD_prefix = true; }
|
||||
if (prefix_src == IXpre) { IX_prefix = true; }
|
||||
if (prefix_src == IYpre) { IY_prefix = true; }
|
||||
if (prefix_src == IXCBpre) { IXCB_prefix = true; IXCB_prefetch = true; }
|
||||
if (prefix_src == IYCBpre) { IYCB_prefix = true; IYCB_prefetch = true; }
|
||||
|
||||
FetchInstruction(ReadMemory(RegPC++));
|
||||
instr_pntr = 0;
|
||||
// only the first prefix in a double prefix increases R, although I don't know how / why
|
||||
if (prefix_src < 4)
|
||||
{
|
||||
temp_R = (byte)(Regs[R] & 0x7F);
|
||||
temp_R++;
|
||||
temp_R &= 0x7F;
|
||||
Regs[R] = (byte)((Regs[R] & 0x80) | temp_R);
|
||||
}
|
||||
break;
|
||||
case ASGN:
|
||||
ASGN_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case ADDS:
|
||||
ADDS_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case EI_RETI:
|
||||
// NOTE: This is needed for systems using multiple interrupt sources, it triggers the next interrupt
|
||||
// Not currently implemented here
|
||||
iff1 = iff2;
|
||||
break;
|
||||
case EI_RETN:
|
||||
iff1 = iff2;
|
||||
break;
|
||||
case OUT:
|
||||
OUT_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case IN:
|
||||
IN_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case NEG:
|
||||
NEG_8_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case INT_MODE:
|
||||
interruptMode = cur_instr[instr_pntr++];
|
||||
break;
|
||||
case RRD:
|
||||
RRD_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case RLD:
|
||||
RLD_Func(cur_instr[instr_pntr++], cur_instr[instr_pntr++]);
|
||||
break;
|
||||
case SET_FL_LD:
|
||||
SET_FL_LD_Func();
|
||||
break;
|
||||
case SET_FL_CP:
|
||||
SET_FL_CP_Func();
|
||||
break;
|
||||
case SET_FL_IR:
|
||||
SET_FL_IR_Func(cur_instr[instr_pntr++]);
|
||||
break;
|
||||
}
|
||||
totalExecutedCycles++;
|
||||
}
|
||||
|
||||
// tracer stuff
|
||||
public Action<TraceInfo> TraceCallback;
|
||||
|
||||
public string TraceHeader
|
||||
{
|
||||
get { return "Z80A: PC, machine code, mnemonic, operands, registers (AF, BC, DE, HL, IX, IY, SP, Cy), flags (CNP3H5ZS)"; }
|
||||
}
|
||||
|
||||
public TraceInfo State(bool disassemble = true)
|
||||
{
|
||||
ushort bytes_read = 0;
|
||||
|
||||
string disasm = disassemble ? Disassemble(RegPC, ReadMemory, out bytes_read) : "---";
|
||||
string byte_code = null;
|
||||
|
||||
for (ushort i = 0; i < bytes_read; i++)
|
||||
{
|
||||
byte_code += ReadMemory((ushort)(RegPC + i)).ToHexString(2);
|
||||
if (i < (bytes_read - 1))
|
||||
{
|
||||
byte_code += " ";
|
||||
}
|
||||
}
|
||||
|
||||
return new TraceInfo
|
||||
{
|
||||
Disassembly = string.Format(
|
||||
"{0:X4}: {1} {2}",
|
||||
RegPC,
|
||||
byte_code.PadRight(12),
|
||||
disasm.PadRight(26)),
|
||||
RegisterInfo = string.Format(
|
||||
"AF:{0:X4} BC:{1:X4} DE:{2:X4} HL:{3:X4} IX:{4:X4} IY:{5:X4} SP:{6:X4} Cy:{7} {8}{9}{10}{11}{12}{13}{14}{15}{16}",
|
||||
(Regs[A] << 8) + Regs[F],
|
||||
(Regs[B] << 8) + Regs[C],
|
||||
(Regs[D] << 8) + Regs[E],
|
||||
(Regs[H] << 8) + Regs[L],
|
||||
(Regs[Ixh] << 8) + Regs[Ixl],
|
||||
(Regs[Iyh] << 8) + Regs[Iyl],
|
||||
Regs[SPl] | (Regs[SPh] << 8),
|
||||
TotalExecutedCycles,
|
||||
FlagC ? "C" : "c",
|
||||
FlagN ? "N" : "n",
|
||||
FlagP ? "P" : "p",
|
||||
Flag3 ? "3" : "-",
|
||||
FlagH ? "H" : "h",
|
||||
Flag5 ? "5" : "-",
|
||||
FlagZ ? "Z" : "z",
|
||||
FlagS ? "S" : "s",
|
||||
FlagI ? "E" : "e")
|
||||
};
|
||||
}
|
||||
// State Save/Load
|
||||
|
||||
public void SyncState(Serializer ser)
|
||||
{
|
||||
ser.BeginSection("Z80A");
|
||||
ser.Sync("Regs", ref Regs, false);
|
||||
ser.Sync("NMI", ref nonMaskableInterrupt);
|
||||
ser.Sync("NMIPending", ref nonMaskableInterruptPending);
|
||||
ser.Sync("IM", ref interruptMode);
|
||||
ser.Sync("IFF1", ref iff1);
|
||||
ser.Sync("IFF2", ref iff2);
|
||||
ser.Sync("Halted", ref halted);
|
||||
ser.Sync("ExecutedCycles", ref totalExecutedCycles);
|
||||
ser.Sync("EI_pending", ref EI_pending);
|
||||
|
||||
ser.Sync("instruction_pointer", ref instr_pntr);
|
||||
ser.Sync("current instruction", ref cur_instr, false);
|
||||
ser.Sync("opcode", ref opcode);
|
||||
ser.Sync("FlagI", ref FlagI);
|
||||
|
||||
ser.Sync("NO Preifx", ref NO_prefix);
|
||||
ser.Sync("CB Preifx", ref CB_prefix);
|
||||
ser.Sync("IX_prefix", ref IX_prefix);
|
||||
ser.Sync("IY_prefix", ref IY_prefix);
|
||||
ser.Sync("IXCB_prefix", ref IXCB_prefix);
|
||||
ser.Sync("IYCB_prefix", ref IYCB_prefix);
|
||||
ser.Sync("EXTD_prefix", ref EXTD_prefix);
|
||||
ser.Sync("IXCB_prefetch", ref IXCB_prefetch);
|
||||
ser.Sync("IYCB_prefetch", ref IYCB_prefetch);
|
||||
ser.Sync("PF", ref PF);
|
||||
|
||||
ser.EndSection();
|
||||
}
|
||||
}
|
||||
}
|
|
@ -12,36 +12,36 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
{
|
||||
return new Dictionary<string, RegisterValue>
|
||||
{
|
||||
["A"] = _cpu.RegisterA,
|
||||
["AF"] = _cpu.RegisterAF,
|
||||
["B"] = _cpu.RegisterB,
|
||||
["BC"] = _cpu.RegisterBC,
|
||||
["C"] = _cpu.RegisterC,
|
||||
["D"] = _cpu.RegisterD,
|
||||
["DE"] = _cpu.RegisterDE,
|
||||
["E"] = _cpu.RegisterE,
|
||||
["F"] = _cpu.RegisterF,
|
||||
["H"] = _cpu.RegisterH,
|
||||
["HL"] = _cpu.RegisterHL,
|
||||
["I"] = _cpu.RegisterI,
|
||||
["IX"] = _cpu.RegisterIX,
|
||||
["IY"] = _cpu.RegisterIY,
|
||||
["L"] = _cpu.RegisterL,
|
||||
["PC"] = _cpu.RegisterPC,
|
||||
["R"] = _cpu.RegisterR,
|
||||
["Shadow AF"] = _cpu.RegisterShadowAF,
|
||||
["Shadow BC"] = _cpu.RegisterShadowBC,
|
||||
["Shadow DE"] = _cpu.RegisterShadowDE,
|
||||
["Shadow HL"] = _cpu.RegisterShadowHL,
|
||||
["SP"] = _cpu.RegisterSP,
|
||||
["Flag C"] = _cpu.RegisterF.Bit(0),
|
||||
["Flag N"] = _cpu.RegisterF.Bit(1),
|
||||
["Flag P/V"] = _cpu.RegisterF.Bit(2),
|
||||
["Flag 3rd"] = _cpu.RegisterF.Bit(3),
|
||||
["Flag H"] = _cpu.RegisterF.Bit(4),
|
||||
["Flag 5th"] = _cpu.RegisterF.Bit(5),
|
||||
["Flag Z"] = _cpu.RegisterF.Bit(6),
|
||||
["Flag S"] = _cpu.RegisterF.Bit(7)
|
||||
["A"] = _cpu.Regs[_cpu.A],
|
||||
["AF"] = _cpu.Regs[_cpu.F] + (_cpu.Regs[_cpu.A] << 8),
|
||||
["B"] = _cpu.Regs[_cpu.B],
|
||||
["BC"] = _cpu.Regs[_cpu.C] + (_cpu.Regs[_cpu.B] << 8),
|
||||
["C"] = _cpu.Regs[_cpu.C],
|
||||
["D"] = _cpu.Regs[_cpu.D],
|
||||
["DE"] = _cpu.Regs[_cpu.E] + (_cpu.Regs[_cpu.D] << 8),
|
||||
["E"] = _cpu.Regs[_cpu.E],
|
||||
["F"] = _cpu.Regs[_cpu.F],
|
||||
["H"] = _cpu.Regs[_cpu.H],
|
||||
["HL"] = _cpu.Regs[_cpu.L] + (_cpu.Regs[_cpu.H] << 8),
|
||||
["I"] = _cpu.Regs[_cpu.I],
|
||||
["IX"] = _cpu.Regs[_cpu.Ixl] + (_cpu.Regs[_cpu.Ixh] << 8),
|
||||
["IY"] = _cpu.Regs[_cpu.Iyl] + (_cpu.Regs[_cpu.Iyh] << 8),
|
||||
["L"] = _cpu.Regs[_cpu.L],
|
||||
["PC"] = _cpu.Regs[_cpu.PCl] + (_cpu.Regs[_cpu.PCh] << 8),
|
||||
["R"] = _cpu.Regs[_cpu.R],
|
||||
["Shadow AF"] = _cpu.Regs[_cpu.F_s] + (_cpu.Regs[_cpu.A_s] << 8),
|
||||
["Shadow BC"] = _cpu.Regs[_cpu.C_s] + (_cpu.Regs[_cpu.B_s] << 8),
|
||||
["Shadow DE"] = _cpu.Regs[_cpu.E_s] + (_cpu.Regs[_cpu.D_s] << 8),
|
||||
["Shadow HL"] = _cpu.Regs[_cpu.L_s] + (_cpu.Regs[_cpu.H_s] << 8),
|
||||
["SP"] = _cpu.Regs[_cpu.Iyl] + (_cpu.Regs[_cpu.Iyh] << 8),
|
||||
["Flag C"] = _cpu.FlagC,
|
||||
["Flag N"] = _cpu.FlagN,
|
||||
["Flag P/V"] = _cpu.FlagP,
|
||||
["Flag 3rd"] = _cpu.Flag3,
|
||||
["Flag H"] = _cpu.FlagH,
|
||||
["Flag 5th"] = _cpu.Flag5,
|
||||
["Flag Z"] = _cpu.FlagZ,
|
||||
["Flag S"] = _cpu.FlagS
|
||||
};
|
||||
}
|
||||
|
||||
|
@ -49,73 +49,85 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
{
|
||||
switch (register)
|
||||
{
|
||||
default:
|
||||
throw new InvalidOperationException();
|
||||
case "A":
|
||||
_cpu.RegisterA = (byte)value;
|
||||
break;
|
||||
case "AF":
|
||||
_cpu.RegisterAF = (byte)value;
|
||||
break;
|
||||
case "B":
|
||||
_cpu.RegisterB = (byte)value;
|
||||
break;
|
||||
case "BC":
|
||||
_cpu.RegisterBC = (byte)value;
|
||||
break;
|
||||
case "C":
|
||||
_cpu.RegisterC = (byte)value;
|
||||
break;
|
||||
case "D":
|
||||
_cpu.RegisterD = (byte)value;
|
||||
break;
|
||||
case "DE":
|
||||
_cpu.RegisterDE = (byte)value;
|
||||
break;
|
||||
case "E":
|
||||
_cpu.RegisterE = (byte)value;
|
||||
break;
|
||||
case "F":
|
||||
_cpu.RegisterF = (byte)value;
|
||||
break;
|
||||
case "H":
|
||||
_cpu.RegisterH = (byte)value;
|
||||
break;
|
||||
case "HL":
|
||||
_cpu.RegisterHL = (byte)value;
|
||||
break;
|
||||
case "I":
|
||||
_cpu.RegisterI = (byte)value;
|
||||
break;
|
||||
case "IX":
|
||||
_cpu.RegisterIX = (byte)value;
|
||||
break;
|
||||
case "IY":
|
||||
_cpu.RegisterIY = (byte)value;
|
||||
break;
|
||||
case "L":
|
||||
_cpu.RegisterL = (byte)value;
|
||||
break;
|
||||
case "PC":
|
||||
_cpu.RegisterPC = (ushort)value;
|
||||
break;
|
||||
case "R":
|
||||
_cpu.RegisterR = (byte)value;
|
||||
break;
|
||||
case "Shadow AF":
|
||||
_cpu.RegisterShadowAF = (byte)value;
|
||||
break;
|
||||
case "Shadow BC":
|
||||
_cpu.RegisterShadowBC = (byte)value;
|
||||
break;
|
||||
case "Shadow DE":
|
||||
_cpu.RegisterShadowDE = (byte)value;
|
||||
break;
|
||||
case "Shadow HL":
|
||||
_cpu.RegisterShadowHL = (byte)value;
|
||||
break;
|
||||
case "SP":
|
||||
_cpu.RegisterSP = (byte)value;
|
||||
default:
|
||||
throw new InvalidOperationException();
|
||||
case "A":
|
||||
_cpu.Regs[_cpu.A] = (ushort)value;
|
||||
break;
|
||||
case "AF":
|
||||
_cpu.Regs[_cpu.F] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.A] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "B":
|
||||
_cpu.Regs[_cpu.B] = (ushort)value;
|
||||
break;
|
||||
case "BC":
|
||||
_cpu.Regs[_cpu.C] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.B] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "C":
|
||||
_cpu.Regs[_cpu.C] = (ushort)value;
|
||||
break;
|
||||
case "D":
|
||||
_cpu.Regs[_cpu.D] = (ushort)value;
|
||||
break;
|
||||
case "DE":
|
||||
_cpu.Regs[_cpu.E] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.D] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "E":
|
||||
_cpu.Regs[_cpu.E] = (ushort)value;
|
||||
break;
|
||||
case "F":
|
||||
_cpu.Regs[_cpu.F] = (ushort)value;
|
||||
break;
|
||||
case "H":
|
||||
_cpu.Regs[_cpu.H] = (ushort)value;
|
||||
break;
|
||||
case "HL":
|
||||
_cpu.Regs[_cpu.L] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.H] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "I":
|
||||
_cpu.Regs[_cpu.I] = (ushort)value;
|
||||
break;
|
||||
case "IX":
|
||||
_cpu.Regs[_cpu.Ixl] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.Ixh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "IY":
|
||||
_cpu.Regs[_cpu.Iyl] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.Iyh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "L":
|
||||
_cpu.Regs[_cpu.L] = (ushort)value;
|
||||
break;
|
||||
case "PC":
|
||||
_cpu.Regs[_cpu.PCl] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.PCh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "R":
|
||||
_cpu.Regs[_cpu.R] = (ushort)value;
|
||||
break;
|
||||
case "Shadow AF":
|
||||
_cpu.Regs[_cpu.F_s] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.A_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "Shadow BC":
|
||||
_cpu.Regs[_cpu.C_s] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.B_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "Shadow DE":
|
||||
_cpu.Regs[_cpu.E_s] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.D_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "Shadow HL":
|
||||
_cpu.Regs[_cpu.L_s] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.H_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "SP":
|
||||
_cpu.Regs[_cpu.SPl] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.SPh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -13,21 +13,42 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
_controller = controller;
|
||||
_lagged = true;
|
||||
|
||||
_cpu.Debug = _tracer.Enabled;
|
||||
|
||||
if (_cpu.Debug && _cpu.Logger == null) // TODO, lets not do this on each frame. But lets refactor CoreComm/CoreComm first
|
||||
if (_tracer.Enabled)
|
||||
{
|
||||
_cpu.Logger = s => _tracer.Put(s);
|
||||
_cpu.TraceCallback = s => _tracer.Put(s);
|
||||
}
|
||||
else
|
||||
{
|
||||
_cpu.TraceCallback = null;
|
||||
}
|
||||
|
||||
// I eyeballed this speed
|
||||
for (int i = 0; i < 5; i++)
|
||||
{
|
||||
_onPressed = controller.IsPressed("ON");
|
||||
_onPressed = controller.IsPressed("ON");
|
||||
|
||||
// and this was derived from other emus
|
||||
_cpu.ExecuteCycles(10000);
|
||||
_cpu.Interrupt = true;
|
||||
if (_onPressed && ON_key_int_EN && !ON_key_int)
|
||||
{
|
||||
ON_key_int = true;
|
||||
_cpu.FlagI = true;
|
||||
}
|
||||
|
||||
// see: http://wikiti.brandonw.net/index.php?title=83:Ports:04
|
||||
// for timer interrupt frequency
|
||||
|
||||
// CPU frequency is 6MHz
|
||||
for (int i = 0; i < 100000; i++)
|
||||
{
|
||||
_cpu.ExecuteOne();
|
||||
|
||||
TIM_count++;
|
||||
if (TIM_count >= TIM_hit)
|
||||
{
|
||||
TIM_count = 0;
|
||||
|
||||
if (TIM_1_int_EN)
|
||||
{
|
||||
TIM_1_int = true;
|
||||
_cpu.FlagI = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Frame++;
|
||||
|
|
|
@ -7,55 +7,52 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
{
|
||||
public partial class TI83 : IStatable
|
||||
{
|
||||
private byte[] _stateBuffer;
|
||||
public bool BinarySaveStatesPreferred
|
||||
{
|
||||
get { return true; }
|
||||
}
|
||||
|
||||
public bool BinarySaveStatesPreferred => false;
|
||||
public void SaveStateText(TextWriter writer)
|
||||
{
|
||||
SyncState(new Serializer(writer));
|
||||
}
|
||||
|
||||
public void LoadStateText(TextReader reader)
|
||||
{
|
||||
SyncState(new Serializer(reader));
|
||||
}
|
||||
|
||||
public void SaveStateBinary(BinaryWriter bw)
|
||||
{
|
||||
SyncState(Serializer.CreateBinaryWriter(bw));
|
||||
SyncState(new Serializer(bw));
|
||||
}
|
||||
|
||||
public void LoadStateBinary(BinaryReader br)
|
||||
{
|
||||
SyncState(Serializer.CreateBinaryReader(br));
|
||||
}
|
||||
|
||||
public void SaveStateText(TextWriter tw)
|
||||
{
|
||||
SyncState(Serializer.CreateTextWriter(tw));
|
||||
}
|
||||
|
||||
public void LoadStateText(TextReader tr)
|
||||
{
|
||||
SyncState(Serializer.CreateTextReader(tr));
|
||||
SyncState(new Serializer(br));
|
||||
}
|
||||
|
||||
public byte[] SaveStateBinary()
|
||||
{
|
||||
if (_stateBuffer == null)
|
||||
{
|
||||
var stream = new MemoryStream();
|
||||
var writer = new BinaryWriter(stream);
|
||||
SaveStateBinary(writer);
|
||||
_stateBuffer = stream.ToArray();
|
||||
writer.Close();
|
||||
return _stateBuffer;
|
||||
}
|
||||
else
|
||||
{
|
||||
var stream = new MemoryStream(_stateBuffer);
|
||||
var writer = new BinaryWriter(stream);
|
||||
SaveStateBinary(writer);
|
||||
writer.Close();
|
||||
return _stateBuffer;
|
||||
}
|
||||
MemoryStream ms = new MemoryStream();
|
||||
BinaryWriter bw = new BinaryWriter(ms);
|
||||
SaveStateBinary(bw);
|
||||
bw.Flush();
|
||||
return ms.ToArray();
|
||||
}
|
||||
|
||||
private void SyncState(Serializer ser)
|
||||
{
|
||||
ser.BeginSection("TI83");
|
||||
byte[] core = null;
|
||||
if (ser.IsWriter)
|
||||
{
|
||||
var ms = new MemoryStream();
|
||||
ms.Close();
|
||||
core = ms.ToArray();
|
||||
}
|
||||
_cpu.SyncState(ser);
|
||||
|
||||
ser.BeginSection("TI83");
|
||||
ser.Sync("RAM", ref _ram, false);
|
||||
ser.Sync("romPageLow3Bits", ref _romPageLow3Bits);
|
||||
ser.Sync("romPageHighBit", ref _romPageHighBit);
|
||||
|
@ -72,6 +69,15 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
ser.Sync("Frame", ref _frame);
|
||||
ser.Sync("LagCount", ref _lagCount);
|
||||
ser.Sync("IsLag", ref _isLag);
|
||||
ser.Sync("ON_key_int", ref ON_key_int);
|
||||
ser.Sync("ON_key_int_EN", ref ON_key_int_EN);
|
||||
ser.Sync("TIM_1_int", ref TIM_1_int);
|
||||
ser.Sync("TIM_1_int_EN", ref TIM_1_int_EN);
|
||||
ser.Sync("TIM_frq", ref TIM_frq);
|
||||
ser.Sync("TIM_mult", ref TIM_mult);
|
||||
ser.Sync("TIM_count", ref TIM_count);
|
||||
ser.Sync("TIM_hit", ref TIM_hit);
|
||||
|
||||
ser.EndSection();
|
||||
|
||||
if (ser.IsReader)
|
||||
|
|
|
@ -2,7 +2,7 @@ using System;
|
|||
using System.Globalization;
|
||||
|
||||
using BizHawk.Emulation.Common;
|
||||
using BizHawk.Emulation.Cores.Components.Z80;
|
||||
using BizHawk.Emulation.Cores.Components.Z80A;
|
||||
|
||||
// http://www.ticalc.org/pub/text/calcinfo/
|
||||
namespace BizHawk.Emulation.Cores.Calculators
|
||||
|
@ -34,21 +34,13 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
_rom = rom;
|
||||
LinkPort = new TI83LinkPort(this);
|
||||
|
||||
// different calculators (different revisions?) have different initPC. we track this in the game database by rom hash
|
||||
// if( *(unsigned long *)(m_pRom + 0x6ce) == 0x04D3163E ) m_Regs.PC.W = 0x6ce; //KNOWN
|
||||
// else if( *(unsigned long *)(m_pRom + 0x6f6) == 0x04D3163E ) m_Regs.PC.W = 0x6f6; //UNKNOWN
|
||||
if (game["initPC"])
|
||||
{
|
||||
_startPC = ushort.Parse(game.OptionValue("initPC"), NumberStyles.HexNumber);
|
||||
}
|
||||
|
||||
HardReset();
|
||||
SetupMemoryDomains();
|
||||
|
||||
_tracer = new TraceBuffer { Header = _cpu.TraceHeader };
|
||||
|
||||
ser.Register<ITraceable>(_tracer);
|
||||
ser.Register<IDisassemblable>(new Disassembler());
|
||||
ser.Register<IDisassemblable>(_cpu);
|
||||
}
|
||||
|
||||
private readonly TraceBuffer _tracer;
|
||||
|
@ -57,8 +49,6 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
private readonly byte[] _rom;
|
||||
|
||||
// configuration
|
||||
private readonly ushort _startPC;
|
||||
|
||||
private IController _controller;
|
||||
|
||||
private byte[] _ram;
|
||||
|
@ -75,6 +65,10 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
private bool _cursorMoved;
|
||||
private int _frame;
|
||||
|
||||
public bool ON_key_int, ON_key_int_EN;
|
||||
public bool TIM_1_int, TIM_1_int_EN;
|
||||
public int TIM_frq, TIM_mult, TIM_count, TIM_hit;
|
||||
|
||||
// Link Cable
|
||||
public TI83LinkPort LinkPort { get; }
|
||||
|
||||
|
@ -151,7 +145,7 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
if (LinkActive)
|
||||
{
|
||||
// Prevent rom calls from disturbing link port activity
|
||||
if (LinkActive && _cpu.RegisterPC < 0x4000)
|
||||
if (LinkActive && _cpu.RegPC < 0x4000)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
@ -169,7 +163,60 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
_romPageLow3Bits = value & 0x7;
|
||||
break;
|
||||
case 3: // PORT_STATUS
|
||||
_maskOn = (byte)(value & 1);
|
||||
// controls ON key interrupts
|
||||
if ((value & 0x1) == 0)
|
||||
{
|
||||
ON_key_int = false;
|
||||
ON_key_int_EN = false;
|
||||
}
|
||||
else
|
||||
{
|
||||
ON_key_int_EN = true;
|
||||
}
|
||||
|
||||
// controls first timer interrupts
|
||||
if ((value & 0x2) == 0)
|
||||
{
|
||||
TIM_1_int = false;
|
||||
TIM_1_int_EN = false;
|
||||
}
|
||||
else
|
||||
{
|
||||
TIM_1_int_EN = true;
|
||||
}
|
||||
|
||||
// controls second timer, not yet implemented and unclear how to differentiate
|
||||
if ((value & 0x4) == 0)
|
||||
{
|
||||
}
|
||||
else
|
||||
{
|
||||
}
|
||||
|
||||
// controls low power mode, not yet implemeneted
|
||||
if ((value & 0x8) == 0)
|
||||
{
|
||||
}
|
||||
else
|
||||
{
|
||||
}
|
||||
break;
|
||||
case 4: // PORT_INTCTRL
|
||||
// controls ON key interrupts
|
||||
TIM_frq = value & 6;
|
||||
|
||||
TIM_mult = ((value & 0x10) == 0x10) ? 1800 : 1620;
|
||||
|
||||
TIM_hit = (int)Math.Floor((double)TIM_mult / (3 + TIM_frq * 2));
|
||||
|
||||
TIM_hit = (int)Math.Floor((double)6000000 / TIM_hit);
|
||||
|
||||
// Bit 0 is some form of memory mapping
|
||||
|
||||
// Bit 5 controls reset
|
||||
|
||||
// Bit 6-7 controls battery power compare (not implemented, will always return full power)
|
||||
|
||||
break;
|
||||
case 16: // PORT_DISPCTRL
|
||||
////Console.WriteLine("write PORT_DISPCTRL {0}",value);
|
||||
|
@ -198,22 +245,23 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
{
|
||||
// Console.WriteLine("read PORT_STATUS");
|
||||
// Bits:
|
||||
// 0 - Set if ON key is down and ON key is trapped
|
||||
// 0 - Set if ON key Interrupt generated
|
||||
// 1 - Update things (keyboard etc)
|
||||
// 2 - Unknown, but used
|
||||
// 3 - Set if ON key is up
|
||||
// 4-7 - Unknown
|
||||
////if (onPressed && maskOn) ret |= 1;
|
||||
////if (!onPressed) ret |= 0x8;
|
||||
return (byte)((_controller.IsPressed("ON") ? _maskOn : 8) | (LinkActive ? 0 : 2));
|
||||
|
||||
return (byte)((_controller.IsPressed("ON") ? 0 : 8) |
|
||||
(TIM_1_int ? 2 : 0) |
|
||||
(ON_key_int ? 1 : 0));
|
||||
}
|
||||
|
||||
case 4: // PORT_INTCTRL
|
||||
////Console.WriteLine("read PORT_INTCTRL");
|
||||
return 0xFF;
|
||||
// returns mirror of link port
|
||||
return (byte)((_romPageHighBit << 4) | (LinkState << 2) | LinkOutput);
|
||||
|
||||
case 16: // PORT_DISPCTRL
|
||||
////Console.WriteLine("read DISPCTRL");
|
||||
// Console.WriteLine("read DISPCTRL");
|
||||
break;
|
||||
|
||||
case 17: // PORT_DISPDATA
|
||||
|
@ -428,13 +476,13 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
|
||||
private void IRQCallback()
|
||||
{
|
||||
// Console.WriteLine("IRQ with vec {0} and cpu.InterruptMode {1}", cpu.RegisterI, cpu.InterruptMode);
|
||||
_cpu.Interrupt = false;
|
||||
//Console.WriteLine("IRQ with vec {0} and cpu.InterruptMode {1}", _cpu.Regs[_cpu.I], _cpu.InterruptMode);
|
||||
_cpu.FlagI = false;
|
||||
}
|
||||
|
||||
private void NMICallback()
|
||||
{
|
||||
Console.WriteLine("NMI");
|
||||
//Console.WriteLine("NMI");
|
||||
_cpu.NonMaskableInterrupt = false;
|
||||
}
|
||||
|
||||
|
@ -447,7 +495,7 @@ namespace BizHawk.Emulation.Cores.Calculators
|
|||
_ram[i] = 0xFF;
|
||||
}
|
||||
|
||||
_cpu.RegisterPC = _startPC;
|
||||
_cpu.RegPC = 0;
|
||||
|
||||
_cpu.IFF1 = false;
|
||||
_cpu.IFF2 = false;
|
||||
|
|
|
@ -38,7 +38,7 @@ namespace BizHawk.Emulation.Cores.ColecoVision
|
|||
|
||||
public byte Read(IController c, bool left_mode, int wheel)
|
||||
{
|
||||
return 0; // needs checking
|
||||
return 0x7F; // needs checking
|
||||
}
|
||||
|
||||
public ControllerDefinition Definition { get; }
|
||||
|
|
|
@ -12,36 +12,36 @@ namespace BizHawk.Emulation.Cores.ColecoVision
|
|||
{
|
||||
return new Dictionary<string, RegisterValue>
|
||||
{
|
||||
["A"] = _cpu.RegisterA,
|
||||
["AF"] = _cpu.RegisterAF,
|
||||
["B"] = _cpu.RegisterB,
|
||||
["BC"] = _cpu.RegisterBC,
|
||||
["C"] = _cpu.RegisterC,
|
||||
["D"] = _cpu.RegisterD,
|
||||
["DE"] = _cpu.RegisterDE,
|
||||
["E"] = _cpu.RegisterE,
|
||||
["F"] = _cpu.RegisterF,
|
||||
["H"] = _cpu.RegisterH,
|
||||
["HL"] = _cpu.RegisterHL,
|
||||
["I"] = _cpu.RegisterI,
|
||||
["IX"] = _cpu.RegisterIX,
|
||||
["IY"] = _cpu.RegisterIY,
|
||||
["L"] = _cpu.RegisterL,
|
||||
["PC"] = _cpu.RegisterPC,
|
||||
["R"] = _cpu.RegisterR,
|
||||
["Shadow AF"] = _cpu.RegisterShadowAF,
|
||||
["Shadow BC"] = _cpu.RegisterShadowBC,
|
||||
["Shadow DE"] = _cpu.RegisterShadowDE,
|
||||
["Shadow HL"] = _cpu.RegisterShadowHL,
|
||||
["SP"] = _cpu.RegisterSP,
|
||||
["Flag C"] = _cpu.RegisterF.Bit(0),
|
||||
["Flag N"] = _cpu.RegisterF.Bit(1),
|
||||
["Flag P/V"] = _cpu.RegisterF.Bit(2),
|
||||
["Flag 3rd"] = _cpu.RegisterF.Bit(3),
|
||||
["Flag H"] = _cpu.RegisterF.Bit(4),
|
||||
["Flag 5th"] = _cpu.RegisterF.Bit(5),
|
||||
["Flag Z"] = _cpu.RegisterF.Bit(6),
|
||||
["Flag S"] = _cpu.RegisterF.Bit(7)
|
||||
["A"] = _cpu.Regs[_cpu.A],
|
||||
["AF"] = _cpu.Regs[_cpu.F] + (_cpu.Regs[_cpu.A] << 8),
|
||||
["B"] = _cpu.Regs[_cpu.B],
|
||||
["BC"] = _cpu.Regs[_cpu.C] + (_cpu.Regs[_cpu.B] << 8),
|
||||
["C"] = _cpu.Regs[_cpu.C],
|
||||
["D"] = _cpu.Regs[_cpu.D],
|
||||
["DE"] = _cpu.Regs[_cpu.E] + (_cpu.Regs[_cpu.D] << 8),
|
||||
["E"] = _cpu.Regs[_cpu.E],
|
||||
["F"] = _cpu.Regs[_cpu.F],
|
||||
["H"] = _cpu.Regs[_cpu.H],
|
||||
["HL"] = _cpu.Regs[_cpu.L] + (_cpu.Regs[_cpu.H] << 8),
|
||||
["I"] = _cpu.Regs[_cpu.I],
|
||||
["IX"] = _cpu.Regs[_cpu.Ixl] + (_cpu.Regs[_cpu.Ixh] << 8),
|
||||
["IY"] = _cpu.Regs[_cpu.Iyl] + (_cpu.Regs[_cpu.Iyh] << 8),
|
||||
["L"] = _cpu.Regs[_cpu.L],
|
||||
["PC"] = _cpu.Regs[_cpu.PCl] + (_cpu.Regs[_cpu.PCh] << 8),
|
||||
["R"] = _cpu.Regs[_cpu.R],
|
||||
["Shadow AF"] = _cpu.Regs[_cpu.F_s] + (_cpu.Regs[_cpu.A_s] << 8),
|
||||
["Shadow BC"] = _cpu.Regs[_cpu.C_s] + (_cpu.Regs[_cpu.B_s] << 8),
|
||||
["Shadow DE"] = _cpu.Regs[_cpu.E_s] + (_cpu.Regs[_cpu.D_s] << 8),
|
||||
["Shadow HL"] = _cpu.Regs[_cpu.L_s] + (_cpu.Regs[_cpu.H_s] << 8),
|
||||
["SP"] = _cpu.Regs[_cpu.Iyl] + (_cpu.Regs[_cpu.Iyh] << 8),
|
||||
["Flag C"] = _cpu.FlagC,
|
||||
["Flag N"] = _cpu.FlagN,
|
||||
["Flag P/V"] = _cpu.FlagP,
|
||||
["Flag 3rd"] = _cpu.Flag3,
|
||||
["Flag H"] = _cpu.FlagH,
|
||||
["Flag 5th"] = _cpu.Flag5,
|
||||
["Flag Z"] = _cpu.FlagZ,
|
||||
["Flag S"] = _cpu.FlagS
|
||||
};
|
||||
}
|
||||
|
||||
|
@ -52,70 +52,82 @@ namespace BizHawk.Emulation.Cores.ColecoVision
|
|||
default:
|
||||
throw new InvalidOperationException();
|
||||
case "A":
|
||||
_cpu.RegisterA = (byte)value;
|
||||
_cpu.Regs[_cpu.A] = (ushort)value;
|
||||
break;
|
||||
case "AF":
|
||||
_cpu.RegisterAF = (byte)value;
|
||||
_cpu.Regs[_cpu.F] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.A] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "B":
|
||||
_cpu.RegisterB = (byte)value;
|
||||
_cpu.Regs[_cpu.B] = (ushort)value;
|
||||
break;
|
||||
case "BC":
|
||||
_cpu.RegisterBC = (byte)value;
|
||||
_cpu.Regs[_cpu.C] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.B] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "C":
|
||||
_cpu.RegisterC = (byte)value;
|
||||
_cpu.Regs[_cpu.C] = (ushort)value;
|
||||
break;
|
||||
case "D":
|
||||
_cpu.RegisterD = (byte)value;
|
||||
_cpu.Regs[_cpu.D] = (ushort)value;
|
||||
break;
|
||||
case "DE":
|
||||
_cpu.RegisterDE = (byte)value;
|
||||
_cpu.Regs[_cpu.E] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.D] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "E":
|
||||
_cpu.RegisterE = (byte)value;
|
||||
_cpu.Regs[_cpu.E] = (ushort)value;
|
||||
break;
|
||||
case "F":
|
||||
_cpu.RegisterF = (byte)value;
|
||||
_cpu.Regs[_cpu.F] = (ushort)value;
|
||||
break;
|
||||
case "H":
|
||||
_cpu.RegisterH = (byte)value;
|
||||
_cpu.Regs[_cpu.H] = (ushort)value;
|
||||
break;
|
||||
case "HL":
|
||||
_cpu.RegisterHL = (byte)value;
|
||||
_cpu.Regs[_cpu.L] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.H] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "I":
|
||||
_cpu.RegisterI = (byte)value;
|
||||
_cpu.Regs[_cpu.I] = (ushort)value;
|
||||
break;
|
||||
case "IX":
|
||||
_cpu.RegisterIX = (byte)value;
|
||||
_cpu.Regs[_cpu.Ixl] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.Ixh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "IY":
|
||||
_cpu.RegisterIY = (byte)value;
|
||||
_cpu.Regs[_cpu.Iyl] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.Iyh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "L":
|
||||
_cpu.RegisterL = (byte)value;
|
||||
_cpu.Regs[_cpu.L] = (ushort)value;
|
||||
break;
|
||||
case "PC":
|
||||
_cpu.RegisterPC = (ushort)value;
|
||||
_cpu.Regs[_cpu.PCl] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.PCh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "R":
|
||||
_cpu.RegisterR = (byte)value;
|
||||
_cpu.Regs[_cpu.R] = (ushort)value;
|
||||
break;
|
||||
case "Shadow AF":
|
||||
_cpu.RegisterShadowAF = (byte)value;
|
||||
_cpu.Regs[_cpu.F_s] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.A_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "Shadow BC":
|
||||
_cpu.RegisterShadowBC = (byte)value;
|
||||
_cpu.Regs[_cpu.C_s] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.B_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "Shadow DE":
|
||||
_cpu.RegisterShadowDE = (byte)value;
|
||||
_cpu.Regs[_cpu.E_s] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.D_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "Shadow HL":
|
||||
_cpu.RegisterShadowHL = (byte)value;
|
||||
_cpu.Regs[_cpu.L_s] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.H_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "SP":
|
||||
_cpu.RegisterSP = (byte)value;
|
||||
_cpu.Regs[_cpu.SPl] = (ushort)(value & 0xFF);
|
||||
_cpu.Regs[_cpu.SPh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -24,16 +24,18 @@ namespace BizHawk.Emulation.Cores.ColecoVision
|
|||
SoftReset();
|
||||
}
|
||||
|
||||
_cpu.Debug = _tracer.Enabled;
|
||||
_frame++;
|
||||
_isLag = true;
|
||||
PSG.BeginFrame(_cpu.TotalExecutedCycles);
|
||||
|
||||
if (_cpu.Debug && _cpu.Logger == null) // TODO, lets not do this on each frame. But lets refactor CoreComm/CoreComm first
|
||||
if (_tracer.Enabled)
|
||||
{
|
||||
_cpu.Logger = (s) => _tracer.Put(s);
|
||||
_cpu.TraceCallback = s => _tracer.Put(s);
|
||||
}
|
||||
else
|
||||
{
|
||||
_cpu.TraceCallback = null;
|
||||
}
|
||||
|
||||
byte tempRet1 = ControllerDeck.ReadPort1(controller, true, true);
|
||||
byte tempRet2 = ControllerDeck.ReadPort2(controller, true, true);
|
||||
|
||||
|
|
|
@ -7,53 +7,52 @@ namespace BizHawk.Emulation.Cores.ColecoVision
|
|||
{
|
||||
public partial class ColecoVision : IStatable
|
||||
{
|
||||
public bool BinarySaveStatesPreferred => false;
|
||||
public bool BinarySaveStatesPreferred
|
||||
{
|
||||
get { return true; }
|
||||
}
|
||||
|
||||
public void SaveStateText(TextWriter writer)
|
||||
{
|
||||
SyncState(new Serializer(writer));
|
||||
}
|
||||
|
||||
public void LoadStateText(TextReader reader)
|
||||
{
|
||||
SyncState(new Serializer(reader));
|
||||
}
|
||||
|
||||
public void SaveStateBinary(BinaryWriter bw)
|
||||
{
|
||||
SyncState(Serializer.CreateBinaryWriter(bw));
|
||||
SyncState(new Serializer(bw));
|
||||
}
|
||||
|
||||
public void LoadStateBinary(BinaryReader br)
|
||||
{
|
||||
SyncState(Serializer.CreateBinaryReader(br));
|
||||
}
|
||||
|
||||
public void SaveStateText(TextWriter tw)
|
||||
{
|
||||
SyncState(Serializer.CreateTextWriter(tw));
|
||||
}
|
||||
|
||||
public void LoadStateText(TextReader tr)
|
||||
{
|
||||
SyncState(Serializer.CreateTextReader(tr));
|
||||
SyncState(new Serializer(br));
|
||||
}
|
||||
|
||||
public byte[] SaveStateBinary()
|
||||
{
|
||||
if (_stateBuffer == null)
|
||||
{
|
||||
var stream = new MemoryStream();
|
||||
var writer = new BinaryWriter(stream);
|
||||
SaveStateBinary(writer);
|
||||
_stateBuffer = stream.ToArray();
|
||||
writer.Close();
|
||||
return _stateBuffer;
|
||||
}
|
||||
else
|
||||
{
|
||||
var stream = new MemoryStream(_stateBuffer);
|
||||
var writer = new BinaryWriter(stream);
|
||||
SaveStateBinary(writer);
|
||||
writer.Close();
|
||||
return _stateBuffer;
|
||||
}
|
||||
MemoryStream ms = new MemoryStream();
|
||||
BinaryWriter bw = new BinaryWriter(ms);
|
||||
SaveStateBinary(bw);
|
||||
bw.Flush();
|
||||
return ms.ToArray();
|
||||
}
|
||||
|
||||
private void SyncState(Serializer ser)
|
||||
{
|
||||
ser.BeginSection("Coleco");
|
||||
byte[] core = null;
|
||||
if (ser.IsWriter)
|
||||
{
|
||||
var ms = new MemoryStream();
|
||||
ms.Close();
|
||||
core = ms.ToArray();
|
||||
}
|
||||
_cpu.SyncState(ser);
|
||||
|
||||
ser.BeginSection("Coleco");
|
||||
_vdp.SyncState(ser);
|
||||
PSG.SyncState(ser);
|
||||
ser.Sync("RAM", ref _ram, false);
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
using BizHawk.Emulation.Common;
|
||||
using BizHawk.Emulation.Cores.Components;
|
||||
using BizHawk.Emulation.Cores.Components.Z80;
|
||||
using BizHawk.Emulation.Cores.Components.Z80A;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.ColecoVision
|
||||
{
|
||||
|
@ -53,7 +53,7 @@ namespace BizHawk.Emulation.Cores.ColecoVision
|
|||
SetupMemoryDomains();
|
||||
|
||||
_tracer.Header = _cpu.TraceHeader;
|
||||
ser.Register<IDisassemblable>(new Disassembler());
|
||||
ser.Register<IDisassemblable>(_cpu);
|
||||
ser.Register<ITraceable>(_tracer);
|
||||
}
|
||||
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
|
||||
using BizHawk.Common;
|
||||
using BizHawk.Emulation.Common;
|
||||
using BizHawk.Emulation.Cores.Components.Z80;
|
||||
using BizHawk.Emulation.Cores.Components.Z80A;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.ColecoVision
|
||||
{
|
||||
|
@ -54,14 +54,17 @@ namespace BizHawk.Emulation.Cores.ColecoVision
|
|||
Cpu.NonMaskableInterrupt = true;
|
||||
}
|
||||
|
||||
Cpu.ExecuteCycles(228);
|
||||
|
||||
Cpu.Interrupt = false;
|
||||
for (int i = 0; i < 228; i++)
|
||||
{
|
||||
Cpu.ExecuteOne();
|
||||
}
|
||||
|
||||
Cpu.FlagI = false;
|
||||
if (Int_pending && scanLine==50)
|
||||
{
|
||||
if (EnableInterrupts)
|
||||
{
|
||||
Cpu.Interrupt = true;
|
||||
Cpu.FlagI = true;
|
||||
Int_pending = false;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -255,19 +255,29 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES
|
|||
case 0x0020:
|
||||
timerlatch &= 0xff00;
|
||||
timerlatch |= value;
|
||||
//timerirq = false;
|
||||
break;
|
||||
case 0x0021:
|
||||
timerlatch &= 0x00ff;
|
||||
timerlatch |= value << 8;
|
||||
//timerirq = false;
|
||||
break;
|
||||
case 0x0022:
|
||||
timerreg = (byte)(value & 3);
|
||||
timervalue = timerlatch;
|
||||
if (diskenable)
|
||||
{
|
||||
timerreg = (byte)(value & 3);
|
||||
if ((value & 0x02) == 0x02)
|
||||
{
|
||||
timervalue = timerlatch;
|
||||
}
|
||||
else
|
||||
{
|
||||
_timerirq = false;
|
||||
}
|
||||
}
|
||||
|
||||
break;
|
||||
case 0x0023:
|
||||
diskenable = (value & 1) != 0;
|
||||
if (!diskenable) { _timerirq = false; }
|
||||
soundenable = (value & 2) != 0;
|
||||
break;
|
||||
case 0x0024:
|
||||
|
@ -344,7 +354,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES
|
|||
|
||||
public override void ClockCPU()
|
||||
{
|
||||
if ((timerreg & 2) != 0)// && timervalue > 0)
|
||||
if ((timerreg & 2) != 0 && diskenable)
|
||||
{
|
||||
if (timervalue!=0)
|
||||
{
|
||||
|
@ -352,22 +362,12 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES
|
|||
}
|
||||
if (timervalue == 0)
|
||||
{
|
||||
|
||||
/*
|
||||
if ((timerreg & 1) != 0)
|
||||
{
|
||||
timervalue = timerlatch;
|
||||
//timervalue = 0xFFFF;
|
||||
}
|
||||
else
|
||||
{
|
||||
timerreg &= unchecked((byte)~2);
|
||||
timervalue = 0;
|
||||
timerlatch = 0;
|
||||
}
|
||||
*/
|
||||
timervalue = timerlatch;
|
||||
timerirq = true;
|
||||
if ((timerreg & 1) == 0)
|
||||
{
|
||||
timerreg -= 2;
|
||||
}
|
||||
}
|
||||
}
|
||||
audio.Clock();
|
||||
|
|
|
@ -67,14 +67,17 @@ namespace BizHawk.Emulation.Cores.Nintendo.NES
|
|||
byte[] fileheader = br.ReadBytes(16);
|
||||
if (fileheader[0] != 0x03)
|
||||
{
|
||||
throw new Exception("Corrupt FDS block 3");
|
||||
// Instead of exceptions, display strong warnings
|
||||
Console.WriteLine("WARNING: INVALID FILE, BLOCK 3 ERROR");
|
||||
//throw new Exception("Corrupt FDS block 3");
|
||||
}
|
||||
int filesize = fileheader[13] + fileheader[14] * 256;
|
||||
|
||||
byte[] file = br.ReadBytes(filesize + 1);
|
||||
if (file[0] != 0x04)
|
||||
{
|
||||
throw new Exception("Corrupt FDS block 4");
|
||||
Console.WriteLine("WARNING: INVALID FILE, BLOCK 4 ERROR");
|
||||
//throw new Exception("Corrupt FDS block 4");
|
||||
}
|
||||
|
||||
WriteBlock(ret, fileheader, 122);
|
||||
|
|
|
@ -0,0 +1,233 @@
|
|||
using System;
|
||||
using BizHawk.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
||||
{
|
||||
class EEPROM93c46
|
||||
{
|
||||
enum EEPROMWriteMode
|
||||
{
|
||||
Instruction,
|
||||
WriteData,
|
||||
WriteAll,
|
||||
Read
|
||||
}
|
||||
|
||||
enum EEPROMReadMode
|
||||
{
|
||||
ReadNew,
|
||||
ReadOld,
|
||||
Hold
|
||||
}
|
||||
|
||||
[Flags]
|
||||
enum EEPROMFlags : byte
|
||||
{
|
||||
Ready = 1,
|
||||
Clock = 2,
|
||||
ChipSelect = 4
|
||||
}
|
||||
|
||||
ushort Address = 0;
|
||||
ushort Value = 0;
|
||||
int BitsWritten = 0;
|
||||
int BitsRead = 0;
|
||||
bool WriteEnable = false;
|
||||
EEPROMWriteMode WriteMode = EEPROMWriteMode.Instruction;
|
||||
EEPROMReadMode ReadMode = EEPROMReadMode.Hold;
|
||||
EEPROMFlags Flags = 0;
|
||||
|
||||
public byte Read(byte[] saveRAM)
|
||||
{
|
||||
switch (ReadMode)
|
||||
{
|
||||
case EEPROMReadMode.ReadNew:
|
||||
// A new value clocking out
|
||||
|
||||
ReadMode = EEPROMReadMode.ReadOld;
|
||||
byte ret = Read(saveRAM);
|
||||
|
||||
if (++BitsRead == 8)
|
||||
{
|
||||
// Increment address
|
||||
BitsRead = 0;
|
||||
++Address;
|
||||
|
||||
if(Address % 2 == 0)
|
||||
{
|
||||
WriteMode = EEPROMWriteMode.Instruction;
|
||||
ReadMode = EEPROMReadMode.Hold;
|
||||
BitsWritten = 0;
|
||||
Value = 0;
|
||||
}
|
||||
}
|
||||
|
||||
return ret;
|
||||
case EEPROMReadMode.ReadOld:
|
||||
// repeat old value
|
||||
|
||||
byte bit = (byte)((saveRAM[Address % saveRAM.Length] >> (7 - BitsRead)) & 1);
|
||||
|
||||
return (byte)((byte)(Flags | EEPROMFlags.Clock) | bit);
|
||||
default:
|
||||
// ready/busy flag is always ready in this emulation
|
||||
return (byte)(Flags | EEPROMFlags.Clock | EEPROMFlags.Ready);
|
||||
}
|
||||
}
|
||||
|
||||
public void Write(byte bit, byte[] saveRAM)
|
||||
{
|
||||
// new instruction?
|
||||
if ((bit & 4) == 0)
|
||||
{
|
||||
WriteMode = EEPROMWriteMode.Instruction;
|
||||
ReadMode = EEPROMReadMode.Hold;
|
||||
BitsWritten = 0;
|
||||
Value = 0;
|
||||
|
||||
Flags = (EEPROMFlags)bit & ~EEPROMFlags.Ready;
|
||||
return;
|
||||
}
|
||||
|
||||
// clock low to high?
|
||||
if ((bit & (byte)EEPROMFlags.Clock) != 0 && (Flags & EEPROMFlags.Clock) == 0)
|
||||
{
|
||||
// all modes shift in a larger value
|
||||
Value = (ushort)((Value << 1) | (bit & 1));
|
||||
++BitsWritten;
|
||||
|
||||
switch (WriteMode)
|
||||
{
|
||||
case EEPROMWriteMode.Instruction:
|
||||
// Process opcode including start bit
|
||||
|
||||
// check start bit
|
||||
if ((Value & 0x100) == 0)
|
||||
return;
|
||||
|
||||
byte op = (byte)Value;
|
||||
Value = 0;
|
||||
BitsWritten = 0;
|
||||
|
||||
switch (op & 0xC0)
|
||||
{
|
||||
case 0x00:
|
||||
// non-addressed commands
|
||||
switch (op & 0xF0)
|
||||
{
|
||||
case 0x00:
|
||||
// EWDS: write disable
|
||||
WriteEnable = false;
|
||||
return;
|
||||
case 0x10:
|
||||
// WRAL: write to all addresses (silly)
|
||||
WriteMode = EEPROMWriteMode.WriteAll;
|
||||
ReadMode = EEPROMReadMode.Hold;
|
||||
return;
|
||||
case 0x20:
|
||||
// ERAL: erase all addresses
|
||||
if (WriteEnable)
|
||||
{
|
||||
for (int i = 0; i < saveRAM.Length; ++i)
|
||||
{
|
||||
saveRAM[i] = 0xFF;
|
||||
}
|
||||
}
|
||||
ReadMode = EEPROMReadMode.Hold;
|
||||
return;
|
||||
case 0x30:
|
||||
// EWEN: write enable
|
||||
WriteEnable = true;
|
||||
return;
|
||||
default:
|
||||
// impossible
|
||||
return;
|
||||
}
|
||||
case 0x40:
|
||||
// WRITE
|
||||
Address = (ushort)((op & 0x3F) << 1);
|
||||
WriteMode = EEPROMWriteMode.WriteData;
|
||||
ReadMode = EEPROMReadMode.Hold;
|
||||
return;
|
||||
case 0x80:
|
||||
// READ
|
||||
Address = (ushort)((op & 0x3F) << 1);
|
||||
ReadMode = EEPROMReadMode.Hold;
|
||||
WriteMode = EEPROMWriteMode.Read;
|
||||
BitsRead = 0;
|
||||
return;
|
||||
case 0xC0:
|
||||
// ERASE
|
||||
Address = (ushort)((op & 0x3F) << 1);
|
||||
if (WriteEnable)
|
||||
{
|
||||
saveRAM[Address % saveRAM.Length] = 0xFF;
|
||||
saveRAM[(Address + 1) % saveRAM.Length] = 0xFF;
|
||||
}
|
||||
ReadMode = EEPROMReadMode.Hold;
|
||||
return;
|
||||
default:
|
||||
// impossible
|
||||
return;
|
||||
}
|
||||
case EEPROMWriteMode.WriteData:
|
||||
// Write bits
|
||||
|
||||
if (BitsWritten < 16)
|
||||
return;
|
||||
|
||||
if (WriteEnable)
|
||||
{
|
||||
saveRAM[Address % saveRAM.Length] = (byte)(Value >> 8);
|
||||
saveRAM[(Address + 1) % saveRAM.Length] = (byte)Value;
|
||||
}
|
||||
WriteMode = EEPROMWriteMode.Instruction;
|
||||
|
||||
Value = 0;
|
||||
BitsWritten = 0;
|
||||
return;
|
||||
case EEPROMWriteMode.WriteAll:
|
||||
// write to ALL addresses
|
||||
|
||||
if (BitsWritten < 16)
|
||||
return;
|
||||
|
||||
Value = 0;
|
||||
BitsWritten = 0;
|
||||
|
||||
if (WriteEnable)
|
||||
{
|
||||
for (int i = 0; i < saveRAM.Length; i += 2)
|
||||
{
|
||||
saveRAM[i % saveRAM.Length] = (byte)Value;
|
||||
saveRAM[(i + 1) % saveRAM.Length] = (byte)(Value >> 8);
|
||||
}
|
||||
}
|
||||
WriteMode = EEPROMWriteMode.Instruction;
|
||||
return;
|
||||
case EEPROMWriteMode.Read:
|
||||
// Clock a new value out
|
||||
ReadMode = EEPROMReadMode.ReadNew;
|
||||
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
Flags = (EEPROMFlags)bit & ~EEPROMFlags.Ready;
|
||||
}
|
||||
|
||||
public void SyncState(Serializer ser)
|
||||
{
|
||||
ser.BeginSection("93c46");
|
||||
ser.Sync("Address", ref Address);
|
||||
ser.Sync("Value", ref Value);
|
||||
ser.Sync("BitsWritten", ref BitsWritten);
|
||||
ser.Sync("BitsRead", ref BitsRead);
|
||||
ser.Sync("WriteEnable", ref WriteEnable);
|
||||
ser.SyncEnum("WriteMode", ref WriteMode);
|
||||
ser.SyncEnum("ReadMode", ref ReadMode);
|
||||
ser.SyncEnum("Flags", ref Flags);
|
||||
ser.EndSection();
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,136 @@
|
|||
using System;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
||||
{
|
||||
public partial class SMS
|
||||
{
|
||||
// The 93c46-connected mapper is assumed to be equivalent to the Sega mapper except for $8000-..
|
||||
// The Sega memory mapper layout looks like so:
|
||||
// $0000-$03FF - ROM (unpaged)
|
||||
// $0400-$3FFF - ROM mapper slot 0
|
||||
// $4000-$7FFF - ROM mapper slot 1
|
||||
// $8000-$BFFF - ROM mapper slot 2 - OR - EEPROM
|
||||
// $C000-$DFFF - System RAM
|
||||
// $E000-$FFFF - System RAM (mirror)
|
||||
// $FFFC - SaveRAM mapper control
|
||||
// $FFFD - Mapper slot 0 control
|
||||
// $FFFE - Mapper slot 1 control
|
||||
// $FFFF - Mapper slot 2 control
|
||||
|
||||
EEPROM93c46 EEPROM;
|
||||
|
||||
byte ReadMemoryEEPROM(ushort address)
|
||||
{
|
||||
byte ret = 0xFF;
|
||||
|
||||
if (address < 0xC000)
|
||||
{
|
||||
if ((Port3E & 0x48) == 0x48) // cart and bios disabled, return empty bus
|
||||
ret = 0xFF;
|
||||
else if (BiosMapped && BiosRom != null)
|
||||
ret = BiosRom[address & 0x1FFF];
|
||||
else if (address < 1024)
|
||||
ret = RomData[address];
|
||||
else if (address < 0x4000)
|
||||
ret = RomData[(RomBank0 * BankSize) + address];
|
||||
else if (address < 0x8000)
|
||||
ret = RomData[(RomBank1 * BankSize) + (address & BankSizeMask)];
|
||||
else
|
||||
{
|
||||
switch (SaveRamBank)
|
||||
{
|
||||
case 0: ret = RomData[(RomBank2 * BankSize) + (address & BankSizeMask)]; break;
|
||||
case 1: if (SaveRAM != null && EEPROM != null) ret = EEPROM.Read(SaveRAM); break;
|
||||
default:
|
||||
ret = SystemRam[address & RamSizeMask];
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
ret = SystemRam[address & RamSizeMask];
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
CDLog_MapResults MapMemoryEEPROM(ushort address, bool write)
|
||||
{
|
||||
if (address < 0xC000)
|
||||
{
|
||||
if ((Port3E & 0x48) == 0x48) // cart and bios disabled, return empty bus
|
||||
return new CDLog_MapResults();
|
||||
else if (BiosMapped && BiosRom != null)
|
||||
return new CDLog_MapResults(); //bios tracking of CDL is not supported
|
||||
else if (address < 1024)
|
||||
return new CDLog_MapResults() { Type = CDLog_AddrType.ROM, Address = address };
|
||||
else if (address < 0x4000)
|
||||
return new CDLog_MapResults() { Type = CDLog_AddrType.ROM, Address = (RomBank0 * BankSize) + address };
|
||||
else if (address < 0x8000)
|
||||
return new CDLog_MapResults() { Type = CDLog_AddrType.ROM, Address = (RomBank1 * BankSize) + (address & BankSizeMask) };
|
||||
else
|
||||
{
|
||||
switch (SaveRamBank)
|
||||
{
|
||||
case 0: return new CDLog_MapResults() { Type = CDLog_AddrType.ROM, Address = (RomBank2 * BankSize) + (address & BankSizeMask) };
|
||||
case 1: return new CDLog_MapResults(); // a serial IO port
|
||||
case 2: return new CDLog_MapResults(); // a serial IO port
|
||||
default:
|
||||
return new CDLog_MapResults() { Type = CDLog_AddrType.MainRAM, Address = address & RamSizeMask };
|
||||
}
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
return new CDLog_MapResults() { Type = CDLog_AddrType.MainRAM, Address = address & RamSizeMask };
|
||||
}
|
||||
}
|
||||
|
||||
void WriteMemoryEEPROM(ushort address, byte value)
|
||||
{
|
||||
if (address >= 0xC000)
|
||||
SystemRam[address & RamSizeMask] = value;
|
||||
|
||||
else if (address >= 0x8000)
|
||||
{
|
||||
if (SaveRAM != null)
|
||||
{
|
||||
SaveRamModified = true;
|
||||
EEPROM.Write(value, SaveRAM);
|
||||
return;
|
||||
}
|
||||
else System.Console.WriteLine("Game attempt to use SRAM but SRAM not present");
|
||||
}
|
||||
|
||||
if (address >= 0xFFFC)
|
||||
{
|
||||
if (address == 0xFFFC)
|
||||
{
|
||||
if ((value & 8) != 0)
|
||||
SaveRamBank = (byte)((value & 4) == 0 ? 1 : 2); // SaveRAM selected
|
||||
else
|
||||
SaveRamBank = 0; // ROM bank selected
|
||||
|
||||
}
|
||||
else if (address == 0xFFFD) RomBank0 = (byte)(value % RomBanks);
|
||||
else if (address == 0xFFFE) RomBank1 = (byte)(value % RomBanks);
|
||||
else if (address == 0xFFFF) RomBank2 = (byte)(value % RomBanks);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
void InitEEPROMMapper()
|
||||
{
|
||||
ReadMemory = ReadMemoryEEPROM;
|
||||
WriteMemory = WriteMemoryEEPROM;
|
||||
MapMemory = MapMemoryEEPROM;
|
||||
WriteMemoryEEPROM(0xFFFC, 0);
|
||||
WriteMemoryEEPROM(0xFFFD, 0);
|
||||
WriteMemoryEEPROM(0xFFFE, 1);
|
||||
WriteMemoryEEPROM(0xFFFF, 2);
|
||||
|
||||
EEPROM = new EEPROM93c46();
|
||||
}
|
||||
}
|
||||
}
|
|
@ -12,36 +12,36 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
{
|
||||
return new Dictionary<string, RegisterValue>
|
||||
{
|
||||
["A"] = Cpu.RegisterA,
|
||||
["AF"] = Cpu.RegisterAF,
|
||||
["B"] = Cpu.RegisterB,
|
||||
["BC"] = Cpu.RegisterBC,
|
||||
["C"] = Cpu.RegisterC,
|
||||
["D"] = Cpu.RegisterD,
|
||||
["DE"] = Cpu.RegisterDE,
|
||||
["E"] = Cpu.RegisterE,
|
||||
["F"] = Cpu.RegisterF,
|
||||
["H"] = Cpu.RegisterH,
|
||||
["HL"] = Cpu.RegisterHL,
|
||||
["I"] = Cpu.RegisterI,
|
||||
["IX"] = Cpu.RegisterIX,
|
||||
["IY"] = Cpu.RegisterIY,
|
||||
["L"] = Cpu.RegisterL,
|
||||
["PC"] = Cpu.RegisterPC,
|
||||
["R"] = Cpu.RegisterR,
|
||||
["Shadow AF"] = Cpu.RegisterShadowAF,
|
||||
["Shadow BC"] = Cpu.RegisterShadowBC,
|
||||
["Shadow DE"] = Cpu.RegisterShadowDE,
|
||||
["Shadow HL"] = Cpu.RegisterShadowHL,
|
||||
["SP"] = Cpu.RegisterSP,
|
||||
["Flag C"] = Cpu.RegisterF.Bit(0),
|
||||
["Flag N"] = Cpu.RegisterF.Bit(1),
|
||||
["Flag P/V"] = Cpu.RegisterF.Bit(2),
|
||||
["Flag 3rd"] = Cpu.RegisterF.Bit(3),
|
||||
["Flag H"] = Cpu.RegisterF.Bit(4),
|
||||
["Flag 5th"] = Cpu.RegisterF.Bit(5),
|
||||
["Flag Z"] = Cpu.RegisterF.Bit(6),
|
||||
["Flag S"] = Cpu.RegisterF.Bit(7)
|
||||
["A"] = Cpu.Regs[Cpu.A],
|
||||
["AF"] = Cpu.Regs[Cpu.F] + (Cpu.Regs[Cpu.A] << 8),
|
||||
["B"] = Cpu.Regs[Cpu.B],
|
||||
["BC"] = Cpu.Regs[Cpu.C] + (Cpu.Regs[Cpu.B] << 8),
|
||||
["C"] = Cpu.Regs[Cpu.C],
|
||||
["D"] = Cpu.Regs[Cpu.D],
|
||||
["DE"] = Cpu.Regs[Cpu.E] + (Cpu.Regs[Cpu.D] << 8),
|
||||
["E"] = Cpu.Regs[Cpu.E],
|
||||
["F"] = Cpu.Regs[Cpu.F],
|
||||
["H"] = Cpu.Regs[Cpu.H],
|
||||
["HL"] = Cpu.Regs[Cpu.L] + (Cpu.Regs[Cpu.H] << 8),
|
||||
["I"] = Cpu.Regs[Cpu.I],
|
||||
["IX"] = Cpu.Regs[Cpu.Ixl] + (Cpu.Regs[Cpu.Ixh] << 8),
|
||||
["IY"] = Cpu.Regs[Cpu.Iyl] + (Cpu.Regs[Cpu.Iyh] << 8),
|
||||
["L"] = Cpu.Regs[Cpu.L],
|
||||
["PC"] = Cpu.Regs[Cpu.PCl] + (Cpu.Regs[Cpu.PCh] << 8),
|
||||
["R"] = Cpu.Regs[Cpu.R],
|
||||
["Shadow AF"] = Cpu.Regs[Cpu.F_s] + (Cpu.Regs[Cpu.A_s] << 8),
|
||||
["Shadow BC"] = Cpu.Regs[Cpu.C_s] + (Cpu.Regs[Cpu.B_s] << 8),
|
||||
["Shadow DE"] = Cpu.Regs[Cpu.E_s] + (Cpu.Regs[Cpu.D_s] << 8),
|
||||
["Shadow HL"] = Cpu.Regs[Cpu.L_s] + (Cpu.Regs[Cpu.H_s] << 8),
|
||||
["SP"] = Cpu.Regs[Cpu.Iyl] + (Cpu.Regs[Cpu.Iyh] << 8),
|
||||
["Flag C"] = Cpu.FlagC,
|
||||
["Flag N"] = Cpu.FlagN,
|
||||
["Flag P/V"] = Cpu.FlagP,
|
||||
["Flag 3rd"] = Cpu.Flag3,
|
||||
["Flag H"] = Cpu.FlagH,
|
||||
["Flag 5th"] = Cpu.Flag5,
|
||||
["Flag Z"] = Cpu.FlagZ,
|
||||
["Flag S"] = Cpu.FlagS
|
||||
};
|
||||
}
|
||||
|
||||
|
@ -52,70 +52,82 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
default:
|
||||
throw new InvalidOperationException();
|
||||
case "A":
|
||||
Cpu.RegisterA = (byte)value;
|
||||
Cpu.Regs[Cpu.A] = (ushort)value;
|
||||
break;
|
||||
case "AF":
|
||||
Cpu.RegisterAF = (byte)value;
|
||||
Cpu.Regs[Cpu.F] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.A] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "B":
|
||||
Cpu.RegisterB = (byte)value;
|
||||
Cpu.Regs[Cpu.B] = (ushort)value;
|
||||
break;
|
||||
case "BC":
|
||||
Cpu.RegisterBC = (byte)value;
|
||||
Cpu.Regs[Cpu.C] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.B] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "C":
|
||||
Cpu.RegisterC = (byte)value;
|
||||
Cpu.Regs[Cpu.C] = (ushort)value;
|
||||
break;
|
||||
case "D":
|
||||
Cpu.RegisterD = (byte)value;
|
||||
Cpu.Regs[Cpu.D] = (ushort)value;
|
||||
break;
|
||||
case "DE":
|
||||
Cpu.RegisterDE = (byte)value;
|
||||
Cpu.Regs[Cpu.E] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.D] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "E":
|
||||
Cpu.RegisterE = (byte)value;
|
||||
Cpu.Regs[Cpu.E] = (ushort)value;
|
||||
break;
|
||||
case "F":
|
||||
Cpu.RegisterF = (byte)value;
|
||||
Cpu.Regs[Cpu.F] = (ushort)value;
|
||||
break;
|
||||
case "H":
|
||||
Cpu.RegisterH = (byte)value;
|
||||
Cpu.Regs[Cpu.H] = (ushort)value;
|
||||
break;
|
||||
case "HL":
|
||||
Cpu.RegisterHL = (byte)value;
|
||||
Cpu.Regs[Cpu.L] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.H] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "I":
|
||||
Cpu.RegisterI = (byte)value;
|
||||
Cpu.Regs[Cpu.I] = (ushort)value;
|
||||
break;
|
||||
case "IX":
|
||||
Cpu.RegisterIX = (byte)value;
|
||||
Cpu.Regs[Cpu.Ixl] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.Ixh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "IY":
|
||||
Cpu.RegisterIY = (byte)value;
|
||||
Cpu.Regs[Cpu.Iyl] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.Iyh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "L":
|
||||
Cpu.RegisterL = (byte)value;
|
||||
Cpu.Regs[Cpu.L] = (ushort)value;
|
||||
break;
|
||||
case "PC":
|
||||
Cpu.RegisterPC = (ushort)value;
|
||||
Cpu.Regs[Cpu.PCl] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.PCh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "R":
|
||||
Cpu.RegisterR = (byte)value;
|
||||
Cpu.Regs[Cpu.R] = (ushort)value;
|
||||
break;
|
||||
case "Shadow AF":
|
||||
Cpu.RegisterShadowAF = (byte)value;
|
||||
Cpu.Regs[Cpu.F_s] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.A_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "Shadow BC":
|
||||
Cpu.RegisterShadowBC = (byte)value;
|
||||
Cpu.Regs[Cpu.C_s] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.B_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "Shadow DE":
|
||||
Cpu.RegisterShadowDE = (byte)value;
|
||||
Cpu.Regs[Cpu.E_s] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.D_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "Shadow HL":
|
||||
Cpu.RegisterShadowHL = (byte)value;
|
||||
Cpu.Regs[Cpu.L_s] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.H_s] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
case "SP":
|
||||
Cpu.RegisterSP = (byte)value;
|
||||
Cpu.Regs[Cpu.SPl] = (ushort)(value & 0xFF);
|
||||
Cpu.Regs[Cpu.SPh] = (ushort)(value & 0xFF00);
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,95 +1,99 @@
|
|||
using BizHawk.Emulation.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
||||
{
|
||||
public sealed partial class SMS : IEmulator
|
||||
{
|
||||
public IEmulatorServiceProvider ServiceProvider { get; }
|
||||
|
||||
public ControllerDefinition ControllerDefinition
|
||||
{
|
||||
get
|
||||
{
|
||||
if (IsGameGear)
|
||||
{
|
||||
return GGController;
|
||||
}
|
||||
|
||||
switch(Settings.ControllerType)
|
||||
{
|
||||
case "Paddle":
|
||||
return SMSPaddleController;
|
||||
case "Light Phaser":
|
||||
// scale the vertical to the display mode
|
||||
SMSLightPhaserController.FloatRanges[1] = new ControllerDefinition.FloatRange(0, Vdp.FrameHeight / 2, Vdp.FrameHeight - 1);
|
||||
|
||||
return SMSLightPhaserController;
|
||||
default:
|
||||
return SmsController;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void FrameAdvance(IController controller, bool render, bool rendersound)
|
||||
{
|
||||
_controller = controller;
|
||||
_lagged = true;
|
||||
_frame++;
|
||||
PSG.BeginFrame(Cpu.TotalExecutedCycles);
|
||||
Cpu.Debug = Tracer.Enabled;
|
||||
if (!IsGameGear)
|
||||
{
|
||||
PSG.StereoPanning = Settings.ForceStereoSeparation ? ForceStereoByte : (byte)0xFF;
|
||||
}
|
||||
|
||||
if (Cpu.Debug && Cpu.Logger == null) // TODO, lets not do this on each frame. But lets refactor CoreComm/CoreComm first
|
||||
{
|
||||
Cpu.Logger = s => Tracer.Put(s);
|
||||
}
|
||||
|
||||
if (IsGameGear == false)
|
||||
{
|
||||
Cpu.NonMaskableInterrupt = controller.IsPressed("Pause");
|
||||
}
|
||||
|
||||
if (IsGame3D && Settings.Fix3D)
|
||||
{
|
||||
Vdp.ExecFrame((Frame & 1) == 0);
|
||||
}
|
||||
else
|
||||
{
|
||||
Vdp.ExecFrame(render);
|
||||
}
|
||||
|
||||
PSG.EndFrame(Cpu.TotalExecutedCycles);
|
||||
if (_lagged)
|
||||
{
|
||||
_lagCount++;
|
||||
_isLag = true;
|
||||
}
|
||||
else
|
||||
{
|
||||
_isLag = false;
|
||||
}
|
||||
}
|
||||
|
||||
public int Frame => _frame;
|
||||
|
||||
public string SystemId => "SMS";
|
||||
|
||||
public bool DeterministicEmulation => true;
|
||||
|
||||
public void ResetCounters()
|
||||
{
|
||||
_frame = 0;
|
||||
_lagCount = 0;
|
||||
_isLag = false;
|
||||
}
|
||||
|
||||
public CoreComm CoreComm { get; }
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
}
|
||||
}
|
||||
}
|
||||
using BizHawk.Emulation.Common;
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
||||
{
|
||||
public sealed partial class SMS : IEmulator
|
||||
{
|
||||
public IEmulatorServiceProvider ServiceProvider { get; }
|
||||
|
||||
public ControllerDefinition ControllerDefinition
|
||||
{
|
||||
get
|
||||
{
|
||||
if (IsGameGear)
|
||||
{
|
||||
return GGController;
|
||||
}
|
||||
|
||||
switch(Settings.ControllerType)
|
||||
{
|
||||
case "Paddle":
|
||||
return SMSPaddleController;
|
||||
case "Light Phaser":
|
||||
// scale the vertical to the display mode
|
||||
SMSLightPhaserController.FloatRanges[1] = new ControllerDefinition.FloatRange(0, Vdp.FrameHeight / 2, Vdp.FrameHeight - 1);
|
||||
|
||||
return SMSLightPhaserController;
|
||||
default:
|
||||
return SmsController;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
public void FrameAdvance(IController controller, bool render, bool rendersound)
|
||||
{
|
||||
_controller = controller;
|
||||
_lagged = true;
|
||||
_frame++;
|
||||
PSG.BeginFrame(Cpu.TotalExecutedCycles);
|
||||
|
||||
if (!IsGameGear)
|
||||
{
|
||||
PSG.StereoPanning = Settings.ForceStereoSeparation ? ForceStereoByte : (byte)0xFF;
|
||||
}
|
||||
|
||||
if (Tracer.Enabled)
|
||||
{
|
||||
Cpu.TraceCallback = s => Tracer.Put(s);
|
||||
}
|
||||
else
|
||||
{
|
||||
Cpu.TraceCallback = null;
|
||||
}
|
||||
|
||||
if (IsGameGear == false)
|
||||
{
|
||||
Cpu.NonMaskableInterrupt = controller.IsPressed("Pause");
|
||||
}
|
||||
|
||||
if (IsGame3D && Settings.Fix3D)
|
||||
{
|
||||
Vdp.ExecFrame((Frame & 1) == 0);
|
||||
}
|
||||
else
|
||||
{
|
||||
Vdp.ExecFrame(render);
|
||||
}
|
||||
|
||||
PSG.EndFrame(Cpu.TotalExecutedCycles);
|
||||
if (_lagged)
|
||||
{
|
||||
_lagCount++;
|
||||
_isLag = true;
|
||||
}
|
||||
else
|
||||
{
|
||||
_isLag = false;
|
||||
}
|
||||
}
|
||||
|
||||
public int Frame => _frame;
|
||||
|
||||
public string SystemId => "SMS";
|
||||
|
||||
public bool DeterministicEmulation => true;
|
||||
|
||||
public void ResetCounters()
|
||||
{
|
||||
_frame = 0;
|
||||
_lagCount = 0;
|
||||
_isLag = false;
|
||||
}
|
||||
|
||||
public CoreComm CoreComm { get; }
|
||||
|
||||
public void Dispose()
|
||||
{
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1,106 +1,105 @@
|
|||
using System.IO;
|
||||
|
||||
using BizHawk.Common;
|
||||
using BizHawk.Emulation.Common;
|
||||
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
||||
{
|
||||
public sealed partial class SMS : IStatable
|
||||
{
|
||||
public bool BinarySaveStatesPreferred
|
||||
{
|
||||
get { return false; }
|
||||
}
|
||||
|
||||
public void SaveStateBinary(BinaryWriter bw)
|
||||
{
|
||||
SyncState(Serializer.CreateBinaryWriter(bw));
|
||||
}
|
||||
|
||||
public void LoadStateBinary(BinaryReader br)
|
||||
{
|
||||
SyncState(Serializer.CreateBinaryReader(br));
|
||||
}
|
||||
|
||||
public void SaveStateText(TextWriter tw)
|
||||
{
|
||||
SyncState(Serializer.CreateTextWriter(tw));
|
||||
}
|
||||
|
||||
public void LoadStateText(TextReader tr)
|
||||
{
|
||||
SyncState(Serializer.CreateTextReader(tr));
|
||||
}
|
||||
|
||||
public byte[] SaveStateBinary()
|
||||
{
|
||||
if (_stateBuffer == null)
|
||||
{
|
||||
var stream = new MemoryStream();
|
||||
var writer = new BinaryWriter(stream);
|
||||
SaveStateBinary(writer);
|
||||
_stateBuffer = stream.ToArray();
|
||||
writer.Close();
|
||||
return _stateBuffer;
|
||||
}
|
||||
else
|
||||
{
|
||||
var stream = new MemoryStream(_stateBuffer);
|
||||
var writer = new BinaryWriter(stream);
|
||||
SaveStateBinary(writer);
|
||||
writer.Close();
|
||||
return _stateBuffer;
|
||||
}
|
||||
}
|
||||
|
||||
private byte[] _stateBuffer;
|
||||
|
||||
private void SyncState(Serializer ser)
|
||||
{
|
||||
ser.BeginSection("SMS");
|
||||
Cpu.SyncState(ser);
|
||||
Vdp.SyncState(ser);
|
||||
PSG.SyncState(ser);
|
||||
ser.Sync("RAM", ref SystemRam, false);
|
||||
ser.Sync("RomBank0", ref RomBank0);
|
||||
ser.Sync("RomBank1", ref RomBank1);
|
||||
ser.Sync("RomBank2", ref RomBank2);
|
||||
ser.Sync("RomBank3", ref RomBank3);
|
||||
ser.Sync("Port01", ref Port01);
|
||||
ser.Sync("Port02", ref Port02);
|
||||
ser.Sync("Port3E", ref Port3E);
|
||||
ser.Sync("Port3F", ref Port3F);
|
||||
ser.Sync("Paddle1High", ref Paddle1High);
|
||||
ser.Sync("Paddle2High", ref Paddle2High);
|
||||
ser.Sync("LatchLightPhaser", ref LatchLightPhaser);
|
||||
|
||||
if (SaveRAM != null)
|
||||
{
|
||||
ser.Sync("SaveRAM", ref SaveRAM, false);
|
||||
ser.Sync("SaveRamBank", ref SaveRamBank);
|
||||
}
|
||||
|
||||
if (ExtRam != null)
|
||||
{
|
||||
ser.Sync("ExtRAM", ref ExtRam, true);
|
||||
}
|
||||
|
||||
if (HasYM2413)
|
||||
{
|
||||
YM2413.SyncState(ser);
|
||||
}
|
||||
|
||||
ser.Sync("Frame", ref _frame);
|
||||
ser.Sync("LagCount", ref _lagCount);
|
||||
ser.Sync("IsLag", ref _isLag);
|
||||
|
||||
ser.EndSection();
|
||||
|
||||
if (ser.IsReader)
|
||||
{
|
||||
SyncAllByteArrayDomains();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
using System.IO;
|
||||
|
||||
using BizHawk.Common;
|
||||
using BizHawk.Emulation.Common;
|
||||
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
||||
{
|
||||
public sealed partial class SMS : IStatable
|
||||
{
|
||||
public bool BinarySaveStatesPreferred
|
||||
{
|
||||
get { return true; }
|
||||
}
|
||||
|
||||
public void SaveStateText(TextWriter writer)
|
||||
{
|
||||
SyncState(new Serializer(writer));
|
||||
}
|
||||
|
||||
public void LoadStateText(TextReader reader)
|
||||
{
|
||||
SyncState(new Serializer(reader));
|
||||
}
|
||||
|
||||
public void SaveStateBinary(BinaryWriter bw)
|
||||
{
|
||||
SyncState(new Serializer(bw));
|
||||
}
|
||||
|
||||
public void LoadStateBinary(BinaryReader br)
|
||||
{
|
||||
SyncState(new Serializer(br));
|
||||
}
|
||||
|
||||
public byte[] SaveStateBinary()
|
||||
{
|
||||
MemoryStream ms = new MemoryStream();
|
||||
BinaryWriter bw = new BinaryWriter(ms);
|
||||
SaveStateBinary(bw);
|
||||
bw.Flush();
|
||||
return ms.ToArray();
|
||||
}
|
||||
|
||||
private void SyncState(Serializer ser)
|
||||
{
|
||||
byte[] core = null;
|
||||
if (ser.IsWriter)
|
||||
{
|
||||
var ms = new MemoryStream();
|
||||
ms.Close();
|
||||
core = ms.ToArray();
|
||||
}
|
||||
Cpu.SyncState(ser);
|
||||
|
||||
ser.BeginSection("SMS");
|
||||
Vdp.SyncState(ser);
|
||||
PSG.SyncState(ser);
|
||||
ser.Sync("RAM", ref SystemRam, false);
|
||||
ser.Sync("RomBank0", ref RomBank0);
|
||||
ser.Sync("RomBank1", ref RomBank1);
|
||||
ser.Sync("RomBank2", ref RomBank2);
|
||||
ser.Sync("RomBank3", ref RomBank3);
|
||||
ser.Sync("Port01", ref Port01);
|
||||
ser.Sync("Port02", ref Port02);
|
||||
ser.Sync("Port3E", ref Port3E);
|
||||
ser.Sync("Port3F", ref Port3F);
|
||||
ser.Sync("Paddle1High", ref Paddle1High);
|
||||
ser.Sync("Paddle2High", ref Paddle2High);
|
||||
ser.Sync("LatchLightPhaser", ref LatchLightPhaser);
|
||||
|
||||
if (SaveRAM != null)
|
||||
{
|
||||
ser.Sync("SaveRAM", ref SaveRAM, false);
|
||||
ser.Sync("SaveRamBank", ref SaveRamBank);
|
||||
}
|
||||
|
||||
if (ExtRam != null)
|
||||
{
|
||||
ser.Sync("ExtRAM", ref ExtRam, true);
|
||||
}
|
||||
|
||||
if (HasYM2413)
|
||||
{
|
||||
YM2413.SyncState(ser);
|
||||
}
|
||||
|
||||
if (EEPROM != null)
|
||||
{
|
||||
EEPROM.SyncState(ser);
|
||||
}
|
||||
|
||||
ser.Sync("Frame", ref _frame);
|
||||
ser.Sync("LagCount", ref _lagCount);
|
||||
ser.Sync("IsLag", ref _isLag);
|
||||
|
||||
ser.EndSection();
|
||||
|
||||
if (ser.IsReader)
|
||||
{
|
||||
SyncAllByteArrayDomains();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -4,7 +4,7 @@ using BizHawk.Common.StringExtensions;
|
|||
using BizHawk.Emulation.Common;
|
||||
using BizHawk.Emulation.Common.Components;
|
||||
using BizHawk.Emulation.Cores.Components;
|
||||
using BizHawk.Emulation.Cores.Components.Z80;
|
||||
using BizHawk.Emulation.Cores.Components.Z80A;
|
||||
|
||||
/*****************************************************
|
||||
TODO:
|
||||
|
@ -75,11 +75,12 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
HasYM2413 = true;
|
||||
}
|
||||
|
||||
Cpu = new Z80A
|
||||
Cpu = new Z80A()
|
||||
{
|
||||
RegisterSP = 0xDFF0,
|
||||
ReadHardware = ReadPort,
|
||||
WriteHardware = WritePort,
|
||||
ReadMemory = ReadMemory,
|
||||
WriteMemory = WriteMemory,
|
||||
MemoryCallbacks = MemoryCallbacks
|
||||
};
|
||||
|
||||
|
@ -113,6 +114,8 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
InitNemesisMapper();
|
||||
else if (game["TerebiOekaki"])
|
||||
InitTerebiOekaki();
|
||||
else if (game["EEPROM"])
|
||||
InitEEPROMMapper();
|
||||
else
|
||||
InitSegaMapper();
|
||||
|
||||
|
@ -160,7 +163,10 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
}
|
||||
|
||||
if (game["SRAM"])
|
||||
{
|
||||
SaveRAM = new byte[int.Parse(game.OptionValue("SRAM"))];
|
||||
Console.WriteLine(SaveRAM.Length);
|
||||
}
|
||||
else if (game.NotInDatabase)
|
||||
SaveRAM = new byte[0x8000];
|
||||
|
||||
|
@ -175,8 +181,11 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
|
||||
var serviceProvider = ServiceProvider as BasicServiceProvider;
|
||||
serviceProvider.Register<ITraceable>(Tracer);
|
||||
serviceProvider.Register<IDisassemblable>(new Disassembler());
|
||||
serviceProvider.Register<IDisassemblable>(Cpu);
|
||||
Vdp.ProcessOverscan();
|
||||
|
||||
Cpu.ReadMemory = ReadMemory;
|
||||
Cpu.WriteMemory = WriteMemory;
|
||||
}
|
||||
|
||||
// Constants
|
||||
|
|
|
@ -4,7 +4,7 @@ using System.IO;
|
|||
|
||||
using BizHawk.Common;
|
||||
using BizHawk.Emulation.Common;
|
||||
using BizHawk.Emulation.Cores.Components.Z80;
|
||||
using BizHawk.Emulation.Cores.Components.Z80A;
|
||||
|
||||
|
||||
namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
||||
|
@ -113,7 +113,7 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
StatusByte &= 0x1F;
|
||||
HIntPending = false;
|
||||
VIntPending = false;
|
||||
Cpu.Interrupt = false;
|
||||
Cpu.FlagI = false;
|
||||
return returnValue;
|
||||
}
|
||||
|
||||
|
@ -291,13 +291,13 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
{
|
||||
case 0: // Mode Control Register 1
|
||||
CheckVideoMode();
|
||||
Cpu.Interrupt = (EnableLineInterrupts && HIntPending);
|
||||
Cpu.Interrupt |= (EnableFrameInterrupts && VIntPending);
|
||||
Cpu.FlagI = (EnableLineInterrupts && HIntPending);
|
||||
Cpu.FlagI |= (EnableFrameInterrupts && VIntPending);
|
||||
break;
|
||||
case 1: // Mode Control Register 2
|
||||
CheckVideoMode();
|
||||
Cpu.Interrupt = (EnableFrameInterrupts && VIntPending);
|
||||
Cpu.Interrupt |= (EnableLineInterrupts && HIntPending);
|
||||
Cpu.FlagI = (EnableFrameInterrupts && VIntPending);
|
||||
Cpu.FlagI |= (EnableLineInterrupts && HIntPending);
|
||||
break;
|
||||
case 2: // Name Table Base Address
|
||||
NameTableBase = CalcNameTableBase();
|
||||
|
@ -347,7 +347,7 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
|
||||
if (VIntPending && EnableFrameInterrupts)
|
||||
{
|
||||
Cpu.Interrupt = true;
|
||||
Cpu.FlagI = true;
|
||||
}
|
||||
|
||||
}
|
||||
|
@ -361,7 +361,7 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
HIntPending = true;
|
||||
if (EnableLineInterrupts)
|
||||
{;
|
||||
Cpu.Interrupt = true;
|
||||
Cpu.FlagI = true;
|
||||
}
|
||||
lineIntLinesRemaining = Registers[0x0A];
|
||||
}
|
||||
|
@ -383,7 +383,14 @@ namespace BizHawk.Emulation.Cores.Sega.MasterSystem
|
|||
ProcessLineInterrupt();
|
||||
Sms.ProcessLineControls();
|
||||
|
||||
Cpu.ExecuteCycles(IPeriod);
|
||||
//Console.Write(Cpu.cur_instr.Length);
|
||||
//Console.Write(" ");
|
||||
//Console.WriteLine(Cpu.instr_pntr);
|
||||
for (int j = 0; j < IPeriod; j++)
|
||||
{
|
||||
Cpu.ExecuteOne();
|
||||
}
|
||||
|
||||
|
||||
if (ScanLine == scanlinesPerFrame - 1)
|
||||
{
|
||||
|
|
|
@ -70,6 +70,7 @@
|
|||
<Compile Include="DiscFormats\Blobs\Blob_ZeroPadAdapter.cs" />
|
||||
<Compile Include="DiscFormats\Blobs\IBlob.cs" />
|
||||
<Compile Include="DiscFormats\Blobs\RiffMaster.cs" />
|
||||
<Compile Include="DiscFormats\MDS_Format.cs" />
|
||||
<Compile Include="DiscFormats\CCD_format.cs" />
|
||||
<Compile Include="DiscFormats\CUE\CueFileResolver.cs" />
|
||||
<Compile Include="DiscFormats\CUE\CUE_Compile.cs" />
|
||||
|
|
|
@ -1,5 +1,6 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Text;
|
||||
|
||||
namespace BizHawk.Emulation.DiscSystem
|
||||
|
@ -114,5 +115,49 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
}
|
||||
|
||||
#endregion
|
||||
|
||||
// (asni 20171013) - Some methods I wrote that have been shoehorned in from another project to speed up development time
|
||||
// If these are offensive in any way, tell me I suck and that I need to do more work with existing methods
|
||||
#region Misc
|
||||
|
||||
/// <summary>
|
||||
/// Returns a byte array of any length
|
||||
/// Not really anything endian going on, but I struggled to find a better place for it
|
||||
/// </summary>
|
||||
/// <param name="buffer"></param>
|
||||
/// <param name="offset"></param>
|
||||
/// <param name="length"></param>
|
||||
/// <returns></returns>
|
||||
public byte[] ReadBytes(byte[] buffer, int offset, int length)
|
||||
{
|
||||
return buffer.Skip(offset).Take(length).ToArray();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Returns an int32 value from any size byte array
|
||||
/// (careful, data may/will be truncated)
|
||||
/// </summary>
|
||||
/// <param name="buffer"></param>
|
||||
/// <param name="offset"></param>
|
||||
/// <param name="length"></param>
|
||||
/// <returns></returns>
|
||||
public int ReadIntValue(byte[] buffer, int offset, int length)
|
||||
{
|
||||
var bytes = buffer.Skip(offset).Take(length).ToArray();
|
||||
|
||||
if (swap)
|
||||
Array.Reverse(bytes);
|
||||
|
||||
if (length == 1)
|
||||
return bytes.FirstOrDefault();
|
||||
|
||||
if (length == 2)
|
||||
return BitConverter.ToInt16(bytes, 0);
|
||||
|
||||
int result = BitConverter.ToInt32(bytes, 0);
|
||||
return result;
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
||||
|
|
|
@ -48,21 +48,24 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
s.Seek(this.Offset * ISOFile.SECTOR_SIZE, SeekOrigin.Begin);
|
||||
|
||||
List<ISONodeRecord> records = new List<ISONodeRecord>();
|
||||
|
||||
// Read the directory entries
|
||||
while (s.Position < ((this.Offset * ISOFile.SECTOR_SIZE) + this.Length))
|
||||
|
||||
// Read the directory entries
|
||||
while (s.Position < ((this.Offset * ISOFile.SECTOR_SIZE) + this.Length))
|
||||
{
|
||||
ISONode node;
|
||||
ISONodeRecord record;
|
||||
|
||||
|
||||
// Read the record
|
||||
record = new ISONodeRecord();
|
||||
record.Parse(s);
|
||||
if (ISOFile.Format == ISOFile.ISOFormat.CDInteractive)
|
||||
record.ParseCDInteractive(s);
|
||||
if (ISOFile.Format == ISOFile.ISOFormat.ISO9660)
|
||||
record.ParseISO9660(s);
|
||||
|
||||
|
||||
//zero 24-jun-2013 - improved validity checks
|
||||
//theres nothing here!
|
||||
if (record.Length == 0)
|
||||
//zero 24-jun-2013 - improved validity checks
|
||||
//theres nothing here!
|
||||
if (record.Length == 0)
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
@ -95,7 +98,8 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
}
|
||||
|
||||
// Add the node as a child
|
||||
this.Children.Add(record.Name, node);
|
||||
if (!this.Children.ContainsKey(record.Name))
|
||||
this.Children.Add(record.Name, node);
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -2,27 +2,51 @@
|
|||
using System.Collections.Generic;
|
||||
using System.Text;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
|
||||
namespace BizHawk.Emulation.DiscSystem
|
||||
{
|
||||
/// <summary>
|
||||
/// This class is meant to parse disk images as specified by ISO9660.
|
||||
/// Specifically, it should work for most disk images that are created
|
||||
/// by the stanard disk imaging software. This class is by no means
|
||||
/// robust to all variations of ISO9660.
|
||||
/// Also, this class does not currently support the UDF file system.
|
||||
///
|
||||
/// TODO: Add functions to enumerate a directory or visit a file...
|
||||
///
|
||||
/// The information for building class came from three primary sources:
|
||||
/// 1. The ISO9660 wikipedia article:
|
||||
/// http://en.wikipedia.org/wiki/ISO_9660
|
||||
/// 2. ISO9660 Simplified for DOS/Windows
|
||||
/// http://alumnus.caltech.edu/~pje/iso9660.html
|
||||
/// 3. The ISO 9660 File System
|
||||
/// http://users.telenet.be/it3.consultants.bvba/handouts/ISO9960.html
|
||||
/// </summary>
|
||||
public class ISOFile
|
||||
/// <summary>
|
||||
/// This class is meant to parse disk images as specified by:
|
||||
///
|
||||
/// ISO9660
|
||||
/// -------
|
||||
/// It should work for most disk images that are created
|
||||
/// by the stanard disk imaging software. This class is by no means
|
||||
/// robust to all variations of ISO9660.
|
||||
/// Also, this class does not currently support the UDF file system.
|
||||
///
|
||||
/// The information for building class came from three primary sources:
|
||||
/// 1. The ISO9660 wikipedia article:
|
||||
/// http://en.wikipedia.org/wiki/ISO_9660
|
||||
/// 2. ISO9660 Simplified for DOS/Windows
|
||||
/// http://alumnus.caltech.edu/~pje/iso9660.html
|
||||
/// 3. The ISO 9660 File System
|
||||
/// http://users.telenet.be/it3.consultants.bvba/handouts/ISO9960.html
|
||||
///
|
||||
///
|
||||
/// CD-I
|
||||
/// ----
|
||||
/// (asni - 20171013) - Class modified to be able to detect and consume Green
|
||||
/// Book disc images.
|
||||
///
|
||||
/// The implemtation of CD-I in this class adds some (but not all) additional
|
||||
/// properties to the class structures that CD-I brings. This means that
|
||||
/// the same ISO class structures can be returned for both standards.
|
||||
/// These small additions are readily found in ISOVolumeDescriptor.cs
|
||||
///
|
||||
/// ISOFile.cs also now contains a public 'ISOFormat' enum that is set
|
||||
/// during disc parsing.
|
||||
///
|
||||
/// The main reference source for this implementation:
|
||||
/// 1. The CD-I Full Functional Specification (aka Green Book)
|
||||
/// https://www.lscdweb.com/data/downloadables/2/8/cdi_may94_r2.pdf
|
||||
///
|
||||
///
|
||||
/// TODO: Add functions to enumerate a directory or visit a file...
|
||||
///
|
||||
/// </summary>
|
||||
public class ISOFile
|
||||
{
|
||||
#region Constants
|
||||
|
||||
|
@ -31,21 +55,37 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
/// </summary>
|
||||
public const int SECTOR_SIZE = 2048;
|
||||
|
||||
#endregion
|
||||
#endregion
|
||||
|
||||
#region Public Members
|
||||
#region Static Members
|
||||
|
||||
/// <summary>
|
||||
/// This is a list of all the volume descriptors in the disk image.
|
||||
/// NOTE: The first entry should be the primary volume.
|
||||
/// </summary>
|
||||
public List<ISOVolumeDescriptor> VolumeDescriptors;
|
||||
/// <summary>
|
||||
/// Making this a static for now. Every other way I tried was fairly ineligant (asni)
|
||||
/// </summary>
|
||||
public static ISOFormat Format;
|
||||
|
||||
public static List<CDIPathNode> CDIPathTable;
|
||||
|
||||
#endregion
|
||||
|
||||
#region Public Members
|
||||
|
||||
/// <summary>
|
||||
/// This is a list of all the volume descriptors in the disk image.
|
||||
/// NOTE: The first entry should be the primary volume.
|
||||
/// </summary>
|
||||
public List<ISOVolumeDescriptor> VolumeDescriptors;
|
||||
|
||||
/// <summary>
|
||||
/// The Directory that is the root of this file system
|
||||
/// </summary>
|
||||
public ISODirectoryNode Root;
|
||||
|
||||
/// <summary>
|
||||
/// The type of CDFS format detected
|
||||
/// </summary>
|
||||
public ISOFormat CDFSType;
|
||||
|
||||
#endregion
|
||||
|
||||
#region Construction
|
||||
|
@ -77,9 +117,9 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
|
||||
// Seek through the first volume descriptor
|
||||
s.Seek(startPosition + (SECTOR_SIZE * startSector), SeekOrigin.Begin);
|
||||
|
||||
// Read one of more volume descriptors
|
||||
do
|
||||
|
||||
// Read one of more volume descriptors
|
||||
do
|
||||
{
|
||||
//zero 24-jun-2013 - improved validity checks
|
||||
|
||||
|
@ -87,6 +127,8 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
bool isValid = desc.Parse(s);
|
||||
if (!isValid) return false;
|
||||
|
||||
this.CDFSType = Format;
|
||||
|
||||
if (desc.IsTerminator())
|
||||
break;
|
||||
else if (desc.Type < 4)
|
||||
|
@ -98,16 +140,17 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
|
||||
} while (true);
|
||||
|
||||
//zero 24-jun-2013 - well, my very first test iso had 2 volume descriptors.
|
||||
// Check to make sure we only read one volume descriptor
|
||||
// Finding more could be an error with the disk.
|
||||
//if (this.VolumeDescriptors.Count != 1) {
|
||||
// Console.WriteLine("Strange ISO format...");
|
||||
// return;
|
||||
//}
|
||||
//zero 24-jun-2013 - well, my very first test iso had 2 volume descriptors.
|
||||
// Check to make sure we only read one volume descriptor
|
||||
// Finding more could be an error with the disk.
|
||||
//if (this.VolumeDescriptors.Count != 1) {
|
||||
// Console.WriteLine("Strange ISO format...");
|
||||
// return;
|
||||
//}
|
||||
|
||||
//zero 24-jun-2013 - if theres no volume descriptors, we're gonna call this not a cdfs
|
||||
if (VolumeDescriptors.Count == 0) return false;
|
||||
|
||||
//zero 24-jun-2013 - if theres no volume descriptors, we're gonna call this not a cdfs
|
||||
if (VolumeDescriptors.Count == 0) return false;
|
||||
|
||||
// Visit all the directories and get the offset of each directory/file
|
||||
|
||||
|
@ -116,12 +159,68 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
|
||||
// Create (and visit) the root node
|
||||
this.Root = new ISODirectoryNode(this.VolumeDescriptors[0].RootDirectoryRecord);
|
||||
visitedNodes.Add(this.Root.Offset, this.Root);
|
||||
this.Root.Parse(s, visitedNodes);
|
||||
|
||||
visitedNodes.Add(this.Root.Offset, this.Root);
|
||||
this.Root.Parse(s, visitedNodes);
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
private List<KeyValuePair<string, ISOFileNode>> fileNodes;
|
||||
private List<ISODirectoryNode> dirsParsed;
|
||||
|
||||
/// <summary>
|
||||
/// Returns a flat list of all recursed files
|
||||
/// </summary>
|
||||
/// <returns></returns>
|
||||
public List<KeyValuePair<string, ISOFileNode>> EnumerateAllFilesRecursively()
|
||||
{
|
||||
fileNodes = new List<KeyValuePair<string, ISOFileNode>>();
|
||||
dirsParsed = new List<ISODirectoryNode>();
|
||||
|
||||
if (Root.Children == null)
|
||||
return fileNodes;
|
||||
|
||||
// get all folders
|
||||
List<KeyValuePair<string, ISONode>> dirs = (from a in Root.Children
|
||||
where a.Value.GetType() == typeof(ISODirectoryNode)
|
||||
select a).ToList();
|
||||
// iterate through each folder
|
||||
foreach (var d in dirs)
|
||||
{
|
||||
// process all files in this directory (and recursively process files in sub folders
|
||||
ISODirectoryNode idn = d.Value as ISODirectoryNode;
|
||||
if (dirsParsed.Where(a => a == idn).Count() > 0)
|
||||
continue;
|
||||
|
||||
dirsParsed.Add(idn);
|
||||
ProcessDirectoryFiles(idn.Children);
|
||||
}
|
||||
|
||||
return fileNodes.Distinct().ToList();
|
||||
}
|
||||
|
||||
private void ProcessDirectoryFiles(Dictionary<string, ISONode> idn)
|
||||
{
|
||||
foreach (var n in idn)
|
||||
{
|
||||
if (n.Value.GetType() == typeof(ISODirectoryNode))
|
||||
{
|
||||
if (dirsParsed.Where(a => a == n.Value).Count() > 0)
|
||||
continue;
|
||||
|
||||
dirsParsed.Add(n.Value as ISODirectoryNode);
|
||||
ProcessDirectoryFiles((n.Value as ISODirectoryNode).Children);
|
||||
}
|
||||
else
|
||||
{
|
||||
KeyValuePair<string, ISOFileNode> f = new KeyValuePair<string, ISOFileNode>(n.Key, n.Value as ISOFileNode);
|
||||
fileNodes.Add(f);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Printing
|
||||
|
@ -135,6 +234,17 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
this.Root.Print(0);
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
#endregion
|
||||
|
||||
#region Misc
|
||||
|
||||
public enum ISOFormat
|
||||
{
|
||||
Unknown,
|
||||
ISO9660,
|
||||
CDInteractive
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
||||
|
|
|
@ -31,10 +31,16 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
/// </summary>
|
||||
public byte Length;
|
||||
|
||||
/// <summary>
|
||||
/// The file offset of the data for this file/directory (in sectors).
|
||||
/// </summary>
|
||||
public long OffsetOfData;
|
||||
/// <summary>
|
||||
/// This is the number of blocks at the beginning of the file reserved for extended attribute information
|
||||
/// The format of the extended attribute record is not defined and is reserved for application use
|
||||
/// </summary>
|
||||
public byte ExtendedAttribRecordLength;
|
||||
|
||||
/// <summary>
|
||||
/// The file offset of the data for this file/directory (in sectors).
|
||||
/// </summary>
|
||||
public long OffsetOfData;
|
||||
/// <summary>
|
||||
/// The length of the data for this file/directory (in bytes).
|
||||
/// </summary>
|
||||
|
@ -148,14 +154,19 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
// Put the array into a memory stream and pass to the main parsing function
|
||||
MemoryStream s = new MemoryStream(data);
|
||||
s.Seek(cursor, SeekOrigin.Begin);
|
||||
this.Parse(s);
|
||||
|
||||
if (ISOFile.Format == ISOFile.ISOFormat.ISO9660)
|
||||
this.ParseISO9660(s);
|
||||
|
||||
if (ISOFile.Format == ISOFile.ISOFormat.CDInteractive)
|
||||
this.ParseCDInteractive(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Parse the node record from the given stream.
|
||||
/// Parse the node record from the given ISO9660 stream.
|
||||
/// </summary>
|
||||
/// <param name="s">The stream to parse from.</param>
|
||||
public void Parse(Stream s)
|
||||
public void ParseISO9660(Stream s)
|
||||
{
|
||||
EndianBitConverter bc = EndianBitConverter.CreateForLittleEndian();
|
||||
long startPosition = s.Position;
|
||||
|
@ -212,6 +223,100 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
s.Seek(startPosition + this.Length, SeekOrigin.Begin);
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
/// <summary>
|
||||
/// Parse the node record from the given CD-I stream.
|
||||
/// </summary>
|
||||
/// <param name="s">The stream to parse from.</param>
|
||||
public void ParseCDInteractive(Stream s)
|
||||
{
|
||||
/*
|
||||
BP Size in bytes Description
|
||||
1 1 Record length
|
||||
2 1 Extended Attribute record length
|
||||
3 4 Reserved
|
||||
7 4 File beginning LBN
|
||||
11 4 Reserved
|
||||
15 4 File size
|
||||
19 6 Creation date
|
||||
25 1 Reserved
|
||||
26 1 File flags
|
||||
27 2 Interleave
|
||||
29 2 Reserved
|
||||
31 2 Album Set Sequence number
|
||||
33 1 File name size
|
||||
34 (n) File name
|
||||
34+n 4 Owner ID
|
||||
38+n 2 Attributes
|
||||
40+n 2 Reserved
|
||||
42+n 1 File number
|
||||
43+n 1 Reserved
|
||||
43+n Total
|
||||
*/
|
||||
|
||||
EndianBitConverter bc = EndianBitConverter.CreateForLittleEndian();
|
||||
EndianBitConverter bcBig = EndianBitConverter.CreateForBigEndian();
|
||||
long startPosition = s.Position;
|
||||
byte[] buffer = new byte[ISOFile.SECTOR_SIZE];
|
||||
|
||||
// Read the entire structure
|
||||
s.Read(buffer, 0, ISOFile.SECTOR_SIZE);
|
||||
s.Position -= ISOFile.SECTOR_SIZE;
|
||||
|
||||
// Get the record length
|
||||
this.Length = buffer[0];
|
||||
|
||||
// extended attribute record length
|
||||
this.ExtendedAttribRecordLength = buffer[1];
|
||||
|
||||
// Read Data Offset
|
||||
this.OffsetOfData = bcBig.ReadIntValue(buffer, 6, 4);
|
||||
|
||||
// Read Data Length
|
||||
this.LengthOfData = bcBig.ReadIntValue(buffer, 14, 4);
|
||||
|
||||
// Read the time
|
||||
var ti = bc.ReadBytes(buffer, 18, 6);
|
||||
this.Year = ti[0];
|
||||
this.Month = ti[1];
|
||||
this.Day = ti[2];
|
||||
this.Hour = ti[3];
|
||||
this.Minute = ti[4];
|
||||
this.Second = ti[5];
|
||||
|
||||
// read interleave - still to do
|
||||
|
||||
// read album (volume) set sequence number (we are ignoring this)
|
||||
|
||||
// Read the name length
|
||||
this.NameLength = buffer[32];
|
||||
|
||||
// Read the file/directory name
|
||||
var name = bc.ReadBytes(buffer, 33, this.NameLength);
|
||||
if (this.NameLength == 1 && (name[0] == 0 || name[0] == 1))
|
||||
{
|
||||
if (name[0] == 0)
|
||||
this.Name = ISONodeRecord.CURRENT_DIRECTORY;
|
||||
else
|
||||
this.Name = ISONodeRecord.PARENT_DIRECTORY;
|
||||
}
|
||||
else
|
||||
{
|
||||
this.Name = ASCIIEncoding.ASCII.GetString(name, 0, this.NameLength);
|
||||
}
|
||||
|
||||
// skip ownerID for now
|
||||
|
||||
// read the flags - only really interested in the directory attribute (bit 15)
|
||||
// (confusingly these are called 'attributes' in CD-I. the CD-I 'File Flags' entry is something else entirely)
|
||||
this.Flags = buffer[37 + this.NameLength];
|
||||
|
||||
// skip filenumber
|
||||
//this.FileNumber = buffer[41 + this.NameLength];
|
||||
|
||||
// Seek to end
|
||||
s.Seek(startPosition + this.Length, SeekOrigin.Begin);
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
||||
|
|
|
@ -25,14 +25,21 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
private const int LENGTH_TIME = 17;
|
||||
private const int LENGTH_RESERVED = 512;
|
||||
|
||||
#endregion
|
||||
#endregion
|
||||
|
||||
#region Public Properties
|
||||
#region Private Properties
|
||||
|
||||
/// <summary>
|
||||
/// The type of this volume description, only 1 and 255 are supported
|
||||
/// </summary>
|
||||
public byte Type;
|
||||
private EndianBitConverter bc = EndianBitConverter.CreateForLittleEndian();
|
||||
private EndianBitConverter bcBig = EndianBitConverter.CreateForBigEndian();
|
||||
|
||||
#endregion
|
||||
|
||||
#region Public Properties
|
||||
|
||||
/// <summary>
|
||||
/// The type of this volume description, only 1 and 255 are supported
|
||||
/// </summary>
|
||||
public byte Type;
|
||||
|
||||
/// <summary>
|
||||
/// The system identifier
|
||||
|
@ -66,21 +73,21 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
/// </summary>
|
||||
public int PathTableSize;
|
||||
/// <summary>
|
||||
/// Sector offset of the first path table
|
||||
/// (ISO9660 only) Sector offset of the first path table
|
||||
/// </summary>
|
||||
public int OffsetOfFirstLittleEndianPathTable;
|
||||
/// <summary>
|
||||
/// Sector offset of the second path table
|
||||
/// </summary>
|
||||
public int OffsetOfSecondLittleEndianPathTable;
|
||||
/// <summary>
|
||||
/// Sector offset of the first path table
|
||||
/// </summary>
|
||||
public int OffsetOfFirstBigEndianPathTable;
|
||||
/// <summary>
|
||||
/// Sector offset of the second path table
|
||||
/// </summary>
|
||||
public int OffsetOfSecondBigEndianPathTable;
|
||||
/// <summary>
|
||||
/// (ISO9660 only) Sector offset of the second path table
|
||||
/// </summary>
|
||||
public int OffsetOfSecondLittleEndianPathTable;
|
||||
/// <summary>
|
||||
/// (ISO9660 only) Sector offset of the first path table
|
||||
/// </summary>
|
||||
public int OffsetOfFirstBigEndianPathTable;
|
||||
/// <summary>
|
||||
/// (ISO9660 only) Sector offset of the second path table
|
||||
/// </summary>
|
||||
public int OffsetOfSecondBigEndianPathTable;
|
||||
|
||||
/// <summary>
|
||||
/// The root directory record
|
||||
|
@ -134,19 +141,43 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
/// </summary>
|
||||
public byte[] EffectiveDateTime;
|
||||
|
||||
/// <summary>
|
||||
/// Extra reserved data
|
||||
/// </summary>
|
||||
public byte[] Reserved;
|
||||
/// <summary>
|
||||
/// (ISO9660 only) Extra reserved data
|
||||
/// </summary>
|
||||
public byte[] Reserved;
|
||||
|
||||
#endregion
|
||||
|
||||
#region Construction
|
||||
// CD-Interactive only
|
||||
|
||||
/// <summary>
|
||||
/// The bits of this field are numbered from 0 to 7 starting with the least significant bit
|
||||
/// BitPosition 0: A value of 0 = the coded character set identifier field specifies only an escape sequence registered according to ISO 2375
|
||||
/// A value of 1 = the coded character set identifier field specifies only an escape sequence NOT registered according to ISO 2375
|
||||
/// BitPostion 1-7: All bits are 0 (reserved for future standardization)
|
||||
/// </summary>
|
||||
public byte VolumeFlags;
|
||||
/// <summary>
|
||||
/// This field specifies one escape sequence according to the International Register of Coded Character Sets to be used with escape
|
||||
/// sequence standards for recording.The ESC character, which is the first character of all sequences, shall be omitted when recording this field
|
||||
/// </summary>
|
||||
public byte[] CodedCharSetIdent;
|
||||
/// <summary>
|
||||
/// The block address of the first block of the system Path Table is kept in this field
|
||||
/// </summary>
|
||||
public int AddressOfPathTable;
|
||||
/// <summary>
|
||||
/// This number is used to indicate the revision number of the file structure standard to which the
|
||||
/// directory search files conform. It is set to one
|
||||
/// </summary>
|
||||
|
||||
/// <summary>
|
||||
/// Constructor.
|
||||
/// </summary>
|
||||
public ISOVolumeDescriptor()
|
||||
#endregion
|
||||
|
||||
#region Construction
|
||||
|
||||
/// <summary>
|
||||
/// Constructor.
|
||||
/// </summary>
|
||||
public ISOVolumeDescriptor()
|
||||
{
|
||||
// Set everything to the default value
|
||||
this.Type = 0;
|
||||
|
@ -183,6 +214,11 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
this.EffectiveDateTime = new byte[LENGTH_TIME];
|
||||
|
||||
this.Reserved = new byte[LENGTH_RESERVED];
|
||||
|
||||
// CD-I specific
|
||||
this.VolumeFlags = 0;
|
||||
this.CodedCharSetIdent = new byte[LENGTH_SHORT_IDENTIFIER];
|
||||
this.AddressOfPathTable = 0;
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
@ -190,137 +226,391 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
#region Parsing
|
||||
|
||||
/// <summary>
|
||||
/// Parse the volume descriptor header.
|
||||
/// Start parsing the volume descriptor header.
|
||||
/// </summary>
|
||||
/// <param name="s">The stream to parse from.</param>
|
||||
public bool Parse(Stream s)
|
||||
{
|
||||
EndianBitConverter bc = EndianBitConverter.CreateForLittleEndian();
|
||||
EndianBitConverter bcBig = EndianBitConverter.CreateForBigEndian();
|
||||
long startPosition = s.Position;
|
||||
byte[] buffer = new byte[ISOFile.SECTOR_SIZE];
|
||||
|
||||
// Read the entire structure
|
||||
s.Read(buffer, 0, ISOFile.SECTOR_SIZE);
|
||||
|
||||
// Get the type
|
||||
this.Type = buffer[0];
|
||||
// Parse based on format
|
||||
byte[] header = bc.ReadBytes(buffer, 0, ISOFile.SECTOR_SIZE);
|
||||
if (GetISO9660(header))
|
||||
{
|
||||
ParseISO9660(s);
|
||||
return true;
|
||||
}
|
||||
if (GetCDI(header))
|
||||
{
|
||||
ParseCDInteractive(s);
|
||||
return true;
|
||||
}
|
||||
|
||||
//zero 24-jun-2013 - validate
|
||||
// "CD001" + 0x01
|
||||
if (buffer[1] == 'C' && buffer[2] == 'D' && buffer[3] == '0' && buffer[4] == '0' && buffer[5] == '1' && buffer[6] == 0x01)
|
||||
{
|
||||
//it seems to be a valid volume descriptor
|
||||
}
|
||||
else
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
// Handle the primary volume information
|
||||
if (this.Type == 1)
|
||||
{
|
||||
int cursor = 8;
|
||||
// Get the system identifier
|
||||
Array.Copy(buffer, cursor,
|
||||
this.SystemIdentifier, 0, LENGTH_SHORT_IDENTIFIER);
|
||||
cursor += LENGTH_SHORT_IDENTIFIER;
|
||||
|
||||
// Get the volume identifier
|
||||
Array.Copy(buffer, cursor,
|
||||
this.VolumeIdentifier, 0, LENGTH_SHORT_IDENTIFIER);
|
||||
cursor += LENGTH_SHORT_IDENTIFIER;
|
||||
|
||||
cursor += 8;
|
||||
|
||||
// Get the total number of sectors
|
||||
this.NumberOfSectors = bc.ToInt32(buffer, cursor);
|
||||
cursor += 8;
|
||||
|
||||
cursor += 32;
|
||||
|
||||
this.VolumeSetSize = bc.ToInt16(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.VolumeSequenceNumber = bc.ToInt16(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.SectorSize = bc.ToInt16(buffer, cursor);
|
||||
cursor += 4;
|
||||
|
||||
this.PathTableSize = bc.ToInt32(buffer, cursor);
|
||||
cursor += 8;
|
||||
this.OffsetOfFirstLittleEndianPathTable = bc.ToInt32(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.OffsetOfSecondLittleEndianPathTable = bc.ToInt32(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.OffsetOfFirstLittleEndianPathTable = bcBig.ToInt32(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.OffsetOfSecondLittleEndianPathTable = bcBig.ToInt32(buffer, cursor);
|
||||
cursor += 4;
|
||||
|
||||
this.RootDirectoryRecord.Parse(buffer, cursor);
|
||||
cursor += LENGTH_ROOT_DIRECTORY_RECORD;
|
||||
|
||||
Array.Copy(buffer, cursor,
|
||||
this.VolumeSetIdentifier, 0, LENGTH_LONG_IDENTIFIER);
|
||||
cursor += LENGTH_LONG_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.PublisherIdentifier, 0, LENGTH_LONG_IDENTIFIER);
|
||||
cursor += LENGTH_LONG_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.DataPreparerIdentifier, 0, LENGTH_LONG_IDENTIFIER);
|
||||
cursor += LENGTH_LONG_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.ApplicationIdentifier, 0, LENGTH_LONG_IDENTIFIER);
|
||||
cursor += LENGTH_LONG_IDENTIFIER;
|
||||
|
||||
Array.Copy(buffer, cursor,
|
||||
this.CopyrightFileIdentifier, 0, LENGTH_IDENTIFIER);
|
||||
cursor += LENGTH_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.AbstractFileIdentifier, 0, LENGTH_IDENTIFIER);
|
||||
cursor += LENGTH_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.BibliographicalFileIdentifier, 0, LENGTH_IDENTIFIER);
|
||||
cursor += LENGTH_IDENTIFIER;
|
||||
|
||||
Array.Copy(buffer, cursor,
|
||||
this.VolumeCreationDateTime, 0, LENGTH_TIME);
|
||||
cursor += LENGTH_TIME;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.LastModifiedDateTime, 0, LENGTH_TIME);
|
||||
cursor += LENGTH_TIME;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.ExpirationDateTime, 0, LENGTH_TIME);
|
||||
cursor += LENGTH_TIME;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.EffectiveDateTime, 0, LENGTH_TIME);
|
||||
cursor += LENGTH_TIME;
|
||||
|
||||
cursor += 1;
|
||||
|
||||
cursor += 1;
|
||||
|
||||
Array.Copy(buffer, cursor,
|
||||
this.Reserved, 0, LENGTH_RESERVED);
|
||||
cursor += LENGTH_RESERVED;
|
||||
}
|
||||
|
||||
return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
#endregion
|
||||
public void ParseISO9660(Stream s)
|
||||
{
|
||||
long startPosition = s.Position;
|
||||
byte[] buffer = new byte[ISOFile.SECTOR_SIZE];
|
||||
s.Position = startPosition - ISOFile.SECTOR_SIZE;
|
||||
|
||||
#region Type Information
|
||||
// Read the entire structure
|
||||
s.Read(buffer, 0, ISOFile.SECTOR_SIZE);
|
||||
|
||||
/// <summary>
|
||||
/// Returns true if this is the terminator volume descriptor.
|
||||
/// </summary>
|
||||
/// <returns>True if the terminator.</returns>
|
||||
public bool IsTerminator()
|
||||
// Get the type
|
||||
this.Type = buffer[0];
|
||||
|
||||
// Handle the primary volume information
|
||||
if (this.Type == 1)
|
||||
{
|
||||
int cursor = 8;
|
||||
// Get the system identifier
|
||||
Array.Copy(buffer, cursor,
|
||||
this.SystemIdentifier, 0, LENGTH_SHORT_IDENTIFIER);
|
||||
cursor += LENGTH_SHORT_IDENTIFIER;
|
||||
|
||||
// Get the volume identifier
|
||||
Array.Copy(buffer, cursor,
|
||||
this.VolumeIdentifier, 0, LENGTH_SHORT_IDENTIFIER);
|
||||
cursor += LENGTH_SHORT_IDENTIFIER;
|
||||
|
||||
cursor += 8;
|
||||
|
||||
// Get the total number of sectors
|
||||
this.NumberOfSectors = bc.ToInt32(buffer, cursor);
|
||||
cursor += 8;
|
||||
|
||||
cursor += 32;
|
||||
|
||||
this.VolumeSetSize = bc.ToInt16(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.VolumeSequenceNumber = bc.ToInt16(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.SectorSize = bc.ToInt16(buffer, cursor);
|
||||
cursor += 4;
|
||||
|
||||
this.PathTableSize = bc.ToInt32(buffer, cursor);
|
||||
cursor += 8;
|
||||
this.OffsetOfFirstLittleEndianPathTable = bc.ToInt32(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.OffsetOfSecondLittleEndianPathTable = bc.ToInt32(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.OffsetOfFirstLittleEndianPathTable = bcBig.ToInt32(buffer, cursor);
|
||||
cursor += 4;
|
||||
this.OffsetOfSecondLittleEndianPathTable = bcBig.ToInt32(buffer, cursor);
|
||||
cursor += 4;
|
||||
|
||||
this.RootDirectoryRecord.Parse(buffer, cursor);
|
||||
cursor += LENGTH_ROOT_DIRECTORY_RECORD;
|
||||
|
||||
Array.Copy(buffer, cursor,
|
||||
this.VolumeSetIdentifier, 0, LENGTH_LONG_IDENTIFIER);
|
||||
cursor += LENGTH_LONG_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.PublisherIdentifier, 0, LENGTH_LONG_IDENTIFIER);
|
||||
cursor += LENGTH_LONG_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.DataPreparerIdentifier, 0, LENGTH_LONG_IDENTIFIER);
|
||||
cursor += LENGTH_LONG_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.ApplicationIdentifier, 0, LENGTH_LONG_IDENTIFIER);
|
||||
cursor += LENGTH_LONG_IDENTIFIER;
|
||||
|
||||
Array.Copy(buffer, cursor,
|
||||
this.CopyrightFileIdentifier, 0, LENGTH_IDENTIFIER);
|
||||
cursor += LENGTH_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.AbstractFileIdentifier, 0, LENGTH_IDENTIFIER);
|
||||
cursor += LENGTH_IDENTIFIER;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.BibliographicalFileIdentifier, 0, LENGTH_IDENTIFIER);
|
||||
cursor += LENGTH_IDENTIFIER;
|
||||
|
||||
Array.Copy(buffer, cursor,
|
||||
this.VolumeCreationDateTime, 0, LENGTH_TIME);
|
||||
cursor += LENGTH_TIME;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.LastModifiedDateTime, 0, LENGTH_TIME);
|
||||
cursor += LENGTH_TIME;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.ExpirationDateTime, 0, LENGTH_TIME);
|
||||
cursor += LENGTH_TIME;
|
||||
Array.Copy(buffer, cursor,
|
||||
this.EffectiveDateTime, 0, LENGTH_TIME);
|
||||
cursor += LENGTH_TIME;
|
||||
|
||||
cursor += 1;
|
||||
|
||||
cursor += 1;
|
||||
|
||||
Array.Copy(buffer, cursor,
|
||||
this.Reserved, 0, LENGTH_RESERVED);
|
||||
cursor += LENGTH_RESERVED;
|
||||
}
|
||||
}
|
||||
|
||||
public void ParseCDInteractive(Stream s)
|
||||
{
|
||||
/* From the Green Book Spec
|
||||
* BP (byte position) obviously is n+1
|
||||
|
||||
BP Size in Bytes Description
|
||||
1 1 Disc Label Record Type
|
||||
2 5 Volume Structure Standard ID
|
||||
7 1 Volume Structure Version number
|
||||
8 1 Volume flags
|
||||
9 32 System identifier
|
||||
41 32 Volume identifier
|
||||
73 12 Reserved
|
||||
85 4 Volume space size
|
||||
89 32 Coded Character Set identifier
|
||||
121 2 Reserved
|
||||
123 2 Number of Volumes in Album
|
||||
125 2 Reserved
|
||||
127 2 Album Set Sequence number
|
||||
129 2 Reserved
|
||||
131 2 Logical Block size
|
||||
133 4 Reserved
|
||||
137 4 Path Table size
|
||||
141 8 Reserved
|
||||
149 4 Address of Path Table
|
||||
153 38 Reserved
|
||||
191 128 Album identifier
|
||||
319 128 Publisher identifier
|
||||
447 128 Data Preparer identifier
|
||||
575 128 Application identifier
|
||||
703 32 Copyright file name
|
||||
735 5 Reserved
|
||||
740 32 Abstract file name
|
||||
772 5 Reserved
|
||||
777 32 Bibliographic file name
|
||||
809 5 Reserved
|
||||
814 16 Creation date and time
|
||||
830 1 Reserved
|
||||
831 16 Modification date and time
|
||||
847 1 Reserved
|
||||
848 16 Expiration date and time
|
||||
864 1 Reserved
|
||||
865 16 Effective date and time
|
||||
881 1 Reserved
|
||||
882 1 File Structure Standard Version number
|
||||
883 1 Reserved
|
||||
884 512 Application use
|
||||
1396 653 Reserved */
|
||||
|
||||
long startPosition = s.Position;
|
||||
byte[] buffer = new byte[ISOFile.SECTOR_SIZE];
|
||||
s.Position = startPosition - ISOFile.SECTOR_SIZE;
|
||||
|
||||
// Read the entire structure
|
||||
s.Read(buffer, 0, ISOFile.SECTOR_SIZE);
|
||||
|
||||
// Get the type
|
||||
this.Type = buffer[0];
|
||||
|
||||
// Handle the primary volume information
|
||||
if (this.Type == 1)
|
||||
{
|
||||
this.VolumeFlags = buffer[7];
|
||||
this.SystemIdentifier = bc.ReadBytes(buffer, 8, LENGTH_SHORT_IDENTIFIER);
|
||||
this.VolumeIdentifier = bc.ReadBytes(buffer, 40, LENGTH_SHORT_IDENTIFIER);
|
||||
this.NumberOfSectors = bcBig.ReadIntValue(buffer, 84, 4);
|
||||
this.CodedCharSetIdent = bc.ReadBytes(buffer, 88, LENGTH_SHORT_IDENTIFIER);
|
||||
this.VolumeSetSize = bcBig.ReadIntValue(buffer, 122, 2);
|
||||
this.VolumeSequenceNumber = bcBig.ReadIntValue(buffer, 126, 2);
|
||||
this.SectorSize = bcBig.ReadIntValue(buffer, 130, 2);
|
||||
this.PathTableSize = bcBig.ReadIntValue(buffer, 136, 4);
|
||||
this.AddressOfPathTable = bcBig.ReadIntValue(buffer, 148, 4);
|
||||
|
||||
this.VolumeSetIdentifier = bc.ReadBytes(buffer, 190, LENGTH_LONG_IDENTIFIER);
|
||||
this.PublisherIdentifier = bc.ReadBytes(buffer, 318, LENGTH_LONG_IDENTIFIER);
|
||||
this.DataPreparerIdentifier = bc.ReadBytes(buffer, 446, LENGTH_LONG_IDENTIFIER);
|
||||
this.ApplicationIdentifier = bc.ReadBytes(buffer, 574, LENGTH_LONG_IDENTIFIER);
|
||||
|
||||
this.CopyrightFileIdentifier = bc.ReadBytes(buffer, 702, LENGTH_SHORT_IDENTIFIER);
|
||||
this.AbstractFileIdentifier = bc.ReadBytes(buffer, 739, LENGTH_SHORT_IDENTIFIER);
|
||||
this.BibliographicalFileIdentifier = bc.ReadBytes(buffer, 776, LENGTH_SHORT_IDENTIFIER);
|
||||
|
||||
this.VolumeCreationDateTime = bc.ReadBytes(buffer, 813, 16);
|
||||
this.LastModifiedDateTime = bc.ReadBytes(buffer, 830, 16);
|
||||
this.ExpirationDateTime = bc.ReadBytes(buffer, 847, 16);
|
||||
this.EffectiveDateTime = bc.ReadBytes(buffer, 864, 16);
|
||||
|
||||
// save current position
|
||||
long pos = s.Position;
|
||||
|
||||
// get path table records
|
||||
s.Position = ISOFile.SECTOR_SIZE * this.AddressOfPathTable;
|
||||
ISOFile.CDIPathTable = CDIPathNode.ParsePathTable(s, this.PathTableSize);
|
||||
|
||||
// read the root dir record
|
||||
s.Position = ISOFile.SECTOR_SIZE * ISOFile.CDIPathTable[0].DirectoryBlockAddress;
|
||||
s.Read(buffer, 0, ISOFile.SECTOR_SIZE);
|
||||
this.RootDirectoryRecord.Parse(buffer, 0);
|
||||
|
||||
// go back to where we were
|
||||
s.Position = pos;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Detect ISO9660
|
||||
/// </summary>
|
||||
/// <param name="buffer"></param>
|
||||
/// <returns></returns>
|
||||
public bool GetISO9660(byte[] buffer)
|
||||
{
|
||||
//zero 24-jun-2013 - validate ISO9660
|
||||
// "CD001" + 0x01
|
||||
if (buffer[1] == 'C' && buffer[2] == 'D' && buffer[3] == '0' && buffer[4] == '0' && buffer[5] == '1' && buffer[6] == 0x01)
|
||||
{
|
||||
ISOFile.Format = ISOFile.ISOFormat.ISO9660;
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Detect CD-I
|
||||
/// </summary>
|
||||
/// <param name="buffer"></param>
|
||||
/// <returns></returns>
|
||||
public bool GetCDI(byte[] buffer)
|
||||
{
|
||||
// CD-Interactive
|
||||
if (Encoding.ASCII.GetString(bc.ReadBytes(buffer, 1, 5)).Contains("CD-I"))
|
||||
{
|
||||
ISOFile.Format = ISOFile.ISOFormat.CDInteractive;
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Type Information
|
||||
|
||||
/// <summary>
|
||||
/// Returns true if this is the terminator volume descriptor.
|
||||
/// </summary>
|
||||
/// <returns>True if the terminator.</returns>
|
||||
public bool IsTerminator()
|
||||
{
|
||||
return (this.Type == 255);
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Represents a Directory Path Table entry on a CD-I disc
|
||||
/// </summary>
|
||||
public class CDIPathNode
|
||||
{
|
||||
#region Public Properties
|
||||
|
||||
/// <summary>
|
||||
/// The length of the directory name.
|
||||
/// </summary>
|
||||
public byte NameLength;
|
||||
|
||||
/// <summary>
|
||||
/// This is the length of the Extended Attribute record
|
||||
/// </summary>
|
||||
public byte ExtendedAttribRecordLength;
|
||||
|
||||
/// <summary>
|
||||
/// This field contains the beginning logical block number (LBN) of the directory file on disc
|
||||
/// </summary>
|
||||
public int DirectoryBlockAddress;
|
||||
|
||||
/// <summary>
|
||||
/// This is the number (relative to the beginning of the Path Table) of this directory's parent
|
||||
/// </summary>
|
||||
public int ParentDirectoryNumber;
|
||||
|
||||
/// <summary>
|
||||
/// The directory name.
|
||||
/// This variable length field is used to store the actual text representing the name of the directory.
|
||||
/// If the length of the file name is odd, a null padding byte is added to make the size of the Path Table record even.
|
||||
/// The padding byte is not included in the name size field.
|
||||
/// </summary>
|
||||
public string Name;
|
||||
|
||||
#endregion
|
||||
|
||||
#region Construction
|
||||
|
||||
/// <summary>
|
||||
/// Empty Constructor
|
||||
/// </summary>
|
||||
public CDIPathNode()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region Parsing
|
||||
|
||||
/*
|
||||
BP Size in bytes Description
|
||||
1 1 Name size
|
||||
2 1 Extended Attribute record length
|
||||
3 4 Directory block address
|
||||
7 2 Parent Directory number
|
||||
9 n Directory file name
|
||||
*/
|
||||
|
||||
public static List<CDIPathNode> ParsePathTable(Stream s, int PathTableSize)
|
||||
{
|
||||
EndianBitConverter bc = EndianBitConverter.CreateForLittleEndian();
|
||||
EndianBitConverter bcBig = EndianBitConverter.CreateForBigEndian();
|
||||
|
||||
byte[] buffer = new byte[ISOFile.SECTOR_SIZE];
|
||||
|
||||
// Read the entire structure
|
||||
s.Read(buffer, 0, ISOFile.SECTOR_SIZE);
|
||||
|
||||
int startCursor = 0;
|
||||
|
||||
List<CDIPathNode> pathNodes = new List<CDIPathNode>();
|
||||
|
||||
int pad = 0;
|
||||
|
||||
do
|
||||
{
|
||||
CDIPathNode node = new CDIPathNode();
|
||||
byte[] data = bc.ReadBytes(buffer, startCursor, ISOFile.SECTOR_SIZE - startCursor);
|
||||
node.NameLength = data[0];
|
||||
|
||||
node.ExtendedAttribRecordLength = data[1];
|
||||
node.DirectoryBlockAddress = bcBig.ReadIntValue(data, 2, 4);
|
||||
node.ParentDirectoryNumber = bcBig.ReadIntValue(data, 6, 2);
|
||||
node.Name = Encoding.ASCII.GetString(bc.ReadBytes(data, 8, data[0]));
|
||||
|
||||
// if nameLength is odd a padding byte must be added
|
||||
|
||||
if (node.NameLength % 2 != 0)
|
||||
pad = 1;
|
||||
|
||||
pathNodes.Add(node);
|
||||
|
||||
startCursor += node.NameLength + 8;
|
||||
|
||||
} while (startCursor < PathTableSize + pad);
|
||||
|
||||
|
||||
return pathNodes;
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,907 @@
|
|||
using System;
|
||||
using System.Text;
|
||||
using System.IO;
|
||||
using System.Globalization;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
using System.Runtime.InteropServices;
|
||||
|
||||
namespace BizHawk.Emulation.DiscSystem
|
||||
{
|
||||
/// <summary>
|
||||
/// Parsing Alcohol 120% files
|
||||
/// Info taken from:
|
||||
/// * http://forum.redump.org/post/41803/#p41803
|
||||
/// * Libmirage image-mds parser - https://sourceforge.net/projects/cdemu/files/libmirage/
|
||||
/// * DiscImageChef - https://github.com/claunia/DiscImageChef/blob/master/DiscImageChef.DiscImages/Alcohol120.cs
|
||||
/// </summary>
|
||||
public class MDS_Format
|
||||
{
|
||||
/// <summary>
|
||||
/// A loose representation of an Alcohol 120 .mds file (with a few extras)
|
||||
/// </summary>
|
||||
public class AFile
|
||||
{
|
||||
/// <summary>
|
||||
/// Full path to the MDS file
|
||||
/// </summary>
|
||||
public string MDSPath;
|
||||
|
||||
/// <summary>
|
||||
/// MDS Header
|
||||
/// </summary>
|
||||
public AHeader Header = new AHeader();
|
||||
|
||||
/// <summary>
|
||||
/// List of MDS session blocks
|
||||
/// </summary>
|
||||
public List<ASession> Sessions = new List<ASession>();
|
||||
|
||||
/// <summary>
|
||||
/// List of track blocks
|
||||
/// </summary>
|
||||
public List<ATrack> Tracks = new List<ATrack>();
|
||||
|
||||
/// <summary>
|
||||
/// Current parsed session objects
|
||||
/// </summary>
|
||||
public List<Session> ParsedSession = new List<Session>();
|
||||
|
||||
/// <summary>
|
||||
/// Calculated MDS TOC entries (still to be parsed into BizHawk)
|
||||
/// </summary>
|
||||
public List<ATOCEntry> TOCEntries = new List<ATOCEntry>();
|
||||
|
||||
}
|
||||
|
||||
public class AHeader
|
||||
{
|
||||
/// <summary>
|
||||
/// Standard alcohol 120% signature - usually "MEDIA DESCRIPTOR"
|
||||
/// </summary>
|
||||
public string Signature; // 16 bytes
|
||||
|
||||
/// <summary>
|
||||
/// Alcohol version?
|
||||
/// </summary>
|
||||
public byte[] Version; // 2 bytes
|
||||
|
||||
/// <summary>
|
||||
/// The medium type
|
||||
/// * 0x00 - CD
|
||||
/// * 0x01 - CD-R
|
||||
/// * 0x02 - CD-RW
|
||||
/// * 0x10 - DVD
|
||||
/// * 0x12 - DVD-R
|
||||
/// </summary>
|
||||
public int Medium;
|
||||
|
||||
/// <summary>
|
||||
/// Number of sessions
|
||||
/// </summary>
|
||||
public int SessionCount;
|
||||
|
||||
/// <summary>
|
||||
/// Burst Cutting Area length
|
||||
/// </summary>
|
||||
public int BCALength;
|
||||
|
||||
/// <summary>
|
||||
/// Burst Cutting Area data offset
|
||||
/// </summary>
|
||||
public Int64 BCAOffset;
|
||||
|
||||
/// <summary>
|
||||
/// Offset to disc (DVD?) structures
|
||||
/// </summary>
|
||||
public Int64 StructureOffset;
|
||||
|
||||
/// <summary>
|
||||
/// Offset to the first session block
|
||||
/// </summary>
|
||||
public Int64 SessionOffset;
|
||||
|
||||
/// <summary>
|
||||
/// Data Position Measurement offset
|
||||
/// </summary>
|
||||
public Int64 DPMOffset;
|
||||
|
||||
/// <summary>
|
||||
/// Parse mds stream for the header
|
||||
/// </summary>
|
||||
/// <param name="stream"></param>
|
||||
/// <returns></returns>
|
||||
public AHeader Parse(Stream stream)
|
||||
{
|
||||
EndianBitConverter bc = EndianBitConverter.CreateForLittleEndian();
|
||||
EndianBitConverter bcBig = EndianBitConverter.CreateForBigEndian();
|
||||
|
||||
byte[] header = new byte[88];
|
||||
stream.Read(header, 0, 88);
|
||||
|
||||
this.Signature = Encoding.ASCII.GetString(header.Take(16).ToArray());
|
||||
this.Version = header.Skip(16).Take(2).ToArray();
|
||||
this.Medium = bc.ToInt16(header.Skip(18).Take(2).ToArray());
|
||||
this.SessionCount = bc.ToInt16(header.Skip(20).Take(2).ToArray());
|
||||
this.BCALength = bc.ToInt16(header.Skip(26).Take(2).ToArray());
|
||||
this.BCAOffset = bc.ToInt32(header.Skip(36).Take(4).ToArray());
|
||||
this.StructureOffset = bc.ToInt32(header.Skip(64).Take(4).ToArray());
|
||||
this.SessionOffset = bc.ToInt32(header.Skip(80).Take(4).ToArray());
|
||||
this.DPMOffset = bc.ToInt32(header.Skip(84).Take(4).ToArray());
|
||||
|
||||
return this;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// MDS session block representation
|
||||
/// </summary>
|
||||
public class ASession
|
||||
{
|
||||
public int SessionStart; /* Session's start address */
|
||||
public int SessionEnd; /* Session's end address */
|
||||
public int SessionNumber; /* Session number */
|
||||
public byte AllBlocks; /* Number of all data blocks. */
|
||||
public byte NonTrackBlocks; /* Number of lead-in data blocks */
|
||||
public int FirstTrack; /* First track in session */
|
||||
public int LastTrack; /* Last track in session */
|
||||
public Int64 TrackOffset; /* Offset of lead-in+regular track data blocks. */
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Representation of an MDS track block
|
||||
/// For convenience (and extra confusion) this also holds the track extrablock, filename(footer) block infos
|
||||
/// as well as the calculated image filepath as specified in the MDS file
|
||||
/// </summary>
|
||||
public class ATrack
|
||||
{
|
||||
/// <summary>
|
||||
/// The specified data mode
|
||||
/// 0x00 - None (no data)
|
||||
/// 0x02 - DVD
|
||||
/// 0xA9 - Audio
|
||||
/// 0xAA - Mode1
|
||||
/// 0xAB - Mode2
|
||||
/// 0xAC - Mode2 Form1
|
||||
/// 0xAD - Mode2 Form2
|
||||
/// </summary>
|
||||
public byte Mode; /* Track mode */
|
||||
|
||||
/// <summary>
|
||||
/// Subchannel mode for the track (0x00 = None, 0x08 = Interleaved)
|
||||
/// </summary>
|
||||
public byte SubMode; /* Subchannel mode */
|
||||
|
||||
/* These are the fields from Sub-channel Q information, which are
|
||||
also returned in full TOC by READ TOC/PMA/ATIP command */
|
||||
public int ADR_Control; /* Adr/Ctl */
|
||||
public int TrackNo; /* Track number field */
|
||||
public int Point; /* Point field (= track number for track entries) */
|
||||
public int AMin; /* Min */
|
||||
public int ASec; /* Sec */
|
||||
public int AFrame; /* Frame */
|
||||
public int Zero; /* Zero */
|
||||
public int PMin; /* PMin */
|
||||
public int PSec; /* PSec */
|
||||
public int PFrame; /* PFrame */
|
||||
|
||||
public Int64 ExtraOffset; /* Start offset of this track's extra block. */
|
||||
public int SectorSize; /* Sector size. */
|
||||
public Int64 PLBA; /* Track start sector (PLBA). */
|
||||
public ulong StartOffset; /* Track start offset (from beginning of MDS file) */
|
||||
public Int64 Files; /* Number of filenames for this track */
|
||||
public Int64 FooterOffset; /* Start offset of footer (from beginning of MDS file) */
|
||||
|
||||
/// <summary>
|
||||
/// Track extra block
|
||||
/// </summary>
|
||||
public ATrackExtra ExtraBlock = new ATrackExtra();
|
||||
|
||||
/// <summary>
|
||||
/// List of footer(filename) blocks for this track
|
||||
/// </summary>
|
||||
public List<AFooter> FooterBlocks = new List<AFooter>();
|
||||
|
||||
/// <summary>
|
||||
/// List of the calculated full paths to this track's image file
|
||||
/// The MDS file itself may contain a filename, or just an *.extension
|
||||
/// </summary>
|
||||
public List<string> ImageFileNamePaths = new List<string>();
|
||||
|
||||
public int BlobIndex;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Extra track block
|
||||
/// </summary>
|
||||
public class ATrackExtra
|
||||
{
|
||||
public Int64 Pregap; /* Number of sectors in pregap. */
|
||||
public Int64 Sectors; /* Number of sectors in track. */
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Footer (filename) block - potentially one for every track
|
||||
/// </summary>
|
||||
public class AFooter
|
||||
{
|
||||
public Int64 FilenameOffset; /* Start offset of image filename string (from beginning of mds file) */
|
||||
public Int64 WideChar; /* Seems to be set to 1 if widechar filename is used */
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Represents a parsed MDS TOC entry
|
||||
/// </summary>
|
||||
public class ATOCEntry
|
||||
{
|
||||
public ATOCEntry(int entryNum)
|
||||
{
|
||||
EntryNum = entryNum;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// these should be 0-indexed
|
||||
/// </summary>
|
||||
public int EntryNum;
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// 1-indexed - the session that this entry belongs to
|
||||
/// </summary>
|
||||
public int Session;
|
||||
|
||||
/// <summary>
|
||||
/// this seems just to be the LBA corresponding to AMIN:ASEC:AFRAME (give or take 150). It's not stored on the disc, and it's redundant.
|
||||
/// </summary>
|
||||
//public int ALBA;
|
||||
|
||||
/// <summary>
|
||||
/// this seems just to be the LBA corresponding to PMIN:PSEC:PFRAME (give or take 150).
|
||||
/// </summary>
|
||||
public int PLBA;
|
||||
|
||||
//these correspond pretty directly to values in the Q subchannel fields
|
||||
//NOTE: they're specified as absolute MSF. That means, they're 2 seconds off from what they should be when viewed as final TOC values
|
||||
public int ADR_Control;
|
||||
public int TrackNo;
|
||||
public int Point;
|
||||
public int AMin;
|
||||
public int ASec;
|
||||
public int AFrame;
|
||||
public int Zero;
|
||||
public int PMin;
|
||||
public int PSec;
|
||||
public int PFrame;
|
||||
|
||||
|
||||
public int SectorSize;
|
||||
public long TrackOffset;
|
||||
|
||||
/// <summary>
|
||||
/// List of the calculated full paths to this track's image file
|
||||
/// The MDS file itself may contain a filename, or just an *.extension
|
||||
/// </summary>
|
||||
public List<string> ImageFileNamePaths = new List<string>();
|
||||
|
||||
/// <summary>
|
||||
/// Track extra block
|
||||
/// </summary>
|
||||
public ATrackExtra ExtraBlock = new ATrackExtra();
|
||||
|
||||
public int BlobIndex;
|
||||
}
|
||||
|
||||
public AFile Parse(Stream stream)
|
||||
{
|
||||
EndianBitConverter bc = EndianBitConverter.CreateForLittleEndian();
|
||||
EndianBitConverter bcBig = EndianBitConverter.CreateForBigEndian();
|
||||
bool isDvd = false;
|
||||
|
||||
AFile aFile = new AFile();
|
||||
|
||||
aFile.MDSPath = (stream as FileStream).Name;
|
||||
|
||||
stream.Seek(0, SeekOrigin.Begin);
|
||||
|
||||
// check whether the header in the mds file is long enough
|
||||
if (stream.Length < 88) throw new MDSParseException("Malformed MDS format: The descriptor file does not appear to be long enough.");
|
||||
|
||||
// parse header
|
||||
aFile.Header = aFile.Header.Parse(stream);
|
||||
|
||||
// parse sessions
|
||||
Dictionary<int, ASession> aSessions = new Dictionary<int, ASession>();
|
||||
|
||||
stream.Seek(aFile.Header.SessionOffset, SeekOrigin.Begin);
|
||||
for (int se = 0; se < aFile.Header.SessionCount; se++)
|
||||
{
|
||||
byte[] sessionHeader = new byte[24];
|
||||
stream.Read(sessionHeader, 0, 24);
|
||||
//sessionHeader.Reverse().ToArray();
|
||||
|
||||
ASession session = new ASession();
|
||||
|
||||
session.SessionStart = bc.ToInt32(sessionHeader.Take(4).ToArray());
|
||||
session.SessionEnd = bc.ToInt32(sessionHeader.Skip(4).Take(4).ToArray());
|
||||
session.SessionNumber = bc.ToInt16(sessionHeader.Skip(8).Take(2).ToArray());
|
||||
session.AllBlocks = sessionHeader[10];
|
||||
session.NonTrackBlocks = sessionHeader[11];
|
||||
session.FirstTrack = bc.ToInt16(sessionHeader.Skip(12).Take(2).ToArray());
|
||||
session.LastTrack = bc.ToInt16(sessionHeader.Skip(14).Take(2).ToArray());
|
||||
session.TrackOffset = bc.ToInt32(sessionHeader.Skip(20).Take(4).ToArray());
|
||||
|
||||
|
||||
//mdsf.Sessions.Add(session);
|
||||
aSessions.Add(session.SessionNumber, session);
|
||||
}
|
||||
|
||||
long footerOffset = 0;
|
||||
|
||||
// parse track blocks
|
||||
Dictionary<int, ATrack> aTracks = new Dictionary<int, ATrack>();
|
||||
|
||||
// iterate through each session block
|
||||
foreach (ASession session in aSessions.Values)
|
||||
{
|
||||
stream.Seek(session.TrackOffset, SeekOrigin.Begin);
|
||||
//Dictionary<int, ATrack> sessionToc = new Dictionary<int, ATrack>();
|
||||
|
||||
// iterate through every block specified in each session
|
||||
for (int bl = 0; bl < session.AllBlocks; bl++)
|
||||
{
|
||||
byte[] trackHeader;
|
||||
ATrack track = new ATrack();
|
||||
|
||||
trackHeader = new byte[80];
|
||||
|
||||
stream.Read(trackHeader, 0, 80);
|
||||
|
||||
track.Mode = trackHeader[0];
|
||||
track.SubMode = trackHeader[1];
|
||||
track.ADR_Control = trackHeader[2];
|
||||
track.TrackNo = trackHeader[3];
|
||||
track.Point = trackHeader[4];
|
||||
track.AMin = trackHeader[5];
|
||||
track.ASec = trackHeader[6];
|
||||
track.AFrame = trackHeader[7];
|
||||
track.Zero = trackHeader[8];
|
||||
track.PMin = trackHeader[9];
|
||||
track.PSec = trackHeader[10];
|
||||
track.PFrame = trackHeader[11];
|
||||
track.ExtraOffset = bc.ToInt32(trackHeader.Skip(12).Take(4).ToArray());
|
||||
track.SectorSize = bc.ToInt16(trackHeader.Skip(16).Take(2).ToArray());
|
||||
track.PLBA = bc.ToInt32(trackHeader.Skip(36).Take(4).ToArray());
|
||||
track.StartOffset = BitConverter.ToUInt64(trackHeader.Skip(40).Take(8).ToArray(), 0);
|
||||
track.Files = bc.ToInt32(trackHeader.Skip(48).Take(4).ToArray());
|
||||
track.FooterOffset = bc.ToInt32(trackHeader.Skip(52).Take(4).ToArray());
|
||||
|
||||
if (track.Mode == 0x02)
|
||||
{
|
||||
isDvd = true;
|
||||
throw new MDSParseException("DVD Detected. Not currently supported!");
|
||||
}
|
||||
|
||||
|
||||
// check for track extra block - this can probably be handled in a separate loop,
|
||||
// but I'll just store the current stream position then seek forward to the extra block for this track
|
||||
Int64 currPos = stream.Position;
|
||||
|
||||
// Only CDs have extra blocks - for DVDs ExtraOffset = track length
|
||||
if (track.ExtraOffset > 0 && !isDvd)
|
||||
{
|
||||
byte[] extHeader = new byte[8];
|
||||
stream.Seek(track.ExtraOffset, SeekOrigin.Begin);
|
||||
stream.Read(extHeader, 0, 8);
|
||||
track.ExtraBlock.Pregap = bc.ToInt32(extHeader.Take(4).ToArray());
|
||||
track.ExtraBlock.Sectors = bc.ToInt32(extHeader.Skip(4).Take(4).ToArray());
|
||||
stream.Seek(currPos, SeekOrigin.Begin);
|
||||
}
|
||||
else if (isDvd == true)
|
||||
{
|
||||
track.ExtraBlock.Sectors = track.ExtraOffset;
|
||||
}
|
||||
|
||||
// read the footer/filename block for this track
|
||||
currPos = stream.Position;
|
||||
long numOfFilenames = track.Files;
|
||||
for (long fi = 1; fi <= numOfFilenames; fi++)
|
||||
{
|
||||
// skip leadin/out info tracks
|
||||
if (track.FooterOffset == 0)
|
||||
continue;
|
||||
|
||||
byte[] foot = new byte[16];
|
||||
stream.Seek(track.FooterOffset, SeekOrigin.Begin);
|
||||
stream.Read(foot, 0, 16);
|
||||
|
||||
AFooter f = new AFooter();
|
||||
f.FilenameOffset = bc.ToInt32(foot.Take(4).ToArray());
|
||||
f.WideChar = bc.ToInt32(foot.Skip(4).Take(4).ToArray());
|
||||
track.FooterBlocks.Add(f);
|
||||
track.FooterBlocks = track.FooterBlocks.Distinct().ToList();
|
||||
|
||||
// parse the filename string
|
||||
string fileName = "*.mdf";
|
||||
if (f.FilenameOffset > 0)
|
||||
{
|
||||
// filename offset is present
|
||||
stream.Seek(f.FilenameOffset, SeekOrigin.Begin);
|
||||
byte[] fname;
|
||||
|
||||
if (numOfFilenames == 1)
|
||||
{
|
||||
if (aFile.Header.DPMOffset == 0)
|
||||
{
|
||||
// filename is in the remaining space to EOF
|
||||
fname = new byte[stream.Length - stream.Position];
|
||||
}
|
||||
else
|
||||
{
|
||||
// filename is in the remaining space to EOF + dpm offset
|
||||
fname = new byte[aFile.Header.DPMOffset - stream.Position];
|
||||
}
|
||||
}
|
||||
|
||||
else
|
||||
{
|
||||
// looks like each filename string is 6 bytes with a trailing \0
|
||||
fname = new byte[6];
|
||||
}
|
||||
|
||||
|
||||
// read the filename
|
||||
stream.Read(fname, 0, fname.Length);
|
||||
|
||||
// if widechar is 1 filename is stored using 16-bit, otherwise 8-bit is used
|
||||
if (f.WideChar == 1)
|
||||
fileName = Encoding.Unicode.GetString(fname).TrimEnd('\0');
|
||||
else
|
||||
fileName = Encoding.Default.GetString(fname).TrimEnd('\0');
|
||||
}
|
||||
|
||||
else
|
||||
{
|
||||
// assume an MDF file with the same name as the MDS
|
||||
}
|
||||
|
||||
string dir = Path.GetDirectoryName(aFile.MDSPath);
|
||||
|
||||
if (f.FilenameOffset == 0 ||
|
||||
string.Compare(fileName, "*.mdf", StringComparison.InvariantCultureIgnoreCase) == 0)
|
||||
{
|
||||
fileName = dir + @"\" + Path.GetFileNameWithoutExtension(aFile.MDSPath) + ".mdf";
|
||||
}
|
||||
else
|
||||
{
|
||||
fileName = dir + @"\" + fileName;
|
||||
}
|
||||
|
||||
track.ImageFileNamePaths.Add(fileName);
|
||||
track.ImageFileNamePaths = track.ImageFileNamePaths.Distinct().ToList();
|
||||
}
|
||||
|
||||
stream.Position = currPos;
|
||||
|
||||
|
||||
aTracks.Add(track.Point, track);
|
||||
aFile.Tracks.Add(track);
|
||||
|
||||
if (footerOffset == 0)
|
||||
footerOffset = track.FooterOffset;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
// build custom session object
|
||||
aFile.ParsedSession = new List<Session>();
|
||||
foreach (var s in aSessions.Values)
|
||||
{
|
||||
Session session = new Session();
|
||||
ATrack startTrack;
|
||||
ATrack endTrack;
|
||||
|
||||
if (!aTracks.TryGetValue(s.FirstTrack, out startTrack))
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
||||
if (!aTracks.TryGetValue(s.LastTrack, out endTrack))
|
||||
{
|
||||
break;
|
||||
}
|
||||
|
||||
session.StartSector = startTrack.PLBA;
|
||||
session.StartTrack = s.FirstTrack;
|
||||
session.SessionSequence = s.SessionNumber;
|
||||
session.EndSector = endTrack.PLBA + endTrack.ExtraBlock.Sectors - 1;
|
||||
session.EndTrack = s.LastTrack;
|
||||
|
||||
aFile.ParsedSession.Add(session);
|
||||
}
|
||||
|
||||
// now build the TOC object
|
||||
foreach (var se in aFile.ParsedSession)
|
||||
{
|
||||
// get the first and last tracks
|
||||
int sTrack = se.StartTrack;
|
||||
int eTrack = se.EndTrack;
|
||||
|
||||
// get list of all tracks from aTracks for this session
|
||||
var tracks = (from a in aTracks.Values
|
||||
where a.TrackNo >= sTrack || a.TrackNo <= eTrack
|
||||
orderby a.TrackNo
|
||||
select a).ToList();
|
||||
|
||||
// create the TOC entries
|
||||
foreach (var t in tracks)
|
||||
{
|
||||
ATOCEntry toc = new ATOCEntry(t.Point);
|
||||
toc.ADR_Control = t.ADR_Control;
|
||||
toc.AFrame = t.AFrame;
|
||||
toc.AMin = t.AMin;
|
||||
toc.ASec = t.ASec;
|
||||
toc.EntryNum = t.TrackNo;
|
||||
toc.PFrame = t.PFrame;
|
||||
toc.PLBA = Convert.ToInt32(t.PLBA);
|
||||
toc.PMin = t.PMin;
|
||||
toc.Point = t.Point;
|
||||
toc.PSec = t.PSec;
|
||||
toc.SectorSize = t.SectorSize;
|
||||
toc.Zero = t.Zero;
|
||||
toc.TrackOffset = Convert.ToInt64(t.StartOffset);
|
||||
toc.Session = se.SessionSequence;
|
||||
toc.ImageFileNamePaths = t.ImageFileNamePaths;
|
||||
toc.ExtraBlock = t.ExtraBlock;
|
||||
toc.BlobIndex = t.BlobIndex;
|
||||
aFile.TOCEntries.Add(toc);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
return aFile;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Custom session object
|
||||
/// </summary>
|
||||
public class Session
|
||||
{
|
||||
public long StartSector;
|
||||
public int StartTrack;
|
||||
public int SessionSequence;
|
||||
public long EndSector;
|
||||
public int EndTrack;
|
||||
}
|
||||
|
||||
|
||||
public class MDSParseException : Exception
|
||||
{
|
||||
public MDSParseException(string message) : base(message) { }
|
||||
}
|
||||
|
||||
|
||||
public class LoadResults
|
||||
{
|
||||
public List<RawTOCEntry> RawTOCEntries;
|
||||
public AFile ParsedMDSFile;
|
||||
public bool Valid;
|
||||
public Exception FailureException;
|
||||
public string MdsPath;
|
||||
}
|
||||
|
||||
public static LoadResults LoadMDSPath(string path)
|
||||
{
|
||||
LoadResults ret = new LoadResults();
|
||||
ret.MdsPath = path;
|
||||
//ret.MdfPath = Path.ChangeExtension(path, ".mdf");
|
||||
try
|
||||
{
|
||||
if (!File.Exists(path)) throw new MDSParseException("Malformed MDS format: nonexistent MDS file!");
|
||||
|
||||
AFile mdsf;
|
||||
using (var infMDS = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read))
|
||||
mdsf = new MDS_Format().Parse(infMDS);
|
||||
|
||||
ret.ParsedMDSFile = mdsf;
|
||||
|
||||
ret.Valid = true;
|
||||
}
|
||||
catch (MDSParseException ex)
|
||||
{
|
||||
ret.FailureException = ex;
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
Dictionary<int, IBlob> MountBlobs(AFile mdsf, Disc disc)
|
||||
{
|
||||
Dictionary<int, IBlob> BlobIndex = new Dictionary<int, IBlob>();
|
||||
|
||||
int count = 0;
|
||||
foreach (var track in mdsf.Tracks)
|
||||
{
|
||||
foreach (var file in track.ImageFileNamePaths.Distinct())
|
||||
{
|
||||
if (!File.Exists(file))
|
||||
throw new MDSParseException("Malformed MDS format: nonexistent image file: " + file);
|
||||
|
||||
IBlob mdfBlob = null;
|
||||
long mdfLen = -1;
|
||||
|
||||
//mount the file
|
||||
if (mdfBlob == null)
|
||||
{
|
||||
var mdfFile = new Disc.Blob_RawFile() { PhysicalPath = file };
|
||||
mdfLen = mdfFile.Length;
|
||||
mdfBlob = mdfFile;
|
||||
}
|
||||
|
||||
bool dupe = false;
|
||||
foreach (var re in disc.DisposableResources)
|
||||
{
|
||||
if (re.ToString() == mdfBlob.ToString())
|
||||
dupe = true;
|
||||
}
|
||||
|
||||
if (!dupe)
|
||||
{
|
||||
// wrap in zeropadadapter
|
||||
disc.DisposableResources.Add(mdfBlob);
|
||||
BlobIndex[count] = mdfBlob;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return BlobIndex;
|
||||
}
|
||||
|
||||
RawTOCEntry EmitRawTOCEntry(ATOCEntry entry)
|
||||
{
|
||||
BCD2 tno, ino;
|
||||
|
||||
//this should actually be zero. im not sure if this is stored as BCD2 or not
|
||||
tno = BCD2.FromDecimal(entry.TrackNo);
|
||||
|
||||
//these are special values.. I think, taken from this:
|
||||
//http://www.staff.uni-mainz.de/tacke/scsi/SCSI2-14.html
|
||||
//the CCD will contain Points as decimal values except for these specially converted decimal values which should stay as BCD.
|
||||
//Why couldn't they all be BCD? I don't know. I guess because BCD is inconvenient, but only A0 and friends have special meaning. It's confusing.
|
||||
ino = BCD2.FromDecimal(entry.Point);
|
||||
if (entry.Point == 0xA0) ino.BCDValue = 0xA0;
|
||||
else if (entry.Point == 0xA1) ino.BCDValue = 0xA1;
|
||||
else if (entry.Point == 0xA2) ino.BCDValue = 0xA2;
|
||||
|
||||
// get ADR & Control from ADR_Control byte
|
||||
byte adrc = Convert.ToByte(entry.ADR_Control);
|
||||
var Control = adrc & 0x0F;
|
||||
var ADR = adrc >> 4;
|
||||
|
||||
var q = new SubchannelQ
|
||||
{
|
||||
q_status = SubchannelQ.ComputeStatus(ADR, (EControlQ)(Control & 0xF)),
|
||||
q_tno = tno,
|
||||
q_index = ino,
|
||||
min = BCD2.FromDecimal(entry.AMin),
|
||||
sec = BCD2.FromDecimal(entry.ASec),
|
||||
frame = BCD2.FromDecimal(entry.AFrame),
|
||||
zero = (byte)entry.Zero,
|
||||
ap_min = BCD2.FromDecimal(entry.PMin),
|
||||
ap_sec = BCD2.FromDecimal(entry.PSec),
|
||||
ap_frame = BCD2.FromDecimal(entry.PFrame),
|
||||
q_crc = 0, //meaningless
|
||||
};
|
||||
|
||||
return new RawTOCEntry { QData = q };
|
||||
}
|
||||
|
||||
|
||||
/// <summary>
|
||||
/// Loads a MDS at the specified path to a Disc object
|
||||
/// </summary>
|
||||
public Disc LoadMDSToDisc(string mdsPath, DiscMountPolicy IN_DiscMountPolicy)
|
||||
{
|
||||
var loadResults = LoadMDSPath(mdsPath);
|
||||
if (!loadResults.Valid)
|
||||
throw loadResults.FailureException;
|
||||
|
||||
Disc disc = new Disc();
|
||||
|
||||
// load all blobs
|
||||
Dictionary<int, IBlob> BlobIndex = MountBlobs(loadResults.ParsedMDSFile, disc);
|
||||
|
||||
var mdsf = loadResults.ParsedMDSFile;
|
||||
|
||||
//generate DiscTOCRaw items from the ones specified in the MDS file
|
||||
disc.RawTOCEntries = new List<RawTOCEntry>();
|
||||
foreach (var entry in mdsf.TOCEntries)
|
||||
{
|
||||
disc.RawTOCEntries.Add(EmitRawTOCEntry(entry));
|
||||
}
|
||||
|
||||
//analyze the RAWTocEntries to figure out what type of track track 1 is
|
||||
var tocSynth = new Synthesize_DiscTOC_From_RawTOCEntries_Job() { Entries = disc.RawTOCEntries };
|
||||
tocSynth.Run();
|
||||
|
||||
// now build the sectors
|
||||
int currBlobIndex = 0;
|
||||
foreach (var session in mdsf.ParsedSession)
|
||||
{
|
||||
for (int i = session.StartTrack; i <= session.EndTrack; i++)
|
||||
{
|
||||
int relMSF = -1;
|
||||
|
||||
var track = mdsf.TOCEntries.Where(t => t.Point == i).FirstOrDefault();
|
||||
if (track == null)
|
||||
break;
|
||||
|
||||
// ignore the info entries
|
||||
if (track.Point == 0xA0 ||
|
||||
track.Point == 0xA1 ||
|
||||
track.Point == 0xA2)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
// get the blob(s) for this track
|
||||
// its probably a safe assumption that there will be only one blob per track,
|
||||
// but i'm still not 100% sure on this
|
||||
var tr = (from a in mdsf.TOCEntries
|
||||
where a.Point == i
|
||||
select a).FirstOrDefault();
|
||||
|
||||
if (tr == null)
|
||||
throw new MDSParseException("BLOB Error!");
|
||||
|
||||
List<string> blobstrings = new List<string>();
|
||||
foreach (var t in tr.ImageFileNamePaths)
|
||||
{
|
||||
if (!blobstrings.Contains(t))
|
||||
blobstrings.Add(t);
|
||||
}
|
||||
|
||||
var tBlobs = (from a in tr.ImageFileNamePaths
|
||||
select a).ToList();
|
||||
|
||||
if (tBlobs.Count < 1)
|
||||
throw new MDSParseException("BLOB Error!");
|
||||
|
||||
// is the currBlob valid for this track, or do we need to increment?
|
||||
string bString = tBlobs.First();
|
||||
|
||||
IBlob mdfBlob = null;
|
||||
|
||||
// check for track pregap and create if neccessary
|
||||
// this is specified in the track extras block
|
||||
if (track.ExtraBlock.Pregap > 0)
|
||||
{
|
||||
CUE.CueTrackType pregapTrackType = CUE.CueTrackType.Audio;
|
||||
if (tocSynth.Result.TOCItems[1].IsData)
|
||||
{
|
||||
if (tocSynth.Result.Session1Format == SessionFormat.Type20_CDXA)
|
||||
pregapTrackType = CUE.CueTrackType.Mode2_2352;
|
||||
else if (tocSynth.Result.Session1Format == SessionFormat.Type10_CDI)
|
||||
pregapTrackType = CUE.CueTrackType.CDI_2352;
|
||||
else if (tocSynth.Result.Session1Format == SessionFormat.Type00_CDROM_CDDA)
|
||||
pregapTrackType = CUE.CueTrackType.Mode1_2352;
|
||||
}
|
||||
for (int pre = 0; pre < track.ExtraBlock.Pregap; pre++)
|
||||
{
|
||||
relMSF++;
|
||||
|
||||
var ss_gap = new CUE.SS_Gap()
|
||||
{
|
||||
Policy = IN_DiscMountPolicy,
|
||||
TrackType = pregapTrackType
|
||||
};
|
||||
disc._Sectors.Add(ss_gap);
|
||||
|
||||
int qRelMSF = pre - Convert.ToInt32(track.ExtraBlock.Pregap);
|
||||
|
||||
//tweak relMSF due to ambiguity/contradiction in yellowbook docs
|
||||
if (!IN_DiscMountPolicy.CUE_PregapContradictionModeA)
|
||||
qRelMSF++;
|
||||
|
||||
//setup subQ
|
||||
byte ADR = 1; //absent some kind of policy for how to set it, this is a safe assumption:
|
||||
ss_gap.sq.SetStatus(ADR, tocSynth.Result.TOCItems[1].Control);
|
||||
ss_gap.sq.q_tno = BCD2.FromDecimal(1);
|
||||
ss_gap.sq.q_index = BCD2.FromDecimal(0);
|
||||
ss_gap.sq.AP_Timestamp = pre;
|
||||
ss_gap.sq.Timestamp = qRelMSF;
|
||||
|
||||
//setup subP
|
||||
ss_gap.Pause = true;
|
||||
}
|
||||
// pregap processing completed
|
||||
}
|
||||
|
||||
|
||||
|
||||
// create track sectors
|
||||
long currBlobOffset = track.TrackOffset;
|
||||
for (long sector = session.StartSector; sector <= session.EndSector; sector++)
|
||||
{
|
||||
CUE.SS_Base sBase = null;
|
||||
|
||||
// get the current blob from the BlobIndex
|
||||
Disc.Blob_RawFile currBlob = BlobIndex[currBlobIndex] as Disc.Blob_RawFile;
|
||||
long currBlobLength = currBlob.Length;
|
||||
long currBlobPosition = sector;
|
||||
if (currBlobPosition == currBlobLength)
|
||||
currBlobIndex++;
|
||||
mdfBlob = disc.DisposableResources[currBlobIndex] as Disc.Blob_RawFile;
|
||||
|
||||
int userSector = 2048;
|
||||
switch (track.SectorSize)
|
||||
{
|
||||
case 2448:
|
||||
sBase = new CUE.SS_2352()
|
||||
{
|
||||
Policy = IN_DiscMountPolicy
|
||||
};
|
||||
userSector = 2352;
|
||||
break;
|
||||
case 2048:
|
||||
default:
|
||||
sBase = new CUE.SS_Mode1_2048()
|
||||
{
|
||||
Policy = IN_DiscMountPolicy
|
||||
};
|
||||
userSector = 2048;
|
||||
break;
|
||||
|
||||
//throw new Exception("Not supported: Sector Size " + track.SectorSize);
|
||||
}
|
||||
|
||||
// configure blob
|
||||
sBase.Blob = mdfBlob;
|
||||
sBase.BlobOffset = currBlobOffset;
|
||||
|
||||
currBlobOffset += track.SectorSize; // userSector;
|
||||
|
||||
// add subchannel data
|
||||
relMSF++;
|
||||
BCD2 tno, ino;
|
||||
|
||||
//this should actually be zero. im not sure if this is stored as BCD2 or not
|
||||
tno = BCD2.FromDecimal(track.TrackNo);
|
||||
|
||||
//these are special values.. I think, taken from this:
|
||||
//http://www.staff.uni-mainz.de/tacke/scsi/SCSI2-14.html
|
||||
//the CCD will contain Points as decimal values except for these specially converted decimal values which should stay as BCD.
|
||||
//Why couldn't they all be BCD? I don't know. I guess because BCD is inconvenient, but only A0 and friends have special meaning. It's confusing.
|
||||
ino = BCD2.FromDecimal(track.Point);
|
||||
if (track.Point == 0xA0) ino.BCDValue = 0xA0;
|
||||
else if (track.Point == 0xA1) ino.BCDValue = 0xA1;
|
||||
else if (track.Point == 0xA2) ino.BCDValue = 0xA2;
|
||||
|
||||
// get ADR & Control from ADR_Control byte
|
||||
byte adrc = Convert.ToByte(track.ADR_Control);
|
||||
var Control = adrc & 0x0F;
|
||||
var ADR = adrc >> 4;
|
||||
|
||||
var q = new SubchannelQ
|
||||
{
|
||||
q_status = SubchannelQ.ComputeStatus(ADR, (EControlQ)(Control & 0xF)),
|
||||
q_tno = BCD2.FromDecimal(track.Point),
|
||||
q_index = ino,
|
||||
AP_Timestamp = disc._Sectors.Count,
|
||||
Timestamp = relMSF - Convert.ToInt32(track.ExtraBlock.Pregap)
|
||||
};
|
||||
|
||||
sBase.sq = q;
|
||||
|
||||
disc._Sectors.Add(sBase);
|
||||
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return disc;
|
||||
}
|
||||
|
||||
} //class MDS_Format
|
||||
}
|
||||
|
||||
|
|
@ -1,5 +1,6 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Linq;
|
||||
|
||||
//disc type identification logic
|
||||
|
||||
|
@ -8,7 +9,7 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
public enum DiscType
|
||||
{
|
||||
/// <summary>
|
||||
/// Disc contains audio in track 1. Nothing more can readily be determined
|
||||
/// Disc contains audio in track 1. This may be a PCFX or PCECD game, but if not it is assumed AudioDisc
|
||||
/// </summary>
|
||||
AudioDisc,
|
||||
|
||||
|
@ -38,7 +39,7 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
SegaSaturn,
|
||||
|
||||
/// <summary>
|
||||
/// Its not clear whether we can ever have enough info to ID a turboCD disc (we're using hashes)
|
||||
/// PC Engine CD
|
||||
/// </summary>
|
||||
TurboCD,
|
||||
|
||||
|
@ -50,7 +51,47 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
/// <summary>
|
||||
/// By NEC.
|
||||
/// </summary>
|
||||
PCFX
|
||||
PCFX,
|
||||
|
||||
/// <summary>
|
||||
/// By Panasonic
|
||||
/// </summary>
|
||||
Panasonic3DO,
|
||||
|
||||
/// <summary>
|
||||
/// Philips
|
||||
/// </summary>
|
||||
CDi,
|
||||
|
||||
/// <summary>
|
||||
/// Nintendo Gamecube
|
||||
/// </summary>
|
||||
GameCube,
|
||||
|
||||
/// <summary>
|
||||
/// Nintendo Wii
|
||||
/// </summary>
|
||||
Wii,
|
||||
|
||||
/// <summary>
|
||||
/// SNK NeoGeo
|
||||
/// </summary>
|
||||
NeoGeoCD,
|
||||
|
||||
/// <summary>
|
||||
/// Bandai Playdia
|
||||
/// </summary>
|
||||
Playdia,
|
||||
|
||||
/// <summary>
|
||||
/// Either CDTV or CD32 (I havent found a reliable way of distinguishing between them yet -asni)
|
||||
/// </summary>
|
||||
Amiga,
|
||||
|
||||
/// <summary>
|
||||
/// Sega Dreamcast
|
||||
/// </summary>
|
||||
Dreamcast
|
||||
}
|
||||
|
||||
public class DiscIdentifier
|
||||
|
@ -76,24 +117,26 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
/// </summary>
|
||||
public DiscType DetectDiscType()
|
||||
{
|
||||
// not fully tested yet
|
||||
// PCFX & TurboCD sometimes (if not alltimes) have audio on track 1 - run these before the AudioDisc detection (asni)
|
||||
if (DetectPCFX())
|
||||
return DiscType.PCFX;
|
||||
|
||||
//check track 1's data type. if it's an audio track, further data-track testing is useless
|
||||
//furthermore, it's probably senseless (no binary data there to read)
|
||||
//NOTE: PCE-CD detection goes through here (no good way to detect PCE cd)
|
||||
if (!_disc.TOC.TOCItems[1].IsData)
|
||||
if (DetectTurboCD())
|
||||
return DiscType.TurboCD;
|
||||
|
||||
//check track 1's data type. if it's an audio track, further data-track testing is useless
|
||||
//furthermore, it's probably senseless (no binary data there to read)
|
||||
if (!_disc.TOC.TOCItems[1].IsData)
|
||||
return DiscType.AudioDisc;
|
||||
|
||||
// if (_dsr.ReadLBA_Mode(_disc.TOC.TOCItems[1].LBA) == 0)
|
||||
// return DiscType.AudioDisc;
|
||||
|
||||
// sega doesnt put anything identifying in the cdfs volume info. but its consistent about putting its own header here in sector 0
|
||||
//asni - this isn't strictly true - SystemIdentifier in volume descriptor has been observed on occasion (see below)
|
||||
if (DetectSegaSaturn())
|
||||
return DiscType.SegaSaturn;
|
||||
|
||||
|
||||
// not fully tested yet
|
||||
if (DetectMegaCD())
|
||||
return DiscType.MegaCD;
|
||||
|
@ -102,31 +145,79 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
if (DetectPSX())
|
||||
return DiscType.SonyPSX;
|
||||
|
||||
//we dont know how to detect TurboCD.
|
||||
//an emulator frontend will likely just guess TurboCD if the disc is UnknownFormat
|
||||
//(we can also have a gameDB!)
|
||||
if (Detect3DO())
|
||||
return DiscType.Panasonic3DO;
|
||||
|
||||
var discView = EDiscStreamView.DiscStreamView_Mode1_2048;
|
||||
if (DetectCDi())
|
||||
return DiscType.CDi;
|
||||
|
||||
if (DetectGameCube())
|
||||
return DiscType.GameCube;
|
||||
|
||||
if (DetectWii())
|
||||
return DiscType.Wii;
|
||||
|
||||
var discView = EDiscStreamView.DiscStreamView_Mode1_2048;
|
||||
if (_disc.TOC.Session1Format == SessionFormat.Type20_CDXA)
|
||||
discView = EDiscStreamView.DiscStreamView_Mode2_Form1_2048;
|
||||
|
||||
var iso = new ISOFile();
|
||||
bool isIso = iso.Parse(new DiscStream(_disc, discView, 0));
|
||||
|
||||
if (!isIso)
|
||||
{
|
||||
// its much quicker to detect dreamcast from ISO data. Only do this if ISO is not detected
|
||||
if (DetectDreamcast())
|
||||
return DiscType.Dreamcast;
|
||||
}
|
||||
|
||||
//*** asni - 20171011 - Suggestion: move this to the beginning of the DetectDiscType() method before any longer running lookups?
|
||||
//its a cheap win for a lot of systems, but ONLY if the iso.Parse() method is quick running (might have to time it)
|
||||
if (isIso)
|
||||
{
|
||||
var appId = System.Text.Encoding.ASCII.GetString(iso.VolumeDescriptors[0].ApplicationIdentifier).TrimEnd('\0', ' ');
|
||||
var sysId = System.Text.Encoding.ASCII.GetString(iso.VolumeDescriptors[0].SystemIdentifier).TrimEnd('\0', ' ');
|
||||
|
||||
//for example: PSX magical drop F (JP SLPS_02337) doesn't have the correct iso PVD fields
|
||||
//but, some PSX games (junky rips) don't have the 'licensed by string' so we'll hope they get caught here
|
||||
if (appId == "PLAYSTATION")
|
||||
//for example: PSX magical drop F (JP SLPS_02337) doesn't have the correct iso PVD fields
|
||||
//but, some PSX games (junky rips) don't have the 'licensed by string' so we'll hope they get caught here
|
||||
if (appId == "PLAYSTATION")
|
||||
return DiscType.SonyPSX;
|
||||
|
||||
if (appId == "PSP GAME")
|
||||
return DiscType.SonyPSP;
|
||||
// in case the appId is not set correctly...
|
||||
if (iso.Root.Children.Where(a => a.Key == "PSP_GAME").FirstOrDefault().Value as ISODirectoryNode != null)
|
||||
return DiscType.SonyPSP;
|
||||
|
||||
return DiscType.UnknownCDFS;
|
||||
}
|
||||
if (sysId == "SEGA SEGASATURN")
|
||||
return DiscType.SegaSaturn;
|
||||
|
||||
if (sysId.Contains("SEGAKATANA"))
|
||||
return DiscType.Dreamcast;
|
||||
|
||||
if (sysId == "MEGA_CD")
|
||||
return DiscType.MegaCD;
|
||||
|
||||
if (sysId == "ASAHI-CDV")
|
||||
return DiscType.Playdia;
|
||||
|
||||
if (sysId == "CDTV" || sysId == "AMIGA")
|
||||
return DiscType.Amiga;
|
||||
foreach (var f in iso.Root.Children)
|
||||
if (f.Key.ToLower().Contains("cd32"))
|
||||
return DiscType.Amiga;
|
||||
|
||||
// NeoGeoCD Check
|
||||
var absTxt = iso.Root.Children.Where(a => a.Key.Contains("ABS.TXT")).ToList();
|
||||
if (absTxt.Count > 0)
|
||||
{
|
||||
if (SectorContains("abstracted by snk", Convert.ToInt32(absTxt.First().Value.Offset)))
|
||||
return DiscType.NeoGeoCD;
|
||||
}
|
||||
|
||||
|
||||
return DiscType.UnknownCDFS;
|
||||
}
|
||||
|
||||
return DiscType.UnknownFormat;
|
||||
}
|
||||
|
@ -165,13 +256,87 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
t++)
|
||||
{
|
||||
var track = _disc.TOC.TOCItems[t];
|
||||
if (track.IsData && StringAt("PC-FX:Hu_CD-ROM", 0, track.LBA))
|
||||
//asni - this search is less specific - turns out there are discs where 'Hu:' is not present
|
||||
if (track.IsData && SectorContains("pc-fx", track.LBA))
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
private byte[] ReadSectorCached(int lba)
|
||||
//asni 20171011 - this ONLY works if a valid cuefile/ccd is passed into DiscIdentifier.
|
||||
//if an .iso is presented, the internally manufactured cue data does not work - possibly something to do with
|
||||
//track 01 being Audio. Not tested, but presumably PCFX has the same issue
|
||||
bool DetectTurboCD()
|
||||
{
|
||||
var toc = _disc.TOC;
|
||||
for (int t = toc.FirstRecordedTrackNumber;
|
||||
t <= toc.LastRecordedTrackNumber;
|
||||
t++)
|
||||
{
|
||||
var track = _disc.TOC.TOCItems[t];
|
||||
//asni - pcfx games also contain the 'PC Engine' string
|
||||
if ((track.IsData && SectorContains("pc engine", track.LBA + 1) && !SectorContains("pc-fx", track.LBA + 1)))
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
bool Detect3DO()
|
||||
{
|
||||
var toc = _disc.TOC;
|
||||
for (int t = toc.FirstRecordedTrackNumber;
|
||||
t <= toc.LastRecordedTrackNumber;
|
||||
t++)
|
||||
{
|
||||
var track = _disc.TOC.TOCItems[t];
|
||||
if (track.IsData && SectorContains("iamaduckiamaduck", track.LBA))
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
//asni - slightly longer running than the others due to its brute-force nature. Should run later in the method
|
||||
bool DetectDreamcast()
|
||||
{
|
||||
for (int i = 0; i < 1000; i++)
|
||||
{
|
||||
if (SectorContains("segakatana", i))
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
bool DetectCDi()
|
||||
{
|
||||
return StringAt("CD-RTOS", 8, 16);
|
||||
}
|
||||
|
||||
bool DetectGameCube()
|
||||
{
|
||||
var data = ReadSectorCached(0);
|
||||
if (data == null) return false;
|
||||
byte[] magic = data.Skip(28).Take(4).ToArray();
|
||||
string hexString = "";
|
||||
foreach (var b in magic)
|
||||
hexString += b.ToString("X2");
|
||||
|
||||
return hexString == "C2339F3D";
|
||||
}
|
||||
|
||||
bool DetectWii()
|
||||
{
|
||||
var data = ReadSectorCached(0);
|
||||
if (data == null) return false;
|
||||
byte[] magic = data.Skip(24).Take(4).ToArray();
|
||||
string hexString = "";
|
||||
foreach (var b in magic)
|
||||
hexString += b.ToString("X2");
|
||||
|
||||
return hexString == "5D1C9EA3";
|
||||
}
|
||||
|
||||
private byte[] ReadSectorCached(int lba)
|
||||
{
|
||||
//read it if we dont have it cached
|
||||
//we wont be caching very much here, it's no big deal
|
||||
|
@ -197,5 +362,12 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
Buffer.BlockCopy(data, n, cmp2, 0, cmp.Length);
|
||||
return System.Linq.Enumerable.SequenceEqual(cmp, cmp2);
|
||||
}
|
||||
|
||||
private bool SectorContains(string s, int lba = 0)
|
||||
{
|
||||
var data = ReadSectorCached(lba);
|
||||
if (data == null) return false;
|
||||
return System.Text.Encoding.ASCII.GetString(data).ToLower().Contains(s.ToLower());
|
||||
}
|
||||
}
|
||||
}
|
|
@ -187,6 +187,11 @@ namespace BizHawk.Emulation.DiscSystem
|
|||
CCD_Format ccdLoader = new CCD_Format();
|
||||
OUT_Disc = ccdLoader.LoadCCDToDisc(IN_FromPath, IN_DiscMountPolicy);
|
||||
}
|
||||
else if (ext == ".mds")
|
||||
{
|
||||
MDS_Format mdsLoader = new MDS_Format();
|
||||
OUT_Disc = mdsLoader.LoadMDSToDisc(IN_FromPath, IN_DiscMountPolicy);
|
||||
}
|
||||
|
||||
|
||||
DONE:
|
||||
|
|
Binary file not shown.
|
@ -193,7 +193,9 @@ void InputDevice_Memcard::SyncState(bool isReader, EW::NewState *ns)
|
|||
//HOWEVER - we clear the dirty flag. that way, a user wont accidentally `clobber` his savestates when loading a state.
|
||||
//instead, the state will only be dirtied when the game actually modifies the contents
|
||||
NSS(card_data);
|
||||
dirty_count = 0;
|
||||
|
||||
if(isReader)
|
||||
dirty_count = 0;
|
||||
}
|
||||
//
|
||||
//int InputDevice_Memcard::StateAction(StateMem* sm, int load, int data_only, const char* section_name)
|
||||
|
|
Loading…
Reference in New Issue