diff --git a/BizHawk.Client.EmuHawk/MainForm.cs b/BizHawk.Client.EmuHawk/MainForm.cs
index 319a8436b1..c4fc2a6829 100644
--- a/BizHawk.Client.EmuHawk/MainForm.cs
+++ b/BizHawk.Client.EmuHawk/MainForm.cs
@@ -1411,70 +1411,76 @@ namespace BizHawk.Client.EmuHawk
private static void LoadSaveRam()
{
- try // zero says: this is sort of sketchy... but this is no time for rearchitecting
+ if (Global.Emulator.HasSaveRam())
{
- byte[] sram;
-
- // GBA meteor core might not know how big the saveram ought to be, so just send it the whole file
- // GBA vba-next core will try to eat anything, regardless of size
- if (Global.Emulator is GBA || Global.Emulator is VBANext)
+ try // zero says: this is sort of sketchy... but this is no time for rearchitecting
{
- sram = File.ReadAllBytes(PathManager.SaveRamPath(Global.Game));
- }
- else
- {
- var oldram = Global.Emulator.CloneSaveRam();
- if (oldram == null)
- {
- // we're eating this one now. the possible negative consequence is that a user could lose
- // their saveram and not know why
- // MessageBox.Show("Error: tried to load saveram, but core would not accept it?");
- return;
- }
- // why do we silently truncate\pad here instead of warning\erroring?
- sram = new byte[oldram.Length];
- using (var reader = new BinaryReader(
- new FileStream(PathManager.SaveRamPath(Global.Game), FileMode.Open, FileAccess.Read)))
- {
- reader.Read(sram, 0, sram.Length);
- }
- }
+ byte[] sram;
- Global.Emulator.StoreSaveRam(sram);
- }
- catch (IOException)
- {
- GlobalWin.OSD.AddMessage("An error occurred while loading Sram");
+ // GBA meteor core might not know how big the saveram ought to be, so just send it the whole file
+ // GBA vba-next core will try to eat anything, regardless of size
+ if (Global.Emulator is GBA || Global.Emulator is VBANext)
+ {
+ sram = File.ReadAllBytes(PathManager.SaveRamPath(Global.Game));
+ }
+ else
+ {
+ var oldram = (Global.Emulator as ISaveRam).CloneSaveRam();
+ if (oldram == null)
+ {
+ // we're eating this one now. the possible negative consequence is that a user could lose
+ // their saveram and not know why
+ // MessageBox.Show("Error: tried to load saveram, but core would not accept it?");
+ return;
+ }
+ // why do we silently truncate\pad here instead of warning\erroring?
+ sram = new byte[oldram.Length];
+ using (var reader = new BinaryReader(
+ new FileStream(PathManager.SaveRamPath(Global.Game), FileMode.Open, FileAccess.Read)))
+ {
+ reader.Read(sram, 0, sram.Length);
+ }
+ }
+
+ (Global.Emulator as ISaveRam).StoreSaveRam(sram);
+ }
+ catch (IOException)
+ {
+ GlobalWin.OSD.AddMessage("An error occurred while loading Sram");
+ }
}
}
private static void SaveRam()
{
- var path = PathManager.SaveRamPath(Global.Game);
- var f = new FileInfo(path);
- if (f.Directory != null && f.Directory.Exists == false)
+ if (Global.Emulator.HasSaveRam())
{
- f.Directory.Create();
- }
-
- // Make backup first
- if (Global.Config.BackupSaveram && f.Exists)
- {
- var backup = path + ".bak";
- var backupFile = new FileInfo(backup);
- if (backupFile.Exists)
+ var path = PathManager.SaveRamPath(Global.Game);
+ var f = new FileInfo(path);
+ if (f.Directory != null && f.Directory.Exists == false)
{
- backupFile.Delete();
+ f.Directory.Create();
}
- f.CopyTo(backup);
+ // Make backup first
+ if (Global.Config.BackupSaveram && f.Exists)
+ {
+ var backup = path + ".bak";
+ var backupFile = new FileInfo(backup);
+ if (backupFile.Exists)
+ {
+ backupFile.Delete();
+ }
+
+ f.CopyTo(backup);
+ }
+
+ var writer = new BinaryWriter(new FileStream(path, FileMode.Create, FileAccess.Write));
+ var saveram = (Global.Emulator as ISaveRam).CloneSaveRam();
+
+ writer.Write(saveram, 0, saveram.Length);
+ writer.Close();
}
-
- var writer = new BinaryWriter(new FileStream(path, FileMode.Create, FileAccess.Write));
- var saveram = Global.Emulator.CloneSaveRam();
-
- writer.Write(saveram, 0, saveram.Length);
- writer.Close();
}
private void SelectSlot(int num)
@@ -3425,7 +3431,7 @@ namespace BizHawk.Client.EmuHawk
GlobalWin.OSD.AddMessage("SRAM cleared.");
}
}
- else if (Global.Emulator.SaveRamModified)
+ else if (Global.Emulator.HasSaveRam() && (Global.Emulator as ISaveRam).SaveRamModified)
{
SaveRam();
}
diff --git a/BizHawk.Emulation.Common/BizHawk.Emulation.Common.csproj b/BizHawk.Emulation.Common/BizHawk.Emulation.Common.csproj
index 9cac77e364..6dd1e917de 100644
--- a/BizHawk.Emulation.Common/BizHawk.Emulation.Common.csproj
+++ b/BizHawk.Emulation.Common/BizHawk.Emulation.Common.csproj
@@ -65,6 +65,7 @@
+
diff --git a/BizHawk.Emulation.Common/Extensions.cs b/BizHawk.Emulation.Common/Extensions.cs
index 51845394d3..077742ee89 100644
--- a/BizHawk.Emulation.Common/Extensions.cs
+++ b/BizHawk.Emulation.Common/Extensions.cs
@@ -17,6 +17,11 @@ namespace BizHawk.Emulation.Common.IEmulatorExtensions
return core is IMemoryDomains;
}
+ public static bool HasSaveRam(this IEmulator core)
+ {
+ return core is ISaveRam;
+ }
+
public static bool IsNull(this IEmulator core)
{
return core == null || core is NullEmulator;
diff --git a/BizHawk.Emulation.Common/Interfaces/IEmulator.cs b/BizHawk.Emulation.Common/Interfaces/IEmulator.cs
index 60e41a4502..a17faee739 100644
--- a/BizHawk.Emulation.Common/Interfaces/IEmulator.cs
+++ b/BizHawk.Emulation.Common/Interfaces/IEmulator.cs
@@ -78,21 +78,6 @@ namespace BizHawk.Emulation.Common
///
string BoardName { get; }
- ///
- /// return a copy of the saveram. editing it won't do you any good unless you later call StoreSaveRam()
- ///
- byte[] CloneSaveRam();
-
- ///
- /// store new saveram to the emu core. the data should be the same size as the return from ReadSaveRam()
- ///
- void StoreSaveRam(byte[] data);
-
- ///
- /// Whether or not Save ram has been modified since the last save
- ///
- bool SaveRamModified { get; set; }
-
///
/// Resets the Frame and Lag counters, and any other similar counters a core might implement
///
diff --git a/BizHawk.Emulation.Common/Interfaces/ISaveRam.cs b/BizHawk.Emulation.Common/Interfaces/ISaveRam.cs
new file mode 100644
index 0000000000..11c487a943
--- /dev/null
+++ b/BizHawk.Emulation.Common/Interfaces/ISaveRam.cs
@@ -0,0 +1,20 @@
+namespace BizHawk.Emulation.Common
+{
+ public interface ISaveRam : IEmulator, ICoreService
+ {
+ ///
+ /// return a copy of the saveram. editing it won't do you any good unless you later call StoreSaveRam()
+ ///
+ byte[] CloneSaveRam();
+
+ ///
+ /// store new saveram to the emu core. the data should be the same size as the return from ReadSaveRam()
+ ///
+ void StoreSaveRam(byte[] data);
+
+ ///
+ /// Whether or not Save ram has been modified since the last save
+ ///
+ bool SaveRamModified { get; }
+ }
+}
diff --git a/BizHawk.Emulation.Cores/Calculator/TI83.cs b/BizHawk.Emulation.Cores/Calculator/TI83.cs
index 6db50f9f58..b08f5c312b 100644
--- a/BizHawk.Emulation.Cores/Calculator/TI83.cs
+++ b/BizHawk.Emulation.Cores/Calculator/TI83.cs
@@ -524,15 +524,6 @@ namespace BizHawk.Emulation.Cores.Calculators
public bool DeterministicEmulation { get { return true; } }
- public byte[] CloneSaveRam() { return null; }
- public void StoreSaveRam(byte[] data) { }
-
- public bool SaveRamModified
- {
- get { return false; }
- set { }
- }
-
public bool BinarySaveStatesPreferred { get { return false; } }
public void SaveStateBinary(BinaryWriter bw) { SyncState(Serializer.CreateBinaryWriter(bw)); }
public void LoadStateBinary(BinaryReader br) { SyncState(Serializer.CreateBinaryReader(br)); }
diff --git a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Savestate.cs b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Savestate.cs
index 28b7a694be..ea377f233d 100644
--- a/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Savestate.cs
+++ b/BizHawk.Emulation.Cores/Computers/Commodore64/C64.Savestate.cs
@@ -17,23 +17,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64
SyncState(new Serializer(reader));
}
- public byte[] CloneSaveRam()
- {
- return null;
- }
-
- // TODO: when disk support is finished, set this flag according to if any writes to disk were done
- public bool SaveRamModified
- {
- get
- {
- return false;
- }
- set
- {
- }
- }
-
public void SaveStateBinary(BinaryWriter bw)
{
SyncState(new Serializer(bw));
@@ -44,10 +27,6 @@ namespace BizHawk.Emulation.Cores.Computers.Commodore64
SyncState(new Serializer(writer));
}
- public void StoreSaveRam(byte[] data)
- {
- }
-
void SyncState(Serializer ser)
{
ser.BeginSection("core");
diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.cs b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.cs
index e206246bdc..cfc8d10b69 100644
--- a/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.cs
+++ b/BizHawk.Emulation.Cores/Consoles/Atari/2600/Atari2600.cs
@@ -73,8 +73,6 @@ namespace BizHawk.Emulation.Cores.Atari.Atari2600
public bool IsLagFrame { get { return _islag; } }
- public bool SaveRamModified { get; set; }
-
public bool DeterministicEmulation { get; set; }
public bool BinarySaveStatesPreferred { get { return false; } }
@@ -152,13 +150,6 @@ namespace BizHawk.Emulation.Cores.Atari.Atari2600
ser.EndSection();
}
- public byte[] CloneSaveRam()
- {
- return null;
- }
-
- public void StoreSaveRam(byte[] data) { }
-
public void SaveStateText(TextWriter writer)
{
SyncState(Serializer.CreateTextWriter(writer));
diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/7800/Atari7800.cs b/BizHawk.Emulation.Cores/Consoles/Atari/7800/Atari7800.cs
index d085e1ef51..7b738104eb 100644
--- a/BizHawk.Emulation.Cores/Consoles/Atari/7800/Atari7800.cs
+++ b/BizHawk.Emulation.Cores/Consoles/Atari/7800/Atari7800.cs
@@ -16,7 +16,7 @@ namespace BizHawk.Emulation.Cores.Atari.Atari7800
portedVersion: "v1.5",
portedUrl: "http://emu7800.sourceforge.net/"
)]
- public partial class Atari7800 : IEmulator, IMemoryDomains, IDebuggable
+ public partial class Atari7800 : IEmulator, IMemoryDomains, ISaveRam, IDebuggable
{
// TODO:
// some things don't work when you try to plug in a 2600 game
@@ -73,6 +73,7 @@ namespace BizHawk.Emulation.Cores.Atari.Atari7800
private int _frame = 0;
#region saveram
+
public byte[] CloneSaveRam()
{
return (byte[])hsram.Clone();
@@ -88,11 +89,8 @@ namespace BizHawk.Emulation.Cores.Atari.Atari7800
{
return GameInfo.MachineType == MachineType.A7800PAL || GameInfo.MachineType == MachineType.A7800NTSC;
}
- set
- {
- throw new Exception("No one ever uses this, and it won't work with the way MainForm is set up.");
- }
}
+
#endregion
public void Dispose()
diff --git a/BizHawk.Emulation.Cores/Consoles/Atari/lynx/Lynx.cs b/BizHawk.Emulation.Cores/Consoles/Atari/lynx/Lynx.cs
index e0cafa8338..f5972fbd6e 100644
--- a/BizHawk.Emulation.Cores/Consoles/Atari/lynx/Lynx.cs
+++ b/BizHawk.Emulation.Cores/Consoles/Atari/lynx/Lynx.cs
@@ -12,7 +12,7 @@ using Newtonsoft.Json;
namespace BizHawk.Emulation.Cores.Atari.Lynx
{
[CoreAttributes("Handy", "K. Wilkins", true, true, "mednafen 0-9-34-1", "http://mednafen.sourceforge.net/")]
- public class Lynx : IEmulator, IVideoProvider, ISyncSoundProvider, IMemoryDomains
+ public class Lynx : IEmulator, IVideoProvider, ISyncSoundProvider, IMemoryDomains, ISaveRam
{
IntPtr Core;
@@ -312,10 +312,6 @@ namespace BizHawk.Emulation.Cores.Atari.Lynx
IntPtr unused2;
return LibLynx.GetSaveRamPtr(Core, out unused, out unused2);
}
- set
- {
- throw new InvalidOperationException();
- }
}
#endregion
diff --git a/BizHawk.Emulation.Cores/Consoles/Coleco/ColecoVision.cs b/BizHawk.Emulation.Cores/Consoles/Coleco/ColecoVision.cs
index 58ff024192..bf128d0b40 100644
--- a/BizHawk.Emulation.Cores/Consoles/Coleco/ColecoVision.cs
+++ b/BizHawk.Emulation.Cores/Consoles/Coleco/ColecoVision.cs
@@ -170,10 +170,6 @@ namespace BizHawk.Emulation.Cores.ColecoVision
}
}
- public byte[] CloneSaveRam() { return null; }
- public void StoreSaveRam(byte[] data) { }
- public bool SaveRamModified { get; set; }
-
public bool DeterministicEmulation { get { return true; } }
public bool BinarySaveStatesPreferred { get { return false; } }
diff --git a/BizHawk.Emulation.Cores/Consoles/Intellivision/Intellivision.cs b/BizHawk.Emulation.Cores/Consoles/Intellivision/Intellivision.cs
index e826f9469e..f4c6b023d0 100644
--- a/BizHawk.Emulation.Cores/Consoles/Intellivision/Intellivision.cs
+++ b/BizHawk.Emulation.Cores/Consoles/Intellivision/Intellivision.cs
@@ -152,27 +152,6 @@ namespace BizHawk.Emulation.Cores.Intellivision
public bool DeterministicEmulation { get { return true; } }
- [FeatureNotImplemented]
- public byte[] CloneSaveRam()
- {
- return null;
- }
-
- [FeatureNotImplemented]
- public void StoreSaveRam(byte[] data)
- {
-
- }
-
- public bool SaveRamModified
- {
- [FeatureNotImplemented]
- get { return false; }
-
- [FeatureNotImplemented]
- set { }
- }
-
public void ResetCounters()
{
Frame = 0;
diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/Meteor.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/Meteor.cs
index 56f1391913..ea15acabe0 100644
--- a/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/Meteor.cs
+++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/Meteor.cs
@@ -14,7 +14,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBA
isPorted: true,
isReleased: false
)]
- public class GBA : IEmulator, IVideoProvider, ISyncSoundProvider, IGBAGPUViewable, IMemoryDomains, IDebuggable
+ public class GBA : IEmulator, IVideoProvider, ISyncSoundProvider, IGBAGPUViewable, IMemoryDomains, ISaveRam, IDebuggable
{
public IDictionary GetCpuFlagsAndRegisters()
{
@@ -140,8 +140,6 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBA
throw new ObjectDisposedException(this.GetType().ToString());
return LibMeteor.libmeteor_hassaveram();
}
- set
- { }
}
#endregion
diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/VBANext.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/VBANext.cs
index 9d7fe274f6..cdeee037eb 100644
--- a/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/VBANext.cs
+++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/GBA/VBANext.cs
@@ -15,7 +15,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.GBA
{
[CoreAttributes("VBA-Next", "many authors", true, true, "cd508312a29ed8c29dacac1b11c2dce56c338a54", "https://github.com/libretro/vba-next")]
public class VBANext : IEmulator, IVideoProvider, ISyncSoundProvider,
- IGBAGPUViewable, IMemoryDomains, IDebuggable, ISettable