AxibugEmuOnline/AxibugEmuOnline.Client/Assets/Script/NesEmulator/VideoProvider.cs

174 lines
5.7 KiB
C#
Raw Normal View History

using AxibugEmuOnline.Client.Common;
2024-07-30 18:53:36 +08:00
using System;
2024-07-31 17:40:32 +08:00
using System.Runtime.InteropServices;
using System.Text;
2024-07-30 11:57:09 +08:00
using UnityEngine;
using UnityEngine.UI;
using VirtualNes.Core;
using static UnityEngine.UI.CanvasScaler;
2024-07-30 11:57:09 +08:00
namespace AxibugEmuOnline.Client
{
public class VideoProvider : MonoBehaviour
{
#region UI_REF
2024-08-06 16:03:17 +08:00
public NesEmulator NesEmu;
public Canvas DrawCanvas;
2024-07-30 11:57:09 +08:00
public RawImage Image;
#endregion
#region GPU_TURBO
//ͼ<><CDBC><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ֽ<EFBFBD><D6BD><EFBFBD>
private int TexBufferSize_gpu;
//ͼ<><CDBC><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ָ<EFBFBD><D6B8>
private IntPtr wrapTexBufferPointer_gpu;
//Unity 2D<32><44><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>,<2C><><EFBFBD><EFBFBD>UI<55>ϻ<EFBFBD><CFBB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
private Texture2D wrapTex_gpu;
//nes<65><73>ɫ<EFBFBD><C9AB><EFBFBD><EFBFBD><EFBFBD><EFBFBD>,<2C><>ת<EFBFBD><D7AA>Ϊunity<74><79><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
private Texture2D pPal_gpu;
private Material GPUTurboMat_gpu;
#endregion
2024-07-30 11:57:09 +08:00
#region CPU
//ͼ<><CDBC><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ֽ<EFBFBD><D6BD><EFBFBD>
private int TexBufferSize_cpu;
//ͼ<><CDBC><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ָ<EFBFBD><D6B8>
private GCHandle wrapTexBufferGH;
private IntPtr wrapTexBufferPointer_cpu;
//Unity 2D<32><44><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>,<2C><><EFBFBD><EFBFBD>UI<55>ϻ<EFBFBD><CFBB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
private Texture2D wrapTex_cpu;
#endregion
2024-07-30 11:57:09 +08:00
public bool GPUTurbo = true;
2024-08-29 17:20:01 +08:00
private void Awake()
{
DrawCanvas.worldCamera = Camera.main;
GPUTurboMat_gpu = Image.material;
}
private void OnDestroy()
{
if (wrapTexBufferGH.IsAllocated)
wrapTexBufferGH.Free();
}
public unsafe void SetDrawData(uint* screenData)
2024-07-30 11:57:09 +08:00
{
PrepareUI(screenData);
if (GPUTurbo) PrepareForGPU(screenData);
else PrepareForCPU(screenData);
if (GPUTurbo)
{
wrapTex_gpu.LoadRawTextureData(wrapTexBufferPointer_gpu, TexBufferSize_gpu);
wrapTex_gpu.Apply();
}
else
{
wrapTex_cpu.LoadRawTextureData(wrapTexBufferPointer_cpu, TexBufferSize_cpu);
wrapTex_cpu.Apply();
}
}
private unsafe void PrepareUI(uint* screenData)
{
if (GPUTurbo)
2024-07-31 17:40:32 +08:00
{
if (Image.material != GPUTurboMat_gpu) Image.material = GPUTurboMat_gpu;
if (wrapTex_gpu == null)
{
wrapTex_gpu = new Texture2D(PPU.SCREEN_WIDTH, PPU.SCREEN_HEIGHT, TextureFormat.RGBA32, false);
wrapTex_gpu.filterMode = FilterMode.Point;
wrapTexBufferPointer_gpu = (IntPtr)screenData;
2024-08-14 13:09:22 +08:00
TexBufferSize_gpu = wrapTex_gpu.width * wrapTex_gpu.height * 4;
}
if (Image.texture != wrapTex_gpu) Image.texture = wrapTex_gpu;
}
else
{
if (Image.material == GPUTurboMat_gpu) Image.material = null;
if (wrapTex_cpu == null)
{
wrapTex_cpu = new Texture2D(PPU.SCREEN_WIDTH - 16, PPU.SCREEN_HEIGHT, TextureFormat.RGBA32, false);
wrapTex_cpu.filterMode = FilterMode.Point;
2024-07-30 18:53:36 +08:00
uint[] cpuTexBuffer = new uint[wrapTex_cpu.width * wrapTex_cpu.height];
wrapTexBufferGH = GCHandle.Alloc(cpuTexBuffer, GCHandleType.Pinned);
wrapTexBufferPointer_cpu = wrapTexBufferGH.AddrOfPinnedObject();
TexBufferSize_cpu = cpuTexBuffer.Length * 4;
}
if (Image.texture != wrapTex_cpu) Image.texture = wrapTex_cpu;
}
}
private unsafe void PrepareForGPU(uint* screenData)
{
if (pPal_gpu == null)
{
var palRaw = PaletteDefine.m_cnPalette[0];
pPal_gpu = new Texture2D(palRaw.Length, 1, TextureFormat.RGBA32, false);
pPal_gpu.filterMode = FilterMode.Point;
for (int i = 0; i < palRaw.Length; i++)
2024-07-30 18:53:36 +08:00
{
uint colorRaw = palRaw[i];
var argbColor = BitConverter.GetBytes(colorRaw);
Color temp = Color.white;
2024-08-14 13:09:22 +08:00
temp.r = argbColor[2] / 255f;
temp.g = argbColor[1] / 255f;
temp.b = argbColor[0] / 255f;
temp.a = 1;
pPal_gpu.SetPixel(i, 0, temp);
2024-07-30 18:53:36 +08:00
}
pPal_gpu.Apply();
GPUTurboMat_gpu.SetTexture("_PalTex", pPal_gpu);
2024-07-30 18:53:36 +08:00
}
}
private unsafe void PrepareForCPU(uint* screenData)
{
int pScn = 0;
int width;
var Dst = (uint*)wrapTexBufferPointer_cpu;
var pDst = 0;
var palRaw = PaletteDefine.m_cnPalette[0];
2024-07-30 18:53:36 +08:00
for (int line = 0; line < PPU.SCREEN_HEIGHT; line++)
{
width = PPU.SCREEN_WIDTH - 16;
while (width > 0)
{
var edx = screenData[pScn + 8];
uint index = edx & 0xFF;
var colorData = palRaw[index];
//dst<73><74><EFBFBD><EFBFBD>ɫ<EFBFBD><C9AB><EFBFBD><EFBFBD>Ϊabgr,<2C><>colorData<74><61><EFBFBD><EFBFBD>Ϊargb
uint r = (colorData & 0x00FF0000) >> 16; // <20><>ȡRedͨ<64><CDA8>
uint g = (colorData & 0x0000FF00) >> 8; // <20><>ȡGreenͨ<6E><CDA8>
uint b = (colorData & 0x000000FF); // <20><>ȡBlueͨ<65><CDA8>
uint abgr = 0xFF000000 | (b << 16) | (g << 8) | (r << 0);
Dst[pDst] = abgr;
pScn += 1;
pDst += 1;
width -= 1;
}
pScn += 16;
}
2024-07-30 11:57:09 +08:00
}
}
}