diff --git a/BizHawk.Client.Common/BizHawk.Client.Common.csproj b/BizHawk.Client.Common/BizHawk.Client.Common.csproj
index 6c0d64b248..c35c7ad9d1 100644
--- a/BizHawk.Client.Common/BizHawk.Client.Common.csproj
+++ b/BizHawk.Client.Common/BizHawk.Client.Common.csproj
@@ -154,6 +154,7 @@
Bk2Movie.cs
+
@@ -229,7 +230,6 @@
-
diff --git a/BizHawk.Client.Common/config/Config.cs b/BizHawk.Client.Common/config/Config.cs
index 9150f211f0..ea8487d026 100644
--- a/BizHawk.Client.Common/config/Config.cs
+++ b/BizHawk.Client.Common/config/Config.cs
@@ -77,6 +77,7 @@ namespace BizHawk.Client.Common
public bool ForbidUD_LR = false;
public bool ShowContextMenu = true;
public bool EnableBackupMovies = true;
+ public bool MoviesOnDisk = false;
public bool HotkeyConfigAutoTab = true;
public bool InputConfigAutoTab = true;
public bool ShowLogWindow = false;
diff --git a/BizHawk.Client.Common/inputAdapters/InputAdapters.cs b/BizHawk.Client.Common/inputAdapters/InputAdapters.cs
index 414f229943..4042aaa89d 100644
--- a/BizHawk.Client.Common/inputAdapters/InputAdapters.cs
+++ b/BizHawk.Client.Common/inputAdapters/InputAdapters.cs
@@ -121,8 +121,11 @@ namespace BizHawk.Client.Common
//" C " is for N64 "P1 C Up" and the like, which should not be subject to mutexing
+ //regarding the unpressing and UDLR logic...... don't think about it. don't question it. don't look at it.
+
if (button.Contains("Down") && !button.Contains(" C "))
{
+ if (!Source.IsPressed(button)) Unpresses.Remove(button);
prefix = button.GetPrecedingString("Down");
string other = prefix + "Up";
if (Source.IsPressed(other))
@@ -136,6 +139,7 @@ namespace BizHawk.Client.Common
if (button.Contains("Up") && !button.Contains(" C "))
{
+ if (!Source.IsPressed(button)) Unpresses.Remove(button);
prefix = button.GetPrecedingString("Up");
string other = prefix + "Down";
if (Source.IsPressed(other))
@@ -150,6 +154,7 @@ namespace BizHawk.Client.Common
if (button.Contains("Right") && !button.Contains(" C "))
{
+ if (!Source.IsPressed(button)) Unpresses.Remove(button);
prefix = button.GetPrecedingString("Right");
string other = prefix + "Left";
if (Source.IsPressed(other))
@@ -163,6 +168,7 @@ namespace BizHawk.Client.Common
if (button.Contains("Left") && !button.Contains(" C "))
{
+ if (!Source.IsPressed(button)) Unpresses.Remove(button);
prefix = button.GetPrecedingString("Left");
string other = prefix + "Right";
if (Source.IsPressed(other))
diff --git a/BizHawk.Client.Common/movie/bk2/Bk2Movie.InputLog.cs b/BizHawk.Client.Common/movie/bk2/Bk2Movie.InputLog.cs
index 2deacda209..99425ac9ed 100644
--- a/BizHawk.Client.Common/movie/bk2/Bk2Movie.InputLog.cs
+++ b/BizHawk.Client.Common/movie/bk2/Bk2Movie.InputLog.cs
@@ -8,7 +8,7 @@ namespace BizHawk.Client.Common
{
public partial class Bk2Movie
{
- protected List _log = new List();
+ protected IStringLog _log;
protected string LogKey = string.Empty;
public string GetInputLog()
@@ -58,7 +58,7 @@ namespace BizHawk.Client.Common
// We are in record mode so replace the movie log with the one from the savestate
if (!Global.MovieSession.MultiTrack.IsActive)
{
- if (Global.Config.EnableBackupMovies && MakeBackup && _log.Any())
+ if (Global.Config.EnableBackupMovies && MakeBackup && _log.Count != 0)
{
SaveBackup();
MakeBackup = false;
diff --git a/BizHawk.Client.Common/movie/bk2/Bk2Movie.cs b/BizHawk.Client.Common/movie/bk2/Bk2Movie.cs
index 12bffc939e..ddbd4f4dd0 100644
--- a/BizHawk.Client.Common/movie/bk2/Bk2Movie.cs
+++ b/BizHawk.Client.Common/movie/bk2/Bk2Movie.cs
@@ -27,6 +27,8 @@ namespace BizHawk.Client.Common
MakeBackup = true;
Header[HeaderKeys.MOVIEVERSION] = "BizHawk v2.0.0";
+
+ _log = StringLogUtil.MakeStringLog();
}
private string _filename;
diff --git a/BizHawk.Client.Common/movie/bk2/StringLogs.cs b/BizHawk.Client.Common/movie/bk2/StringLogs.cs
new file mode 100644
index 0000000000..355efdc9ef
--- /dev/null
+++ b/BizHawk.Client.Common/movie/bk2/StringLogs.cs
@@ -0,0 +1,196 @@
+using System;
+using System.Collections.Generic;
+using System.Globalization;
+using System.IO;
+using System.Linq;
+using System.Text;
+
+using BizHawk.Common;
+
+namespace BizHawk.Client.Common
+{
+ public static class StringLogUtil
+ {
+ public static bool DefaultToDisk;
+ public static IStringLog MakeStringLog()
+ {
+ if (DefaultToDisk)
+ return new DiskStringLog();
+ else return new ListStringLog();
+ }
+ }
+
+ public interface IStringLog : IDisposable, IEnumerable
+ {
+ void RemoveAt(int index);
+ int Count { get; }
+ void Clear();
+ void Add(string str);
+ string this[int index] { get; set; }
+ void Insert(int index, string val);
+ void InsertRange(int index, IEnumerable collection);
+ void AddRange(IEnumerable collection);
+ void RemoveRange(int index, int count);
+ IStringLog Clone();
+ void CopyTo(string[] array);
+ void CopyTo(int index, string[] array, int arrayIndex, int count);
+ }
+
+ class ListStringLog : List, IStringLog
+ {
+ public IStringLog Clone()
+ {
+ ListStringLog ret = new ListStringLog();
+ ret.AddRange(this);
+ return ret;
+ }
+
+ public void Dispose() { }
+ }
+
+ ///
+ /// A dumb-ish IStringLog with storage on disk with no provision for recovering lost space, except upon Clear()
+ /// The purpose here is to avoid having too complicated buggy logic or a dependency on sqlite or such.
+ /// It should be faster than those alternatives, but wasteful of disk space.
+ /// It should also be easier to add new IList-like methods than dealing with a database
+ ///
+ class DiskStringLog : IStringLog
+ {
+ List Offsets = new List();
+ long cursor = 0;
+ BinaryWriter bw;
+ BinaryReader br;
+
+ FileStream stream;
+ public DiskStringLog()
+ {
+ var path = TempFileCleaner.GetTempFilename("movieOnDisk");
+ stream = new FileStream(path, FileMode.Create, System.Security.AccessControl.FileSystemRights.FullControl, FileShare.None, 4 * 1024, FileOptions.DeleteOnClose);
+ bw = new BinaryWriter(stream);
+ br = new BinaryReader(stream);
+ }
+
+ public IStringLog Clone()
+ {
+ DiskStringLog ret = new DiskStringLog();
+ for (int i = 0; i < Count; i++)
+ ret.Add(this[i]);
+ return ret;
+ }
+
+ public void Dispose()
+ {
+ stream.Dispose();
+ }
+
+ public int Count { get { return Offsets.Count; } }
+
+ public void Clear()
+ {
+ stream.SetLength(0);
+ Offsets.Clear();
+ cursor = 0;
+ }
+
+ public void Add(string str)
+ {
+ Offsets.Add(stream.Position);
+ bw.Write(str);
+ bw.Flush();
+ }
+
+ public void RemoveAt(int index)
+ {
+ Offsets.RemoveAt(index);
+ //no garbage collection in the disk file... oh well.
+ }
+
+ public string this[int index]
+ {
+ get
+ {
+ stream.Position = Offsets[index];
+ return br.ReadString();
+ }
+ set
+ {
+ stream.Position = stream.Length;
+ Offsets[index] = stream.Position;
+ bw.Write(value);
+ bw.Flush();
+ }
+ }
+
+ public void Insert(int index, string val)
+ {
+ Offsets.Insert(index, stream.Position);
+ bw.Write(val);
+ bw.Flush();
+ }
+
+ public void InsertRange(int index, IEnumerable collection)
+ {
+ foreach(var item in collection)
+ Insert(index++,item);
+ }
+
+ public void AddRange(IEnumerable collection)
+ {
+ foreach (var item in collection)
+ Add(item);
+ }
+
+ class Enumerator : IEnumerator
+ {
+ public DiskStringLog log;
+ int index;
+ public string Current { get { return log[index]; } }
+ object System.Collections.IEnumerator.Current { get { return log[index]; } }
+ bool System.Collections.IEnumerator.MoveNext()
+ {
+ index++;
+ if (index >= log.Count)
+ {
+ index = log.Count;
+ return false;
+ }
+ return true;
+ }
+ void System.Collections.IEnumerator.Reset() { index = 0; }
+
+ public void Dispose() { }
+ }
+
+ IEnumerator IEnumerable.GetEnumerator()
+ {
+ return new Enumerator() { log = this };
+ }
+
+ System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
+ {
+ return new Enumerator() { log = this };
+ }
+
+ public void RemoveRange(int index, int count)
+ {
+ int end = index + count - 1;
+ for (int i = 0; i < count; i++)
+ {
+ RemoveAt(end);
+ end--;
+ }
+ }
+
+ public void CopyTo(string[] array)
+ {
+ for (int i = 0; i < Count; i++)
+ array[i] = this[i];
+ }
+
+ public void CopyTo(int index, string[] array, int arrayIndex, int count)
+ {
+ for (int i = 0; i < count; i++)
+ array[i + arrayIndex] = this[index + i];
+ }
+ }
+}
diff --git a/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs b/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs
index 0ddee97697..30e94dd5a9 100644
--- a/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs
+++ b/BizHawk.Client.Common/movie/conversions/MovieConversionExtensions.cs
@@ -145,7 +145,7 @@ namespace BizHawk.Client.Common.MovieConversionExtensions
tas.BinarySavestate = savestate;
tas.ClearLagLog();
- List entries = old.GetLogEntries();
+ var entries = old.GetLogEntries();
tas.CopyLog(entries.Skip(frame));
tas.CopyVerificationLog(old.VerificationLog);
@@ -220,7 +220,7 @@ namespace BizHawk.Client.Common.MovieConversionExtensions
tas.TasStateManager.Clear();
tas.ClearLagLog();
- List entries = old.GetLogEntries();
+ var entries = old.GetLogEntries();
tas.CopyVerificationLog(old.VerificationLog);
tas.CopyVerificationLog(entries);
diff --git a/BizHawk.Client.Common/movie/tasproj/TasBranch.cs b/BizHawk.Client.Common/movie/tasproj/TasBranch.cs
index b290831ff5..0e684efc06 100644
--- a/BizHawk.Client.Common/movie/tasproj/TasBranch.cs
+++ b/BizHawk.Client.Common/movie/tasproj/TasBranch.cs
@@ -11,8 +11,8 @@ namespace BizHawk.Client.Common
public class TasBranch
{
public int Frame { get; set; }
- public byte[] CoreData { get; set; }
- public List InputLog { get; set; }
+ public byte[] CoreData { get; set; }
+ public IStringLog InputLog { get; set; }
public BitmapBuffer OSDFrameBuffer { get; set; }
public TasLagLog LagLog { get; set; }
public TasMovieChangeLog ChangeLog { get; set; }
@@ -63,9 +63,10 @@ namespace BizHawk.Client.Common
});
bs.PutLump(ninput, delegate(TextWriter tw)
- {
- foreach (var line in b.InputLog)
- tw.WriteLine(line);
+ {
+ int todo = b.InputLog.Count;
+ for (int i = 0; i < todo; i++)
+ tw.WriteLine(b.InputLog[i]);
});
bs.PutLump(nframebuffer, delegate(Stream s)
@@ -145,8 +146,8 @@ namespace BizHawk.Client.Common
});
bl.GetLump(ninput, true, delegate(TextReader tr)
- {
- b.InputLog = new List();
+ {
+ b.InputLog = StringLogUtil.MakeStringLog();
string line;
while ((line = tr.ReadLine()) != null)
b.InputLog.Add(line);
diff --git a/BizHawk.Client.Common/movie/tasproj/TasMovie.IO.cs b/BizHawk.Client.Common/movie/tasproj/TasMovie.IO.cs
index 58ce372164..51f979c27f 100644
--- a/BizHawk.Client.Common/movie/tasproj/TasMovie.IO.cs
+++ b/BizHawk.Client.Common/movie/tasproj/TasMovie.IO.cs
@@ -319,7 +319,7 @@ namespace BizHawk.Client.Common
ChangeLog.ClearLog();
}
- private static string InputLogToString(List log)
+ private static string InputLogToString(IStringLog log)
{
var sb = new StringBuilder();
foreach (var record in log)
diff --git a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs
index b78016fdf1..4bfdeeb103 100644
--- a/BizHawk.Client.Common/movie/tasproj/TasMovie.cs
+++ b/BizHawk.Client.Common/movie/tasproj/TasMovie.cs
@@ -21,7 +21,7 @@ namespace BizHawk.Client.Common
public readonly TasSession Session;
private readonly TasLagLog LagLog = new TasLagLog();
private readonly Dictionary InputStateCache = new Dictionary();
- public readonly List VerificationLog = new List(); // For movies that do not begin with power-on, this is the input required to get into the initial state
+ public readonly IStringLog VerificationLog = StringLogUtil.MakeStringLog(); // For movies that do not begin with power-on, this is the input required to get into the initial state
public readonly TasBranchCollection Branches = new TasBranchCollection();
private BackgroundWorker _progressReportWorker = null;
@@ -81,7 +81,7 @@ namespace BizHawk.Client.Common
}
public TasLagLog TasLagLog { get { return LagLog; } }
- public List InputLog { get { return _log; } }
+ public IStringLog InputLog { get { return _log; } }
public TasMovieMarkerList Markers { get; set; }
public bool BindMarkersToInput { get; set; }
public bool UseInputCache { get; set; }
@@ -331,7 +331,7 @@ namespace BizHawk.Client.Common
}
}
- public List GetLogEntries()
+ public IStringLog GetLogEntries()
{
return _log;
}
@@ -350,7 +350,7 @@ namespace BizHawk.Client.Common
{
TimelineBranchFrame = null;
- if (Global.Config.EnableBackupMovies && MakeBackup && _log.Any())
+ if (Global.Config.EnableBackupMovies && MakeBackup && _log.Count != 0)
{
SaveBackup();
MakeBackup = false;
@@ -500,7 +500,8 @@ namespace BizHawk.Client.Common
{
int? divergentPoint = DivergentPoint(_log, branch.InputLog);
- _log = branch.InputLog.ToList();
+ if (_log != null) _log.Dispose();
+ _log = branch.InputLog.Clone();
//_changes = true;
// if there are branch states, they will be loaded anyway
@@ -523,7 +524,7 @@ namespace BizHawk.Client.Common
}
// TODO: use LogGenerators rather than string comparisons
- private int? DivergentPoint(List currentLog, List newLog)
+ private int? DivergentPoint(IStringLog currentLog, IStringLog newLog)
{
int max = newLog.Count;
if (currentLog.Count < newLog.Count)
diff --git a/BizHawk.Client.Common/rewind/StreamBlobDatabase.cs b/BizHawk.Client.Common/rewind/StreamBlobDatabase.cs
index 1ec2f9e157..50dbc74333 100644
--- a/BizHawk.Client.Common/rewind/StreamBlobDatabase.cs
+++ b/BizHawk.Client.Common/rewind/StreamBlobDatabase.cs
@@ -2,6 +2,8 @@
using System.Collections.Generic;
using System.IO;
+using BizHawk.Common;
+
namespace BizHawk.Client.Common
{
///
@@ -23,7 +25,7 @@ namespace BizHawk.Client.Common
_mCapacity = capacity;
if (onDisk)
{
- var path = Path.Combine(Path.GetTempPath(), "bizhawk.rewindbuf-pid" + System.Diagnostics.Process.GetCurrentProcess().Id + "-" + Guid.NewGuid());
+ var path = TempFileCleaner.GetTempFilename("rewindbuf");
// I checked the DeleteOnClose operation to make sure it cleans up when the process is aborted, and it seems to.
// Otherwise we would have a more complex tempfile management problem here.
diff --git a/BizHawk.Client.EmuHawk/Program.cs b/BizHawk.Client.EmuHawk/Program.cs
index e1ed1c6eeb..96c37c97c6 100644
--- a/BizHawk.Client.EmuHawk/Program.cs
+++ b/BizHawk.Client.EmuHawk/Program.cs
@@ -59,13 +59,16 @@ namespace BizHawk.Client.EmuHawk
}
}
- BizHawk.Client.Common.TempFileCleaner.Start();
+ BizHawk.Common.TempFileCleaner.Start();
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
+
+ HawkFile.ArchiveHandlerFactory = new SevenZipSharpArchiveHandler();
+
string iniPath = Path.Combine(Path.GetDirectoryName(Assembly.GetExecutingAssembly().Location), "config.ini");
Global.Config = ConfigService.Load(iniPath);
Global.Config.ResolveDefaults();
- HawkFile.ArchiveHandlerFactory = new SevenZipSharpArchiveHandler();
+ BizHawk.Client.Common.StringLogUtil.DefaultToDisk = Global.Config.MoviesOnDisk;
//super hacky! this needs to be done first. still not worth the trouble to make this system fully proper
for (int i = 0; i < args.Length; i++)
diff --git a/BizHawk.Client.EmuHawk/tools/TAStudio/BookmarksBranchesBox.cs b/BizHawk.Client.EmuHawk/tools/TAStudio/BookmarksBranchesBox.cs
index 9379ade84a..d0ee563907 100644
--- a/BizHawk.Client.EmuHawk/tools/TAStudio/BookmarksBranchesBox.cs
+++ b/BizHawk.Client.EmuHawk/tools/TAStudio/BookmarksBranchesBox.cs
@@ -230,7 +230,7 @@ namespace BizHawk.Client.EmuHawk
{
Frame = Global.Emulator.Frame,
CoreData = (byte[])((Global.Emulator as IStatable).SaveStateBinary().Clone()),
- InputLog = Movie.InputLog.ToList(),
+ InputLog = Movie.InputLog.Clone(),
OSDFrameBuffer = GlobalWin.MainForm.CaptureOSD(),
LagLog = Movie.TasLagLog.Clone(),
ChangeLog = new TasMovieChangeLog(Movie),
diff --git a/BizHawk.Common/BizHawk.Common.csproj b/BizHawk.Common/BizHawk.Common.csproj
index 825675481d..eaeabf9830 100644
--- a/BizHawk.Common/BizHawk.Common.csproj
+++ b/BizHawk.Common/BizHawk.Common.csproj
@@ -76,6 +76,7 @@
+
diff --git a/BizHawk.Common/InstanceDll.cs b/BizHawk.Common/InstanceDll.cs
index 5f4ae2b744..e3be750a65 100644
--- a/BizHawk.Common/InstanceDll.cs
+++ b/BizHawk.Common/InstanceDll.cs
@@ -9,7 +9,7 @@ namespace BizHawk.Common
public InstanceDll(string dllPath)
{
//copy the dll to a temp directory
- var path = Path.Combine(Path.GetTempPath(), "instancedll-pid" + System.Diagnostics.Process.GetCurrentProcess().Id + "-" + Guid.NewGuid()) + "-" + Path.GetFileName(dllPath);
+ var path = TempFileCleaner.GetTempFilename(string.Format("{0}", Path.GetFileNameWithoutExtension(dllPath)),".dll",false);
using (var stream = new FileStream(path, FileMode.Create, System.Security.AccessControl.FileSystemRights.FullControl, FileShare.ReadWrite | FileShare.Delete, 4 * 1024, FileOptions.None))
using (var sdll = File.OpenRead(dllPath))
sdll.CopyTo(stream);
@@ -24,10 +24,8 @@ namespace BizHawk.Common
string envpath_new = Path.GetDirectoryName(path) + ";" + envpath;
Environment.SetEnvironmentVariable("PATH", envpath_new, EnvironmentVariableTarget.Process);
_hModule = LoadLibrary(path); //consider using LoadLibraryEx instead of shenanigans?
- var newfname = Path.GetFileName(path);
- newfname = "bizhawk.bizdelete-" + newfname;
- var newpath = Path.Combine(Path.GetDirectoryName(path), newfname);
- File.Move(path, newpath);
+ var newfname = TempFileCleaner.RenameTempFilenameForDelete(path);
+ File.Move(path, newfname);
}
finally
{
diff --git a/BizHawk.Client.Common/TempFileCleaner.cs b/BizHawk.Common/TempFileManager.cs
similarity index 60%
rename from BizHawk.Client.Common/TempFileCleaner.cs
rename to BizHawk.Common/TempFileManager.cs
index b6a7f0eda6..2c927f6ab5 100644
--- a/BizHawk.Client.Common/TempFileCleaner.cs
+++ b/BizHawk.Common/TempFileManager.cs
@@ -1,7 +1,7 @@
using System;
using System.IO;
-namespace BizHawk.Client.Common
+namespace BizHawk.Common
{
///
/// Starts a thread which cleans any filenames in %temp% beginning with bizhawk.bizdelete.
@@ -12,6 +12,23 @@ namespace BizHawk.Client.Common
{
//todo - manage paths other than %temp%, make not static, or allow adding multiple paths to static instance
+ public static string GetTempFilename(string friendlyname, string extension = null, bool delete = true)
+ {
+ string guidPart = Guid.NewGuid().ToString();
+ var fname = string.Format("biz-{0}-{1}-{2}{3}", System.Diagnostics.Process.GetCurrentProcess().Id, friendlyname, guidPart, extension ?? "");
+ if (delete) fname = RenameTempFilenameForDelete(fname);
+ return Path.Combine(Path.GetTempPath(), fname);
+ }
+
+ public static string RenameTempFilenameForDelete(string path)
+ {
+ string filename = Path.GetFileName(path);
+ string dir = Path.GetDirectoryName(path);
+ if (!filename.StartsWith("biz-")) throw new InvalidOperationException();
+ filename = "bizdelete-" + filename.Remove(0, 4);
+ return Path.Combine(dir, filename);
+ }
+
public static void Start()
{
lock (typeof(TempFileCleaner))
@@ -31,7 +48,7 @@ namespace BizHawk.Client.Common
var di = new DirectoryInfo(Path.GetTempPath());
for (; ; )
{
- var fis = di.GetFiles("bizhawk.bizdelete*");
+ var fis = di.GetFiles("bizdelete-*");
foreach (var fi in fis)
{
try
diff --git a/BizHawk.Emulation.Cores/Libretro/LibRetro.cs b/BizHawk.Emulation.Cores/Libretro/LibRetro.cs
index 30cb95fbd9..3119682383 100644
--- a/BizHawk.Emulation.Cores/Libretro/LibRetro.cs
+++ b/BizHawk.Emulation.Cores/Libretro/LibRetro.cs
@@ -5,6 +5,8 @@ using System.Text;
using System.Runtime.InteropServices;
using System.Reflection;
+#pragma warning disable 649, 169
+
using BizHawk.Common;
namespace BizHawk.Emulation.Cores
diff --git a/Dist/changelog.txt b/Dist/changelog.txt
index c9ddda8ddb..6f74db9a89 100644
--- a/Dist/changelog.txt
+++ b/Dist/changelog.txt
@@ -3,8 +3,10 @@ next
=========================================
*EmuHawk
+**Add libretro player, compatible with selected cores
+**Support Code-Data Logger for GB/GBC, SMS/GG, SNES, and Genesis
+**Add GameShark cheat converter
**Add custom exception display box, so exception info can be clipboarded out
-**Support Code-Data Logger for GB/GBC, SMS/GG, and Genesis
**Cheat Dialog: Fix flakiness in value-editing
**Stop FP precision conflicts between lua scripts and D3D Display method
**DispMethod D3D: More leniency in compilation of optional shaders (so it's able to run on more low spec systems)
@@ -12,8 +14,13 @@ next
**Validate user shaders at selection time
**Support user custom AR selection
**Add --load-state commandline
+**Streamline editing RAM Watches
+**Tidy main form context menu
+**Add more options for U+D/L+R forbid/mutex
**Fix #525 - Memorywatch hex textbox now remembers values across memdomain switches
**Fix #526 - Hex editor repainting fails and garbage rendering
+**Fix #535 - domain list does not update when changing cores
+**Fix #537 - Annoyance with "always on top"
**Tastudio (TODO - editorialize this section)
***color stated frames on WasLag too.
@@ -33,6 +40,8 @@ next
**Lua
**Fix gameExtraPadding coordinate translation
**Clarify script pause/stop state in UI and logic
+**Fix forms.destroyall() and call it when lua console closes
+**Fix error in sizing of lua draw buffers with SetGameExtraPadding (and probably ClientExtraPadding) use
*PSX
**Fix #530 "AV Resizing shows black screen with PSX"
@@ -40,6 +49,7 @@ next
*SNES
**Fix crashes in GFX debugger (including fix #529)
**Recommend proper SNES PAR
+**Build dlls without msvcrt14 dependencies (to run on more systems)
*Genesis
**Fix missing scrollbars in VDP viewer