From d053a0b4148a583100f942d8ba72778b3838973d Mon Sep 17 00:00:00 2001
From: CasualPokePlayer <50538166+CasualPokePlayer@users.noreply.github.com>
Date: Fri, 19 Nov 2021 19:59:27 -0800
Subject: [PATCH] Refactor GambatteLink (#3002)
* refactor gambattelink to be more modular and add support for 3x/4x
---
.../config/GB/DGBPrefs.Designer.cs | 58 +++-
.../config/GB/DGBPrefs.cs | 26 +-
.../Base Implementations/LinkedDebuggable.cs | 57 ++++
.../LinkedMemoryDomains.cs | 60 ++++
.../Base Implementations/LinkedSaveRam.cs | 66 +++++
.../Nintendo/Gameboy/Gambatte.ICodeDataLog.cs | 18 +-
.../Nintendo/Gameboy/Gambatte.IDebuggable.cs | 15 +-
.../Gameboy/GambatteLink.ICodeDataLog.cs | 25 +-
.../Gameboy/GambatteLink.IDebuggable.cs | 46 ---
.../Gameboy/GambatteLink.IEmulator.cs | 273 ++++++++++++------
.../Gameboy/GambatteLink.IMemoryDomains.cs | 56 ----
.../Nintendo/Gameboy/GambatteLink.ISaveRam.cs | 31 --
.../Gameboy/GambatteLink.ISettable.cs | 58 ++--
.../Gameboy/GambatteLink.ISoundProvider.cs | 263 ++++++++++-------
.../Gameboy/GambatteLink.IStatable.cs | 129 ++++++---
.../Gameboy/GambatteLink.IVideoProvider.cs | 22 +-
.../Consoles/Nintendo/Gameboy/GambatteLink.cs | 138 ++++++---
17 files changed, 851 insertions(+), 490 deletions(-)
create mode 100644 src/BizHawk.Emulation.Common/Base Implementations/LinkedDebuggable.cs
create mode 100644 src/BizHawk.Emulation.Common/Base Implementations/LinkedMemoryDomains.cs
create mode 100644 src/BizHawk.Emulation.Common/Base Implementations/LinkedSaveRam.cs
delete mode 100644 src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IDebuggable.cs
delete mode 100644 src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IMemoryDomains.cs
delete mode 100644 src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISaveRam.cs
diff --git a/src/BizHawk.Client.EmuHawk/config/GB/DGBPrefs.Designer.cs b/src/BizHawk.Client.EmuHawk/config/GB/DGBPrefs.Designer.cs
index 0dc2c0218a..b9dcce34ed 100644
--- a/src/BizHawk.Client.EmuHawk/config/GB/DGBPrefs.Designer.cs
+++ b/src/BizHawk.Client.EmuHawk/config/GB/DGBPrefs.Designer.cs
@@ -33,11 +33,17 @@
this.gbPrefControl1 = new GBPrefControl();
this.tabPage2 = new System.Windows.Forms.TabPage();
this.gbPrefControl2 = new GBPrefControl();
+ this.tabPage3 = new System.Windows.Forms.TabPage();
+ this.gbPrefControl3 = new GBPrefControl();
+ this.tabPage4 = new System.Windows.Forms.TabPage();
+ this.gbPrefControl4 = new GBPrefControl();
this.buttonCancel = new System.Windows.Forms.Button();
this.buttonOK = new System.Windows.Forms.Button();
this.tabControl1.SuspendLayout();
this.tabPage1.SuspendLayout();
this.tabPage2.SuspendLayout();
+ this.tabPage3.SuspendLayout();
+ this.tabPage4.SuspendLayout();
this.SuspendLayout();
//
// tabControl1
@@ -47,6 +53,8 @@
| System.Windows.Forms.AnchorStyles.Right)));
this.tabControl1.Controls.Add(this.tabPage1);
this.tabControl1.Controls.Add(this.tabPage2);
+ this.tabControl1.Controls.Add(this.tabPage3);
+ this.tabControl1.Controls.Add(this.tabPage4);
this.tabControl1.Location = new System.Drawing.Point(12, 12);
this.tabControl1.Name = "tabControl1";
this.tabControl1.SelectedIndex = 0;
@@ -61,7 +69,7 @@
this.tabPage1.Padding = new System.Windows.Forms.Padding(3);
this.tabPage1.Size = new System.Drawing.Size(509, 308);
this.tabPage1.TabIndex = 0;
- this.tabPage1.Text = "Left Gameboy";
+ this.tabPage1.Text = "Player 1 Gameboy";
this.tabPage1.UseVisualStyleBackColor = true;
//
// gbPrefControl1
@@ -81,7 +89,7 @@
this.tabPage2.Padding = new System.Windows.Forms.Padding(3);
this.tabPage2.Size = new System.Drawing.Size(509, 308);
this.tabPage2.TabIndex = 1;
- this.tabPage2.Text = "Right Gameboy";
+ this.tabPage2.Text = "Player 2 Gameboy";
this.tabPage2.UseVisualStyleBackColor = true;
//
// gbPrefControl2
@@ -93,6 +101,46 @@
this.gbPrefControl2.Size = new System.Drawing.Size(503, 302);
this.gbPrefControl2.TabIndex = 0;
//
+ // tabPage3
+ //
+ this.tabPage3.Controls.Add(this.gbPrefControl3);
+ this.tabPage3.Location = new System.Drawing.Point(4, 22);
+ this.tabPage3.Name = "tabPage3";
+ this.tabPage3.Padding = new System.Windows.Forms.Padding(3);
+ this.tabPage3.Size = new System.Drawing.Size(509, 308);
+ this.tabPage3.TabIndex = 2;
+ this.tabPage3.Text = "Player 3 Gameboy";
+ this.tabPage3.UseVisualStyleBackColor = true;
+ //
+ // gbPrefControl3
+ //
+ this.gbPrefControl3.ColorGameBoy = false;
+ this.gbPrefControl3.Dock = System.Windows.Forms.DockStyle.Fill;
+ this.gbPrefControl3.Location = new System.Drawing.Point(3, 3);
+ this.gbPrefControl3.Name = "gbPrefControl3";
+ this.gbPrefControl3.Size = new System.Drawing.Size(503, 302);
+ this.gbPrefControl3.TabIndex = 0;
+ //
+ // tabPage4
+ //
+ this.tabPage4.Controls.Add(this.gbPrefControl4);
+ this.tabPage4.Location = new System.Drawing.Point(4, 22);
+ this.tabPage4.Name = "tabPage4";
+ this.tabPage4.Padding = new System.Windows.Forms.Padding(3);
+ this.tabPage4.Size = new System.Drawing.Size(509, 308);
+ this.tabPage4.TabIndex = 3;
+ this.tabPage4.Text = "Player 4 Gameboy";
+ this.tabPage4.UseVisualStyleBackColor = true;
+ //
+ // gbPrefControl4
+ //
+ this.gbPrefControl4.ColorGameBoy = false;
+ this.gbPrefControl4.Dock = System.Windows.Forms.DockStyle.Fill;
+ this.gbPrefControl4.Location = new System.Drawing.Point(3, 3);
+ this.gbPrefControl4.Name = "gbPrefControl4";
+ this.gbPrefControl4.Size = new System.Drawing.Size(503, 302);
+ this.gbPrefControl4.TabIndex = 0;
+ //
// buttonCancel
//
this.buttonCancel.Anchor = ((System.Windows.Forms.AnchorStyles)((System.Windows.Forms.AnchorStyles.Bottom | System.Windows.Forms.AnchorStyles.Right)));
@@ -130,6 +178,8 @@
this.tabControl1.ResumeLayout(false);
this.tabPage1.ResumeLayout(false);
this.tabPage2.ResumeLayout(false);
+ this.tabPage3.ResumeLayout(false);
+ this.tabPage4.ResumeLayout(false);
this.ResumeLayout(false);
}
@@ -139,10 +189,14 @@
private System.Windows.Forms.TabControl tabControl1;
private System.Windows.Forms.TabPage tabPage1;
private System.Windows.Forms.TabPage tabPage2;
+ private System.Windows.Forms.TabPage tabPage3;
+ private System.Windows.Forms.TabPage tabPage4;
private System.Windows.Forms.Button buttonCancel;
private System.Windows.Forms.Button buttonOK;
private GBPrefControl gbPrefControl1;
private GBPrefControl gbPrefControl2;
+ private GBPrefControl gbPrefControl3;
+ private GBPrefControl gbPrefControl4;
}
}
\ No newline at end of file
diff --git a/src/BizHawk.Client.EmuHawk/config/GB/DGBPrefs.cs b/src/BizHawk.Client.EmuHawk/config/GB/DGBPrefs.cs
index 968ea24e75..9d4375964d 100644
--- a/src/BizHawk.Client.EmuHawk/config/GB/DGBPrefs.cs
+++ b/src/BizHawk.Client.EmuHawk/config/GB/DGBPrefs.cs
@@ -22,25 +22,31 @@ namespace BizHawk.Client.EmuHawk
InitializeComponent();
gbPrefControl1.DialogParent = this;
gbPrefControl2.DialogParent = this;
+ gbPrefControl3.DialogParent = this;
+ gbPrefControl4.DialogParent = this;
Icon = Properties.Resources.DualIcon;
}
private void PutSettings(GambatteLink.GambatteLinkSettings s, GambatteLink.GambatteLinkSyncSettings ss)
{
- gbPrefControl1.PutSettings(_config, _game, _movieSession, s.L, ss.L);
- gbPrefControl2.PutSettings(_config, _game, _movieSession, s.R, ss.R);
+ gbPrefControl1.PutSettings(_config, _game, _movieSession, s._linkedSettings[0], ss._linkedSyncSettings[0]);
+ gbPrefControl2.PutSettings(_config, _game, _movieSession, s._linkedSettings[1], ss._linkedSyncSettings[1]);
+ gbPrefControl3.PutSettings(_config, _game, _movieSession, s._linkedSettings[2], ss._linkedSyncSettings[2]);
+ gbPrefControl4.PutSettings(_config, _game, _movieSession, s._linkedSettings[3], ss._linkedSyncSettings[3]);
}
private void GetSettings(out GambatteLink.GambatteLinkSettings s, out GambatteLink.GambatteLinkSyncSettings ss)
{
- gbPrefControl1.GetSettings(out var sl, out var ssl);
- gbPrefControl2.GetSettings(out var sr, out var ssr);
+ gbPrefControl1.GetSettings(out var s1, out var ss1);
+ gbPrefControl2.GetSettings(out var s2, out var ss2);
+ gbPrefControl3.GetSettings(out var s3, out var ss3);
+ gbPrefControl4.GetSettings(out var s4, out var ss4);
- s = new GambatteLink.GambatteLinkSettings(sl, sr);
- ss = new GambatteLink.GambatteLinkSyncSettings(ssl, ssr);
+ s = new GambatteLink.GambatteLinkSettings(s1, s2, s3, s4);
+ ss = new GambatteLink.GambatteLinkSyncSettings(ss1, ss2, ss3, ss4);
}
- private bool SyncSettingsChanged => gbPrefControl1.SyncSettingsChanged || gbPrefControl2.SyncSettingsChanged;
+ private bool SyncSettingsChanged => gbPrefControl1.SyncSettingsChanged || gbPrefControl2.SyncSettingsChanged || gbPrefControl3.SyncSettingsChanged || gbPrefControl4.SyncSettingsChanged;
public static void DoDGBPrefsDialog(IMainFormForConfig mainForm, Config config, IGameInfo game, IMovieSession movieSession, GambatteLink gambatte)
{
@@ -50,8 +56,10 @@ namespace BizHawk.Client.EmuHawk
using var dlg = new DGBPrefs(mainForm.DialogController, config, game, movieSession);
dlg.PutSettings(s, ss);
- dlg.gbPrefControl1.ColorGameBoy = gambatte.IsCGBMode(false);
- dlg.gbPrefControl2.ColorGameBoy = gambatte.IsCGBMode(true);
+ dlg.gbPrefControl1.ColorGameBoy = gambatte.IsCGBMode(0);
+ dlg.gbPrefControl2.ColorGameBoy = gambatte.IsCGBMode(1);
+ dlg.gbPrefControl3.ColorGameBoy = gambatte.IsCGBMode(2);
+ dlg.gbPrefControl4.ColorGameBoy = gambatte.IsCGBMode(3);
if (mainForm.ShowDialogAsChild(dlg) == DialogResult.OK)
{
diff --git a/src/BizHawk.Emulation.Common/Base Implementations/LinkedDebuggable.cs b/src/BizHawk.Emulation.Common/Base Implementations/LinkedDebuggable.cs
new file mode 100644
index 0000000000..d6dde3a173
--- /dev/null
+++ b/src/BizHawk.Emulation.Common/Base Implementations/LinkedDebuggable.cs
@@ -0,0 +1,57 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+
+namespace BizHawk.Emulation.Common
+{
+ ///
+ /// A generic linked implementation of IDebuggable that can be used by any link core
+ ///
+ ///
+ public class LinkedDebuggable : IDebuggable
+ {
+ private readonly IEmulator[] _linkedCores;
+ private readonly int _numCores;
+
+ public LinkedDebuggable(IEmulator[] linkedCores, int numCores, MemoryCallbackSystem memoryCallbacks)
+ {
+ _linkedCores = linkedCores;
+ _numCores = numCores;
+ MemoryCallbacks = memoryCallbacks;
+ }
+
+ public IDictionary GetCpuFlagsAndRegisters()
+ {
+ var ret = new List>();
+
+ for (int i = 0; i < _numCores; i++)
+ {
+ ret.AddRange(_linkedCores[i].AsDebuggable().GetCpuFlagsAndRegisters()
+ .Select(reg => new KeyValuePair($"P{i + 1} " + reg.Key, reg.Value)).ToList());
+ }
+
+ return ret.ToDictionary(pair => pair.Key, pair => pair.Value);
+ }
+
+ public void SetCpuRegister(string register, int value)
+ {
+ for (int i = 0; i < _numCores; i++)
+ {
+ if (register.StartsWith($"P{i + 1} "))
+ {
+ _linkedCores[i].AsDebuggable().SetCpuRegister(register.Replace($"P{i + 1} ", ""), value);
+ }
+ }
+ }
+
+ public IMemoryCallbackSystem MemoryCallbacks { get; }
+
+ public bool CanStep(StepType type) => false;
+
+ [FeatureNotImplemented]
+ public void Step(StepType type) => throw new NotImplementedException();
+
+ [FeatureNotImplemented]
+ public long TotalExecutedCycles => throw new NotImplementedException();
+ }
+}
diff --git a/src/BizHawk.Emulation.Common/Base Implementations/LinkedMemoryDomains.cs b/src/BizHawk.Emulation.Common/Base Implementations/LinkedMemoryDomains.cs
new file mode 100644
index 0000000000..ab4dc731be
--- /dev/null
+++ b/src/BizHawk.Emulation.Common/Base Implementations/LinkedMemoryDomains.cs
@@ -0,0 +1,60 @@
+#nullable disable
+
+using System.Collections.Generic;
+
+namespace BizHawk.Emulation.Common
+{
+ ///
+ /// A generic linked implementation of IMemoryDomains that can be used by any link core
+ ///
+ ///
+ public class LinkedMemoryDomains : MemoryDomainList
+ {
+ public LinkedMemoryDomains(IEmulator[] linkedCores, int numCores)
+ : base(LinkMemoryDomains(linkedCores, numCores))
+ {
+ SystemBus = linkedCores[0].AsMemoryDomains().SystemBus;
+ }
+
+ private static List LinkMemoryDomains(IEmulator[] linkedCores, int numCores)
+ {
+ var mm = new List();
+
+ for (int i = 0; i < numCores; i++)
+ {
+ foreach (var md in linkedCores[i].AsMemoryDomains() as MemoryDomainList)
+ {
+ mm.Add(new WrappedMemoryDomain($"P{i + 1} " + md.Name, md));
+ }
+ }
+
+ return mm;
+ }
+
+ private class WrappedMemoryDomain : MemoryDomain
+ {
+ private readonly MemoryDomain _m;
+
+ public WrappedMemoryDomain(string name, MemoryDomain m)
+ {
+ _m = m;
+
+ Name = name;
+ Size = m.Size;
+ WordSize = m.WordSize;
+ EndianType = m.EndianType;
+ Writable = m.Writable;
+ }
+
+ public override byte PeekByte(long addr)
+ {
+ return _m.PeekByte(addr);
+ }
+
+ public override void PokeByte(long addr, byte val)
+ {
+ _m.PokeByte(addr, val);
+ }
+ }
+ }
+}
diff --git a/src/BizHawk.Emulation.Common/Base Implementations/LinkedSaveRam.cs b/src/BizHawk.Emulation.Common/Base Implementations/LinkedSaveRam.cs
new file mode 100644
index 0000000000..c2d344ef35
--- /dev/null
+++ b/src/BizHawk.Emulation.Common/Base Implementations/LinkedSaveRam.cs
@@ -0,0 +1,66 @@
+using System;
+using System.Collections.Generic;
+
+namespace BizHawk.Emulation.Common
+{
+ ///
+ /// A generic linked implementation of ISaveRam that can be used by any link core
+ ///
+ ///
+ public class LinkedSaveRam : ISaveRam
+ {
+ private readonly IEmulator[] _linkedCores;
+ private readonly int _numCores;
+
+ public LinkedSaveRam(IEmulator[] linkedCores, int numCores)
+ {
+ _linkedCores = linkedCores;
+ _numCores = numCores;
+ }
+
+ public bool SaveRamModified => LinkedSaveRamModified();
+
+ private bool LinkedSaveRamModified()
+ {
+ for (int i = 0; i < _numCores; i++)
+ {
+ if (_linkedCores[i].AsSaveRam().SaveRamModified)
+ {
+ return true;
+ }
+ }
+ return false;
+ }
+
+ public byte[] CloneSaveRam()
+ {
+ var linkedBuffers = new List();
+ int len = 0;
+ for (int i = 0; i < _numCores; i++)
+ {
+ linkedBuffers.Add(_linkedCores[i].AsSaveRam().CloneSaveRam()!);
+ len += linkedBuffers[i].Length;
+ }
+ byte[] ret = new byte[len];
+ int pos = 0;
+ for (int i = 0; i < _numCores; i++)
+ {
+ Buffer.BlockCopy(linkedBuffers[i], 0, ret, pos, linkedBuffers[i].Length);
+ pos += linkedBuffers[i].Length;
+ }
+ return ret;
+ }
+
+ public void StoreSaveRam(byte[] data)
+ {
+ int pos = 0;
+ for (int i = 0; i < _numCores; i++)
+ {
+ var b = new byte[_linkedCores[i].AsSaveRam().CloneSaveRam()!.Length];
+ Buffer.BlockCopy(data, pos, b, 0, b.Length);
+ pos += b.Length;
+ _linkedCores[i].AsSaveRam().StoreSaveRam(b);
+ }
+ }
+ }
+}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/Gambatte.ICodeDataLog.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/Gambatte.ICodeDataLog.cs
index 0bcfe5e57a..5fdbb85c06 100644
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/Gambatte.ICodeDataLog.cs
+++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/Gambatte.ICodeDataLog.cs
@@ -7,21 +7,26 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
{
public partial class Gameboy : ICodeDataLogger
{
- public void SetCDL(ICodeDataLog cdl)
+ public void SetCDL(ICodeDataLog cdl) => SetCDL(cdl, "");
+
+ internal void SetCDL(ICodeDataLog cdl, string which)
{
_cdl = cdl;
+ _which = which;
LibGambatte.gambatte_setcdcallback(GambatteState, cdl == null ? null : _cdCallback);
}
- public void NewCDL(ICodeDataLog cdl)
+ public void NewCDL(ICodeDataLog cdl) => NewCDL(cdl, "");
+
+ internal void NewCDL(ICodeDataLog cdl, string which)
{
- cdl["ROM"] = new byte[MemoryDomains["ROM"]!.Size];
+ cdl[which + "ROM"] = new byte[MemoryDomains["ROM"]!.Size];
// cdl["HRAM"] = new byte[_memoryDomains["HRAM"]!.Size]; //this is probably useless, but it's here if someone needs it
- cdl["WRAM"] = new byte[MemoryDomains["WRAM"]!.Size];
+ cdl[which + "WRAM"] = new byte[MemoryDomains["WRAM"]!.Size];
var found = MemoryDomains["CartRAM"];
- if (found is not null) cdl["CartRAM"] = new byte[found.Size];
+ if (found is not null) cdl[which + "CartRAM"] = new byte[found.Size];
cdl.SubType = "GB";
cdl.SubVer = 0;
@@ -33,6 +38,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
}
private ICodeDataLog _cdl;
+ private string _which;
private readonly LibGambatte.CDCallback _cdCallback;
private void CDCallbackProc(int addr, LibGambatte.CDLog_AddrType addrtype, LibGambatte.CDLog_Flags flags)
@@ -51,7 +57,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
_ => throw new InvalidOperationException("Juniper lightbulb proxy"),
};
- _cdl[key][addr] |= (byte)flags;
+ _cdl[_which + key][addr] |= (byte)flags;
}
}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/Gambatte.IDebuggable.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/Gambatte.IDebuggable.cs
index 5b7968e819..68ed64345b 100644
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/Gambatte.IDebuggable.cs
+++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/Gambatte.IDebuggable.cs
@@ -60,6 +60,7 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
internal void ConnectMemoryCallbackSystem(MemoryCallbackSystem mcs)
{
_memorycallbacks = mcs;
+ _memorycallbacks.ActiveChanged += SetMemoryCallbacks;
}
private void InitMemoryCallbacks()
@@ -78,12 +79,14 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
_writecb = CreateCallback(MemoryCallbackFlags.AccessWrite, () => MemoryCallbacks.HasWrites);
_execcb = CreateCallback(MemoryCallbackFlags.AccessExecute, () => MemoryCallbacks.HasExecutes);
- _memorycallbacks.ActiveChanged += () =>
- {
- LibGambatte.gambatte_setreadcallback(GambatteState, MemoryCallbacks.HasReads ? _readcb : null);
- LibGambatte.gambatte_setwritecallback(GambatteState, MemoryCallbacks.HasWrites ? _writecb : null);
- LibGambatte.gambatte_setexeccallback(GambatteState, MemoryCallbacks.HasExecutes ? _execcb : null);
- };
+ _memorycallbacks.ActiveChanged += SetMemoryCallbacks;
+ }
+
+ private void SetMemoryCallbacks()
+ {
+ LibGambatte.gambatte_setreadcallback(GambatteState, MemoryCallbacks.HasReads ? _readcb : null);
+ LibGambatte.gambatte_setwritecallback(GambatteState, MemoryCallbacks.HasWrites ? _writecb : null);
+ LibGambatte.gambatte_setexeccallback(GambatteState, MemoryCallbacks.HasExecutes ? _execcb : null);
}
}
}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ICodeDataLog.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ICodeDataLog.cs
index 00a2bdb93c..6686524720 100644
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ICodeDataLog.cs
+++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ICodeDataLog.cs
@@ -4,19 +4,34 @@ using System.IO;
namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
{
- partial class GambatteLink
+ partial class GambatteLink : ICodeDataLogger
{
void ICodeDataLogger.SetCDL(ICodeDataLog cdl)
{
- ((ICodeDataLogger)L).SetCDL(cdl);
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i].SetCDL(cdl, $"P{i + 1} ");
+ }
}
void ICodeDataLogger.NewCDL(ICodeDataLog cdl)
{
- ((ICodeDataLogger)L).NewCDL(cdl);
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i].NewCDL(cdl, $"P{i + 1} ");
+ }
}
- void ICodeDataLogger.DisassembleCDL(Stream s, ICodeDataLog cdl) { ((ICodeDataLogger)L).DisassembleCDL(s, cdl); }
-
+ [FeatureNotImplemented]
+ void ICodeDataLogger.DisassembleCDL(Stream s, ICodeDataLog cdl)
+ {
+ // this doesn't actually do anything
+ /*
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i].DisassembleCDL(s, cdl);
+ }
+ */
+ }
}
}
\ No newline at end of file
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IDebuggable.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IDebuggable.cs
deleted file mode 100644
index 04783cc4b2..0000000000
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IDebuggable.cs
+++ /dev/null
@@ -1,46 +0,0 @@
-using System;
-using System.Collections.Generic;
-using System.Linq;
-
-using BizHawk.Emulation.Common;
-
-namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
-{
- public partial class GambatteLink : IDebuggable
- {
- public IDictionary GetCpuFlagsAndRegisters()
- {
- var left = L.GetCpuFlagsAndRegisters()
- .Select(reg => new KeyValuePair("Left " + reg.Key, reg.Value));
-
- var right = R.GetCpuFlagsAndRegisters()
- .Select(reg => new KeyValuePair("Right " + reg.Key, reg.Value));
-
- return left.Union(right).ToDictionary(pair => pair.Key, pair => pair.Value);
- }
-
- public void SetCpuRegister(string register, int value)
- {
- if (register.StartsWith("Left "))
- {
- L.SetCpuRegister(register.Replace("Left ", ""), value);
- }
- else if (register.StartsWith("Right "))
- {
- R.SetCpuRegister(register.Replace("Right ", ""), value);
- }
- }
-
- public IMemoryCallbackSystem MemoryCallbacks { get; } = new MemoryCallbackSystem(new[] { "System Bus" });
-
- public bool CanStep(StepType type) => false;
-
- [FeatureNotImplemented]
- public void Step(StepType type) => throw new NotImplementedException();
-
- [FeatureNotImplemented]
- public long TotalExecutedCycles => throw new NotImplementedException();
-
- private readonly MemoryCallbackSystem _memorycallbacks = new MemoryCallbackSystem(new[] { "System Bus" });
- }
-}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IEmulator.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IEmulator.cs
index 4d5a1dc062..d04d74a5fa 100644
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IEmulator.cs
+++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IEmulator.cs
@@ -6,56 +6,82 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
{
public partial class GambatteLink : IEmulator, IBoardInfo
{
- public IEmulatorServiceProvider ServiceProvider { get; }
+ public IEmulatorServiceProvider ServiceProvider => _serviceProvider;
- public ControllerDefinition ControllerDefinition => DualGbController;
+ public ControllerDefinition ControllerDefinition => GBLinkController;
public bool FrameAdvance(IController controller, bool render, bool rendersound = true)
{
- LCont.Clear();
- RCont.Clear();
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedConts[i].Clear();
+ }
- foreach (var s in DualGbController.BoolButtons)
+ foreach (var s in GBLinkController.BoolButtons)
{
if (controller.IsPressed(s))
{
- if (s.Contains("P1 "))
+ for (int i = 0; i < _numCores; i++)
{
- LCont.Set(s.Replace("P1 ", ""));
- }
- else if (s.Contains("P2 "))
- {
- RCont.Set(s.Replace("P2 ", ""));
+ if (s.Contains($"P{i + 1} "))
+ {
+ _linkedConts[i].Set(s.Replace($"P{i + 1} ", ""));
+ }
}
}
}
- bool cablediscosignalNew = controller.IsPressed("Toggle Cable");
- if (cablediscosignalNew && !_cablediscosignal)
+ bool cableDiscoSignalNew = controller.IsPressed("Toggle Cable Connection");
+ if (cableDiscoSignalNew && !_cableDiscoSignal)
{
- _cableconnected ^= true;
- Console.WriteLine("Cable connect status to {0}", _cableconnected);
+ _cableConnected ^= true;
+ Console.WriteLine("Cable connect status to {0}", _cableConnected);
}
- _cablediscosignal = cablediscosignalNew;
+ _cableDiscoSignal = cableDiscoSignalNew;
- L.FrameAdvancePrep(LCont);
- R.FrameAdvancePrep(RCont);
+ if (_numCores > 2)
+ {
+ bool cableShiftSignalNew = controller.IsPressed("Toggle Cable Shift");
+ if (cableShiftSignalNew && !_cableShiftSignal)
+ {
+ _cableShifted ^= true;
+ Console.WriteLine("Cable shift status to {0}", _cableShifted);
+ }
+
+ _cableShifted = cableShiftSignalNew;
+
+ bool cableSpaceSignalNew = controller.IsPressed("Toggle Cable Spacing");
+ if (cableSpaceSignalNew && !_cableSpaceSignal)
+ {
+ _cableSpaced ^= true;
+ Console.WriteLine("Cable spacing status to {0}", _cableSpaceSignal);
+ }
+
+ _cableSpaceSignal = cableSpaceSignalNew;
+ }
+
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i].FrameAdvancePrep(_linkedConts[i]);
+ }
unsafe
{
- fixed (int* leftfbuff = &FrameBuffer[0])
+ fixed (int* fbuff = &FrameBuffer[0])
{
// use pitch to have both cores write to the same frame buffer, interleaved
- int* rightfbuff = leftfbuff + 160;
- const int Pitch = 160 * 2;
+ int Pitch = 160 * _numCores;
- fixed (short* leftsbuff = LeftBuffer, rightsbuff = RightBuffer)
+ fixed (short* sbuff = &SoundBuffer[0])
{
const int Step = 32; // could be 1024 for GB
- int nL = _overflowL;
- int nR = _overflowR;
+ int[] n = new int[_numCores];
+ for (int i = 0; i < _numCores; i++)
+ {
+ n[i] = _linkedOverflow[i];
+ }
// slowly step our way through the frame, while continually checking and resolving link cable status
for (int target = 0; target < SampPerFrame;)
@@ -66,65 +92,94 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
target = SampPerFrame; // don't run for slightly too long depending on step
}
- // gambatte_runfor() aborts early when a frame is produced, but we don't want that, hence the while()
- while (nL < target)
+ for (int i = 0; i < _numCores; i++)
{
- uint nsamp = (uint)(target - nL);
- if (LibGambatte.gambatte_runfor(L.GambatteState, leftfbuff, Pitch, leftsbuff + (nL * 2), ref nsamp) > 0)
+ // gambatte_runfor() aborts early when a frame is produced, but we don't want that, hence the while()
+ while (n[i] < target)
{
- for (int i = 0; i < 144; i++)
+ uint nsamp = (uint)(target - n[i]);
+ if (LibGambatte.gambatte_runfor(_linkedCores[i].GambatteState, fbuff + (i * 160), Pitch, sbuff + (i * MaxSampsPerFrame) + (n[i] * 2), ref nsamp) > 0)
{
- Array.Copy(FrameBuffer, Pitch * i, VideoBuffer, Pitch * i, 160);
+ for (int j = 0; j < 144; j++)
+ {
+ Array.Copy(FrameBuffer, (i * 160) + (j * Pitch), VideoBuffer, (i * 160) + (j * Pitch), 160);
+ }
}
+
+ n[i] += (int)nsamp;
}
-
- nL += (int)nsamp;
- }
-
- while (nR < target)
- {
- uint nsamp = (uint)(target - nR);
- if (LibGambatte.gambatte_runfor(R.GambatteState, rightfbuff, Pitch, rightsbuff + (nR * 2), ref nsamp) > 0)
- {
- for (int i = 0; i < 144; i++)
- {
- Array.Copy(FrameBuffer, 160 + Pitch * i, VideoBuffer, 160 + Pitch * i, 160);
- }
- }
-
- nR += (int)nsamp;
}
// poll link cable statuses, but not when the cable is disconnected
- if (!_cableconnected)
+ if (!_cableConnected)
{
continue;
}
- if (LibGambatte.gambatte_linkstatus(L.GambatteState, 256) != 0) // ClockTrigger
+ switch (_numCores)
{
- LibGambatte.gambatte_linkstatus(L.GambatteState, 257); // ack
- int lo = LibGambatte.gambatte_linkstatus(L.GambatteState, 258); // GetOut
- int ro = LibGambatte.gambatte_linkstatus(R.GambatteState, 258);
- LibGambatte.gambatte_linkstatus(L.GambatteState, ro & 0xff); // ShiftIn
- LibGambatte.gambatte_linkstatus(R.GambatteState, lo & 0xff); // ShiftIn
- }
-
- if (LibGambatte.gambatte_linkstatus(R.GambatteState, 256) != 0) // ClockTrigger
- {
- LibGambatte.gambatte_linkstatus(R.GambatteState, 257); // ack
- int lo = LibGambatte.gambatte_linkstatus(L.GambatteState, 258); // GetOut
- int ro = LibGambatte.gambatte_linkstatus(R.GambatteState, 258);
- LibGambatte.gambatte_linkstatus(L.GambatteState, ro & 0xff); // ShiftIn
- LibGambatte.gambatte_linkstatus(R.GambatteState, lo & 0xff); // ShiftIn
+ case 2:
+ {
+ TryCableBytewiseTransfer(P1, P2);
+ TryCableBytewiseTransfer(P2, P1);
+ break;
+ }
+ case 3:
+ {
+ if (_cableSpaced)
+ {
+ TryCableBytewiseTransfer(P1, P3);
+ TryCableBytewiseTransfer(P3, P1);
+ }
+ else if (_cableShifted)
+ {
+ TryCableBytewiseTransfer(P2, P3);
+ TryCableBytewiseTransfer(P3, P2);
+ }
+ else
+ {
+ TryCableBytewiseTransfer(P1, P2);
+ TryCableBytewiseTransfer(P2, P1);
+ }
+ break;
+ }
+ case 4:
+ {
+ if (_cableSpaced)
+ {
+ TryCableBytewiseTransfer(P1, P3);
+ TryCableBytewiseTransfer(P3, P1);
+ TryCableBytewiseTransfer(P2, P4);
+ TryCableBytewiseTransfer(P4, P2);
+ }
+ else if (_cableShifted)
+ {
+ TryCableBytewiseTransfer(P2, P3);
+ TryCableBytewiseTransfer(P3, P2);
+ TryCableBytewiseTransfer(P4, P1);
+ TryCableBytewiseTransfer(P1, P4);
+ }
+ else
+ {
+ TryCableBytewiseTransfer(P1, P2);
+ TryCableBytewiseTransfer(P2, P1);
+ TryCableBytewiseTransfer(P3, P4);
+ TryCableBytewiseTransfer(P4, P3);
+ }
+ break;
+ }
+ default:
+ throw new Exception();
}
}
- _overflowL = nL - SampPerFrame;
- _overflowR = nR - SampPerFrame;
- if (_overflowL < 0 || _overflowR < 0)
+ for (int i = 0; i < _numCores; i++)
{
- throw new Exception("Timing problem?");
+ _linkedOverflow[i] = n[i] - SampPerFrame;
+ if (_linkedOverflow[i] < 0)
+ {
+ throw new Exception("Timing problem?");
+ }
}
if (rendersound)
@@ -133,22 +188,25 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
}
// copy extra samples back to beginning
- for (int i = 0; i < _overflowL * 2; i++)
+ for (int i = 0; i < _numCores; i++)
{
- LeftBuffer[i] = LeftBuffer[i + (SampPerFrame * 2)];
- }
-
- for (int i = 0; i < _overflowR * 2; i++)
- {
- RightBuffer[i] = RightBuffer[i + (SampPerFrame * 2)];
+ for (int j = 0; j < _linkedOverflow[i] * 2; j++)
+ {
+ SoundBuffer[(i * MaxSampsPerFrame) + j] = SoundBuffer[(i * MaxSampsPerFrame) + j + (SampPerFrame * 2)];
+ }
}
}
}
}
- L.FrameAdvancePost();
- R.FrameAdvancePost();
- IsLagFrame = L.IsLagFrame && R.IsLagFrame;
+ IsLagFrame = true;
+
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i].FrameAdvancePost();
+ IsLagFrame &= _linkedCores[i].IsLagFrame;
+ }
+
if (IsLagFrame)
{
LagCount++;
@@ -159,38 +217,71 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
return true;
}
+ private void TryCableBytewiseTransfer(int master, int slave)
+ {
+ if (LibGambatte.gambatte_linkstatus(_linkedCores[master].GambatteState, 256) != 0) // ClockTrigger
+ {
+ LibGambatte.gambatte_linkstatus(_linkedCores[master].GambatteState, 257); // ack
+ int om = LibGambatte.gambatte_linkstatus(_linkedCores[master].GambatteState, 258); // GetOut
+ int os = LibGambatte.gambatte_linkstatus(_linkedCores[slave].GambatteState, 258);
+ LibGambatte.gambatte_linkstatus(_linkedCores[slave].GambatteState, om & 0xff); // ShiftIn
+ LibGambatte.gambatte_linkstatus(_linkedCores[master].GambatteState, os & 0xff); // ShiftIn
+ }
+ }
+
public int Frame { get; private set; }
public string SystemId => VSystemID.Raw.DGB;
- public bool DeterministicEmulation => L.DeterministicEmulation && R.DeterministicEmulation;
+ public bool DeterministicEmulation => LinkedDeterministicEmulation();
- public string BoardName => L.BoardName + '|' + R.BoardName;
+ private bool LinkedDeterministicEmulation()
+ {
+ bool deterministicEmulation = true;
+ for (int i = 0; i < _numCores; i++)
+ {
+ deterministicEmulation &= _linkedCores[i].DeterministicEmulation;
+ }
+ return deterministicEmulation;
+ }
+
+ public string BoardName => LinkedBoardName();
+
+ private string LinkedBoardName()
+ {
+ string boardName = "";
+ for (int i = 0; i < _numCores; i++)
+ {
+ boardName += _linkedCores[i].BoardName + "|";
+ }
+ return boardName.Remove(boardName.Length - 1);
+ }
public void ResetCounters()
{
Frame = 0;
LagCount = 0;
IsLagFrame = false;
+
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedOverflow[i] = 0;
+ }
}
public void Dispose()
{
- if (!_disposed)
+ if (_numCores > 0)
{
- L.Dispose();
- L = null;
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i].Dispose();
+ _linkedCores[i] = null;
+ _linkedBlips[i].Dispose();
+ _linkedBlips[i] = null;
+ }
- R.Dispose();
- R = null;
-
- _blipLeft.Dispose();
- _blipLeft = null;
-
- _blipRight.Dispose();
- _blipRight = null;
-
- _disposed = true;
+ _numCores = 0;
}
}
}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IMemoryDomains.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IMemoryDomains.cs
deleted file mode 100644
index d6a1d008f1..0000000000
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IMemoryDomains.cs
+++ /dev/null
@@ -1,56 +0,0 @@
-using System.Collections.Generic;
-
-using BizHawk.Emulation.Common;
-
-namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
-{
- public partial class GambatteLink
- {
- private IMemoryDomains _memoryDomains;
-
- private void SetMemoryDomains()
- {
- var mm = new List();
-
- foreach (var md in L.MemoryDomains)
- {
- mm.Add(new WrappedMemoryDomain("L " + md.Name, md));
- }
-
- foreach (var md in R.MemoryDomains)
- {
- mm.Add(new WrappedMemoryDomain("R " + md.Name, md));
- }
-
- _memoryDomains = new MemoryDomainList(mm);
- (ServiceProvider as BasicServiceProvider).Register(_memoryDomains);
- }
-
- // todo: clean this up
- private class WrappedMemoryDomain : MemoryDomain
- {
- private readonly MemoryDomain _m;
-
- public WrappedMemoryDomain(string name, MemoryDomain m)
- {
- _m = m;
-
- Name = name;
- Size = m.Size;
- WordSize = m.WordSize;
- EndianType = m.EndianType;
- Writable = m.Writable;
- }
-
- public override byte PeekByte(long addr)
- {
- return _m.PeekByte(addr);
- }
-
- public override void PokeByte(long addr, byte val)
- {
- _m.PokeByte(addr, val);
- }
- }
- }
-}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISaveRam.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISaveRam.cs
deleted file mode 100644
index 02a991c5f0..0000000000
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISaveRam.cs
+++ /dev/null
@@ -1,31 +0,0 @@
-using System;
-
-using BizHawk.Emulation.Common;
-
-namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
-{
- public partial class GambatteLink : ISaveRam
- {
- public bool SaveRamModified => L.SaveRamModified || R.SaveRamModified;
-
- public byte[] CloneSaveRam()
- {
- var lb = L.CloneSaveRam()!;
- var rb = R.CloneSaveRam()!;
- byte[] ret = new byte[lb.Length + rb.Length];
- Buffer.BlockCopy(lb, 0, ret, 0, lb.Length);
- Buffer.BlockCopy(rb, 0, ret, lb.Length, rb.Length);
- return ret;
- }
-
- public void StoreSaveRam(byte[] data)
- {
- var lb = new byte[L.CloneSaveRam()!.Length];
- var rb = new byte[R.CloneSaveRam()!.Length];
- Buffer.BlockCopy(data, 0, lb, 0, lb.Length);
- Buffer.BlockCopy(data, lb.Length, rb, 0, rb.Length);
- L.StoreSaveRam(lb);
- R.StoreSaveRam(rb);
- }
- }
-}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISettable.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISettable.cs
index 544b748ad1..6557027bae 100644
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISettable.cs
+++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISettable.cs
@@ -4,76 +4,78 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
{
public partial class GambatteLink : ISettable
{
+ private GambatteLinkSettings _settings;
+ private GambatteLinkSyncSettings _syncSettings;
+
public GambatteLinkSettings GetSettings()
{
- return new GambatteLinkSettings
- (
- L.GetSettings(),
- R.GetSettings()
- );
+ return _settings.Clone();
}
+
public GambatteLinkSyncSettings GetSyncSettings()
{
- return new GambatteLinkSyncSettings
- (
- L.GetSyncSettings(),
- R.GetSyncSettings()
- );
+ return _syncSettings.Clone();
}
public PutSettingsDirtyBits PutSettings(GambatteLinkSettings o)
{
- return (PutSettingsDirtyBits)((int)L.PutSettings(o.L) | (int)R.PutSettings(o.R));
+ _settings = o;
+ var ret = PutSettingsDirtyBits.None;
+ for (int i = 0; i < _numCores; i++)
+ {
+ ret |= _linkedCores[i].PutSettings(o._linkedSettings[i]);
+ }
+ return ret;
}
public PutSettingsDirtyBits PutSyncSettings(GambatteLinkSyncSettings o)
{
- return (PutSettingsDirtyBits)((int)L.PutSyncSettings(o.L) | (int)R.PutSyncSettings(o.R));
+ _syncSettings = o;
+ var ret = PutSettingsDirtyBits.None;
+ for (int i = 0; i < _numCores; i++)
+ {
+ ret |= _linkedCores[i].PutSyncSettings(o._linkedSyncSettings[i]);
+ }
+ return ret;
}
public class GambatteLinkSettings
{
- public Gameboy.GambatteSettings L;
- public Gameboy.GambatteSettings R;
+ public Gameboy.GambatteSettings[] _linkedSettings;
public GambatteLinkSettings()
{
- L = new Gameboy.GambatteSettings();
- R = new Gameboy.GambatteSettings();
+ _linkedSettings = new Gameboy.GambatteSettings[MAX_PLAYERS] { new(), new(), new(), new() };
}
- public GambatteLinkSettings(Gameboy.GambatteSettings L, Gameboy.GambatteSettings R)
+ public GambatteLinkSettings(Gameboy.GambatteSettings one, Gameboy.GambatteSettings two, Gameboy.GambatteSettings three, Gameboy.GambatteSettings four)
{
- this.L = L;
- this.R = R;
+ _linkedSettings = new Gameboy.GambatteSettings[MAX_PLAYERS] { one, two, three, four };
}
public GambatteLinkSettings Clone()
{
- return new GambatteLinkSettings(L.Clone(), R.Clone());
+ return new GambatteLinkSettings(_linkedSettings[P1].Clone(), _linkedSettings[P2].Clone(), _linkedSettings[P3].Clone(), _linkedSettings[P4].Clone());
}
}
public class GambatteLinkSyncSettings
{
- public Gameboy.GambatteSyncSettings L;
- public Gameboy.GambatteSyncSettings R;
+ public Gameboy.GambatteSyncSettings[] _linkedSyncSettings;
public GambatteLinkSyncSettings()
{
- L = new Gameboy.GambatteSyncSettings();
- R = new Gameboy.GambatteSyncSettings();
+ _linkedSyncSettings = new Gameboy.GambatteSyncSettings[MAX_PLAYERS] { new(), new(), new(), new() };
}
- public GambatteLinkSyncSettings(Gameboy.GambatteSyncSettings L, Gameboy.GambatteSyncSettings R)
+ public GambatteLinkSyncSettings(Gameboy.GambatteSyncSettings one, Gameboy.GambatteSyncSettings two, Gameboy.GambatteSyncSettings three, Gameboy.GambatteSyncSettings four)
{
- this.L = L;
- this.R = R;
+ _linkedSyncSettings = new Gameboy.GambatteSyncSettings[MAX_PLAYERS] { one, two, three, four };
}
public GambatteLinkSyncSettings Clone()
{
- return new GambatteLinkSyncSettings(L.Clone(), R.Clone());
+ return new GambatteLinkSyncSettings(_linkedSyncSettings[P1].Clone(), _linkedSyncSettings[P2].Clone(), _linkedSyncSettings[P3].Clone(), _linkedSyncSettings[P4].Clone());
}
}
}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISoundProvider.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISoundProvider.cs
index 7c94ec28fa..05310b1e83 100644
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISoundProvider.cs
+++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.ISoundProvider.cs
@@ -1,111 +1,154 @@
using System;
-
-using BizHawk.Emulation.Common;
-
-namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
-{
- public partial class GambatteLink : ISoundProvider
- {
- public bool CanProvideAsync => false;
-
- public void SetSyncMode(SyncSoundMode mode)
- {
- if (mode == SyncSoundMode.Async)
- {
- throw new NotSupportedException("Async mode is not supported.");
- }
- }
-
- public SyncSoundMode SyncMode => SyncSoundMode.Sync;
-
- public void GetSamplesSync(out short[] samples, out int nsamp)
- {
- nsamp = _sampleBufferContains;
- samples = SampleBuffer;
- }
-
- public void GetSamplesAsync(short[] samples)
- {
- throw new InvalidOperationException("Async mode is not supported.");
- }
-
- public void DiscardSamples()
- {
- _sampleBufferContains = 0;
- }
-
- // i tried using the left and right buffers and then mixing them together... it was kind of a mess of code, and slow
- private BlipBuffer _blipLeft;
- private BlipBuffer _blipRight;
-
- private readonly short[] LeftBuffer = new short[(35112 + 2064) * 2];
- private readonly short[] RightBuffer = new short[(35112 + 2064) * 2];
-
- private readonly short[] SampleBuffer = new short[1536];
- private int _sampleBufferContains = 0;
-
- private int _latchLeft;
- private int _latchRight;
-
- private unsafe void PrepSound()
- {
- fixed (short* sl = LeftBuffer, sr = RightBuffer)
- {
- for (uint i = 0; i < SampPerFrame * 2; i += 2)
- {
- int s = (sl[i] + sl[i + 1]) / 2;
- if (s != _latchLeft)
- {
- _blipLeft.AddDelta(i, s - _latchLeft);
- _latchLeft = s;
- }
-
- s = (sr[i] + sr[i + 1]) / 2;
- if (s != _latchRight)
- {
- _blipRight.AddDelta(i, s - _latchRight);
- _latchRight = s;
- }
- }
- }
-
- _blipLeft.EndFrame(SampPerFrame * 2);
- _blipRight.EndFrame(SampPerFrame * 2);
- int count = _blipLeft.SamplesAvailable();
- if (count != _blipRight.SamplesAvailable())
- {
- throw new Exception("Sound problem?");
- }
-
- // calling blip.Clear() causes rounding fractions to be reset,
- // and if only one channel is muted, in subsequent frames we can be off by a sample or two
- // not a big deal, but we didn't account for it. so we actually complete the entire
- // audio read and then stamp it out if muted.
- _blipLeft.ReadSamplesLeft(SampleBuffer, count);
- if (L.Muted)
- {
- fixed (short* p = SampleBuffer)
- {
- for (int i = 0; i < SampleBuffer.Length; i += 2)
- {
- p[i] = 0;
- }
- }
- }
-
- _blipRight.ReadSamplesRight(SampleBuffer, count);
- if (R.Muted)
- {
- fixed (short* p = SampleBuffer)
- {
- for (int i = 1; i < SampleBuffer.Length; i += 2)
- {
- p[i] = 0;
- }
- }
- }
-
- _sampleBufferContains = count;
- }
- }
-}
+
+using BizHawk.Emulation.Common;
+
+namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
+{
+ public partial class GambatteLink : ISoundProvider
+ {
+ public bool CanProvideAsync => false;
+
+ public void SetSyncMode(SyncSoundMode mode)
+ {
+ if (mode == SyncSoundMode.Async)
+ {
+ throw new NotSupportedException("Async mode is not supported.");
+ }
+ }
+
+ public SyncSoundMode SyncMode => SyncSoundMode.Sync;
+
+ public void GetSamplesSync(out short[] samples, out int nsamp)
+ {
+ nsamp = _sampleBufferContains;
+ samples = SampleBuffer;
+ }
+
+ public void GetSamplesAsync(short[] samples)
+ {
+ throw new InvalidOperationException("Async mode is not supported.");
+ }
+
+ public void DiscardSamples()
+ {
+ _sampleBufferContains = 0;
+ }
+
+ // i tried using the left and right buffers and then mixing them together... it was kind of a mess of code, and slow
+ private readonly BlipBuffer[] _linkedBlips;
+
+ private readonly short[] SoundBuffer;
+
+ private readonly short[] ScratchBuffer = new short[1536];
+
+ private readonly short[] SampleBuffer = new short[1536];
+ private int _sampleBufferContains = 0;
+
+ private readonly int[] _linkedLatches;
+
+ private unsafe void PrepSound()
+ {
+ fixed (short* sb = &SoundBuffer[0])
+ {
+ for (int i = 0; i < _numCores; i++)
+ {
+ for (uint j = 0; j < SampPerFrame * 2; j += 2)
+ {
+ int s = (sb[(i * MaxSampsPerFrame) + j] + sb[(i * MaxSampsPerFrame) + j + 1]) / 2;
+ if (s != _linkedLatches[i])
+ {
+ _linkedBlips[i].AddDelta(j, s - _linkedLatches[i]);
+ _linkedLatches[i] = s;
+ }
+ }
+ _linkedBlips[i].EndFrame(SampPerFrame * 2);
+ }
+ }
+
+ int count = _linkedBlips[P1].SamplesAvailable();
+ for (int i = 1; i < _numCores; i++)
+ {
+ if (count != _linkedBlips[i].SamplesAvailable())
+ {
+ throw new Exception("Sound problem?");
+ }
+ }
+
+ // calling blip.Clear() causes rounding fractions to be reset,
+ // and if only one channel is muted, in subsequent frames we can be off by a sample or two
+ // not a big deal, but we didn't account for it. so we actually complete the entire
+ // audio read and then stamp it out if muted.
+
+ switch (_numCores)
+ {
+ case 2:
+ {
+ // no need to do any complicated mixing
+ _linkedBlips[P1].ReadSamplesLeft(_settings._linkedSettings[P1].Muted ? ScratchBuffer : SampleBuffer, count);
+ _linkedBlips[P2].ReadSamplesRight(_settings._linkedSettings[P2].Muted ? ScratchBuffer : SampleBuffer, count);
+ break;
+ }
+ case 3:
+ {
+ // since P2 is center, mix its samples with P1 and P3
+ _linkedBlips[P1].ReadSamplesLeft(_settings._linkedSettings[P1].Muted ? ScratchBuffer : SampleBuffer, count);
+ _linkedBlips[P3].ReadSamplesRight(_settings._linkedSettings[P3].Muted ? ScratchBuffer : SampleBuffer, count);
+ _linkedBlips[P2].ReadSamples(ScratchBuffer, count, false);
+ if (!_settings._linkedSettings[P2].Muted)
+ {
+ fixed (short* p = SampleBuffer, q = ScratchBuffer)
+ {
+ for (int i = 0; i < SampleBuffer.Length; i++)
+ {
+ int s = (p[i] + q[i]) / 2;
+ p[i] = (short)s;
+ }
+ }
+ }
+ break;
+ }
+ case 4:
+ {
+ // since P1 and P2 are left side and P3 and P4 are right side, mix their samples accordingly
+ _linkedBlips[P1].ReadSamplesLeft(_settings._linkedSettings[P1].Muted ? ScratchBuffer : SampleBuffer, count);
+ _linkedBlips[P3].ReadSamplesRight(_settings._linkedSettings[P3].Muted ? ScratchBuffer : SampleBuffer, count);
+ _linkedBlips[P2].ReadSamplesLeft(ScratchBuffer, count);
+ _linkedBlips[P4].ReadSamplesRight(ScratchBuffer, count);
+ if (_settings._linkedSettings[P2].Muted)
+ {
+ fixed (short* p = ScratchBuffer)
+ {
+ for (int i = 0; i < ScratchBuffer.Length; i += 2)
+ {
+ p[i] = 0;
+ }
+ }
+ }
+ if (_settings._linkedSettings[P4].Muted)
+ {
+ fixed (short* p = ScratchBuffer)
+ {
+ for (int i = 1; i < ScratchBuffer.Length; i += 2)
+ {
+ p[i] = 0;
+ }
+ }
+ }
+ fixed (short* p = SampleBuffer, q = ScratchBuffer)
+ {
+ for (int i = 0; i < SampleBuffer.Length; i++)
+ {
+ int s = (p[i] + q[i]) / 2;
+ p[i] = (short)s;
+ }
+ }
+ break;
+ }
+ default:
+ throw new Exception();
+ }
+
+ _sampleBufferContains = count;
+ }
+ }
+}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IStatable.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IStatable.cs
index 03ca3948e8..299f07dbda 100644
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IStatable.cs
+++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IStatable.cs
@@ -1,4 +1,5 @@
-using System.IO;
+using System;
+using System.IO;
using Newtonsoft.Json;
@@ -10,87 +11,119 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
{
public void SaveStateText(TextWriter writer)
{
- var s = new DGBSerialized
- {
- L = L.SaveState(),
- R = R.SaveState(),
- IsLagFrame = IsLagFrame,
- LagCount = LagCount,
- Frame = Frame,
- overflowL = _overflowL,
- overflowR = _overflowR,
- LatchL = _latchLeft,
- LatchR = _latchRight,
- cableconnected = _cableconnected,
- cablediscosignal = _cablediscosignal
- };
- ser.Serialize(writer, s);
+ ser.Serialize(writer, new DGBSerialized(this));
}
public void LoadStateText(TextReader reader)
{
var s = (DGBSerialized)ser.Deserialize(reader, typeof(DGBSerialized));
- L.LoadState(s.L);
- R.LoadState(s.R);
+ if (s.NumCores != _numCores)
+ {
+ throw new InvalidOperationException("Core number mismatch!");
+ }
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i].LoadState(s.LinkedStates[i]);
+ _linkedOverflow[i] = s.LinkedOverflow[i];
+ _linkedLatches[i] = s.LinkedLatches[i];
+ }
IsLagFrame = s.IsLagFrame;
LagCount = s.LagCount;
Frame = s.Frame;
- _overflowL = s.overflowL;
- _overflowR = s.overflowR;
- _latchLeft = s.LatchL;
- _latchRight = s.LatchR;
- _cableconnected = s.cableconnected;
- _cablediscosignal = s.cablediscosignal;
+ _cableConnected = s.CableConnected;
+ _cableDiscoSignal = s.CableDiscoSignal;
+ _cableShifted = s.CableShifted;
+ _cableShiftSignal = s.CableShiftSignal;
+ _cableSpaced = s.CableSpaced;
+ _cableSpaceSignal = s.CableSpaceSignal;
}
public void SaveStateBinary(BinaryWriter writer)
{
- L.SaveStateBinary(writer);
- R.SaveStateBinary(writer);
+ writer.Write(_numCores);
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i].SaveStateBinary(writer);
+ writer.Write(_linkedOverflow[i]);
+ writer.Write(_linkedLatches[i]);
+ }
// other variables
writer.Write(IsLagFrame);
writer.Write(LagCount);
writer.Write(Frame);
- writer.Write(_overflowL);
- writer.Write(_overflowR);
- writer.Write(_latchLeft);
- writer.Write(_latchRight);
- writer.Write(_cableconnected);
- writer.Write(_cablediscosignal);
+ writer.Write(_cableConnected);
+ writer.Write(_cableDiscoSignal);
+ writer.Write(_cableShifted);
+ writer.Write(_cableShiftSignal);
+ writer.Write(_cableSpaced);
+ writer.Write(_cableSpaceSignal);
}
public void LoadStateBinary(BinaryReader reader)
{
- L.LoadStateBinary(reader);
- R.LoadStateBinary(reader);
+ if (_numCores != reader.ReadInt32())
+ {
+ throw new InvalidOperationException("Core number mismatch!");
+ }
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i].LoadStateBinary(reader);
+ _linkedOverflow[i] = reader.ReadInt32();
+ _linkedLatches[i] = reader.ReadInt32();
+ }
// other variables
IsLagFrame = reader.ReadBoolean();
LagCount = reader.ReadInt32();
Frame = reader.ReadInt32();
- _overflowL = reader.ReadInt32();
- _overflowR = reader.ReadInt32();
- _latchLeft = reader.ReadInt32();
- _latchRight = reader.ReadInt32();
- _cableconnected = reader.ReadBoolean();
- _cablediscosignal = reader.ReadBoolean();
+ _cableConnected = reader.ReadBoolean();
+ _cableDiscoSignal = reader.ReadBoolean();
+ _cableShifted = reader.ReadBoolean();
+ _cableShiftSignal = reader.ReadBoolean();
+ _cableSpaced = reader.ReadBoolean();
+ _cableSpaceSignal = reader.ReadBoolean();
}
private readonly JsonSerializer ser = new JsonSerializer { Formatting = Formatting.Indented };
private class DGBSerialized
{
- public TextState L;
- public TextState R;
+ public int NumCores;
+ public TextState[] LinkedStates;
// other data
public bool IsLagFrame;
public int LagCount;
public int Frame;
- public int overflowL;
- public int overflowR;
- public int LatchL;
- public int LatchR;
- public bool cableconnected;
- public bool cablediscosignal;
+ public int[] LinkedOverflow;
+ public int[] LinkedLatches;
+ public bool CableConnected;
+ public bool CableDiscoSignal;
+ public bool CableShifted;
+ public bool CableShiftSignal;
+ public bool CableSpaced;
+ public bool CableSpaceSignal;
+
+ public DGBSerialized(GambatteLink linkcore)
+ {
+ NumCores = linkcore._numCores;
+ LinkedStates = new TextState[NumCores];
+ LinkedOverflow = new int[NumCores];
+ LinkedLatches = new int[NumCores];
+ for (int i = 0; i < NumCores; i++)
+ {
+ LinkedStates[i] = linkcore._linkedCores[i].SaveState();
+ LinkedOverflow[i] = linkcore._linkedOverflow[i];
+ LinkedLatches[i] = linkcore._linkedLatches[i];
+ }
+ IsLagFrame = linkcore.IsLagFrame;
+ LagCount = linkcore.LagCount;
+ Frame = linkcore.Frame;
+ CableConnected = linkcore._cableConnected;
+ CableDiscoSignal = linkcore._cableDiscoSignal;
+ CableShifted = linkcore._cableShifted;
+ CableShiftSignal = linkcore._cableShiftSignal;
+ CableSpaced = linkcore._cableSpaced;
+ CableSpaceSignal = linkcore._cableSpaceSignal;
+ }
}
}
}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IVideoProvider.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IVideoProvider.cs
index d80edf6f39..37d89f58f3 100644
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IVideoProvider.cs
+++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.IVideoProvider.cs
@@ -1,30 +1,32 @@
-using BizHawk.Emulation.Common;
+using System;
+
+using BizHawk.Emulation.Common;
namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
{
public partial class GambatteLink : IVideoProvider
{
- public int VirtualWidth => 320;
+ public int VirtualWidth => 160 * _numCores;
public int VirtualHeight => 144;
- public int BufferWidth => 320;
+ public int BufferWidth => 160 * _numCores;
public int BufferHeight => 144;
- public int VsyncNumerator => L.VsyncNumerator;
+ public int VsyncNumerator => _linkedCores[P1].VsyncNumerator;
- public int VsyncDenominator => L.VsyncDenominator;
+ public int VsyncDenominator => _linkedCores[P1].VsyncDenominator;
public int BackgroundColor => unchecked((int)0xff000000);
- private readonly int[] FrameBuffer = CreateVideoBuffer();
+ private readonly int[] FrameBuffer;
public int[] GetVideoBuffer() => VideoBuffer;
- private readonly int[] VideoBuffer = CreateVideoBuffer();
+ private readonly int[] VideoBuffer;
- private static int[] CreateVideoBuffer()
+ private int[] CreateVideoBuffer()
{
- var b = new int[160 * 2 * 144];
- for (int i = 0; i < (160 * 2 * 144); i++)
+ var b = new int[BufferWidth * BufferHeight];
+ for (int i = 0; i < b.Length; i++)
{
b[i] = -1; // GB/C screen is disabled on bootup, so it always starts as white, not black
}
diff --git a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.cs b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.cs
index 77d0cb3429..59a0f847e7 100644
--- a/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.cs
+++ b/src/BizHawk.Emulation.Cores/Consoles/Nintendo/Gameboy/GambatteLink.cs
@@ -1,4 +1,5 @@
using System;
+using System.Linq;
using BizHawk.Emulation.Common;
@@ -6,32 +7,42 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
{
[PortedCore(CoreNames.DualGambatte, "sinamas/natt")]
[ServiceNotApplicable(new[] { typeof(IDriveLight) })]
- public partial class GambatteLink : IEmulator, IVideoProvider, ISoundProvider, IInputPollable, ISaveRam, IStatable, ILinkable,
- IBoardInfo, IRomInfo, IDebuggable, ISettable, ICodeDataLogger
+ public partial class GambatteLink : ILinkable, IRomInfo
{
[CoreConstructor(VSystemID.Raw.DGB)]
public GambatteLink(CoreLoadParameters lp)
{
- if (lp.Roms.Count != 2)
+ if (lp.Roms.Count < MIN_PLAYERS || lp.Roms.Count > MAX_PLAYERS)
throw new InvalidOperationException("Wrong number of roms");
- ServiceProvider = new BasicServiceProvider(this);
- GambatteLinkSettings linkSettings = lp.Settings ?? new GambatteLinkSettings();
- GambatteLinkSyncSettings linkSyncSettings = lp.SyncSettings ?? new GambatteLinkSyncSettings();
+ _numCores = lp.Roms.Count;
- L = new Gameboy(lp.Comm, lp.Roms[0].Game, lp.Roms[0].RomData, linkSettings.L, linkSyncSettings.L, lp.DeterministicEmulationRequested);
- R = new Gameboy(lp.Comm, lp.Roms[1].Game, lp.Roms[1].RomData, linkSettings.R, linkSyncSettings.R, lp.DeterministicEmulationRequested);
+ _serviceProvider = new BasicServiceProvider(this);
+ _settings = lp.Settings ?? new GambatteLinkSettings();
+ _syncSettings = lp.SyncSettings ?? new GambatteLinkSyncSettings();
- // connect link cable
- LibGambatte.gambatte_linkstatus(L.GambatteState, 259);
- LibGambatte.gambatte_linkstatus(R.GambatteState, 259);
+ _linkedCores = new Gameboy[_numCores];
+ _linkedConts = new SaveController[_numCores];
+ _linkedBlips = new BlipBuffer[_numCores];
+ _linkedLatches = new int[_numCores];
+ _linkedOverflow = new int[_numCores];
- L.ConnectInputCallbackSystem(_inputCallbacks);
- R.ConnectInputCallbackSystem(_inputCallbacks);
- L.ConnectMemoryCallbackSystem(_memorycallbacks);
- R.ConnectMemoryCallbackSystem(_memorycallbacks);
+ RomDetails = "";
+ _memoryCallbacks = new MemoryCallbackSystem(new[] { "System Bus" });
- RomDetails = "LEFT:\r\n" + L.RomDetails + "RIGHT:\r\n" + R.RomDetails;
+ for (int i = 0; i < _numCores; i++)
+ {
+ _linkedCores[i] = new Gameboy(lp.Comm, lp.Roms[i].Game, lp.Roms[i].RomData, _settings._linkedSettings[i], _syncSettings._linkedSyncSettings[i], lp.DeterministicEmulationRequested);
+ LibGambatte.gambatte_linkstatus(_linkedCores[i].GambatteState, 259); // connect link cable
+ _linkedCores[i].ConnectInputCallbackSystem(_inputCallbacks);
+ _linkedCores[i].ConnectMemoryCallbackSystem(_memoryCallbacks);
+ _linkedConts[i] = new SaveController(Gameboy.CreateControllerDefinition(false, false));
+ _linkedBlips[i] = new BlipBuffer(1024);
+ _linkedBlips[i].SetRates(2097152 * 2, 44100);
+ _linkedOverflow[i] = 0;
+ _linkedLatches[i] = 0;
+ RomDetails += $"P{i + 1}:\r\n" + _linkedCores[i].RomDetails;
+ }
LinkConnected = true;
@@ -39,56 +50,99 @@ namespace BizHawk.Emulation.Cores.Nintendo.Gameboy
LagCount = 0;
IsLagFrame = false;
- _blipLeft = new BlipBuffer(1024);
- _blipRight = new BlipBuffer(1024);
- _blipLeft.SetRates(2097152 * 2, 44100);
- _blipRight.SetRates(2097152 * 2, 44100);
+ SoundBuffer = new short[MaxSampsPerFrame * _numCores];
- SetMemoryDomains();
+ FrameBuffer = CreateVideoBuffer();
+ VideoBuffer = CreateVideoBuffer();
+
+ GBLinkController = CreateControllerDefinition();
+
+ _linkedSaveRam = new LinkedSaveRam(_linkedCores, _numCores);
+ _serviceProvider.Register(_linkedSaveRam);
+
+ _linkedMemoryDomains = new LinkedMemoryDomains(_linkedCores, _numCores);
+ _serviceProvider.Register(_linkedMemoryDomains);
+
+ _linkedDebuggable = new LinkedDebuggable(_linkedCores, _numCores, _memoryCallbacks);
+ _serviceProvider.Register(_linkedDebuggable);
}
+ private readonly BasicServiceProvider _serviceProvider;
+
+ private readonly MemoryCallbackSystem _memoryCallbacks;
+
public string RomDetails { get; }
public bool LinkConnected
{
- get => _cableconnected;
- set => _cableconnected = value;
+ get => _cableConnected;
+ set => _cableConnected = value;
}
- private bool _disposed = false;
+ private int _numCores = 0;
+ private readonly Gameboy[] _linkedCores;
- private Gameboy L;
- private Gameboy R;
+ private readonly LinkedSaveRam _linkedSaveRam;
+ private readonly LinkedMemoryDomains _linkedMemoryDomains;
+ private readonly LinkedDebuggable _linkedDebuggable;
- // counter to ensure we do 35112 samples per frame
- private int _overflowL = 0;
- private int _overflowR = 0;
+ // counters to ensure we do 35112 samples per frame
+ private readonly int[] _linkedOverflow;
// if true, the link cable is currently connected
- private bool _cableconnected = true;
+ private bool _cableConnected = true;
+
+ // if true, the link cable is currently shifted (3x/4x only)
+ private bool _cableShifted = false;
+
+ // if true, the link cable is currently spaced outwards (3x/4x only)
+ private bool _cableSpaced = false;
// if true, the link cable toggle signal is currently asserted
- private bool _cablediscosignal = false;
+ private bool _cableDiscoSignal = false;
+
+ // if true, the link cable shift signal is currently asserted
+ private bool _cableShiftSignal = false;
+
+ // if true, the link cable spacing signal is currently asserted
+ private bool _cableSpaceSignal = false;
private const int SampPerFrame = 35112;
+ private const int MaxSampsPerFrame = (SampPerFrame + 2064) * 2;
- private readonly SaveController LCont = new SaveController(Gameboy.CreateControllerDefinition(false, false));
- private readonly SaveController RCont = new SaveController(Gameboy.CreateControllerDefinition(false, false));
+ private readonly SaveController[] _linkedConts;
- public bool IsCGBMode(bool right)
+ public bool IsCGBMode(int which)
{
- return right ? R.IsCGBMode() : L.IsCGBMode();
+ return which < _numCores && _linkedCores[which].IsCGBMode();
}
- private static readonly ControllerDefinition DualGbController = new ControllerDefinition
+ private ControllerDefinition GBLinkController { get; }
+
+ private ControllerDefinition CreateControllerDefinition()
{
- Name = "Dual Gameboy Controller",
- BoolButtons =
+ var ret = new ControllerDefinition { Name = $"GB Link {_numCores}x Controller" };
+ for (int i = 0; i < _numCores; i++)
{
- "P1 Up", "P1 Down", "P1 Left", "P1 Right", "P1 A", "P1 B", "P1 Select", "P1 Start", "P1 Power",
- "P2 Up", "P2 Down", "P2 Left", "P2 Right", "P2 A", "P2 B", "P2 Select", "P2 Start", "P2 Power",
- "Toggle Cable"
+ ret.BoolButtons.AddRange(
+ new[] { "Up", "Down", "Left", "Right", "A", "B", "Select", "Start", "Power" }
+ .Select(s => $"P{i + 1} {s}"));
}
- };
+ ret.BoolButtons.Add("Toggle Cable Connection");
+ if (_numCores > 2)
+ {
+ ret.BoolButtons.Add("Toggle Cable Shift");
+ ret.BoolButtons.Add("Toggle Cable Spacing");
+ }
+ return ret;
+ }
+
+ private const int P1 = 0;
+ private const int P2 = 1;
+ private const int P3 = 2;
+ private const int P4 = 3;
+
+ private const int MIN_PLAYERS = 2;
+ private const int MAX_PLAYERS = 4;
}
}