forked from sin365/AxibugEmuOnline
169 lines
4.5 KiB
C#
169 lines
4.5 KiB
C#
using AxibugEmuOnline.Client.ClientCore;
|
||
using System;
|
||
using System.Diagnostics;
|
||
using System.IO;
|
||
using System.Xml.Linq;
|
||
using UnityEngine;
|
||
using VirtualNes.Core;
|
||
using VirtualNes.Core.Debug;
|
||
|
||
namespace AxibugEmuOnline.Client
|
||
{
|
||
public class NesEmulator : MonoBehaviour, IEmuCore
|
||
{
|
||
public NES NesCore { get; private set; }
|
||
|
||
public VideoProvider VideoProvider;
|
||
public AudioProvider AudioProvider;
|
||
public bool m_bPause;
|
||
|
||
private void Start()
|
||
{
|
||
Application.targetFrameRate = 60;
|
||
VideoProvider.NesEmu = this;
|
||
AudioProvider.NesEmu = this;
|
||
}
|
||
|
||
public void StartGame(RomFile rom)
|
||
{
|
||
StopGame();
|
||
|
||
Supporter.Setup(new CoreSupporter());
|
||
Debuger.Setup(new CoreDebuger());
|
||
|
||
App.nesRomLib.AddRomFile(rom);
|
||
|
||
try
|
||
{
|
||
NesCore = new NES(rom.FileName);
|
||
}
|
||
catch (Exception ex)
|
||
{
|
||
NesCore = null;
|
||
App.log.Error(ex.ToString());
|
||
}
|
||
}
|
||
|
||
public void StopGame()
|
||
{
|
||
NesCore?.Dispose();
|
||
NesCore = null;
|
||
}
|
||
|
||
private void Update()
|
||
{
|
||
if (m_bPause) return;
|
||
|
||
if (NesCore != null)
|
||
{
|
||
PushEmulatorFrame();
|
||
if (InGameUI.Instance.IsNetPlay)
|
||
FixEmulatorFrame();
|
||
|
||
var screenBuffer = NesCore.ppu.GetScreenPtr();
|
||
var lineColorMode = NesCore.ppu.GetLineColorMode();
|
||
VideoProvider.SetDrawData(screenBuffer, lineColorMode, 256, 240);
|
||
}
|
||
}
|
||
|
||
private void FixEmulatorFrame()
|
||
{
|
||
var skipFrameCount = App.roomMgr.netReplay.GetSkipFrameCount();
|
||
|
||
if (skipFrameCount > 0) App.log.Debug($"SKIP FRAME : {skipFrameCount}");
|
||
for (int i = 0; i < skipFrameCount; i++)
|
||
{
|
||
if (!PushEmulatorFrame()) break;
|
||
}
|
||
}
|
||
|
||
ControllerState lastState;
|
||
private bool PushEmulatorFrame()
|
||
{
|
||
Supporter.SampleInput(NesCore.FrameCount);
|
||
var controlState = Supporter.GetControllerState();
|
||
|
||
//<2F><><EFBFBD><EFBFBD>δ<EFBFBD>յ<EFBFBD>Input<75><74><EFBFBD><EFBFBD>,<2C><><EFBFBD><EFBFBD>֡<EFBFBD><D6A1><EFBFBD>ƽ<EFBFBD>
|
||
if (!controlState.valid) return false;
|
||
|
||
#if UNITY_EDITOR
|
||
if (controlState != lastState)
|
||
{
|
||
App.log.Info($"[LOCALDEBUG]{NesCore.FrameCount}-->{controlState}");
|
||
}
|
||
#endif
|
||
|
||
NesCore.pad.Sync(controlState);
|
||
NesCore.EmulateFrame(true);
|
||
|
||
lastState = controlState;
|
||
|
||
return true;
|
||
}
|
||
|
||
public void Pause()
|
||
{
|
||
m_bPause = true;
|
||
}
|
||
|
||
public void Resume()
|
||
{
|
||
m_bPause = false;
|
||
}
|
||
|
||
|
||
|
||
[Conditional("UNITY_EDITOR")]
|
||
[ContextMenu("ImportNesDB")]
|
||
public void ImportNesDB()
|
||
{
|
||
var db = Resources.Load<RomDB>("NES/ROMDB");
|
||
db.Clear();
|
||
|
||
var xmlStr = File.ReadAllText("nes20db.xml");
|
||
var xml = XDocument.Parse(xmlStr);
|
||
var games = xml.Element("nes20db").Elements("game");
|
||
foreach (var game in games)
|
||
{
|
||
var crcStr = game.Element("rom").Attribute("crc32").Value;
|
||
var crc = uint.Parse($"{crcStr}", System.Globalization.NumberStyles.HexNumber);
|
||
|
||
var mapper = int.Parse($"{game.Element("pcb").Attribute("mapper").Value}");
|
||
|
||
if (mapper > 255) continue;
|
||
db.AddInfo(new RomDB.RomInfo { CRC = crc, Mapper = mapper });
|
||
}
|
||
|
||
UnityEditor.EditorUtility.SetDirty(db);
|
||
UnityEditor.AssetDatabase.SaveAssets();
|
||
}
|
||
|
||
public void SetupScheme()
|
||
{
|
||
ControlScheme.Current = ControlSchemeSetts.NES;
|
||
}
|
||
|
||
public void LoadState(object state)
|
||
{
|
||
NesCore.LoadState((State)state);
|
||
}
|
||
|
||
public object GetState()
|
||
{
|
||
return NesCore.GetState();
|
||
}
|
||
|
||
public byte[] GetStateBytes()
|
||
{
|
||
return NesCore.GetState().ToBytes();
|
||
}
|
||
|
||
public void LoadStateFromBytes(byte[] data)
|
||
{
|
||
State st = new State();
|
||
st.FromByte(data);
|
||
NesCore.LoadState(st);
|
||
}
|
||
}
|
||
}
|