From 3bc6fe1b04482ca17df8f620af4a97dfcdefa055 Mon Sep 17 00:00:00 2001 From: nattthebear Date: Thu, 10 Mar 2016 22:00:52 -0500 Subject: [PATCH] Add the elf stuff back to solution (doesn't do anything, don't use, ugly and embarassing, etc). Switch gpgx to use bizinvoker and slash a few hacks there. Bizinvoker - add support for forwarded attributes in Compatibility mode ([Out], [In]), and support enum, ref enum, and enum[]. Misc cleanup. --- BizHawk.Common/BizHawk.Common.csproj | 1 + .../IImportResolver.cs | 2 +- BizHawk.Common/InstanceDll.cs | 7 +- .../BizHawk.Emulation.Common.csproj | 1 - .../BizInvoke/BizInvoker.cs | 37 +- .../BizInvoke/DynamicLibraryImportResolver.cs | 1 + .../BizHawk.Emulation.Cores.csproj | 5 + .../Consoles/Nintendo/QuickNES/LibQuickNES.cs | 253 ----- .../Sega/gpgx/GPGX.ICodeDataLogger.cs | 4 +- .../Consoles/Sega/gpgx/GPGX.IDebuggable.cs | 8 +- .../Consoles/Sega/gpgx/GPGX.IEmulator.cs | 21 +- .../Consoles/Sega/gpgx/GPGX.IMemoryDomains.cs | 12 +- .../Consoles/Sega/gpgx/GPGX.ISaveRam.cs | 70 +- .../Consoles/Sega/gpgx/GPGX.ISettable.cs | 2 +- .../Consoles/Sega/gpgx/GPGX.IStatable.cs | 4 +- .../Consoles/Sega/gpgx/GPGX.IVideoProvider.cs | 2 +- .../Consoles/Sega/gpgx/GPGX.cs | 37 +- .../Sega/gpgx/GPGXControlConverterDynamic.cs | 264 ----- .../Consoles/Sega/gpgx/GPGXDynamic.cs | 900 ------------------ .../Consoles/Sega/gpgx/LibGPGX.cs | 126 +-- .../Consoles/Sega/gpgx/LibGPGXDynamic.cs | 323 ------- BizHawk.Emulation.Cores/ElfRunner.cs | 15 +- BizHawk.Emulation.Cores/MemoryBlock.cs | 101 +- 23 files changed, 258 insertions(+), 1938 deletions(-) rename {BizHawk.Emulation.Common/BizInvoke => BizHawk.Common}/IImportResolver.cs (91%) delete mode 100644 BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGXControlConverterDynamic.cs delete mode 100644 BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGXDynamic.cs delete mode 100644 BizHawk.Emulation.Cores/Consoles/Sega/gpgx/LibGPGXDynamic.cs diff --git a/BizHawk.Common/BizHawk.Common.csproj b/BizHawk.Common/BizHawk.Common.csproj index a39c714414..961f21bbb3 100644 --- a/BizHawk.Common/BizHawk.Common.csproj +++ b/BizHawk.Common/BizHawk.Common.csproj @@ -100,6 +100,7 @@ + diff --git a/BizHawk.Emulation.Common/BizInvoke/IImportResolver.cs b/BizHawk.Common/IImportResolver.cs similarity index 91% rename from BizHawk.Emulation.Common/BizInvoke/IImportResolver.cs rename to BizHawk.Common/IImportResolver.cs index 8a023c380f..db1b5d388d 100644 --- a/BizHawk.Emulation.Common/BizInvoke/IImportResolver.cs +++ b/BizHawk.Common/IImportResolver.cs @@ -3,7 +3,7 @@ using System.Collections.Generic; using System.Linq; using System.Text; -namespace BizHawk.Emulation.Common.BizInvoke +namespace BizHawk.Common { public interface IImportResolver { diff --git a/BizHawk.Common/InstanceDll.cs b/BizHawk.Common/InstanceDll.cs index e8d24f2cbc..91731b43eb 100644 --- a/BizHawk.Common/InstanceDll.cs +++ b/BizHawk.Common/InstanceDll.cs @@ -4,7 +4,7 @@ using System.Runtime.InteropServices; namespace BizHawk.Common { - public class InstanceDll : IDisposable + public class InstanceDll : IDisposable, IImportResolver { public InstanceDll(string dllPath) { @@ -58,6 +58,11 @@ namespace BizHawk.Common return GetProcAddress(_hModule, procName); } + IntPtr IImportResolver.Resolve(string entryPoint) + { + return GetProcAddress(entryPoint); + } + public void Dispose() { if (_hModule != IntPtr.Zero) diff --git a/BizHawk.Emulation.Common/BizHawk.Emulation.Common.csproj b/BizHawk.Emulation.Common/BizHawk.Emulation.Common.csproj index 992b2cffe6..27c250c339 100644 --- a/BizHawk.Emulation.Common/BizHawk.Emulation.Common.csproj +++ b/BizHawk.Emulation.Common/BizHawk.Emulation.Common.csproj @@ -103,7 +103,6 @@ - diff --git a/BizHawk.Emulation.Common/BizInvoke/BizInvoker.cs b/BizHawk.Emulation.Common/BizInvoke/BizInvoker.cs index 475b5c40fb..eeb64a3da7 100644 --- a/BizHawk.Emulation.Common/BizInvoke/BizInvoker.cs +++ b/BizHawk.Emulation.Common/BizInvoke/BizInvoker.cs @@ -5,6 +5,7 @@ using System.Text; using System.Reflection; using System.Reflection.Emit; using System.Runtime.InteropServices; +using BizHawk.Common; namespace BizHawk.Emulation.Common.BizInvoke { @@ -106,7 +107,7 @@ namespace BizHawk.Emulation.Common.BizInvoke // hooks that will be run on the created proxy object var postCreateHooks = new List>(); - var type = ImplModuleBilder.DefineType("Bizhawk.BizInvokeProxy", TypeAttributes.Class | TypeAttributes.Public | TypeAttributes.Sealed, baseType); + var type = ImplModuleBilder.DefineType("Bizhawk.BizInvokeProxy" + baseType.Name, TypeAttributes.Class | TypeAttributes.Public | TypeAttributes.Sealed, baseType); foreach (var mi in baseMethods) { @@ -144,6 +145,21 @@ namespace BizHawk.Emulation.Common.BizInvoke delegateCtor.SetImplementationFlags(MethodImplAttributes.Runtime | MethodImplAttributes.Managed); var delegateInvoke = delegateType.DefineMethod("Invoke", MethodAttributes.Public | MethodAttributes.HideBySig | MethodAttributes.NewSlot | MethodAttributes.Virtual, returnType, paramTypes); + // we have to project all of the attributes from the baseMethod to the delegateInvoke + // so for something like [Out], the interop engine will see it and use it + for (int i = 0; i < paramInfos.Length; i++) + { + var p = delegateInvoke.DefineParameter(i + 1, ParameterAttributes.None, paramInfos[i].Name); + foreach (var a in paramInfos[i].GetCustomAttributes(false)) + p.SetCustomAttribute(GetAttributeBuilder(a)); + } + { + var p = delegateInvoke.DefineParameter(0, ParameterAttributes.Retval, baseMethod.ReturnParameter.Name); + foreach (var a in baseMethod.ReturnParameter.GetCustomAttributes(false)) + p.SetCustomAttribute(GetAttributeBuilder(a)); + } + + delegateInvoke.SetImplementationFlags(MethodImplAttributes.Runtime | MethodImplAttributes.Managed); // add the [UnmanagedFunctionPointer] to the delegate so interop will know how to call it var attr = new CustomAttributeBuilder(typeof(UnmanagedFunctionPointerAttribute).GetConstructor(new[] { typeof(CallingConvention) }), new object[] { nativeCall }); @@ -255,8 +271,8 @@ namespace BizHawk.Emulation.Common.BizInvoke if (type.IsByRef) { var et = type.GetElementType(); - if (!et.IsPrimitive) - throw new InvalidOperationException("Only refs of primitive types are supported!"); + if (!et.IsPrimitive && !et.IsEnum) + throw new InvalidOperationException("Only refs of primitive or enum types are supported!"); var loc = il.DeclareLocal(type, true); il.Emit(OpCodes.Ldarg, (short)idx); il.Emit(OpCodes.Dup); @@ -267,8 +283,8 @@ namespace BizHawk.Emulation.Common.BizInvoke else if (type.IsArray) { var et = type.GetElementType(); - if (!et.IsPrimitive) - throw new InvalidOperationException("Only arrays of primitive types are supported!"); + if (!et.IsPrimitive && !et.IsEnum) + throw new InvalidOperationException("Only arrays of primitive or enum types are supported!"); // these two cases aren't too hard to add if (type.GetArrayRank() > 1) @@ -315,7 +331,7 @@ namespace BizHawk.Emulation.Common.BizInvoke il.MarkLabel(end); return typeof(IntPtr); } - else if (type.IsPrimitive) + else if (type.IsPrimitive || type.IsEnum) { il.Emit(OpCodes.Ldarg, (short)idx); return type; @@ -325,6 +341,15 @@ namespace BizHawk.Emulation.Common.BizInvoke throw new InvalidOperationException("Unrecognized parameter type!"); } } + + private static CustomAttributeBuilder GetAttributeBuilder(object o) + { + // anything more clever we can do here? + var t = o.GetType(); + if (t == typeof(OutAttribute) || t == typeof(InAttribute)) + return new CustomAttributeBuilder(t.GetConstructor(Type.EmptyTypes), new object[0]); + throw new InvalidOperationException("Unknown parameter attribute " + t.Name); + } } /// diff --git a/BizHawk.Emulation.Common/BizInvoke/DynamicLibraryImportResolver.cs b/BizHawk.Emulation.Common/BizInvoke/DynamicLibraryImportResolver.cs index a437e8a6cd..2acf6e8d3a 100644 --- a/BizHawk.Emulation.Common/BizInvoke/DynamicLibraryImportResolver.cs +++ b/BizHawk.Emulation.Common/BizInvoke/DynamicLibraryImportResolver.cs @@ -3,6 +3,7 @@ using System.Collections.Generic; using System.Linq; using System.Text; using System.Runtime.InteropServices; +using BizHawk.Common; namespace BizHawk.Emulation.Common.BizInvoke { diff --git a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj index b516657907..14e107af39 100644 --- a/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj +++ b/BizHawk.Emulation.Cores/BizHawk.Emulation.Cores.csproj @@ -97,6 +97,9 @@ false + + ..\References\ELFSharp.dll + ..\References\EMU7800.dll @@ -967,6 +970,7 @@ + @@ -979,6 +983,7 @@ + diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/QuickNES/LibQuickNES.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/QuickNES/LibQuickNES.cs index 82e573126e..42af3c2f82 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/QuickNES/LibQuickNES.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/QuickNES/LibQuickNES.cs @@ -7,259 +7,6 @@ using BizHawk.Emulation.Common.BizInvoke; namespace BizHawk.Emulation.Cores.Consoles.Nintendo.QuickNES { - public static class LibQuickNESOld - { - public const string dllname = "libquicknes.dll"; - - /// - /// setup extra mappers. should be done before anything else - /// - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_setup_mappers(); - /// - /// create a new quicknes context - /// - /// NULL on failure - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_new(); - /// - /// destroy a quicknes context - /// - /// context previously returned from qn_new() - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_delete(IntPtr e); - /// - /// load an ines file - /// - /// context - /// file - /// length of file - /// - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_loadines(IntPtr e, byte[] data, int length); - /// - /// set audio sample rate - /// - /// context - /// hz - /// string error - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_set_sample_rate(IntPtr e, int rate); - /// - /// emulate a single frame - /// - /// context - /// pad 1 input - /// pad 2 input - /// string error - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_emulate_frame(IntPtr e, int pad1, int pad2); - /// - /// blit to rgb32 - /// - /// Context - /// rgb32 256x240 packed - /// rgb32 colors, 512 of them - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_blit(IntPtr e, int[] dest, int[] colors, int cropleft, int croptop, int cropright, int cropbottom); - /// - /// get quicknes's default palette - /// - /// 1536 bytes suitable for qn_blit - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_get_default_colors(); - /// - /// get number of times joypad was read in most recent frame - /// - /// context - /// 0 means lag - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern int qn_get_joypad_read_count(IntPtr e); - /// - /// get audio info for most recent frame - /// - /// context - /// number of samples actually created - /// 1 for mono, 2 for stereo - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_get_audio_info(IntPtr e, ref int sample_count, ref int chan_count); - /// - /// get audio for most recent frame. must not be called more than once per frame! - /// - /// context - /// sample buffer - /// length to read into sample buffer - /// length actually read - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern int qn_read_audio(IntPtr e, short[] dest, int max_samples); - /// - /// reset the console - /// - /// context - /// true for powercycle, false for reset button - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_reset(IntPtr e, bool hard); - /// - /// get the required byte size of a savestate - /// - /// context - /// size is returned - /// string error - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_state_size(IntPtr e, ref int size); - /// - /// save state to buffer - /// - /// context - /// buffer - /// length of buffer - /// string error - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_state_save(IntPtr e, byte[] dest, int size); - /// - /// load state from buffer - /// - /// context - /// buffer - /// length of buffer - /// string error - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_state_load(IntPtr e, byte[] src, int size); - /// - /// query battery ram state - /// - /// context - /// true if battery backup sram exists - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern bool qn_has_battery_ram(IntPtr e); - /// - /// query battery ram size - /// - /// context - /// size is returned - /// string error - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_battery_ram_size(IntPtr e, ref int size); - /// - /// save battery ram to buffer - /// - /// context - /// buffer - /// size - /// string error - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_battery_ram_save(IntPtr e, byte[] dest, int size); - /// - /// load battery ram from buffer - /// - /// context - /// buffer - /// size - /// string error - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_battery_ram_load(IntPtr e, byte[] src, int size); - /// - /// clear battery ram - /// - /// context - /// string error - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_battery_ram_clear(IntPtr e); - /// - /// set sprite limit; does not affect emulation - /// - /// context - /// 0 to hide, 8 for normal, 64 for all - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_set_sprite_limit(IntPtr e, int n); - /// - /// get memory area for debugging - /// - /// Context - /// - /// - /// - /// - /// - /// - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern bool qn_get_memory_area(IntPtr e, int which, ref IntPtr data, ref int size, ref bool writable, ref IntPtr name); - /// - /// peek the system bus - /// - /// Context - /// 0000:ffff, but non-ram/rom addresses won't work - /// - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern byte qn_peek_prgbus(IntPtr e, int addr); - /// - /// poke the system bus - /// - /// Context - /// 0000:ffff, but non-ram/rom addresses won't work - /// - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_poke_prgbus(IntPtr e, int addr, byte val); - /// - /// get internal registers - /// - /// Context - /// a, x, y, sp, pc, p - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_get_cpuregs(IntPtr e, [Out] int[] dest); - /// - /// get the mapper that's loaded - /// - /// Context - /// recieves mapper number - /// mapper name - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_get_mapper(IntPtr e, ref int number); - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void TraceCallback(IntPtr data); - - /// - /// set a trace callback to be run on each cycle - /// - /// Context - /// - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_set_tracecb(IntPtr e, TraceCallback cb); - - - - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern byte qn_get_reg2000(IntPtr e); - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_get_palmem(IntPtr e); - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern IntPtr qn_get_oammem(IntPtr e); - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern byte qn_peek_ppu(IntPtr e, int addr); - [DllImport(dllname, CallingConvention = CallingConvention.Cdecl)] - public static extern void qn_peek_ppubus(IntPtr e, byte[] dest); - - /// - /// handle "string error" as returned by some quicknes functions - /// - /// - public static void ThrowStringError(IntPtr p) - { - if (p == IntPtr.Zero) - return; - string s = Marshal.PtrToStringAnsi(p); - if (s == "Unsupported mapper" || s == "Not an iNES file") // Not worth making a new exception for the iNES error, they ultimately are the same problem - { - throw new Emulation.Common.UnsupportedGameException("Quicknes unsupported mapper"); - } - else - { - throw new InvalidOperationException("LibQuickNES error: " + s); - } - } - } - public abstract class LibQuickNES { public const string dllname = "libquicknes.dll"; diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ICodeDataLogger.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ICodeDataLogger.cs index 63d4cfe903..13e0849388 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ICodeDataLogger.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ICodeDataLogger.cs @@ -10,8 +10,8 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx public void SetCDL(CodeDataLog cdl) { CDL = cdl; - if (cdl == null) LibGPGX.gpgx_set_cd_callback(null); - else LibGPGX.gpgx_set_cd_callback(CDCallback); + if (cdl == null) Core.gpgx_set_cd_callback(null); + else Core.gpgx_set_cd_callback(CDCallback); } public void NewCDL(CodeDataLog cdl) diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IDebuggable.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IDebuggable.cs index 671d2ff774..6c34b8c53f 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IDebuggable.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IDebuggable.cs @@ -10,9 +10,9 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx { public IDictionary GetCpuFlagsAndRegisters() { - LibGPGX.RegisterInfo[] regs = new LibGPGX.RegisterInfo[LibGPGX.gpgx_getmaxnumregs()]; + LibGPGX.RegisterInfo[] regs = new LibGPGX.RegisterInfo[Core.gpgx_getmaxnumregs()]; - int n = LibGPGX.gpgx_getregs(regs); + int n = Core.gpgx_getregs(regs); if (n > regs.Length) throw new InvalidOperationException("A buffer overrun has occured!"); var ret = new Dictionary(); @@ -63,7 +63,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx private void RefreshMemCallbacks() { - LibGPGX.gpgx_set_mem_callback( + Core.gpgx_set_mem_callback( MemoryCallbacks.HasReads ? ReadCallback : null, MemoryCallbacks.HasWrites ? WriteCallback : null, MemoryCallbacks.HasExecutes ? ExecCallback : null); @@ -71,7 +71,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx private void KillMemCallbacks() { - LibGPGX.gpgx_set_mem_callback(null, null, null); + Core.gpgx_set_mem_callback(null, null, null); } } } diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IEmulator.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IEmulator.cs index a61ab25a4f..0a3bbe623b 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IEmulator.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IEmulator.cs @@ -29,26 +29,26 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx public void FrameAdvance(bool render, bool rendersound = true) { if (Controller["Reset"]) - LibGPGX.gpgx_reset(false); + Core.gpgx_reset(false); if (Controller["Power"]) - LibGPGX.gpgx_reset(true); + Core.gpgx_reset(true); // do we really have to get each time? nothing has changed - if (!LibGPGX.gpgx_get_control(input, inputsize)) + if (!Core.gpgx_get_control(input, inputsize)) throw new Exception("gpgx_get_control() failed!"); ControlConverter.ScreenWidth = vwidth; ControlConverter.ScreenHeight = vheight; ControlConverter.Convert(Controller, input); - if (!LibGPGX.gpgx_put_control(input, inputsize)) + if (!Core.gpgx_put_control(input, inputsize)) throw new Exception("gpgx_put_control() failed!"); IsLagFrame = true; Frame++; _drivelight = false; - LibGPGX.gpgx_advance(); + Core.gpgx_advance(); UpdateVideo(); update_audio(); @@ -89,16 +89,11 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx { if (!disposed) { - if (AttachedCore != this) - throw new Exception(); - if (SaveRamModified) - _disposedSaveRam = CloneSaveRam(); - KillMemCallbacks(); + // KillMemCallbacks(); // not needed when not single instance + if (Dll != null) + Dll.Dispose(); if (CD != null) - { CD.Dispose(); - } - AttachedCore = null; disposed = true; } } diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IMemoryDomains.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IMemoryDomains.cs index 5e9de71638..f89f695962 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IMemoryDomains.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IMemoryDomains.cs @@ -17,7 +17,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx { IntPtr area = IntPtr.Zero; int size = 0; - IntPtr pname = LibGPGX.gpgx_get_memdom(i, ref area, ref size); + IntPtr pname = Core.gpgx_get_memdom(i, ref area, ref size); if (area == IntPtr.Zero || pname == IntPtr.Zero || size == 0) continue; string name = Marshal.PtrToStringAnsi(pname); @@ -36,7 +36,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx { if (addr < 0 || addr >= 65536) throw new ArgumentOutOfRangeException(); - LibGPGX.gpgx_poke_vram(((int)addr) ^ 1, val); + Core.gpgx_poke_vram(((int)addr) ^ 1, val); }, byteSize: 2)); } @@ -53,14 +53,14 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx var a = (uint)addr; if (a >= 0x1000000) throw new ArgumentOutOfRangeException(); - return LibGPGX.gpgx_peek_m68k_bus(a); + return Core.gpgx_peek_m68k_bus(a); }, delegate (long addr, byte val) { var a = (uint)addr; if (a >= 0x1000000) throw new ArgumentOutOfRangeException(); - LibGPGX.gpgx_write_m68k_bus(a, val); + Core.gpgx_write_m68k_bus(a, val); }, 2); mm.Add(m68Bus); @@ -71,14 +71,14 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx var a = (uint)addr; if (a >= 0x1000000) throw new ArgumentOutOfRangeException(); - return LibGPGX.gpgx_peek_s68k_bus(a); + return Core.gpgx_peek_s68k_bus(a); }, delegate (long addr, byte val) { var a = (uint)addr; if (a >= 0x1000000) throw new ArgumentOutOfRangeException(); - LibGPGX.gpgx_write_s68k_bus(a, val); + Core.gpgx_write_s68k_bus(a, val); }, 2); if (IsSegaCD) diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ISaveRam.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ISaveRam.cs index 747c798df6..4e7e0641c0 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ISaveRam.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ISaveRam.cs @@ -11,69 +11,41 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx { public byte[] CloneSaveRam() { - if (disposed) - { - if (_disposedSaveRam != null) - { - return (byte[])_disposedSaveRam.Clone(); - } - + int size = 0; + IntPtr area = IntPtr.Zero; + Core.gpgx_get_sram(ref area, ref size); + if (size <= 0 || area == IntPtr.Zero) return new byte[0]; - } - else - { - int size = 0; - IntPtr area = IntPtr.Zero; - LibGPGX.gpgx_get_sram(ref area, ref size); - if (size <= 0 || area == IntPtr.Zero) - return new byte[0]; - LibGPGX.gpgx_sram_prepread(); + Core.gpgx_sram_prepread(); - byte[] ret = new byte[size]; - Marshal.Copy(area, ret, 0, size); - return ret; - } + byte[] ret = new byte[size]; + Marshal.Copy(area, ret, 0, size); + return ret; } public void StoreSaveRam(byte[] data) { - if (disposed) - { - throw new ObjectDisposedException(typeof(GPGX).ToString()); - } - else - { - int size = 0; - IntPtr area = IntPtr.Zero; - LibGPGX.gpgx_get_sram(ref area, ref size); - if (size <= 0 || area == IntPtr.Zero) - return; - if (size != data.Length) - throw new Exception("Unexpected saveram size"); + int size = 0; + IntPtr area = IntPtr.Zero; + Core.gpgx_get_sram(ref area, ref size); + if (size <= 0 || area == IntPtr.Zero) + return; + if (size != data.Length) + throw new Exception("Unexpected saveram size"); - Marshal.Copy(data, 0, area, size); - LibGPGX.gpgx_sram_commitwrite(); - } + Marshal.Copy(data, 0, area, size); + Core.gpgx_sram_commitwrite(); } public bool SaveRamModified { get { - if (disposed) - { - return _disposedSaveRam != null; - } - else - { - int size = 0; - IntPtr area = IntPtr.Zero; - LibGPGX.gpgx_get_sram(ref area, ref size); - return size > 0 && area != IntPtr.Zero; - } + int size = 0; + IntPtr area = IntPtr.Zero; + Core.gpgx_get_sram(ref area, ref size); + return size > 0 && area != IntPtr.Zero; } } - - private byte[] _disposedSaveRam = null; } } diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ISettable.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ISettable.cs index e5ce32ed4c..7450a6cb53 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ISettable.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.ISettable.cs @@ -23,7 +23,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx { bool ret = GPGXSettings.NeedsReboot(_settings, o); _settings = o; - LibGPGX.gpgx_set_draw_mask(_settings.GetDrawMask()); + Core.gpgx_set_draw_mask(_settings.GetDrawMask()); return ret; } diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IStatable.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IStatable.cs index da6febc406..c7782c82db 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IStatable.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IStatable.cs @@ -48,7 +48,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx } reader.Read(_savebuff, 0, _savebuff.Length); - if (!LibGPGX.gpgx_state_load(_savebuff, _savebuff.Length)) + if (!Core.gpgx_state_load(_savebuff, _savebuff.Length)) { throw new Exception("gpgx_state_load() returned false"); } @@ -62,7 +62,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx public void SaveStateBinary(BinaryWriter writer) { - if (!LibGPGX.gpgx_state_save(_savebuff, _savebuff.Length)) + if (!Core.gpgx_state_save(_savebuff, _savebuff.Length)) throw new Exception("gpgx_state_save() returned false"); writer.Write(_savebuff.Length); diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IVideoProvider.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IVideoProvider.cs index d60ff246dd..024be21c56 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IVideoProvider.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.IVideoProvider.cs @@ -40,7 +40,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx int gppitch, gpwidth, gpheight; IntPtr src = IntPtr.Zero; - LibGPGX.gpgx_get_video(out gpwidth, out gpheight, out gppitch, ref src); + Core.gpgx_get_video(out gpwidth, out gpheight, out gppitch, ref src); vwidth = gpwidth; vheight = gpheight; diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.cs index 3af4241275..048d7e6e0e 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGX.cs @@ -2,6 +2,9 @@ using System.Runtime.InteropServices; using BizHawk.Emulation.Common; +using BizHawk.Common; +using System.IO; +using BizHawk.Emulation.Common.BizInvoke; namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx { @@ -12,12 +15,13 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx isReleased: true, portedVersion: "r874", portedUrl: "https://code.google.com/p/genplus-gx/", - singleInstance: true + singleInstance: false )] public partial class GPGX : IEmulator, ISyncSoundProvider, IVideoProvider, ISaveRam, IStatable, IRegionable, IInputPollable, IDebuggable, IDriveLight, ICodeDataLogger, IDisassemblable { - static GPGX AttachedCore = null; + LibGPGX Core; + InstanceDll Dll; DiscSystem.Disc CD; DiscSystem.DiscSectorReader DiscSectorReader; @@ -50,7 +54,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx public GPGX(CoreComm comm, byte[] rom, DiscSystem.Disc CD, object Settings, object SyncSettings) { ServiceProvider = new BasicServiceProvider(this); - // this can influence some things internally + // this can influence some things internally (autodetect romtype, etc) string romextension = "GEN"; // three or six button? @@ -64,16 +68,13 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx try { + Dll = new InstanceDll(Path.Combine(comm.CoreFileProvider.DllPath(), LibGPGX.DllName)); + Core = BizInvoker.GetInvoker(Dll); + _syncSettings = (GPGXSyncSettings)SyncSettings ?? new GPGXSyncSettings(); _settings = (GPGXSettings)Settings ?? new GPGXSettings(); CoreComm = comm; - if (AttachedCore != null) - { - AttachedCore.Dispose(); - AttachedCore = null; - } - AttachedCore = this; LoadCallback = new LibGPGX.load_archive_cb(load_archive); @@ -119,13 +120,13 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx } - if (!LibGPGX.gpgx_init(romextension, LoadCallback, _syncSettings.UseSixButton, system_a, system_b, _syncSettings.Region, _settings.GetNativeSettings())) + if (!Core.gpgx_init(romextension, LoadCallback, _syncSettings.UseSixButton, system_a, system_b, _syncSettings.Region, _settings.GetNativeSettings())) throw new Exception("gpgx_init() failed"); { int fpsnum = 60; int fpsden = 1; - LibGPGX.gpgx_get_fps(ref fpsnum, ref fpsden); + Core.gpgx_get_fps(ref fpsnum, ref fpsden); CoreComm.VsyncNum = fpsnum; CoreComm.VsyncDen = fpsden; Region = CoreComm.VsyncRate > 55 ? DisplayType.NTSC : DisplayType.PAL; @@ -133,8 +134,8 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx // compute state size { - byte[] tmp = new byte[LibGPGX.gpgx_state_max_size()]; - int size = LibGPGX.gpgx_state_size(tmp, tmp.Length); + byte[] tmp = new byte[Core.gpgx_state_max_size()]; + int size = Core.gpgx_state_size(tmp, tmp.Length); if (size <= 0) throw new Exception("Couldn't Determine GPGX internal state size!"); _savebuff = new byte[size]; @@ -150,7 +151,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx SetMemoryDomains(); InputCallback = new LibGPGX.input_cb(input_callback); - LibGPGX.gpgx_set_input_callback(InputCallback); + Core.gpgx_set_input_callback(InputCallback); if (CD != null) DriveLightEnabled = true; @@ -353,7 +354,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx private void SetControllerDefinition() { inputsize = Marshal.SizeOf(typeof(LibGPGX.InputData)); - if (!LibGPGX.gpgx_get_control(input, inputsize)) + if (!Core.gpgx_get_control(input, inputsize)) throw new Exception("gpgx_get_control() failed"); ControlConverter = new GPGXControlConverter(input); @@ -369,8 +370,8 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx public void UpdateVDPViewContext(LibGPGX.VDPView view) { - LibGPGX.gpgx_get_vdp_view(view); - LibGPGX.gpgx_flush_vram(); // fully regenerate internal caches as needed + Core.gpgx_get_vdp_view(view); + Core.gpgx_flush_vram(); // fully regenerate internal caches as needed } short[] samples = new short[4096]; @@ -391,7 +392,7 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx void update_audio() { IntPtr src = IntPtr.Zero; - LibGPGX.gpgx_get_audio(ref nsamp, ref src); + Core.gpgx_get_audio(ref nsamp, ref src); if (src != IntPtr.Zero) { Marshal.Copy(src, samples, 0, nsamp * 2); diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGXControlConverterDynamic.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGXControlConverterDynamic.cs deleted file mode 100644 index b7f2041578..0000000000 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGXControlConverterDynamic.cs +++ /dev/null @@ -1,264 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Linq; -using System.Text; - -using BizHawk.Common; -using BizHawk.Emulation.Common; - -namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx -{ - public class GPGXControlConverterDynamic - { - // this isn't all done - - struct CName - { - public string Name; - public LibGPGXDynamic.INPUT_KEYS Key; - public CName(string Name, LibGPGXDynamic.INPUT_KEYS Key) - { - this.Name = Name; - this.Key = Key; - } - } - - static CName[] Genesis3 = - { - new CName("Up", LibGPGXDynamic.INPUT_KEYS.INPUT_UP), - new CName("Down", LibGPGXDynamic.INPUT_KEYS.INPUT_DOWN), - new CName("Left", LibGPGXDynamic.INPUT_KEYS.INPUT_LEFT), - new CName("Right", LibGPGXDynamic.INPUT_KEYS.INPUT_RIGHT), - new CName("A", LibGPGXDynamic.INPUT_KEYS.INPUT_A), - new CName("B", LibGPGXDynamic.INPUT_KEYS.INPUT_B), - new CName("C", LibGPGXDynamic.INPUT_KEYS.INPUT_C), - new CName("Start", LibGPGXDynamic.INPUT_KEYS.INPUT_START), - }; - - static CName[] Genesis6 = - { - new CName("Up", LibGPGXDynamic.INPUT_KEYS.INPUT_UP), - new CName("Down", LibGPGXDynamic.INPUT_KEYS.INPUT_DOWN), - new CName("Left", LibGPGXDynamic.INPUT_KEYS.INPUT_LEFT), - new CName("Right", LibGPGXDynamic.INPUT_KEYS.INPUT_RIGHT), - new CName("A", LibGPGXDynamic.INPUT_KEYS.INPUT_A), - new CName("B", LibGPGXDynamic.INPUT_KEYS.INPUT_B), - new CName("C", LibGPGXDynamic.INPUT_KEYS.INPUT_C), - new CName("Start", LibGPGXDynamic.INPUT_KEYS.INPUT_START), - new CName("X", LibGPGXDynamic.INPUT_KEYS.INPUT_X), - new CName("Y", LibGPGXDynamic.INPUT_KEYS.INPUT_Y), - new CName("Z", LibGPGXDynamic.INPUT_KEYS.INPUT_Z), - new CName("Mode", LibGPGXDynamic.INPUT_KEYS.INPUT_MODE), - }; - - static CName[] Mouse = - { - new CName("Mouse Left", LibGPGXDynamic.INPUT_KEYS.INPUT_MOUSE_LEFT), - new CName("Mouse Center", LibGPGXDynamic.INPUT_KEYS.INPUT_MOUSE_CENTER), - new CName("Mouse Right", LibGPGXDynamic.INPUT_KEYS.INPUT_MOUSE_RIGHT), - new CName("Mouse Start", LibGPGXDynamic.INPUT_KEYS.INPUT_MOUSE_START), - }; - - static CName[] Lightgun = - { - new CName("Lightgun Trigger", LibGPGXDynamic.INPUT_KEYS.INPUT_MENACER_TRIGGER), - new CName("Lightgun Start", LibGPGXDynamic.INPUT_KEYS.INPUT_MENACER_START), - }; - - static CName[] Activator = - { - new CName("1L", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_1L), - new CName("1U", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_1U), - new CName("2L", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_2L), - new CName("2U", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_2U), - new CName("3L", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_3L), - new CName("3U", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_3U), - new CName("4L", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_4L), - new CName("4U", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_4U), - new CName("5L", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_5L), - new CName("5U", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_5U), - new CName("6L", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_6L), - new CName("6U", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_6U), - new CName("7L", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_7L), - new CName("7U", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_7U), - new CName("8L", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_8L), - new CName("8U", LibGPGXDynamic.INPUT_KEYS.INPUT_ACTIVATOR_8U), - }; - - static CName[] XEA1P = - { - new CName("XE A", LibGPGXDynamic.INPUT_KEYS.INPUT_XE_A), - new CName("XE B", LibGPGXDynamic.INPUT_KEYS.INPUT_XE_B), - new CName("XE C", LibGPGXDynamic.INPUT_KEYS.INPUT_XE_C), - new CName("XE D", LibGPGXDynamic.INPUT_KEYS.INPUT_XE_D), - new CName("XE Start", LibGPGXDynamic.INPUT_KEYS.INPUT_XE_START), - new CName("XE Select", LibGPGXDynamic.INPUT_KEYS.INPUT_XE_SELECT), - new CName("XE E1", LibGPGXDynamic.INPUT_KEYS.INPUT_XE_E1), - new CName("XE E2", LibGPGXDynamic.INPUT_KEYS.INPUT_XE_E2), - }; - - static ControllerDefinition.FloatRange MouseRange = new ControllerDefinition.FloatRange(-256, 0, 255); - // lightgun needs to be transformed to match the current screen resolution - static ControllerDefinition.FloatRange LightgunRange = new ControllerDefinition.FloatRange(0, 5000, 10000); - - static ControllerDefinition.FloatRange XEA1PRange = new ControllerDefinition.FloatRange(-128, 0, 127); - - LibGPGXDynamic.InputData target = null; - IController source = null; - - List Converts = new List(); - - public ControllerDefinition ControllerDef { get; private set; } - - void AddToController(int idx, int player, IEnumerable Buttons) - { - foreach (var Button in Buttons) - { - string Name = string.Format("P{0} {1}", player, Button.Name); - ControllerDef.BoolButtons.Add(Name); - var ButtonFlag = Button.Key; - Converts.Add(delegate() - { - if (source.IsPressed(Name)) - target.pad[idx] |= ButtonFlag; - }); - } - } - - void DoMouseAnalog(int idx, int player) - { - string NX = string.Format("P{0} Mouse X", player); - string NY = string.Format("P{0} Mouse Y", player); - ControllerDef.FloatControls.Add(NX); - ControllerDef.FloatControls.Add(NY); - ControllerDef.FloatRanges.Add(MouseRange); - ControllerDef.FloatRanges.Add(MouseRange); - Converts.Add(delegate() - { - target.analog[(2 * idx) + 0] = (short)source.GetFloat(NX); - target.analog[(2 * idx) + 1] = (short)source.GetFloat(NY); - }); - } - - void DoLightgunAnalog(int idx, int player) - { - string NX = string.Format("P{0} Lightgun X", player); - string NY = string.Format("P{0} Lightgun Y", player); - ControllerDef.FloatControls.Add(NX); - ControllerDef.FloatControls.Add(NY); - ControllerDef.FloatRanges.Add(LightgunRange); - ControllerDef.FloatRanges.Add(LightgunRange); - Converts.Add(delegate() - { - target.analog[(2 * idx) + 0] = (short)(source.GetFloat(NX) / 10000.0f * (ScreenWidth - 1)); - target.analog[(2 * idx) + 1] = (short)(source.GetFloat(NY) / 10000.0f * (ScreenHeight - 1)); - }); - } - - void DoXEA1PAnalog(int idx, int player) - { - string NX = string.Format("P{0} Stick X", player); - string NY = string.Format("P{0} Stick Y", player); - string NZ = string.Format("P{0} Stick Z", player); - ControllerDef.FloatControls.Add(NX); - ControllerDef.FloatControls.Add(NY); - ControllerDef.FloatControls.Add(NZ); - ControllerDef.FloatRanges.Add(XEA1PRange); - ControllerDef.FloatRanges.Add(XEA1PRange); - ControllerDef.FloatRanges.Add(XEA1PRange); - Converts.Add(delegate() - { - target.analog[(2 * idx) + 0] = (short)(source.GetFloat(NX)); - target.analog[(2 * idx) + 1] = (short)(source.GetFloat(NY)); - // +2 is correct in how gpgx internally does this - target.analog[(2 * idx) + 2] = (short)(source.GetFloat(NZ)); - }); - } - - public GPGXControlConverterDynamic(LibGPGXDynamic.InputData input) - { - Console.WriteLine("Genesis Controller report:"); - foreach (var e in input.system) - Console.WriteLine("S:{0}", e); - foreach (var e in input.dev) - Console.WriteLine("D:{0}", e); - - int player = 1; - - ControllerDef = new ControllerDefinition(); - - ControllerDef.BoolButtons.Add("Power"); - ControllerDef.BoolButtons.Add("Reset"); - - for (int i = 0; i < LibGPGXDynamic.MAX_DEVICES; i++) - { - switch (input.dev[i]) - { - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_PAD3B: - AddToController(i, player, Genesis3); - player++; - break; - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_PAD6B: - AddToController(i, player, Genesis6); - player++; - break; - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_MOUSE: - AddToController(i, player, Mouse); - DoMouseAnalog(i, player); - player++; - break; - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_NONE: - break; - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_LIGHTGUN: - // supports menacers and justifiers - AddToController(i, player, Lightgun); - DoLightgunAnalog(i, player); - player++; - break; - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_PAD2B: - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_PADDLE: - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_SPORTSPAD: - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_TEREBI: - throw new Exception("Master System only device? Something went wrong."); - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_ACTIVATOR: - AddToController(i, player, Activator); - player++; - break; - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_XE_A1P: - AddToController(i, player, XEA1P); - DoXEA1PAnalog(i, player); - player++; - break; - case LibGPGXDynamic.INPUT_DEVICE.DEVICE_PICO: - // PICO isn't finished on the unmanaged side either - throw new Exception("Sega PICO not implemented yet!"); - default: - throw new Exception("Unknown Genesis control device! Something went wrong."); - } - } - - ControllerDef.Name = "GPGX Genesis Controller"; - } - - public void Convert(IController source, LibGPGXDynamic.InputData target) - { - this.source = source; - this.target = target; - target.ClearAllBools(); - foreach (var f in Converts) - f(); - this.source = null; - this.target = null; - } - - /// - /// must be set for proper lightgun operation - /// - public int ScreenWidth { get; set; } - /// - /// must be set for proper lightgun operation - /// - public int ScreenHeight { get; set; } - - } -} diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGXDynamic.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGXDynamic.cs deleted file mode 100644 index 49c6bd2d9c..0000000000 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/GPGXDynamic.cs +++ /dev/null @@ -1,900 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Linq; -using System.Text; - -using BizHawk.Common.BufferExtensions; -using BizHawk.Emulation.Common; -using BizHawk.Common; - -using System.Runtime.InteropServices; - -using System.IO; - -using System.ComponentModel; - - -namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx -{ - [CoreAttributes( - "Genplus-gx", - "", - isPorted: true, - isReleased: true, - portedVersion: "r874", - portedUrl: "https://code.google.com/p/genplus-gx/", - singleInstance: false - )] - public class GPGXDynamic : IEmulator, ISyncSoundProvider, IVideoProvider, ISaveRam, IStatable, - IInputPollable, IDebuggable, ISettable, IDriveLight - { - DiscSystem.Disc CD; - byte[] romfile; - bool drivelight; - - bool disposed = false; - - LibGPGXDynamic gpgx; - ElfRunner elf; - - LibGPGXDynamic.load_archive_cb LoadCallback = null; - LibGPGXDynamic.input_cb InputCallback = null; - - LibGPGXDynamic.InputData input = new LibGPGXDynamic.InputData(); - - public enum ControlType - { - None, - OnePlayer, - Normal, - Xea1p, - Activator, - Teamplayer, - Wayplay, - Mouse - }; - - //[CoreConstructor("GEN")] - public GPGXDynamic(CoreComm comm, byte[] file, object Settings, object SyncSettings) - : this(comm, file, null, Settings, SyncSettings) - { - } - - public GPGXDynamic(CoreComm comm, byte[] rom, DiscSystem.Disc CD, object Settings, object SyncSettings) - { - ServiceProvider = new BasicServiceProvider(this); - (ServiceProvider as BasicServiceProvider).Register(_tracer); - // this can influence some things internally - string romextension = "GEN"; - - // three or six button? - // http://www.sega-16.com/forum/showthread.php?4398-Forgotten-Worlds-giving-you-GAME-OVER-immediately-Fix-inside&highlight=forgotten%20worlds - - //hack, don't use - //romfile = File.ReadAllBytes(@"D:\encodes\bizhawksrc\output\SANIC CD\PierSolar (E).bin"); - if (rom != null && rom.Length > 16 * 1024 * 1024) - { - throw new InvalidOperationException("ROM too big! Did you try to load a CD as a ROM?"); - } - - elf = new ElfRunner(Path.Combine(comm.CoreFileProvider.DllPath(), "gpgx.elf")); - try - { - gpgx = new LibGPGXDynamic(); - elf.PopulateInterface(gpgx); - - _SyncSettings = (GPGXSyncSettings)SyncSettings ?? new GPGXSyncSettings(); - - CoreComm = comm; - - LoadCallback = new LibGPGXDynamic.load_archive_cb(load_archive); - - this.romfile = rom; - this.CD = CD; - - LibGPGXDynamic.INPUT_SYSTEM system_a = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_NONE; - LibGPGXDynamic.INPUT_SYSTEM system_b = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_NONE; - - switch (this._SyncSettings.ControlType) - { - case ControlType.None: - default: - break; - case ControlType.Activator: - system_a = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_ACTIVATOR; - system_b = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_ACTIVATOR; - break; - case ControlType.Normal: - system_a = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_MD_GAMEPAD; - system_b = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_MD_GAMEPAD; - break; - case ControlType.OnePlayer: - system_a = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_MD_GAMEPAD; - break; - case ControlType.Xea1p: - system_a = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_XE_A1P; - break; - case ControlType.Teamplayer: - system_a = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_TEAMPLAYER; - system_b = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_TEAMPLAYER; - break; - case ControlType.Wayplay: - system_a = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_WAYPLAY; - system_b = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_WAYPLAY; - break; - case ControlType.Mouse: - system_a = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_MD_GAMEPAD; - // seems like mouse in port 1 would be supported, but not both at the same time - system_b = LibGPGXDynamic.INPUT_SYSTEM.SYSTEM_MOUSE; - break; - } - - - if (!gpgx.gpgx_init(romextension, LoadCallback, this._SyncSettings.UseSixButton, system_a, system_b, this._SyncSettings.Region)) - throw new Exception("gpgx_init() failed"); - - { - int fpsnum = 60; - int fpsden = 1; - gpgx.gpgx_get_fps(ref fpsnum, ref fpsden); - CoreComm.VsyncNum = fpsnum; - CoreComm.VsyncDen = fpsden; - DisplayType = CoreComm.VsyncRate > 55 ? DisplayType.NTSC : DisplayType.PAL; - } - - // compute state size - { - byte[] tmp = new byte[gpgx.gpgx_state_max_size()]; - int size = gpgx.gpgx_state_size(tmp, tmp.Length); - if (size <= 0) - throw new Exception("Couldn't Determine GPGX internal state size!"); - savebuff = new byte[size]; - savebuff2 = new byte[savebuff.Length + 13]; - Console.WriteLine("GPGX Internal State Size: {0}", size); - } - - SetControllerDefinition(); - - // pull the default video size from the core - update_video_initial(); - - SetMemoryDomains(); - - InputCallback = new LibGPGXDynamic.input_cb(input_callback); - gpgx.gpgx_set_input_callback(InputCallback); - - if (CD != null) - DriveLightEnabled = true; - - PutSettings((GPGXSettings)Settings ?? new GPGXSettings()); - - InitMemCallbacks(); - KillMemCallbacks(); - } - catch - { - Dispose(); - throw; - } - } - - public IEmulatorServiceProvider ServiceProvider { get; private set; } - - public bool DriveLightEnabled { get; private set; } - public bool DriveLightOn { get; private set; } - - /// - /// core callback for file loading - /// - /// string identifying file to be loaded - /// buffer to load file to - /// maximum length buffer can hold - /// actual size loaded, or 0 on failure - int load_archive(string filename, IntPtr buffer, int maxsize) - { - byte[] srcdata = null; - - if (buffer == IntPtr.Zero) - { - Console.WriteLine("Couldn't satisfy firmware request {0} because buffer == NULL", filename); - return 0; - } - - if (filename == "PRIMARY_ROM") - { - if (romfile == null) - { - Console.WriteLine("Couldn't satisfy firmware request PRIMARY_ROM because none was provided."); - return 0; - } - srcdata = romfile; - } - else if (filename == "PRIMARY_CD" || filename == "SECONDARY_CD") - { - if (filename == "PRIMARY_CD" && romfile != null) - { - Console.WriteLine("Declined to satisfy firmware request PRIMARY_CD because PRIMARY_ROM was provided."); - return 0; - } - else - { - if (CD == null) - { - Console.WriteLine("Couldn't satisfy firmware request {0} because none was provided.", filename); - return 0; - } - srcdata = GetCDData(); - if (srcdata.Length != maxsize) - { - Console.WriteLine("Couldn't satisfy firmware request {0} because of struct size.", filename); - return 0; - } - } - } - else - { - // use fromtend firmware interface - - string firmwareID = null; - switch (filename) - { - case "CD_BIOS_EU": firmwareID = "CD_BIOS_EU"; break; - case "CD_BIOS_JP": firmwareID = "CD_BIOS_JP"; break; - case "CD_BIOS_US": firmwareID = "CD_BIOS_US"; break; - default: - break; - } - if (firmwareID != null) - { - // this path will be the most common PEBKAC error, so be a bit more vocal about the problem - srcdata = CoreComm.CoreFileProvider.GetFirmware("GEN", firmwareID, false, "GPGX firmwares are usually required."); - if (srcdata == null) - { - Console.WriteLine("Frontend couldn't satisfy firmware request GEN:{0}", firmwareID); - return 0; - } - } - else - { - Console.WriteLine("Unrecognized firmware request {0}", filename); - return 0; - } - } - - if (srcdata != null) - { - if (srcdata.Length > maxsize) - { - Console.WriteLine("Couldn't satisfy firmware request {0} because {1} > {2}", filename, srcdata.Length, maxsize); - return 0; - } - else - { - Marshal.Copy(srcdata, 0, buffer, srcdata.Length); - Console.WriteLine("Firmware request {0} satisfied at size {1}", filename, srcdata.Length); - return srcdata.Length; - } - } - else - { - throw new Exception(); - //Console.WriteLine("Couldn't satisfy firmware request {0} for unknown reasons", filename); - //return 0; - } - - } - - void CDRead(int lba, IntPtr dest, bool audio) - { - if (audio) - { - byte[] data = new byte[2352]; - if (lba < CD.LBACount) - { - CD.ReadLBA_2352(lba, data, 0); - } - else - { - // audio seems to read slightly past the end of disks; probably innoculous - // just send back 0s. - // Console.WriteLine("!!{0} >= {1}", lba, CD.LBACount); - } - Marshal.Copy(data, 0, dest, 2352); - } - else - { - byte[] data = new byte[2048]; - CD.ReadLBA_2048(lba, data, 0); - Marshal.Copy(data, 0, dest, 2048); - drivelight = true; - } - } - - LibGPGXDynamic.cd_read_cb cd_callback_handle; - - unsafe byte[] GetCDData() - { - LibGPGXDynamic.CDData ret = new LibGPGXDynamic.CDData(); - int size = Marshal.SizeOf(ret); - - ret.readcallback = cd_callback_handle = new LibGPGXDynamic.cd_read_cb(CDRead); - - var ses = CD.Structure.Sessions[0]; - int ntrack = ses.Tracks.Count; - - // bet you a dollar this is all wrong - for (int i = 0; i < LibGPGXDynamic.CD_MAX_TRACKS; i++) - { - if (i < ntrack) - { - ret.tracks[i].start = ses.Tracks[i].Indexes[1].aba - 150; - ret.tracks[i].end = ses.Tracks[i].LengthInSectors + ret.tracks[i].start; - if (i == ntrack - 1) - { - ret.end = ret.tracks[i].end; - ret.last = ntrack; - } - } - else - { - ret.tracks[i].start = 0; - ret.tracks[i].end = 0; - } - } - - byte[] retdata = new byte[size]; - - fixed (byte* p = &retdata[0]) - { - Marshal.StructureToPtr(ret, (IntPtr)p, false); - } - return retdata; - } - - - #region controller - - /// - /// size of native input struct - /// - int inputsize; - - GPGXControlConverterDynamic ControlConverter; - - public ControllerDefinition ControllerDefinition { get; private set; } - public IController Controller { get; set; } - - void SetControllerDefinition() - { - inputsize = Marshal.SizeOf(typeof(LibGPGXDynamic.InputData)); - if (!gpgx.gpgx_get_control(input, inputsize)) - throw new Exception("gpgx_get_control() failed"); - - ControlConverter = new GPGXControlConverterDynamic(input); - ControllerDefinition = ControlConverter.ControllerDef; - } - - public LibGPGXDynamic.INPUT_DEVICE[] GetDevices() - { - return (LibGPGXDynamic.INPUT_DEVICE[])input.dev.Clone(); - } - - // core callback for input - void input_callback() - { - InputCallbacks.Call(); - IsLagFrame = false; - } - - private readonly InputCallbackSystem _inputCallbacks = new InputCallbackSystem(); - - public IInputCallbackSystem InputCallbacks { get { return _inputCallbacks; } } - - private readonly TraceBuffer _tracer = new TraceBuffer(); - - #endregion - - // TODO: use render and rendersound - public void FrameAdvance(bool render, bool rendersound = true) - { - if (Controller["Reset"]) - gpgx.gpgx_reset(false); - if (Controller["Power"]) - gpgx.gpgx_reset(true); - - // do we really have to get each time? nothing has changed - if (!gpgx.gpgx_get_control(input, inputsize)) - throw new Exception("gpgx_get_control() failed!"); - - ControlConverter.ScreenWidth = vwidth; - ControlConverter.ScreenHeight = vheight; - ControlConverter.Convert(Controller, input); - - if (!gpgx.gpgx_put_control(input, inputsize)) - throw new Exception("gpgx_put_control() failed!"); - - IsLagFrame = true; - Frame++; - drivelight = false; - - gpgx.gpgx_advance(); - update_video(); - update_audio(); - - if (IsLagFrame) - LagCount++; - - if (CD != null) - DriveLightOn = drivelight; - } - - public int Frame { get; private set; } - public int LagCount { get; set; } - public bool IsLagFrame { get; private set; } - - public string SystemId { get { return "GEN"; } } - public bool DeterministicEmulation { get { return true; } } - public string BoardName { get { return null; } } - - public CoreComm CoreComm { get; private set; } - - #region saveram - - byte[] DisposedSaveRam = null; - - public byte[] CloneSaveRam() - { - if (disposed) - { - if (DisposedSaveRam != null) - { - return (byte[])DisposedSaveRam.Clone(); - } - else - { - return new byte[0]; - } - } - else - { - int size = 0; - IntPtr area = IntPtr.Zero; - gpgx.gpgx_get_sram(ref area, ref size); - if (size <= 0 || area == IntPtr.Zero) - return new byte[0]; - gpgx.gpgx_sram_prepread(); - - byte[] ret = new byte[size]; - Marshal.Copy(area, ret, 0, size); - return ret; - } - } - - public void StoreSaveRam(byte[] data) - { - if (disposed) - { - throw new ObjectDisposedException(typeof(GPGX).ToString()); - } - else - { - int size = 0; - IntPtr area = IntPtr.Zero; - gpgx.gpgx_get_sram(ref area, ref size); - if (size <= 0 || area == IntPtr.Zero) - return; - if (size != data.Length) - throw new Exception("Unexpected saveram size"); - - Marshal.Copy(data, 0, area, size); - gpgx.gpgx_sram_commitwrite(); - } - } - - public bool SaveRamModified - { - get - { - if (disposed) - { - return DisposedSaveRam != null; - } - else - { - int size = 0; - IntPtr area = IntPtr.Zero; - gpgx.gpgx_get_sram(ref area, ref size); - return size > 0 && area != IntPtr.Zero; - } - } - } - - #endregion - - public void ResetCounters() - { - Frame = 0; - IsLagFrame = false; - LagCount = 0; - } - - #region savestates - - private byte[] savebuff; - private byte[] savebuff2; - - public void SaveStateText(System.IO.TextWriter writer) - { - var temp = SaveStateBinary(); - temp.SaveAsHexFast(writer); - // write extra copy of stuff we don't use - writer.WriteLine("Frame {0}", Frame); - } - - public void LoadStateText(System.IO.TextReader reader) - { - string hex = reader.ReadLine(); - byte[] state = new byte[hex.Length / 2]; - state.ReadFromHexFast(hex); - LoadStateBinary(new System.IO.BinaryReader(new System.IO.MemoryStream(state))); - } - - public void SaveStateBinary(System.IO.BinaryWriter writer) - { - if (!gpgx.gpgx_state_save(savebuff, savebuff.Length)) - throw new Exception("gpgx_state_save() returned false"); - - writer.Write(savebuff.Length); - writer.Write(savebuff); - // other variables - writer.Write(Frame); - writer.Write(LagCount); - writer.Write(IsLagFrame); - } - - public void LoadStateBinary(System.IO.BinaryReader reader) - { - int newlen = reader.ReadInt32(); - if (newlen != savebuff.Length) - throw new Exception("Unexpected state size"); - reader.Read(savebuff, 0, savebuff.Length); - if (!gpgx.gpgx_state_load(savebuff, savebuff.Length)) - throw new Exception("gpgx_state_load() returned false"); - // other variables - Frame = reader.ReadInt32(); - LagCount = reader.ReadInt32(); - IsLagFrame = reader.ReadBoolean(); - update_video(); - } - - public byte[] SaveStateBinary() - { - var ms = new System.IO.MemoryStream(savebuff2, true); - var bw = new System.IO.BinaryWriter(ms); - SaveStateBinary(bw); - bw.Flush(); - ms.Close(); - return savebuff2; - } - - public bool BinarySaveStatesPreferred { get { return true; } } - - #endregion - - #region debugging tools - - private IMemoryDomains MemoryDomains; - - unsafe void SetMemoryDomains() - { - var mm = new List(); - for (int i = LibGPGXDynamic.MIN_MEM_DOMAIN; i <= LibGPGXDynamic.MAX_MEM_DOMAIN; i++) - { - IntPtr area = IntPtr.Zero; - int size = 0; - IntPtr pname = gpgx.gpgx_get_memdom(i, ref area, ref size); - if (area == IntPtr.Zero || pname == IntPtr.Zero || size == 0) - continue; - string name = Marshal.PtrToStringAnsi(pname); - if (name == "VRAM") - { - // vram pokes need to go through hook which invalidates cached tiles - byte* p = (byte*)area; - mm.Add(new MemoryDomain(name, size, MemoryDomain.Endian.Unknown, - delegate(long addr) - { - if (addr < 0 || addr >= 65536) - throw new ArgumentOutOfRangeException(); - return p[addr ^ 1]; - }, - delegate(long addr, byte val) - { - if (addr < 0 || addr >= 65536) - throw new ArgumentOutOfRangeException(); - gpgx.gpgx_poke_vram(((int)addr) ^ 1, val); - }, - byteSize: 2)); - } - - else - { - var byteSize = name.Contains("Z80") ? 1 : 2; - mm.Add(MemoryDomain.FromIntPtrSwap16(name, size, MemoryDomain.Endian.Big, area, writable: true, byteSize: byteSize)); - } - } - MemoryDomains = new MemoryDomainList(mm); - (ServiceProvider as BasicServiceProvider).Register(MemoryDomains); - } - - public IDictionary GetCpuFlagsAndRegisters() - { - LibGPGXDynamic.RegisterInfo[] regs = new LibGPGXDynamic.RegisterInfo[gpgx.gpgx_getmaxnumregs()]; - - int n = gpgx.gpgx_getregs(regs); - if (n > regs.Length) - throw new InvalidOperationException("A buffer overrun has occured!"); - var ret = new Dictionary(); - for (int i = 0; i < n; i++) - { - // el hacko - string name = Marshal.PtrToStringAnsi(regs[i].Name); - byte size = 32; - if (name.Contains("68K SR") || name.StartsWith("Z80")) - size = 16; - ret[Marshal.PtrToStringAnsi(regs[i].Name)] = - new RegisterValue { BitSize = size, Value = (ulong)regs[i].Value }; - } - return ret; - } - - public bool CanStep(StepType type) { return false; } - - [FeatureNotImplemented] - public void Step(StepType type) { throw new NotImplementedException(); } - - [FeatureNotImplemented] - public void SetCpuRegister(string register, int value) - { - throw new NotImplementedException(); - } - - public void UpdateVDPViewContext(LibGPGXDynamic.VDPView view) - { - gpgx.gpgx_get_vdp_view(view); - gpgx.gpgx_flush_vram(); // fully regenerate internal caches as needed - } - - private readonly MemoryCallbackSystem _memoryCallbacks = new MemoryCallbackSystem(); - public IMemoryCallbackSystem MemoryCallbacks { get { return _memoryCallbacks; } } - - LibGPGXDynamic.mem_cb ExecCallback; - LibGPGXDynamic.mem_cb ReadCallback; - LibGPGXDynamic.mem_cb WriteCallback; - - void InitMemCallbacks() - { - ExecCallback = new LibGPGXDynamic.mem_cb(a => MemoryCallbacks.CallExecutes(a)); - ReadCallback = new LibGPGXDynamic.mem_cb(a => MemoryCallbacks.CallReads(a)); - WriteCallback = new LibGPGXDynamic.mem_cb(a => MemoryCallbacks.CallWrites(a)); - _memoryCallbacks.ActiveChanged += RefreshMemCallbacks; - } - - void RefreshMemCallbacks() - { - gpgx.gpgx_set_mem_callback( - MemoryCallbacks.HasReads ? ReadCallback : null, - MemoryCallbacks.HasWrites ? WriteCallback : null, - MemoryCallbacks.HasExecutes ? ExecCallback : null); - } - - void KillMemCallbacks() - { - gpgx.gpgx_set_mem_callback(null, null, null); - } - - #endregion - - public void Dispose() - { - if (!disposed) - { - // if (SaveRamModified) - // DisposedSaveRam = CloneSaveRam(); - // KillMemCallbacks(); - if (CD != null) - { - CD.Dispose(); - } - if (elf != null) - { - elf.Dispose(); - elf = null; - gpgx = null; - } - - disposed = true; - } - } - - #region SoundProvider - - short[] samples = new short[4096]; - int nsamp = 0; - - public ISoundProvider SoundProvider { get { return null; } } - public ISyncSoundProvider SyncSoundProvider { get { return this; } } - public bool StartAsyncSound() { return false; } - public void EndAsyncSound() { } - - public void GetSamples(out short[] samples, out int nsamp) - { - nsamp = this.nsamp; - samples = this.samples; - this.nsamp = 0; - } - - public void DiscardSamples() - { - this.nsamp = 0; - } - - void update_audio() - { - IntPtr src = IntPtr.Zero; - gpgx.gpgx_get_audio(ref nsamp, ref src); - if (src != IntPtr.Zero) - { - Marshal.Copy(src, samples, 0, nsamp * 2); - } - } - - #endregion - - #region VideoProvider - - public DisplayType DisplayType { get; private set; } - - int[] vidbuff = new int[0]; - int vwidth; - int vheight; - public int[] GetVideoBuffer() { return vidbuff; } - public int VirtualWidth { get { return BufferWidth; } } // TODO - public int VirtualHeight { get { return BufferHeight; } } // TODO - public int BufferWidth { get { return vwidth; } } - public int BufferHeight { get { return vheight; } } - public int BackgroundColor { get { return unchecked((int)0xff000000); } } - - void update_video_initial() - { - // hack: you should call update_video() here, but that gives you 256x192 on frame 0 - // and we know that we only use GPGX to emulate genesis games that will always be 320x224 immediately afterwards - - // so instead, just assume a 320x224 size now; if that happens to be wrong, it'll be fixed soon enough. - - vwidth = 320; - vheight = 224; - vidbuff = new int[vwidth * vheight]; - for (int i = 0; i < vidbuff.Length; i++) - vidbuff[i] = unchecked((int)0xff000000); - } - - unsafe void update_video() - { - int pitch = 0; - IntPtr src = IntPtr.Zero; - - gpgx.gpgx_get_video(ref vwidth, ref vheight, ref pitch, ref src); - - if (vidbuff.Length < vwidth * vheight) - vidbuff = new int[vwidth * vheight]; - - int rinc = (pitch / 4) - vwidth; - fixed (int* pdst_ = &vidbuff[0]) - { - int* pdst = pdst_; - int* psrc = (int*)src; - - for (int j = 0; j < vheight; j++) - { - for (int i = 0; i < vwidth; i++) - *pdst++ = *psrc++;// | unchecked((int)0xff000000); - psrc += rinc; - } - } - } - - #endregion - - #region Settings - - GPGXSyncSettings _SyncSettings; - GPGXSettings _Settings; - - public GPGXSettings GetSettings() { return _Settings.Clone(); } - public GPGXSyncSettings GetSyncSettings() { return _SyncSettings.Clone(); } - public bool PutSettings(GPGXSettings o) - { - _Settings = o; - gpgx.gpgx_set_draw_mask(_Settings.GetDrawMask()); - return false; - } - public bool PutSyncSettings(GPGXSyncSettings o) - { - bool ret = GPGXSyncSettings.NeedsReboot(_SyncSettings, o); - _SyncSettings = o; - return ret; - } - - public class GPGXSettings - { - [DisplayName("Background Layer A")] - [Description("True to draw BG layer A")] - [DefaultValue(true)] - public bool DrawBGA { get; set; } - - [DisplayName("Background Layer B")] - [Description("True to draw BG layer B")] - [DefaultValue(true)] - public bool DrawBGB { get; set; } - - [DisplayName("Background Layer W")] - [Description("True to draw BG layer W")] - [DefaultValue(true)] - public bool DrawBGW { get; set; } - - public GPGXSettings() - { - SettingsUtil.SetDefaultValues(this); - } - - public GPGXSettings Clone() - { - return (GPGXSettings)MemberwiseClone(); - } - - public LibGPGXDynamic.DrawMask GetDrawMask() - { - LibGPGXDynamic.DrawMask ret = 0; - if (DrawBGA) ret |= LibGPGXDynamic.DrawMask.BGA; - if (DrawBGB) ret |= LibGPGXDynamic.DrawMask.BGB; - if (DrawBGW) ret |= LibGPGXDynamic.DrawMask.BGW; - return ret; - } - } - - public class GPGXSyncSettings - { - [DisplayName("Use Six Button Controllers")] - [Description("Controls the type of any attached normal controllers; six button controllers are used if true, otherwise three button controllers. Some games don't work correctly with six button controllers. Not relevant if other controller types are connected.")] - [DefaultValue(true)] - public bool UseSixButton { get; set; } - - [DisplayName("Control Type")] - [Description("Sets the type of controls that are plugged into the console. Some games will automatically load with a different control type.")] - [DefaultValue(ControlType.Normal)] - public ControlType ControlType { get; set; } - - [DisplayName("Autodetect Region")] - [Description("Sets the region of the emulated console. Many games can run on multiple regions and will behave differently on different ones. Some games may require a particular region.")] - [DefaultValue(LibGPGXDynamic.Region.Autodetect)] - public LibGPGXDynamic.Region Region { get; set; } - - public GPGXSyncSettings() - { - SettingsUtil.SetDefaultValues(this); - } - - public GPGXSyncSettings Clone() - { - return (GPGXSyncSettings)MemberwiseClone(); - } - - public static bool NeedsReboot(GPGXSyncSettings x, GPGXSyncSettings y) - { - return !DeepEquality.DeepEquals(x, y); - } - } - - #endregion - } -} diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/LibGPGX.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/LibGPGX.cs index 6605d83722..fe1d8cc887 100644 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/LibGPGX.cs +++ b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/LibGPGX.cs @@ -3,22 +3,25 @@ using System.Collections.Generic; using System.Linq; using System.Text; using System.Runtime.InteropServices; +using BizHawk.Emulation.Common.BizInvoke; namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx { - public static class LibGPGX + public abstract class LibGPGX { - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_get_video(out int w, out int h, out int pitch, ref IntPtr buffer); + public const string DllName = "libgenplusgx.dll"; - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_get_audio(ref int n, ref IntPtr buffer); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_get_video(out int w, out int h, out int pitch, ref IntPtr buffer); + + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_get_audio(ref int n, ref IntPtr buffer); [UnmanagedFunctionPointer(CallingConvention.Cdecl)] public delegate int load_archive_cb(string filename, IntPtr buffer, int maxsize); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_advance(); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_advance(); public enum Region : int { @@ -48,50 +51,50 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx public uint BackdropColor; } - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern bool gpgx_init(string feromextension, load_archive_cb feload_archive_cb, bool sixbutton, INPUT_SYSTEM system_a, INPUT_SYSTEM system_b, Region region, [In]InitSettings settings); + [BizImport(CallingConvention.Cdecl, Compatibility=true)] + public abstract bool gpgx_init(string feromextension, load_archive_cb feload_archive_cb, bool sixbutton, INPUT_SYSTEM system_a, INPUT_SYSTEM system_b, Region region, [In]InitSettings settings); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_get_fps(ref int num, ref int den); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_get_fps(ref int num, ref int den); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern int gpgx_state_max_size(); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern int gpgx_state_size(byte[] dest, int size); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern bool gpgx_state_save(byte[] dest, int size); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern bool gpgx_state_load(byte[] src, int size); + [BizImport(CallingConvention.Cdecl)] + public abstract int gpgx_state_max_size(); + [BizImport(CallingConvention.Cdecl)] + public abstract int gpgx_state_size(byte[] dest, int size); + [BizImport(CallingConvention.Cdecl)] + public abstract bool gpgx_state_save(byte[] dest, int size); + [BizImport(CallingConvention.Cdecl)] + public abstract bool gpgx_state_load(byte[] src, int size); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern bool gpgx_get_control([Out]InputData dest, int bytes); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern bool gpgx_put_control([In]InputData src, int bytes); + [BizImport(CallingConvention.Cdecl, Compatibility = true)] + public abstract bool gpgx_get_control([Out]InputData dest, int bytes); + [BizImport(CallingConvention.Cdecl, Compatibility = true)] + public abstract bool gpgx_put_control([In]InputData src, int bytes); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_get_sram(ref IntPtr area, ref int size); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_get_sram(ref IntPtr area, ref int size); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_clear_sram(); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_clear_sram(); public const int MIN_MEM_DOMAIN = 0; public const int MAX_MEM_DOMAIN = 13; - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] + [BizImport(CallingConvention.Cdecl)] // apparently, if you use built in string marshalling, the interop will assume that // the unmanaged char pointer was allocated in hglobal and try to free it that way - public static extern IntPtr gpgx_get_memdom(int which, ref IntPtr area, ref int size); + public abstract IntPtr gpgx_get_memdom(int which, ref IntPtr area, ref int size); // call this before reading sram returned by gpgx_get_sram() - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_sram_prepread(); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_sram_prepread(); // call this after writing sram returned by gpgx_get_sram() - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_sram_commitwrite(); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_sram_commitwrite(); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_reset(bool hard); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_reset(bool hard); public const int MAX_DEVICES = 8; @@ -149,8 +152,8 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx [UnmanagedFunctionPointer(CallingConvention.Cdecl)] public delegate void input_cb(); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_set_input_callback(input_cb cb); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_set_input_callback(input_cb cb); [UnmanagedFunctionPointer(CallingConvention.Cdecl)] public delegate void mem_cb(uint addr); @@ -158,11 +161,11 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx [UnmanagedFunctionPointer(CallingConvention.Cdecl)] public delegate void CDCallback(int addr, CDLog_AddrType addrtype, CDLog_Flags flags); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_set_mem_callback(mem_cb read, mem_cb write, mem_cb exec); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_set_mem_callback(mem_cb read, mem_cb write, mem_cb exec); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_set_cd_callback(CDCallback cd); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_set_cd_callback(CDCallback cd); @@ -307,14 +310,14 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx public VDPNameTable NTW; } - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_get_vdp_view([Out] VDPView view); + [BizImport(CallingConvention.Cdecl, Compatibility = true)] + public abstract void gpgx_get_vdp_view([Out] VDPView view); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_poke_vram(int addr, byte value); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_poke_vram(int addr, byte value); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_flush_vram(); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_flush_vram(); [StructLayout(LayoutKind.Sequential)] public struct RegisterInfo @@ -323,11 +326,11 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx public IntPtr Name; } - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern int gpgx_getmaxnumregs(); + [BizImport(CallingConvention.Cdecl)] + public abstract int gpgx_getmaxnumregs(); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern int gpgx_getregs([Out] RegisterInfo[] regs); + [BizImport(CallingConvention.Cdecl, Compatibility = true)] + public abstract int gpgx_getregs([Out] RegisterInfo[] regs); [Flags] public enum DrawMask : int @@ -339,17 +342,16 @@ namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx Backdrop = 16 } - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_set_draw_mask(DrawMask mask); - - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_write_m68k_bus(uint addr, byte data); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern void gpgx_write_s68k_bus(uint addr, byte data); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern byte gpgx_peek_m68k_bus(uint addr); - [DllImport("libgenplusgx.dll", CallingConvention = CallingConvention.Cdecl)] - public static extern byte gpgx_peek_s68k_bus(uint addr); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_set_draw_mask(DrawMask mask); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_write_m68k_bus(uint addr, byte data); + [BizImport(CallingConvention.Cdecl)] + public abstract void gpgx_write_s68k_bus(uint addr, byte data); + [BizImport(CallingConvention.Cdecl)] + public abstract byte gpgx_peek_m68k_bus(uint addr); + [BizImport(CallingConvention.Cdecl)] + public abstract byte gpgx_peek_s68k_bus(uint addr); } } diff --git a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/LibGPGXDynamic.cs b/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/LibGPGXDynamic.cs deleted file mode 100644 index 7402bde832..0000000000 --- a/BizHawk.Emulation.Cores/Consoles/Sega/gpgx/LibGPGXDynamic.cs +++ /dev/null @@ -1,323 +0,0 @@ -using System; -using System.Collections.Generic; -using System.Linq; -using System.Text; -using System.Runtime.InteropServices; - -namespace BizHawk.Emulation.Cores.Consoles.Sega.gpgx -{ - public class LibGPGXDynamic - { - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_get_video_t(ref int w, ref int h, ref int pitch, ref IntPtr buffer); - public gpgx_get_video_t gpgx_get_video; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_get_audio_t(ref int n, ref IntPtr buffer); - public gpgx_get_audio_t gpgx_get_audio; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate int load_archive_cb(string filename, IntPtr buffer, int maxsize); - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_advance_t(); - public gpgx_advance_t gpgx_advance; - - public enum Region : int - { - Autodetect = 0, - USA = 1, - Europe = 2, - Japan_NTSC = 3, - Japan_PAL = 4 - } - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate bool gpgx_init_t(string feromextension, load_archive_cb feload_archive_cb, bool sixbutton, INPUT_SYSTEM system_a, INPUT_SYSTEM system_b, Region region); - public gpgx_init_t gpgx_init; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_get_fps_t(ref int num, ref int den); - public gpgx_get_fps_t gpgx_get_fps; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate int gpgx_state_max_size_t(); - public gpgx_state_max_size_t gpgx_state_max_size; - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate int gpgx_state_size_t(byte[] dest, int size); - public gpgx_state_size_t gpgx_state_size; - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate bool gpgx_state_save_t(byte[] dest, int size); - public gpgx_state_save_t gpgx_state_save; - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate bool gpgx_state_load_t(byte[] src, int size); - public gpgx_state_load_t gpgx_state_load; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate bool gpgx_get_control_t([Out]InputData dest, int bytes); - public gpgx_get_control_t gpgx_get_control; - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate bool gpgx_put_control_t([In]InputData src, int bytes); - public gpgx_put_control_t gpgx_put_control; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_get_sram_t(ref IntPtr area, ref int size); - public gpgx_get_sram_t gpgx_get_sram; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_clear_sram_t(); - public gpgx_clear_sram_t gpgx_clear_sram; - - public const int MIN_MEM_DOMAIN = 0; - public const int MAX_MEM_DOMAIN = 13; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - // apparently, if you use built in string marshalling, the interop will assume that - // the unmanaged char pointer was allocated in hglobal and try to free it that way - public delegate IntPtr gpgx_get_memdom_t(int which, ref IntPtr area, ref int size); - public gpgx_get_memdom_t gpgx_get_memdom; - - // call this before reading sram returned by gpgx_get_sram() - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_sram_prepread_t(); - public gpgx_sram_prepread_t gpgx_sram_prepread; - - // call this after writing sram returned by gpgx_get_sram() - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_sram_commitwrite_t(); - public gpgx_sram_commitwrite_t gpgx_sram_commitwrite; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_reset_t(bool hard); - public gpgx_reset_t gpgx_reset; - - public const int MAX_DEVICES = 8; - - public enum INPUT_SYSTEM : byte - { - SYSTEM_NONE = 0, // unconnected port - SYSTEM_MD_GAMEPAD = 1, // single 3-buttons or 6-buttons Control Pad - SYSTEM_MOUSE = 2, // Sega Mouse - SYSTEM_MENACER = 3, // Sega Menacer -- port B only - SYSTEM_JUSTIFIER = 4, // Konami Justifiers -- port B only - SYSTEM_XE_A1P = 5, // XE-A1P analog controller -- port A only - SYSTEM_ACTIVATOR = 6, // Sega Activator - SYSTEM_MS_GAMEPAD = 7, // single 2-buttons Control Pad -- Master System - SYSTEM_LIGHTPHASER = 8, // Sega Light Phaser -- Master System - SYSTEM_PADDLE = 9, // Sega Paddle Control -- Master System - SYSTEM_SPORTSPAD = 10, // Sega Sports Pad -- Master System - SYSTEM_TEAMPLAYER = 11, // Multi Tap -- Sega TeamPlayer - SYSTEM_WAYPLAY = 12, // Multi Tap -- EA 4-Way Play -- use both ports - }; - - public enum INPUT_DEVICE : byte - { - DEVICE_NONE = 0xff, // unconnected device = fixed ID for Team Player) - DEVICE_PAD3B = 0x00, // 3-buttons Control Pad = fixed ID for Team Player) - DEVICE_PAD6B = 0x01, // 6-buttons Control Pad = fixed ID for Team Player) - DEVICE_PAD2B = 0x02, // 2-buttons Control Pad - DEVICE_MOUSE = 0x03, // Sega Mouse - DEVICE_LIGHTGUN = 0x04, // Sega Light Phaser, Menacer or Konami Justifiers - DEVICE_PADDLE = 0x05, // Sega Paddle Control - DEVICE_SPORTSPAD = 0x06,// Sega Sports Pad - DEVICE_PICO = 0x07, // PICO tablet - DEVICE_TEREBI = 0x08, // Terebi Oekaki tablet - DEVICE_XE_A1P = 0x09, // XE-A1P analog controller - DEVICE_ACTIVATOR = 0x0a,// Activator - }; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void input_cb(); - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_set_input_callback_t(input_cb cb); - public gpgx_set_input_callback_t gpgx_set_input_callback; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void mem_cb(uint addr); - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_set_mem_callback_t(mem_cb read, mem_cb write, mem_cb exec); - public gpgx_set_mem_callback_t gpgx_set_mem_callback; - - /// - /// not every flag is valid for every device! - /// - [Flags] - public enum INPUT_KEYS : ushort - { - /* Default Input bitmasks */ - INPUT_MODE = 0x0800, - INPUT_X = 0x0400, - INPUT_Y = 0x0200, - INPUT_Z = 0x0100, - INPUT_START = 0x0080, - INPUT_A = 0x0040, - INPUT_C = 0x0020, - INPUT_B = 0x0010, - INPUT_RIGHT = 0x0008, - INPUT_LEFT = 0x0004, - INPUT_DOWN = 0x0002, - INPUT_UP = 0x0001, - - /* Master System specific bitmasks */ - INPUT_BUTTON2 = 0x0020, - INPUT_BUTTON1 = 0x0010, - - /* Mega Mouse specific bitmask */ - INPUT_MOUSE_START = 0x0080, - INPUT_MOUSE_CENTER = 0x0040, - INPUT_MOUSE_RIGHT = 0x0020, - INPUT_MOUSE_LEFT = 0x0010, - - /* Pico hardware specific bitmask */ - INPUT_PICO_PEN = 0x0080, - INPUT_PICO_RED = 0x0010, - - /* XE-1AP specific bitmask */ - INPUT_XE_E1 = 0x0800, - INPUT_XE_E2 = 0x0400, - INPUT_XE_START = 0x0200, - INPUT_XE_SELECT = 0x0100, - INPUT_XE_A = 0x0080, - INPUT_XE_B = 0x0040, - INPUT_XE_C = 0x0020, - INPUT_XE_D = 0x0010, - - /* Activator specific bitmasks */ - INPUT_ACTIVATOR_8U = 0x8000, - INPUT_ACTIVATOR_8L = 0x4000, - INPUT_ACTIVATOR_7U = 0x2000, - INPUT_ACTIVATOR_7L = 0x1000, - INPUT_ACTIVATOR_6U = 0x0800, - INPUT_ACTIVATOR_6L = 0x0400, - INPUT_ACTIVATOR_5U = 0x0200, - INPUT_ACTIVATOR_5L = 0x0100, - INPUT_ACTIVATOR_4U = 0x0080, - INPUT_ACTIVATOR_4L = 0x0040, - INPUT_ACTIVATOR_3U = 0x0020, - INPUT_ACTIVATOR_3L = 0x0010, - INPUT_ACTIVATOR_2U = 0x0008, - INPUT_ACTIVATOR_2L = 0x0004, - INPUT_ACTIVATOR_1U = 0x0002, - INPUT_ACTIVATOR_1L = 0x0001, - - /* Menacer */ - INPUT_MENACER_TRIGGER = 0x0040, - INPUT_MENACER_START = 0x0080, - }; - - [StructLayout(LayoutKind.Sequential)] - public class InputData - { - [MarshalAs(UnmanagedType.ByValArray, SizeConst = 2)] - public readonly INPUT_SYSTEM[] system = new INPUT_SYSTEM[2]; - [MarshalAs(UnmanagedType.ByValArray, SizeConst = MAX_DEVICES)] - public readonly INPUT_DEVICE[] dev = new INPUT_DEVICE[MAX_DEVICES]; - /// - /// digital inputs - /// - [MarshalAs(UnmanagedType.ByValArray, SizeConst = MAX_DEVICES)] - public readonly INPUT_KEYS[] pad = new INPUT_KEYS[MAX_DEVICES]; - /// - /// analog (x/y) - /// - [MarshalAs(UnmanagedType.ByValArray, SizeConst = MAX_DEVICES * 2)] - public readonly short[] analog = new short[MAX_DEVICES * 2]; - /// - /// gun horizontal offset - /// - public int x_offset; - /// - /// gun vertical offset - /// - public int y_offset; - - public void ClearAllBools() - { - for (int i = 0; i < pad.Length; i++) - pad[i] = 0; - } - } - - public const int CD_MAX_TRACKS = 100; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void cd_read_cb(int lba, IntPtr dest, bool audio); - - [StructLayout(LayoutKind.Sequential)] - public struct CDTrack - { - public int start; - public int end; - } - - [StructLayout(LayoutKind.Sequential)] - public class CDData - { - public int end; - public int last; - [MarshalAs(UnmanagedType.ByValArray, SizeConst = CD_MAX_TRACKS)] - public readonly CDTrack[] tracks = new CDTrack[CD_MAX_TRACKS]; - public cd_read_cb readcallback; - } - - [StructLayout(LayoutKind.Sequential)] - public struct VDPNameTable - { - public int Width; // in cells - public int Height; // in cells - public int Baseaddr; - } - - [StructLayout(LayoutKind.Sequential)] - public class VDPView - { - public IntPtr VRAM; - public IntPtr PatternCache; - public IntPtr ColorCache; - public VDPNameTable NTA; - public VDPNameTable NTB; - public VDPNameTable NTW; - } - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_get_vdp_view_t([Out] VDPView view); - public gpgx_get_vdp_view_t gpgx_get_vdp_view; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_poke_vram_t(int addr, byte value); - public gpgx_poke_vram_t gpgx_poke_vram; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_flush_vram_t(); - public gpgx_flush_vram_t gpgx_flush_vram; - - [StructLayout(LayoutKind.Sequential)] - public struct RegisterInfo - { - public int Value; - public IntPtr Name; - } - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate int gpgx_getmaxnumregs_t(); - public gpgx_getmaxnumregs_t gpgx_getmaxnumregs; - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate int gpgx_getregs_t([Out] RegisterInfo[] regs); - public gpgx_getregs_t gpgx_getregs; - - [Flags] - public enum DrawMask : int - { - BGA = 1, - BGB = 2, - BGW = 4 - } - - [UnmanagedFunctionPointer(CallingConvention.Cdecl)] - public delegate void gpgx_set_draw_mask_t(DrawMask mask); - public gpgx_set_draw_mask_t gpgx_set_draw_mask; - } -} diff --git a/BizHawk.Emulation.Cores/ElfRunner.cs b/BizHawk.Emulation.Cores/ElfRunner.cs index b279ec5752..e443eaa91d 100644 --- a/BizHawk.Emulation.Cores/ElfRunner.cs +++ b/BizHawk.Emulation.Cores/ElfRunner.cs @@ -30,7 +30,7 @@ namespace BizHawk.Emulation.Cores long orig_start = loadsegs.Min(s => s.Address); orig_start &= ~(Environment.SystemPageSize - 1); long orig_end = loadsegs.Max(s => s.Address + s.Size); - _base = new MemoryBlock(UIntPtr.Zero, orig_end - orig_start); + _base = new MemoryBlock(0, (ulong)(orig_end - orig_start)); _loadoffset = (long)_base.Start - orig_start; try @@ -40,7 +40,7 @@ namespace BizHawk.Emulation.Cores foreach (var seg in loadsegs) { var data = seg.GetContents(); - Marshal.Copy(data, 0, (IntPtr)(seg.Address + _loadoffset), data.Length); + Marshal.Copy(data, 0, Z.SS(seg.Address + _loadoffset), data.Length); } RegisterSymbols(); ProcessRelocations(); @@ -51,14 +51,14 @@ namespace BizHawk.Emulation.Cores foreach (var sec in _elf.Sections.Where(s => s.Flags.HasFlag(SectionFlags.Allocatable))) { if (sec.Flags.HasFlag(SectionFlags.Executable)) - _base.Set((UIntPtr)(sec.LoadAddress + _loadoffset), sec.Size, MemoryBlock.Protection.RX); + _base.Set((ulong)(sec.LoadAddress + _loadoffset), (ulong)sec.Size, MemoryBlock.Protection.RX); else if (sec.Flags.HasFlag(SectionFlags.Writable)) - _base.Set((UIntPtr)(sec.LoadAddress + _loadoffset), sec.Size, MemoryBlock.Protection.RW); + _base.Set((ulong)(sec.LoadAddress + _loadoffset), (ulong)sec.Size, MemoryBlock.Protection.RW); } //FixupGOT(); ConnectAllClibPatches(); - Console.WriteLine("Loaded {0}@{1:X16}", filename, (long)_base.Start); + Console.WriteLine("Loaded {0}@{1:X16}", filename, _base.Start); } catch { @@ -200,6 +200,7 @@ namespace BizHawk.Emulation.Cores #region clib monkeypatches + // our clib expects a few function pointers to be defined for it /// @@ -222,7 +223,7 @@ namespace BizHawk.Emulation.Cores /// desired exit code [UnmanagedFunctionPointer(CallingConvention.Cdecl)] private delegate void Exit_D(int status); - + /* [CLibPatch("_ZZFree")] private void FreeMem(IntPtr p, IntPtr size) { @@ -242,7 +243,7 @@ namespace BizHawk.Emulation.Cores block.Set(block.Start, block.Size, MemoryBlock.Protection.RW); Console.WriteLine("AllocMem: {0:X8}@{1:X16}", (long)size, (long)block.Start); return p; - } + }*/ [CLibPatch("_ZZExit")] private void Exit(int status) { diff --git a/BizHawk.Emulation.Cores/MemoryBlock.cs b/BizHawk.Emulation.Cores/MemoryBlock.cs index efe4545b2d..a02428de4f 100644 --- a/BizHawk.Emulation.Cores/MemoryBlock.cs +++ b/BizHawk.Emulation.Cores/MemoryBlock.cs @@ -6,41 +6,94 @@ using System.Runtime.InteropServices; namespace BizHawk.Emulation.Cores { + public static class Z + { + public static IntPtr US(ulong l) + { + if (IntPtr.Size == 8) + return (IntPtr)(long)l; + else + return (IntPtr)(int)l; + } + + public static UIntPtr UU(ulong l) + { + if (UIntPtr.Size == 8) + return (UIntPtr)l; + else + return (UIntPtr)(uint)l; + } + + public static IntPtr SS(long l) + { + if (IntPtr.Size == 8) + return (IntPtr)l; + else + return (IntPtr)(int)l; + } + + public static UIntPtr SU(long l) + { + if (UIntPtr.Size == 8) + return (UIntPtr)(ulong)l; + else + return (UIntPtr)(uint)l; + } + } + public sealed class MemoryBlock : IDisposable { - public UIntPtr Start { get; private set; } - public long Size { get; private set; } - public UIntPtr End { get; private set; } + /// + /// starting address of the memory block + /// + public ulong Start { get; private set; } + /// + /// total size of the memory block + /// + public ulong Size { get; private set; } + /// + /// ending address of the memory block; equal to start + size + /// + public ulong End { get; private set; } public int PageSize { get { return Environment.SystemPageSize; } } - public MemoryBlock(long size) - : this(UIntPtr.Zero, size) + /// + /// allocate size bytes at any address + /// + /// + public MemoryBlock(ulong size) + : this(0, size) { } - public MemoryBlock(UIntPtr start, long size) + /// + /// allocate size bytes starting at a particular address + /// + /// + /// + public MemoryBlock(ulong start, ulong size) { #if !MONO - Start = Kernel32.VirtualAlloc(start, checked((UIntPtr)size), + Start = (ulong)Kernel32.VirtualAlloc(Z.UU(start), Z.UU(size), Kernel32.AllocationType.RESERVE | Kernel32.AllocationType.COMMIT, Kernel32.MemoryProtection.NOACCESS); - if (Start == UIntPtr.Zero) + if (Start == 0) { throw new InvalidOperationException("VirtualAlloc() returned NULL"); } - if (start != UIntPtr.Zero) - End = (UIntPtr)((long)start + size); + if (start != 0) + End = start + size; else - End = (UIntPtr)((long)Start + size); - Size = (long)End - (long)Start; + End = Start + size; + Size = End - Start; #else - Start = LibC.mmap(start, checked((UIntPtr)size), 0, LibC.MapType.MAP_ANONYMOUS, -1, IntPtr.Zero); - if (Start == UIntPtr.Zero) + Start = (ulong)LibC.mmap(ZC.U(start), ZC.U(size), 0, LibC.MapType.MAP_ANONYMOUS, -1, IntPtr.Zero); + if (Start == 0) { throw new InvalidOperationException("mmap() returned NULL"); } - End = (UIntPtr)((long)Start + size); - Size = (long)End - (long)Start; + End = Start + size; + Size = End - Start; #endif } @@ -49,12 +102,12 @@ namespace BizHawk.Emulation.Cores R, RW, RX, None } - public void Set(UIntPtr start, long length, Protection prot) + public void Set(ulong start, ulong length, Protection prot) { - if ((ulong)start < (ulong)Start) + if (start < Start) throw new ArgumentOutOfRangeException("start"); - if ((ulong)start + (ulong)length > (ulong)End) + if (start + length > End) throw new ArgumentOutOfRangeException("length"); #if !MONO @@ -68,7 +121,7 @@ namespace BizHawk.Emulation.Cores default: throw new ArgumentOutOfRangeException("prot"); } Kernel32.MemoryProtection old; - if (!Kernel32.VirtualProtect(start, (UIntPtr)length, p, out old)) + if (!Kernel32.VirtualProtect(Z.UU(start), Z.UU(length), p, out old)) throw new InvalidOperationException("VirtualProtect() returned FALSE!"); #else LibC.ProtType p; @@ -95,14 +148,14 @@ namespace BizHawk.Emulation.Cores private void Dispose(bool disposing) { - if (Start != UIntPtr.Zero) + if (Start != 0) { #if !MONO - Kernel32.VirtualFree(Start, UIntPtr.Zero, Kernel32.FreeType.RELEASE); + Kernel32.VirtualFree(Z.UU(Start), UIntPtr.Zero, Kernel32.FreeType.RELEASE); #else - LibC.munmap(Start, (UIntPtr)Size); + LibC.munmap(ZC.U(Start), (UIntPtr)Size); #endif - Start = UIntPtr.Zero; + Start = 0; } }