实现NesCore LoadState和SaveState
This commit is contained in:
parent
fea833734d
commit
12c02781a0
@ -359,7 +359,7 @@ namespace AxibugEmuOnline.Client.Manager
|
|||||||
void RecvHostSyn_RoomFrameAllInputData(byte[] reqData)
|
void RecvHostSyn_RoomFrameAllInputData(byte[] reqData)
|
||||||
{
|
{
|
||||||
Protobuf_Room_Syn_RoomFrameAllInputData msg = ProtoBufHelper.DeSerizlize<Protobuf_Room_Syn_RoomFrameAllInputData>(reqData);
|
Protobuf_Room_Syn_RoomFrameAllInputData msg = ProtoBufHelper.DeSerizlize<Protobuf_Room_Syn_RoomFrameAllInputData>(reqData);
|
||||||
netReplay.InData(new ReplayStep() { FrameStartID = (int)msg.FrameID, InPut = msg.InputData }, (int)msg.ServerFrameID);
|
//netReplay.InData(new ReplayStep() { FrameStartID = (int)msg.FrameID, InPut = msg.InputData }, (int)msg.ServerFrameID);
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SendScreen(byte[] RenderBuffer)
|
public void SendScreen(byte[] RenderBuffer)
|
||||||
|
@ -5,15 +5,16 @@ namespace AxibugEmuOnline.Client
|
|||||||
{
|
{
|
||||||
public class InGameUI : CommandExecuter
|
public class InGameUI : CommandExecuter
|
||||||
{
|
{
|
||||||
|
|
||||||
public static InGameUI Instance { get; private set; }
|
public static InGameUI Instance { get; private set; }
|
||||||
|
|
||||||
public RomFile RomFile => m_rom;
|
public RomFile RomFile => m_rom;
|
||||||
public override bool Enable => gameObject.activeInHierarchy;
|
public override bool Enable => gameObject.activeInHierarchy;
|
||||||
private RomFile m_rom;
|
private RomFile m_rom;
|
||||||
private object m_core;
|
private object m_core;
|
||||||
|
private object m_state;
|
||||||
|
|
||||||
private InGameUI_SaveState m_saveMenu;
|
private InGameUI_SaveState m_saveStateMenu;
|
||||||
|
private InGameUI_LoadState m_loadStateMenu;
|
||||||
|
|
||||||
protected override void Awake()
|
protected override void Awake()
|
||||||
{
|
{
|
||||||
@ -27,15 +28,47 @@ namespace AxibugEmuOnline.Client
|
|||||||
Instance = null;
|
Instance = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// 获取模拟器核心对象
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="T">模拟器核心对象类型</typeparam>
|
||||||
public T GetCore<T>() => (T)m_core;
|
public T GetCore<T>() => (T)m_core;
|
||||||
|
/// <summary> 保存快速快照 </summary>
|
||||||
|
public void SaveQuickState(object state)
|
||||||
|
{
|
||||||
|
m_state = state;
|
||||||
|
}
|
||||||
|
/// <summary>
|
||||||
|
/// 读取快速快照
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="T"></typeparam>
|
||||||
|
/// <returns></returns>
|
||||||
|
public bool GetQuickState<T>(out T state)
|
||||||
|
{
|
||||||
|
state = default(T);
|
||||||
|
|
||||||
|
if (m_state is T)
|
||||||
|
{
|
||||||
|
state = (T)m_state;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
public void Show(RomFile currentRom, object core)
|
public void Show(RomFile currentRom, object core)
|
||||||
{
|
{
|
||||||
m_saveMenu = new InGameUI_SaveState(this);
|
|
||||||
CommandDispatcher.Instance.RegistController(this);
|
CommandDispatcher.Instance.RegistController(this);
|
||||||
|
|
||||||
|
m_saveStateMenu = new InGameUI_SaveState(this);
|
||||||
|
m_loadStateMenu = new InGameUI_LoadState(this);
|
||||||
|
|
||||||
m_rom = currentRom;
|
m_rom = currentRom;
|
||||||
m_core = core;
|
m_core = core;
|
||||||
|
|
||||||
gameObject.SetActiveEx(true);
|
gameObject.SetActiveEx(true);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -43,14 +76,12 @@ namespace AxibugEmuOnline.Client
|
|||||||
{
|
{
|
||||||
CommandDispatcher.Instance.UnRegistController(this);
|
CommandDispatcher.Instance.UnRegistController(this);
|
||||||
|
|
||||||
m_rom = null;
|
|
||||||
m_core = null;
|
|
||||||
gameObject.SetActiveEx(false);
|
gameObject.SetActiveEx(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected override void OnCmdOptionMenu()
|
protected override void OnCmdOptionMenu()
|
||||||
{
|
{
|
||||||
OptionUI.Instance.Pop(new List<OptionMenu> { m_saveMenu });
|
OptionUI.Instance.Pop(new List<OptionMenu> { m_saveStateMenu, m_loadStateMenu });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,32 @@
|
|||||||
|
using AxibugEmuOnline.Client.ClientCore;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using VirtualNes.Core;
|
||||||
|
|
||||||
|
namespace AxibugEmuOnline.Client
|
||||||
|
{
|
||||||
|
public class InGameUI_LoadState : ExecuteMenu
|
||||||
|
{
|
||||||
|
private InGameUI m_gameUI;
|
||||||
|
|
||||||
|
public InGameUI_LoadState(InGameUI gameUI) : base("śÁČĄżěŐŐ", null)
|
||||||
|
{
|
||||||
|
m_gameUI = gameUI;
|
||||||
|
}
|
||||||
|
|
||||||
|
public override void OnExcute()
|
||||||
|
{
|
||||||
|
Stopwatch sw = Stopwatch.StartNew();
|
||||||
|
switch (m_gameUI.RomFile.Platform)
|
||||||
|
{
|
||||||
|
case EnumPlatform.NES:
|
||||||
|
if (m_gameUI.GetQuickState<State>(out var quickState))
|
||||||
|
{
|
||||||
|
m_gameUI.GetCore<NesEmulator>().NesCore.LoadState(quickState);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
sw.Stop();
|
||||||
|
App.log.Info($"{m_gameUI.RomFile.Platform}====>żěŐŐźÓÔŘşÄĘą:{sw.Elapsed.TotalMilliseconds}ms");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,11 @@
|
|||||||
|
fileFormatVersion: 2
|
||||||
|
guid: 522140a3272d84a40b1ff91a3ebdb1b0
|
||||||
|
MonoImporter:
|
||||||
|
externalObjects: {}
|
||||||
|
serializedVersion: 2
|
||||||
|
defaultReferences: []
|
||||||
|
executionOrder: 0
|
||||||
|
icon: {instanceID: 0}
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
@ -19,6 +19,7 @@ namespace AxibugEmuOnline.Client
|
|||||||
{
|
{
|
||||||
case EnumPlatform.NES:
|
case EnumPlatform.NES:
|
||||||
var state = m_gameUI.GetCore<NesEmulator>().NesCore.GetState();
|
var state = m_gameUI.GetCore<NesEmulator>().NesCore.GetState();
|
||||||
|
m_gameUI.SaveQuickState(state);
|
||||||
App.log.Info($"{m_gameUI.RomFile.Platform}===>快照大小{state.ToBytes().Length}");
|
App.log.Info($"{m_gameUI.RomFile.Platform}===>快照大小{state.ToBytes().Length}");
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
@ -168,6 +168,9 @@ namespace AxibugEmuOnline.Client
|
|||||||
{
|
{
|
||||||
public string Name { get; protected set; }
|
public string Name { get; protected set; }
|
||||||
public Sprite Icon { get; protected set; }
|
public Sprite Icon { get; protected set; }
|
||||||
|
public virtual bool Visible => true;
|
||||||
|
public virtual bool Enable => true;
|
||||||
|
|
||||||
public OptionMenu(string name, Sprite icon = null)
|
public OptionMenu(string name, Sprite icon = null)
|
||||||
{
|
{
|
||||||
Name = name;
|
Name = name;
|
||||||
|
@ -543,6 +543,11 @@ namespace VirtualNes.Core
|
|||||||
@internal.GetFrameIRQ(ref Cycle, ref Count, ref Type, ref IRQ, ref Occur);
|
@internal.GetFrameIRQ(ref Cycle, ref Count, ref Type, ref IRQ, ref Occur);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal void SetFrameIRQ(int Cycle, byte Count, byte Type, byte IRQ, byte Occur)
|
||||||
|
{
|
||||||
|
@internal.SetFrameIRQ(Cycle, Count, Type, IRQ, Occur);
|
||||||
|
}
|
||||||
|
|
||||||
internal void SaveState(StateBuffer buffer)
|
internal void SaveState(StateBuffer buffer)
|
||||||
{
|
{
|
||||||
// 時間軸を同期させる為Flushする
|
// 時間軸を同期させる為Flushする
|
||||||
@ -589,6 +594,49 @@ namespace VirtualNes.Core
|
|||||||
buffer.Position += (fme7.GetSize() + 15) & (~0x0F); // Padding
|
buffer.Position += (fme7.GetSize() + 15) & (~0x0F); // Padding
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
@internal.LoadState(buffer);
|
||||||
|
buffer.Skip((@internal.GetSize() + 15) & (~0x0F));
|
||||||
|
|
||||||
|
// VRC6
|
||||||
|
if ((exsound_select & 0x01) != 0)
|
||||||
|
{
|
||||||
|
vrc6.LoadState(buffer);
|
||||||
|
buffer.Skip((int)((vrc6.GetSize() + 15) & (~0x0F))); // Padding
|
||||||
|
}
|
||||||
|
// VRC7 (not support)
|
||||||
|
if ((exsound_select & 0x02) != 0)
|
||||||
|
{
|
||||||
|
vrc7.LoadState(buffer);
|
||||||
|
buffer.Skip((vrc7.GetSize() + 15) & (~0x0F)); // Padding
|
||||||
|
}
|
||||||
|
// FDS
|
||||||
|
if ((exsound_select & 0x04) != 0)
|
||||||
|
{
|
||||||
|
fds.LoadState(buffer);
|
||||||
|
buffer.Skip((fds.GetSize() + 15) & (~0x0F)); // Padding
|
||||||
|
}
|
||||||
|
// MMC5
|
||||||
|
if ((exsound_select & 0x08) != 0)
|
||||||
|
{
|
||||||
|
mmc5.LoadState(buffer);
|
||||||
|
buffer.Skip((mmc5.GetSize() + 15) & (~0x0F)); // Padding
|
||||||
|
}
|
||||||
|
// N106
|
||||||
|
if ((exsound_select & 0x10) != 0)
|
||||||
|
{
|
||||||
|
n106.LoadState(buffer);
|
||||||
|
buffer.Skip((n106.GetSize() + 15) & (~0x0F)); // Padding
|
||||||
|
}
|
||||||
|
// FME7
|
||||||
|
if ((exsound_select & 0x20) != 0)
|
||||||
|
{
|
||||||
|
fme7.LoadState(buffer);
|
||||||
|
buffer.Skip((fme7.GetSize() + 15) & (~0x0F)); // Padding
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public struct QUEUEDATA
|
public struct QUEUEDATA
|
||||||
|
@ -514,6 +514,36 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(now_freq);
|
buffer.Write(now_freq);
|
||||||
buffer.Write(output);
|
buffer.Write(output);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(0x80);
|
||||||
|
volenv_mode = buffer.Read_byte();
|
||||||
|
volenv_gain = buffer.Read_byte();
|
||||||
|
volenv_decay = buffer.Read_byte();
|
||||||
|
volenv_phaseacc = buffer.Read_double();
|
||||||
|
swpenv_mode = buffer.Read_byte();
|
||||||
|
swpenv_gain = buffer.Read_byte();
|
||||||
|
swpenv_decay = buffer.Read_byte();
|
||||||
|
swpenv_phaseacc = buffer.Read_double();
|
||||||
|
envelope_enable = buffer.Read_byte();
|
||||||
|
envelope_speed = buffer.Read_byte();
|
||||||
|
wave_setup = buffer.Read_byte();
|
||||||
|
master_volume = buffer.Read_int();
|
||||||
|
main_wavetable = buffer.Read_ints(64);
|
||||||
|
main_enable = buffer.Read_byte();
|
||||||
|
main_frequency = buffer.Read_int();
|
||||||
|
main_addr = buffer.Read_int();
|
||||||
|
lfo_wavetable = buffer.Read_bytes(64);
|
||||||
|
lfo_enable = buffer.Read_byte();
|
||||||
|
lfo_frequency = buffer.Read_int();
|
||||||
|
lfo_addr = buffer.Read_int();
|
||||||
|
lfo_phaseacc = buffer.Read_double();
|
||||||
|
sweep_bias = buffer.Read_int();
|
||||||
|
now_volume = buffer.Read_int();
|
||||||
|
now_freq = buffer.Read_int();
|
||||||
|
output = buffer.Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -398,6 +398,17 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(envtbl_index);
|
buffer.Write(envtbl_index);
|
||||||
buffer.Write(envstep_index);
|
buffer.Write(envstep_index);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(3);
|
||||||
|
volume = buffer.Read_byte();
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
envadr = buffer.Read_int();
|
||||||
|
envtbl_index = buffer.Read_byte();
|
||||||
|
envstep_index = buffer.Read_byte();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class NOISE : IStateBufferObject
|
public class NOISE : IStateBufferObject
|
||||||
@ -424,6 +435,14 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(noiserange);
|
buffer.Write(noiserange);
|
||||||
buffer.Write(noiseout);
|
buffer.Write(noiseout);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
noiserange = buffer.Read_int();
|
||||||
|
noiseout = buffer.Read_byte();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class CHANNEL : IStateBufferObject
|
public class CHANNEL : IStateBufferObject
|
||||||
@ -469,6 +488,19 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(phaseacc);
|
buffer.Write(phaseacc);
|
||||||
buffer.Write(output_vol);
|
buffer.Write(output_vol);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(3);
|
||||||
|
enable = buffer.Read_byte();
|
||||||
|
env_on = buffer.Read_byte();
|
||||||
|
noise_on = buffer.Read_byte();
|
||||||
|
adder = buffer.Read_byte();
|
||||||
|
volume = buffer.Read_byte();
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
output_vol = buffer.Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -23,7 +23,7 @@ namespace VirtualNes.Core
|
|||||||
public virtual bool Sync(int cycles) { return false; }
|
public virtual bool Sync(int cycles) { return false; }
|
||||||
public virtual int GetFreq(int channel) { return 0; }
|
public virtual int GetFreq(int channel) { return 0; }
|
||||||
public virtual void SaveState(StateBuffer buffer) { }
|
public virtual void SaveState(StateBuffer buffer) { }
|
||||||
public virtual void LoadState(byte[] p) { }
|
public virtual void LoadState(StateReader buffer) { }
|
||||||
|
|
||||||
public static int INT2FIX(int x)
|
public static int INT2FIX(int x)
|
||||||
{
|
{
|
||||||
|
@ -1203,6 +1203,15 @@ namespace VirtualNes.Core
|
|||||||
occur = FrameIRQoccur;
|
occur = FrameIRQoccur;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal void SetFrameIRQ(int cycle, byte count, byte type, byte IRQ, byte occur)
|
||||||
|
{
|
||||||
|
FrameCycle = cycle;
|
||||||
|
FrameCount = count;
|
||||||
|
FrameType = type;
|
||||||
|
FrameIRQ = IRQ;
|
||||||
|
FrameIRQoccur = occur;
|
||||||
|
}
|
||||||
|
|
||||||
public override uint GetSize()
|
public override uint GetSize()
|
||||||
{
|
{
|
||||||
return sizeof(byte) +
|
return sizeof(byte) +
|
||||||
@ -1219,7 +1228,7 @@ namespace VirtualNes.Core
|
|||||||
ch4.GetSize();
|
ch4.GetSize();
|
||||||
}
|
}
|
||||||
|
|
||||||
public unsafe override void SaveState(StateBuffer p)
|
public override void SaveState(StateBuffer p)
|
||||||
{
|
{
|
||||||
p.Write(reg4015);
|
p.Write(reg4015);
|
||||||
p.Write(sync_reg4015);
|
p.Write(sync_reg4015);
|
||||||
@ -1352,6 +1361,39 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(dummy2);
|
buffer.Write(dummy2);
|
||||||
buffer.Write(sync_len_count);
|
buffer.Write(sync_len_count);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(4);
|
||||||
|
enable = buffer.Read_byte();
|
||||||
|
holdnote = buffer.Read_byte();
|
||||||
|
volume = buffer.Read_byte();
|
||||||
|
complement = buffer.Read_byte();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
freqlimit = buffer.Read_int();
|
||||||
|
adder = buffer.Read_int();
|
||||||
|
duty = buffer.Read_int();
|
||||||
|
len_count = buffer.Read_int();
|
||||||
|
nowvolume = buffer.Read_int();
|
||||||
|
env_fixed = buffer.Read_byte();
|
||||||
|
env_decay = buffer.Read_byte();
|
||||||
|
env_count = buffer.Read_byte();
|
||||||
|
dummy0 = buffer.Read_byte();
|
||||||
|
env_vol = buffer.Read_int();
|
||||||
|
swp_on = buffer.Read_byte();
|
||||||
|
swp_inc = buffer.Read_byte();
|
||||||
|
swp_shift = buffer.Read_byte();
|
||||||
|
swp_decay = buffer.Read_byte();
|
||||||
|
swp_count = buffer.Read_byte();
|
||||||
|
dummy1 = buffer.Read_bytes(3);
|
||||||
|
sync_reg = buffer.Read_bytes(4);
|
||||||
|
sync_output_enable = buffer.Read_byte();
|
||||||
|
sync_enable = buffer.Read_byte();
|
||||||
|
sync_holdnote = buffer.Read_byte();
|
||||||
|
dummy2 = buffer.Read_byte();
|
||||||
|
sync_len_count = buffer.Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
public class TRIANGLE : IStateBufferObject
|
public class TRIANGLE : IStateBufferObject
|
||||||
{
|
{
|
||||||
@ -1428,6 +1470,27 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(sync_len_count);
|
buffer.Write(sync_len_count);
|
||||||
buffer.Write(sync_lin_count);
|
buffer.Write(sync_lin_count);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(4);
|
||||||
|
enable = buffer.Read_byte();
|
||||||
|
holdnote = buffer.Read_byte();
|
||||||
|
counter_start = buffer.Read_byte();
|
||||||
|
dummy0 = buffer.Read_byte();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
len_count = buffer.Read_int();
|
||||||
|
lin_count = buffer.Read_int();
|
||||||
|
adder = buffer.Read_int();
|
||||||
|
nowvolume = buffer.Read_int();
|
||||||
|
sync_reg = buffer.Read_bytes(4);
|
||||||
|
sync_enable = buffer.Read_byte();
|
||||||
|
sync_holdnote = buffer.Read_byte();
|
||||||
|
sync_counter_start = buffer.Read_byte();
|
||||||
|
sync_len_count = buffer.Read_int();
|
||||||
|
sync_lin_count = buffer.Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
public class DPCM : IStateBufferObject
|
public class DPCM : IStateBufferObject
|
||||||
{
|
{
|
||||||
@ -1487,6 +1550,35 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(sync_dmalength);
|
buffer.Write(sync_dmalength);
|
||||||
buffer.Write(sync_cache_dmalength);
|
buffer.Write(sync_cache_dmalength);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(4);
|
||||||
|
enable = buffer.Read_byte();
|
||||||
|
looping = buffer.Read_byte();
|
||||||
|
cur_byte = buffer.Read_byte();
|
||||||
|
dpcm_value = buffer.Read_byte();
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
output = buffer.Read_int();
|
||||||
|
address = buffer.Read_ushort();
|
||||||
|
cache_addr = buffer.Read_ushort();
|
||||||
|
dmalength = buffer.Read_int();
|
||||||
|
cache_dmalength = buffer.Read_int();
|
||||||
|
dpcm_output_real = buffer.Read_int();
|
||||||
|
dpcm_output_fake = buffer.Read_int();
|
||||||
|
dpcm_output_old = buffer.Read_int();
|
||||||
|
dpcm_output_offset = buffer.Read_int();
|
||||||
|
sync_reg = buffer.Read_bytes(4);
|
||||||
|
sync_enable = buffer.Read_byte();
|
||||||
|
sync_looping = buffer.Read_byte();
|
||||||
|
sync_irq_gen = buffer.Read_byte();
|
||||||
|
sync_irq_enable = buffer.Read_byte();
|
||||||
|
sync_cycles = buffer.Read_int();
|
||||||
|
sync_cache_cycles = buffer.Read_int();
|
||||||
|
sync_dmalength = buffer.Read_int();
|
||||||
|
sync_cache_dmalength = buffer.Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
public class NOISE : IStateBufferObject
|
public class NOISE : IStateBufferObject
|
||||||
{
|
{
|
||||||
@ -1582,6 +1674,32 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(dummy1);
|
buffer.Write(dummy1);
|
||||||
buffer.Write(sync_len_count);
|
buffer.Write(sync_len_count);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(4);
|
||||||
|
enable = buffer.Read_byte();
|
||||||
|
holdnote = buffer.Read_byte();
|
||||||
|
volume = buffer.Read_byte();
|
||||||
|
xor_tap = buffer.Read_byte();
|
||||||
|
shift_reg = buffer.Read_int();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
len_count = buffer.Read_int();
|
||||||
|
nowvolume = buffer.Read_int();
|
||||||
|
output = buffer.Read_int();
|
||||||
|
env_fixed = buffer.Read_byte();
|
||||||
|
env_decay = buffer.Read_byte();
|
||||||
|
env_count = buffer.Read_byte();
|
||||||
|
dummy0 = buffer.Read_byte();
|
||||||
|
env_vol = buffer.Read_int();
|
||||||
|
sync_reg = buffer.Read_bytes(4);
|
||||||
|
sync_output_enable = buffer.Read_byte();
|
||||||
|
sync_enable = buffer.Read_byte();
|
||||||
|
sync_holdnote = buffer.Read_byte();
|
||||||
|
dummy1 = buffer.Read_byte();
|
||||||
|
sync_len_count = buffer.Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -421,6 +421,15 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(dummy);
|
buffer.Write(dummy);
|
||||||
buffer.Write(vbl_length);
|
buffer.Write(vbl_length);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(4);
|
||||||
|
enable = buffer.Read_byte();
|
||||||
|
holdnote = buffer.Read_byte();
|
||||||
|
dummy = buffer.Read_bytes(2);
|
||||||
|
vbl_length = buffer.Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class RECTANGLE : IStateBufferObject
|
public class RECTANGLE : IStateBufferObject
|
||||||
@ -467,6 +476,24 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(adder);
|
buffer.Write(adder);
|
||||||
buffer.Write(duty_flip);
|
buffer.Write(duty_flip);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(4);
|
||||||
|
enable = buffer.Read_byte();
|
||||||
|
vbl_length = buffer.Read_int();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
output_vol = buffer.Read_int();
|
||||||
|
fixed_envelope = buffer.Read_byte();
|
||||||
|
holdnote = buffer.Read_byte();
|
||||||
|
volume = buffer.Read_byte();
|
||||||
|
env_vol = buffer.Read_byte();
|
||||||
|
env_phase = buffer.Read_int();
|
||||||
|
env_decay = buffer.Read_int();
|
||||||
|
adder = buffer.Read_int();
|
||||||
|
duty_flip = buffer.Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -243,6 +243,19 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(vol);
|
buffer.Write(vol);
|
||||||
buffer.Write(databuf);
|
buffer.Write(databuf);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
freq = buffer.Read_uint();
|
||||||
|
phase = buffer.Read_uint();
|
||||||
|
tonelen = buffer.Read_uint();
|
||||||
|
output = buffer.Read_int();
|
||||||
|
toneadr = buffer.Read_byte();
|
||||||
|
volupdate = buffer.Read_byte();
|
||||||
|
vol = buffer.Read_byte();
|
||||||
|
databuf = buffer.Read_byte();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -304,6 +304,19 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(adder);
|
buffer.Write(adder);
|
||||||
buffer.Write(duty_pos);
|
buffer.Write(duty_pos);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(3);
|
||||||
|
enable = buffer.Read_byte();
|
||||||
|
gate = buffer.Read_byte();
|
||||||
|
volume = buffer.Read_byte();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
output_vol = buffer.Read_int();
|
||||||
|
adder = buffer.Read_byte();
|
||||||
|
duty_pos = buffer.Read_byte();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public class SAWTOOTH : IStateBufferObject
|
public class SAWTOOTH : IStateBufferObject
|
||||||
@ -353,6 +366,19 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(accum);
|
buffer.Write(accum);
|
||||||
buffer.Write(phaseaccum);
|
buffer.Write(phaseaccum);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg = buffer.Read_bytes(3);
|
||||||
|
enable = buffer.Read_byte();
|
||||||
|
volume = buffer.Read_byte();
|
||||||
|
phaseacc = buffer.Read_int();
|
||||||
|
freq = buffer.Read_int();
|
||||||
|
output_vol = buffer.Read_int();
|
||||||
|
adder = buffer.Read_byte();
|
||||||
|
accum = buffer.Read_byte();
|
||||||
|
phaseaccum = buffer.Read_byte();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2022,10 +2022,20 @@ namespace VirtualNes.Core
|
|||||||
r = R;
|
r = R;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal void SetContext(R6502 r)
|
||||||
|
{
|
||||||
|
R = r;
|
||||||
|
}
|
||||||
|
|
||||||
internal int GetDmaCycles()
|
internal int GetDmaCycles()
|
||||||
{
|
{
|
||||||
return DMA_cycles;
|
return DMA_cycles;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal void SetDmaCycles(int cycles)
|
||||||
|
{
|
||||||
|
DMA_cycles = cycles;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public enum StatusFlag6502 : int
|
public enum StatusFlag6502 : int
|
||||||
|
@ -53,6 +53,11 @@ namespace VirtualNes.Core
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void WriteTo(T[] source, int start, int length)
|
||||||
|
{
|
||||||
|
Array.Copy(source, 0, m_rawArray, Offset + start, length);
|
||||||
|
}
|
||||||
|
|
||||||
public Span<T> Span(int start, int length)
|
public Span<T> Span(int start, int length)
|
||||||
{
|
{
|
||||||
return new Span<T>(m_rawArray, start + Offset, length);
|
return new Span<T>(m_rawArray, start + Offset, length);
|
||||||
|
@ -1872,12 +1872,15 @@ namespace VirtualNes.Core
|
|||||||
state.dskBLOCK.BlockVersion = 0x0210;
|
state.dskBLOCK.BlockVersion = 0x0210;
|
||||||
state.dskBLOCK.BlockSize = 0;
|
state.dskBLOCK.BlockSize = 0;
|
||||||
|
|
||||||
|
state.dskdata = new List<uint>();
|
||||||
|
|
||||||
for (int i = 16; i < DiskSize; i++)
|
for (int i = 16; i < DiskSize; i++)
|
||||||
{
|
{
|
||||||
if (lpWrite[i] != 0)
|
if (lpWrite[i] != 0)
|
||||||
{
|
{
|
||||||
state.dskdata = (uint)(i & 0x00FFFFFF);
|
uint data = (uint)(i & 0x00FFFFFF);
|
||||||
state.dskdata |= ((uint)lpDisk[i] & 0xFF) << 24;
|
data |= ((uint)lpDisk[i] & 0xFF) << 24;
|
||||||
|
state.dskdata.Add(data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1898,7 +1901,191 @@ namespace VirtualNes.Core
|
|||||||
|
|
||||||
public void LoadState(State state)
|
public void LoadState(State state)
|
||||||
{
|
{
|
||||||
|
//HEADER
|
||||||
|
{
|
||||||
|
state.HEADER.ID = "VirtuaNES ST";
|
||||||
|
state.HEADER.BlockVersion = 0x0200;
|
||||||
|
|
||||||
|
if (rom.GetMapperNo() != 20)
|
||||||
|
rom.SetPROM_CRC(state.HEADER.Ext0);
|
||||||
|
else
|
||||||
|
{
|
||||||
|
rom.SetGameID(state.HEADER.Ext0);
|
||||||
|
rom.SetMakerID(state.HEADER.Ext1);
|
||||||
|
rom.SetDiskNo(state.HEADER.Ext2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//REGISTER STATE
|
||||||
|
{
|
||||||
|
R6502 R = new R6502();
|
||||||
|
R.PC = state.reg.cpureg.PC;
|
||||||
|
R.A = state.reg.cpureg.A;
|
||||||
|
R.X = state.reg.cpureg.X;
|
||||||
|
R.Y = state.reg.cpureg.Y;
|
||||||
|
R.S = state.reg.cpureg.S;
|
||||||
|
R.P = state.reg.cpureg.P;
|
||||||
|
R.INT_pending = state.reg.cpureg.I;
|
||||||
|
cpu.SetContext(R);
|
||||||
|
|
||||||
|
apu.SetFrameIRQ(
|
||||||
|
state.reg.cpureg.FrameIRQ_cycles,
|
||||||
|
state.reg.cpureg.FrameIRQ_count,
|
||||||
|
state.reg.cpureg.FrameIRQ_type,
|
||||||
|
state.reg.cpureg.FrameIRQ,
|
||||||
|
state.reg.cpureg.FrameIRQ_occur
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
|
cpu.SetDmaCycles(state.reg.cpureg.DMA_cycles);
|
||||||
|
emul_cycles = state.reg.cpureg.emul_cycles;
|
||||||
|
base_cycles = state.reg.cpureg.base_cycles;
|
||||||
|
|
||||||
|
// LOAD PPU STATE
|
||||||
|
MMU.PPUREG[0] = state.reg.ppureg.reg0;
|
||||||
|
MMU.PPUREG[1] = state.reg.ppureg.reg1;
|
||||||
|
MMU.PPUREG[2] = state.reg.ppureg.reg2;
|
||||||
|
MMU.PPUREG[3] = state.reg.ppureg.reg3;
|
||||||
|
MMU.PPU7_Temp = state.reg.ppureg.reg7;
|
||||||
|
MMU.loopy_t = state.reg.ppureg.loopy_t;
|
||||||
|
MMU.loopy_v = state.reg.ppureg.loopy_v;
|
||||||
|
MMU.loopy_x = state.reg.ppureg.loopy_x;
|
||||||
|
MMU.PPU56Toggle = state.reg.ppureg.toggle56;
|
||||||
|
}
|
||||||
|
|
||||||
|
//RAM STATE
|
||||||
|
{
|
||||||
|
// SAVE RAM STATE
|
||||||
|
MemoryUtility.memcpy(MMU.RAM, state.ram.RAM, state.ram.RAM.Length);
|
||||||
|
MemoryUtility.memcpy(MMU.BGPAL, state.ram.BGPAL, state.ram.BGPAL.Length);
|
||||||
|
MemoryUtility.memcpy(MMU.SPPAL, state.ram.SPPAL, state.ram.SPPAL.Length);
|
||||||
|
MemoryUtility.memcpy(MMU.SPRAM, state.ram.SPRAM, state.ram.SPRAM.Length);
|
||||||
|
|
||||||
|
if (rom.IsSAVERAM())
|
||||||
|
{
|
||||||
|
Array.Copy(state.WRAM, MMU.WRAM, SAVERAM_SIZE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//BANK STATE
|
||||||
|
{
|
||||||
|
// SAVE CPU MEMORY BANK DATA
|
||||||
|
// BANK0,1,2はバンクセーブに関係なし
|
||||||
|
// VirtuaNES0.30から
|
||||||
|
// バンク3はSRAM使用に関わらずセーブ
|
||||||
|
for (int i = 3; i < 8; i++)
|
||||||
|
{
|
||||||
|
MMU.CPU_MEM_TYPE[i] = state.mmu.CPU_MEM_TYPE[i];
|
||||||
|
MMU.CPU_MEM_PAGE[i] = state.mmu.CPU_MEM_PAGE[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// SAVE VRAM MEMORY DATA
|
||||||
|
for (int i = 0; i < 12; i++)
|
||||||
|
{
|
||||||
|
MMU.PPU_MEM_TYPE[i] = state.mmu.PPU_MEM_TYPE[i];
|
||||||
|
MMU.PPU_MEM_PAGE[i] = state.mmu.PPU_MEM_PAGE[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < 8; i++)
|
||||||
|
{
|
||||||
|
MMU.CRAM_USED[i] = state.mmu.CRAM_USED[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// WRITE CPU RAM MEMORY BANK
|
||||||
|
|
||||||
|
int stateStep = 0;
|
||||||
|
var stateCPU_MEM_BANK = state.CPU_MEM_BANK.ToArray();
|
||||||
|
|
||||||
|
for (int i = 3; i < 8; i++)
|
||||||
|
{
|
||||||
|
if (state.mmu.CPU_MEM_TYPE[i] != MMU.BANKTYPE_ROM)
|
||||||
|
{
|
||||||
|
var sourceData = new Span<byte>(stateCPU_MEM_BANK, stateStep * 8 * 1024, 8 * 1024);
|
||||||
|
MMU.CPU_MEM_BANK[i].WriteTo(sourceData.ToArray(), 0, 8 * 1024);
|
||||||
|
stateStep++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Array.Copy(state.VRAM, MMU.VRAM, state.VRAM.Length);
|
||||||
|
|
||||||
|
stateStep = 0;
|
||||||
|
var stateCRAM = state.CRAM.ToArray();
|
||||||
|
// LOAD CRAM MEMORY
|
||||||
|
for (int i = 0; i < 8; i++)
|
||||||
|
{
|
||||||
|
if (MMU.CRAM_USED[i] != 0)
|
||||||
|
{
|
||||||
|
var sourceData = stateCRAM.AsSpan(stateStep * 4 * 1024, 4 * 1024).ToArray();
|
||||||
|
Array.Copy(sourceData, 0, MMU.CRAM, 0x1000 * i, 4 * 1024);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// MMC STATE
|
||||||
|
{
|
||||||
|
state.mmc = MMCSTAT.GetDefault();
|
||||||
|
|
||||||
|
// Create Header
|
||||||
|
state.mmcBLOCK.ID = "MMC DATA";
|
||||||
|
state.mmcBLOCK.BlockVersion = 0x0100;
|
||||||
|
state.mmcBLOCK.BlockSize = state.mmc.GetSize();
|
||||||
|
|
||||||
|
if (mapper.IsStateSave())
|
||||||
|
{
|
||||||
|
mapper.LoadState(state.mmc.mmcdata);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//CONTROLLER STATE
|
||||||
|
{
|
||||||
|
pad.pad1bit = state.ctr.pad1bit;
|
||||||
|
pad.pad2bit = state.ctr.pad2bit;
|
||||||
|
pad.pad3bit = state.ctr.pad3bit;
|
||||||
|
pad.pad4bit = state.ctr.pad4bit;
|
||||||
|
pad.SetStrobe(state.ctr.strobe == 0 ? false : true);
|
||||||
|
}
|
||||||
|
|
||||||
|
//SND STATE
|
||||||
|
{
|
||||||
|
state.snd = SNDSTAT.GetDefault();
|
||||||
|
|
||||||
|
var buffer = new StateReader(state.snd.snddata);
|
||||||
|
apu.LoadState(buffer);
|
||||||
|
}
|
||||||
|
|
||||||
|
// DISKIMAGE STATE
|
||||||
|
if (rom.GetMapperNo() == 20)
|
||||||
|
{
|
||||||
|
var lpDisk = rom.GetPROM();
|
||||||
|
var lpWrite = rom.GetDISK();
|
||||||
|
int DiskSize = 16 + 65500 * rom.GetDiskNo();
|
||||||
|
|
||||||
|
Array.Clear(lpWrite, 0, DiskSize);
|
||||||
|
|
||||||
|
for (int i = 0; i < state.dsk.DifferentSize; i++)
|
||||||
|
{
|
||||||
|
var pos = state.dskdata[i];
|
||||||
|
byte data = (byte)(pos >> 24);
|
||||||
|
pos &= 0x00FFFFFF;
|
||||||
|
|
||||||
|
if (pos >= 16 && pos < DiskSize)
|
||||||
|
{
|
||||||
|
lpDisk[pos] = data;
|
||||||
|
lpWrite[pos] = 0xFF;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// EXCTR STATE
|
||||||
|
if (pad.GetExController() != 0)
|
||||||
|
{
|
||||||
|
pad.SetSyncExData(state.exctr.data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
internal void SetZapperPos(int x, int y)
|
||||||
|
{
|
||||||
|
ZapperX = x; ZapperY = y;
|
||||||
}
|
}
|
||||||
|
|
||||||
public enum IRQMETHOD
|
public enum IRQMETHOD
|
||||||
|
@ -515,6 +515,11 @@ namespace VirtualNes.Core
|
|||||||
return bStrobe;
|
return bStrobe;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal void SetStrobe(bool v)
|
||||||
|
{
|
||||||
|
bStrobe = v;
|
||||||
|
}
|
||||||
|
|
||||||
internal uint GetSyncExData()
|
internal uint GetSyncExData()
|
||||||
{
|
{
|
||||||
uint data = 0;
|
uint data = 0;
|
||||||
@ -571,6 +576,63 @@ namespace VirtualNes.Core
|
|||||||
}
|
}
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
internal void SetSyncExData(uint data)
|
||||||
|
{
|
||||||
|
switch ((EXCONTROLLER)excontroller_select)
|
||||||
|
{
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_ZAPPER:
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_PADDLE:
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_SPACESHADOWGUN:
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_OEKAKIDS_TABLET:
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_VSZAPPER:
|
||||||
|
{
|
||||||
|
int x, y;
|
||||||
|
if ((data & 0x80000000) != 0)
|
||||||
|
{
|
||||||
|
x = -1;
|
||||||
|
y = -1;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
x = (int)(data & 0xFF);
|
||||||
|
y = (int)((data & 0xFF00) >> 8);
|
||||||
|
}
|
||||||
|
expad.SetSyncData(0, x);
|
||||||
|
expad.SetSyncData(1, y);
|
||||||
|
nes.SetZapperPos(x, y);
|
||||||
|
}
|
||||||
|
if (excontroller_select != (int)EXCONTROLLER.EXCONTROLLER_SPACESHADOWGUN)
|
||||||
|
{
|
||||||
|
if ((data & 0x0010000) != 0)
|
||||||
|
expad.SetSyncData(2, 1);
|
||||||
|
else
|
||||||
|
expad.SetSyncData(2, 0);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
expad.SetSyncData(2, (byte)(data >> 16));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_CRAZYCLIMBER:
|
||||||
|
expad.SetSyncData(0, (int)data);
|
||||||
|
break;
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_TOPRIDER:
|
||||||
|
expad.SetSyncData(0, (int)data);
|
||||||
|
break;
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_FAMILYTRAINER_A:
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_FAMILYTRAINER_B:
|
||||||
|
expad.SetSyncData(0, (int)data);
|
||||||
|
break;
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_EXCITINGBOXING:
|
||||||
|
expad.SetSyncData(0, (int)data);
|
||||||
|
break;
|
||||||
|
case EXCONTROLLER.EXCONTROLLER_MAHJANG:
|
||||||
|
expad.SetSyncData(0, (int)data);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public enum VSType
|
public enum VSType
|
||||||
|
@ -353,16 +353,31 @@ namespace VirtualNes.Core
|
|||||||
return diskno;
|
return diskno;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal void SetDiskNo(int v)
|
||||||
|
{
|
||||||
|
diskno = v;
|
||||||
|
}
|
||||||
|
|
||||||
internal uint GetGameID()
|
internal uint GetGameID()
|
||||||
{
|
{
|
||||||
return fdsgameID;
|
return fdsgameID;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal void SetGameID(uint id)
|
||||||
|
{
|
||||||
|
fdsgameID = id;
|
||||||
|
}
|
||||||
|
|
||||||
internal uint GetMakerID()
|
internal uint GetMakerID()
|
||||||
{
|
{
|
||||||
return fdsmakerID;
|
return fdsmakerID;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
internal void SetMakerID(uint id)
|
||||||
|
{
|
||||||
|
fdsmakerID = id;
|
||||||
|
}
|
||||||
|
|
||||||
internal bool IsVSUNISYSTEM()
|
internal bool IsVSUNISYSTEM()
|
||||||
{
|
{
|
||||||
return (header.control2 & (byte)EnumRomControlByte2.ROM_VSUNISYSTEM) != 0;
|
return (header.control2 & (byte)EnumRomControlByte2.ROM_VSUNISYSTEM) != 0;
|
||||||
@ -373,6 +388,11 @@ namespace VirtualNes.Core
|
|||||||
return crc;
|
return crc;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void SetPROM_CRC(uint v)
|
||||||
|
{
|
||||||
|
crc = v;
|
||||||
|
}
|
||||||
|
|
||||||
internal byte GetPROM_SIZE()
|
internal byte GetPROM_SIZE()
|
||||||
{
|
{
|
||||||
return header.PRG_PAGE_SIZE;
|
return header.PRG_PAGE_SIZE;
|
||||||
|
@ -3,11 +3,19 @@
|
|||||||
public struct BLOCKHDR : IStateBufferObject
|
public struct BLOCKHDR : IStateBufferObject
|
||||||
{
|
{
|
||||||
public readonly bool Valid => !string.IsNullOrEmpty(ID);
|
public readonly bool Valid => !string.IsNullOrEmpty(ID);
|
||||||
|
/// <summary> 总是8个字节 </summary>
|
||||||
public string ID;
|
public string ID;
|
||||||
public ushort Reserved;
|
public ushort Reserved;
|
||||||
public ushort BlockVersion;
|
public ushort BlockVersion;
|
||||||
public uint BlockSize;
|
public uint BlockSize;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public readonly uint GetSize()
|
||||||
|
{
|
||||||
|
return (uint)(8 + sizeof(ushort) + sizeof(ushort) + sizeof(uint));
|
||||||
|
}
|
||||||
|
|
||||||
public readonly void SaveState(StateBuffer buffer)
|
public readonly void SaveState(StateBuffer buffer)
|
||||||
{
|
{
|
||||||
if (Valid)
|
if (Valid)
|
||||||
@ -19,9 +27,12 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public readonly uint GetSize()
|
public void LoadState(StateReader buffer)
|
||||||
{
|
{
|
||||||
return (uint)(ID.Length + sizeof(ushort) + sizeof(ushort) + sizeof(uint));
|
ID = buffer.Read_string(8);
|
||||||
|
Reserved = buffer.Read_ushort();
|
||||||
|
BlockVersion = buffer.Read_ushort();
|
||||||
|
BlockSize = buffer.Read_uint();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -24,5 +24,14 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(pad4bit);
|
buffer.Write(pad4bit);
|
||||||
buffer.Write(strobe);
|
buffer.Write(strobe);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
pad1bit = buffer.Read_uint();
|
||||||
|
pad2bit = buffer.Read_uint();
|
||||||
|
pad3bit = buffer.Read_uint();
|
||||||
|
pad4bit = buffer.Read_uint();
|
||||||
|
strobe = buffer.Read_byte();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,5 +13,10 @@
|
|||||||
{
|
{
|
||||||
buffer.Write(DifferentSize);
|
buffer.Write(DifferentSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
DifferentSize = buffer.Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -13,5 +13,10 @@
|
|||||||
{
|
{
|
||||||
buffer.Write(data);
|
buffer.Write(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
data = buffer.Read_uint();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -16,6 +16,13 @@ namespace VirtualNes.Core
|
|||||||
/// <summary> 2字节 </summary>
|
/// <summary> 2字节 </summary>
|
||||||
public ushort Ext2;
|
public ushort Ext2;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
public readonly uint GetSize()
|
||||||
|
{
|
||||||
|
return (uint)(ID.Length + sizeof(ushort) + sizeof(uint) + sizeof(ushort) + sizeof(ushort));
|
||||||
|
}
|
||||||
|
|
||||||
public readonly void SaveState(StateBuffer buffer)
|
public readonly void SaveState(StateBuffer buffer)
|
||||||
{
|
{
|
||||||
buffer.Write(ID);
|
buffer.Write(ID);
|
||||||
@ -25,9 +32,13 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(Ext2);
|
buffer.Write(Ext2);
|
||||||
}
|
}
|
||||||
|
|
||||||
public readonly uint GetSize()
|
public void LoadState(StateReader buffer)
|
||||||
{
|
{
|
||||||
return (uint)(ID.Length + sizeof(ushort) + sizeof(uint) + sizeof(ushort) + sizeof(ushort));
|
ID = buffer.Read_string(12);
|
||||||
|
BlockVersion = buffer.Read_ushort();
|
||||||
|
Ext0 = buffer.Read_uint();
|
||||||
|
Ext1 = buffer.Read_ushort();
|
||||||
|
Ext2 = buffer.Read_ushort();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -9,14 +9,19 @@
|
|||||||
return new MMCSTAT() { mmcdata = new byte[256] };
|
return new MMCSTAT() { mmcdata = new byte[256] };
|
||||||
}
|
}
|
||||||
|
|
||||||
public uint GetSize()
|
public readonly uint GetSize()
|
||||||
{
|
{
|
||||||
return (uint)mmcdata.Length;
|
return 256;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SaveState(StateBuffer buffer)
|
public readonly void SaveState(StateBuffer buffer)
|
||||||
{
|
{
|
||||||
buffer.Write(mmcdata);
|
buffer.Write(mmcdata);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
mmcdata = buffer.Read_bytes(256);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -34,5 +34,14 @@
|
|||||||
buffer.Write(PPU_MEM_PAGE);
|
buffer.Write(PPU_MEM_PAGE);
|
||||||
buffer.Write(CRAM_USED);
|
buffer.Write(CRAM_USED);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
CPU_MEM_TYPE = buffer.Read_bytes(8);
|
||||||
|
CPU_MEM_PAGE = buffer.Read_ushorts(8);
|
||||||
|
PPU_MEM_TYPE = buffer.Read_bytes(12);
|
||||||
|
PPU_MEM_PAGE = buffer.Read_ushorts(12);
|
||||||
|
CRAM_USED = buffer.Read_bytes(8);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -34,5 +34,13 @@ namespace VirtualNes.Core
|
|||||||
buffer.Write(SPPAL);
|
buffer.Write(SPPAL);
|
||||||
buffer.Write(SPRAM);
|
buffer.Write(SPRAM);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
RAM = buffer.Read_bytes(2 * 1024);
|
||||||
|
BGPAL = buffer.Read_bytes(16);
|
||||||
|
SPPAL = buffer.Read_bytes(16);
|
||||||
|
SPRAM = buffer.Read_bytes(256);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,15 +5,23 @@
|
|||||||
public CPUSTAT cpureg;
|
public CPUSTAT cpureg;
|
||||||
public PPUSTAT ppureg;
|
public PPUSTAT ppureg;
|
||||||
|
|
||||||
public void SaveState(StateBuffer buffer)
|
|
||||||
|
|
||||||
|
public readonly uint GetSize()
|
||||||
|
{
|
||||||
|
return cpureg.GetSize() + ppureg.GetSize();
|
||||||
|
}
|
||||||
|
|
||||||
|
public readonly void SaveState(StateBuffer buffer)
|
||||||
{
|
{
|
||||||
cpureg.SaveState(buffer);
|
cpureg.SaveState(buffer);
|
||||||
ppureg.SaveState(buffer);
|
ppureg.SaveState(buffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
public uint GetSize()
|
public void LoadState(StateReader buffer)
|
||||||
{
|
{
|
||||||
return cpureg.GetSize() + ppureg.GetSize();
|
cpureg.LoadState(buffer);
|
||||||
|
ppureg.LoadState(buffer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -60,6 +68,25 @@
|
|||||||
buffer.Write(emul_cycles);
|
buffer.Write(emul_cycles);
|
||||||
buffer.Write(base_cycles);
|
buffer.Write(base_cycles);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
PC = buffer.Read_ushort();
|
||||||
|
A = buffer.Read_byte();
|
||||||
|
X = buffer.Read_byte();
|
||||||
|
Y = buffer.Read_byte();
|
||||||
|
S = buffer.Read_byte();
|
||||||
|
P = buffer.Read_byte();
|
||||||
|
I = buffer.Read_byte();
|
||||||
|
FrameIRQ = buffer.Read_byte();
|
||||||
|
FrameIRQ_occur = buffer.Read_byte();
|
||||||
|
FrameIRQ_count = buffer.Read_byte();
|
||||||
|
FrameIRQ_type = buffer.Read_byte();
|
||||||
|
FrameIRQ_cycles = buffer.Read_int();
|
||||||
|
DMA_cycles = buffer.Read_int();
|
||||||
|
emul_cycles = buffer.Read_long();
|
||||||
|
base_cycles = buffer.Read_long();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public struct PPUSTAT : IStateBufferObject
|
public struct PPUSTAT : IStateBufferObject
|
||||||
@ -92,5 +119,18 @@
|
|||||||
buffer.Write(loopy_v);
|
buffer.Write(loopy_v);
|
||||||
buffer.Write(loopy_x);
|
buffer.Write(loopy_x);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
reg0 = buffer.Read_byte();
|
||||||
|
reg1 = buffer.Read_byte();
|
||||||
|
reg2 = buffer.Read_byte();
|
||||||
|
reg3 = buffer.Read_byte();
|
||||||
|
reg7 = buffer.Read_byte();
|
||||||
|
toggle56 = buffer.Read_byte();
|
||||||
|
loopy_t = buffer.Read_ushort();
|
||||||
|
loopy_v = buffer.Read_ushort();
|
||||||
|
loopy_x = buffer.Read_ushort();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -15,14 +15,19 @@ namespace VirtualNes.Core
|
|||||||
return new SNDSTAT() { snddata = new byte[0x800] };
|
return new SNDSTAT() { snddata = new byte[0x800] };
|
||||||
}
|
}
|
||||||
|
|
||||||
public uint GetSize()
|
public readonly uint GetSize()
|
||||||
{
|
{
|
||||||
return (uint)snddata.Length;
|
return (uint)snddata.Length;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void SaveState(StateBuffer buffer)
|
public readonly void SaveState(StateBuffer buffer)
|
||||||
{
|
{
|
||||||
buffer.Write(snddata);
|
buffer.Write(snddata);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public void LoadState(StateReader buffer)
|
||||||
|
{
|
||||||
|
snddata = buffer.Read_bytes(0x800);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -31,7 +31,7 @@ namespace VirtualNes.Core
|
|||||||
|
|
||||||
public BLOCKHDR dskBLOCK;
|
public BLOCKHDR dskBLOCK;
|
||||||
public DISKDATA dsk;
|
public DISKDATA dsk;
|
||||||
public uint dskdata;
|
public List<uint> dskdata;
|
||||||
|
|
||||||
public BLOCKHDR exctrBLOCK;
|
public BLOCKHDR exctrBLOCK;
|
||||||
public EXCTRSTAT exctr;
|
public EXCTRSTAT exctr;
|
||||||
@ -87,7 +87,10 @@ namespace VirtualNes.Core
|
|||||||
{
|
{
|
||||||
dskBLOCK.SaveState(buffer);
|
dskBLOCK.SaveState(buffer);
|
||||||
dsk.SaveState(buffer);
|
dsk.SaveState(buffer);
|
||||||
buffer.Write(dskdata);
|
foreach (var data in dskdata)
|
||||||
|
{
|
||||||
|
buffer.Write(data);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (exctrBLOCK.Valid)
|
if (exctrBLOCK.Valid)
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
|
using System.IO;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
|
||||||
namespace VirtualNes.Core
|
namespace VirtualNes.Core
|
||||||
@ -31,15 +32,11 @@ namespace VirtualNes.Core
|
|||||||
}
|
}
|
||||||
public void Write(sbyte[] sbytes)
|
public void Write(sbyte[] sbytes)
|
||||||
{
|
{
|
||||||
foreach(var value in sbytes)
|
foreach (var value in sbytes)
|
||||||
{
|
{
|
||||||
Write(value);
|
Write(value);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
public void Write(byte[] bytes, int length)
|
|
||||||
{
|
|
||||||
Data.AddRange(bytes);
|
|
||||||
}
|
|
||||||
public void Write(byte value)
|
public void Write(byte value)
|
||||||
{
|
{
|
||||||
Data.Add(value);
|
Data.Add(value);
|
||||||
@ -80,11 +77,134 @@ namespace VirtualNes.Core
|
|||||||
{
|
{
|
||||||
Write(BitConverter.GetBytes(value));
|
Write(BitConverter.GetBytes(value));
|
||||||
}
|
}
|
||||||
|
public void Write(uint value)
|
||||||
|
{
|
||||||
|
Write(BitConverter.GetBytes(value));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public class StateReader
|
||||||
|
{
|
||||||
|
private MemoryStream m_dataStream;
|
||||||
|
public StateReader(byte[] bytes)
|
||||||
|
{
|
||||||
|
m_dataStream = new MemoryStream(bytes);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Skip(uint count)
|
||||||
|
{
|
||||||
|
m_dataStream.Seek(count, SeekOrigin.Current);
|
||||||
|
}
|
||||||
|
public void Skip(long count)
|
||||||
|
{
|
||||||
|
m_dataStream.Seek(count, SeekOrigin.Current);
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte[] Read_bytes(int length)
|
||||||
|
{
|
||||||
|
var result = new byte[length];
|
||||||
|
m_dataStream.Read(result, 0, length);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
public sbyte[] Read_sbytes(int length)
|
||||||
|
{
|
||||||
|
var result = new sbyte[length];
|
||||||
|
for (int i = 0; i < length; i++)
|
||||||
|
{
|
||||||
|
result[i] = (sbyte)m_dataStream.ReadByte();
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte Read_byte()
|
||||||
|
{
|
||||||
|
return (byte)m_dataStream.ReadByte();
|
||||||
|
}
|
||||||
|
public ushort[] Read_ushorts(int length)
|
||||||
|
{
|
||||||
|
ushort[] result = new ushort[length];
|
||||||
|
for (int i = 0; i < length; i++)
|
||||||
|
{
|
||||||
|
int byte1 = m_dataStream.ReadByte();
|
||||||
|
int byte2 = m_dataStream.ReadByte();
|
||||||
|
|
||||||
|
result[i] = (ushort)(byte1 << 8 | byte2);
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
public int[] Read_ints(int length)
|
||||||
|
{
|
||||||
|
int[] result = new int[length];
|
||||||
|
for (int i = 0; i < length; i++)
|
||||||
|
{
|
||||||
|
int byte1 = m_dataStream.ReadByte();
|
||||||
|
int byte2 = m_dataStream.ReadByte();
|
||||||
|
int byte3 = m_dataStream.ReadByte();
|
||||||
|
int byte4 = m_dataStream.ReadByte();
|
||||||
|
|
||||||
|
result[i] = byte1 << 24 | byte2 << 16 | byte3 << 8 | byte4;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public string Read_string(int length)
|
||||||
|
{
|
||||||
|
var result = Read_bytes(length);
|
||||||
|
return Encoding.ASCII.GetString(result);
|
||||||
|
}
|
||||||
|
public double Read_double()
|
||||||
|
{
|
||||||
|
var result = Read_bytes(4);
|
||||||
|
return BitConverter.ToDouble(result, 0);
|
||||||
|
}
|
||||||
|
public ushort Read_ushort()
|
||||||
|
{
|
||||||
|
var b1 = Read_byte();
|
||||||
|
var b2 = Read_byte();
|
||||||
|
return (ushort)(b1 << 8 | b2);
|
||||||
|
}
|
||||||
|
|
||||||
|
public int Read_int()
|
||||||
|
{
|
||||||
|
var b1 = Read_byte();
|
||||||
|
var b2 = Read_byte();
|
||||||
|
var b3 = Read_byte();
|
||||||
|
var b4 = Read_byte();
|
||||||
|
|
||||||
|
return b1 << 24 | b2 << 16 | b3 << 8 | b4;
|
||||||
|
}
|
||||||
|
|
||||||
|
public sbyte Read_sbyte(sbyte value)
|
||||||
|
{
|
||||||
|
return (sbyte)m_dataStream.ReadByte();
|
||||||
|
}
|
||||||
|
public long Read_long()
|
||||||
|
{
|
||||||
|
var b1 = Read_byte();
|
||||||
|
var b2 = Read_byte();
|
||||||
|
var b3 = Read_byte();
|
||||||
|
var b4 = Read_byte();
|
||||||
|
var b5 = Read_byte();
|
||||||
|
var b6 = Read_byte();
|
||||||
|
var b7 = Read_byte();
|
||||||
|
var b8 = Read_byte();
|
||||||
|
|
||||||
|
return b1 << 56 | b2 << 48 | b3 << 40 | b4 << 32 | b5 << 24 | b6 << 16 | b7 << 8 | b8;
|
||||||
|
}
|
||||||
|
|
||||||
|
public uint Read_uint()
|
||||||
|
{
|
||||||
|
return (uint)Read_int();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public interface IStateBufferObject
|
public interface IStateBufferObject
|
||||||
{
|
{
|
||||||
uint GetSize();
|
uint GetSize();
|
||||||
void SaveState(StateBuffer buffer);
|
void SaveState(StateBuffer buffer);
|
||||||
|
void LoadState(StateReader buffer);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user