2024-12-25 01:03:00 +08:00
|
|
|
|
using AxibugEmuOnline.Client.ClientCore;
|
2024-12-11 21:21:27 +08:00
|
|
|
|
using System;
|
|
|
|
|
using System.Runtime.InteropServices;
|
|
|
|
|
using UnityEngine;
|
|
|
|
|
using UnityEngine.UI;
|
|
|
|
|
using VirtualNes.Core;
|
|
|
|
|
|
|
|
|
|
namespace AxibugEmuOnline.Client
|
|
|
|
|
{
|
|
|
|
|
public class VideoProvider : MonoBehaviour
|
|
|
|
|
{
|
|
|
|
|
#region UI_REF
|
|
|
|
|
public NesEmulator NesEmu;
|
|
|
|
|
public Canvas DrawCanvas;
|
|
|
|
|
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;
|
2024-12-25 01:03:00 +08:00
|
|
|
|
[SerializeField]
|
2024-12-11 21:21:27 +08:00
|
|
|
|
private Material GPUTurboMat_gpu;
|
2024-12-25 01:03:00 +08:00
|
|
|
|
private RenderTexture rt_gpu;
|
2024-12-11 21:21:27 +08:00
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
|
|
#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
|
|
|
|
|
|
|
|
|
|
public bool GPUTurbo = true;
|
|
|
|
|
|
|
|
|
|
private void Awake()
|
|
|
|
|
{
|
|
|
|
|
DrawCanvas.worldCamera = Camera.main;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void OnDestroy()
|
|
|
|
|
{
|
|
|
|
|
if (wrapTexBufferGH.IsAllocated)
|
|
|
|
|
wrapTexBufferGH.Free();
|
2024-12-25 01:03:00 +08:00
|
|
|
|
|
|
|
|
|
if (rt_gpu != null)
|
|
|
|
|
{
|
|
|
|
|
RenderTexture.ReleaseTemporary(rt_gpu);
|
|
|
|
|
rt_gpu = null;
|
|
|
|
|
}
|
2024-12-11 21:21:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public unsafe void SetDrawData(uint* screenData)
|
|
|
|
|
{
|
|
|
|
|
PrepareUI(screenData);
|
|
|
|
|
if (GPUTurbo) PrepareForGPU(screenData);//<2F>ж<EFBFBD>ʹ<EFBFBD><CAB9>GPU<50><55><EFBFBD><EFBFBD>CPU
|
|
|
|
|
else PrepareForCPU(screenData);//ʹ<><CAB9>CPU
|
|
|
|
|
|
|
|
|
|
if (GPUTurbo)
|
|
|
|
|
{
|
|
|
|
|
wrapTex_gpu.LoadRawTextureData(wrapTexBufferPointer_gpu, TexBufferSize_gpu);
|
|
|
|
|
wrapTex_gpu.Apply();
|
2024-12-25 01:03:00 +08:00
|
|
|
|
Graphics.Blit(wrapTex_gpu, rt_gpu, GPUTurboMat_gpu);
|
2024-12-11 21:21:27 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
wrapTex_cpu.LoadRawTextureData(wrapTexBufferPointer_cpu, TexBufferSize_cpu);
|
|
|
|
|
wrapTex_cpu.Apply();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-25 01:03:00 +08:00
|
|
|
|
public void ApplyFilterEffect()
|
|
|
|
|
{
|
|
|
|
|
Image.texture = App.filter.ExecuteFilterRender(rt_gpu);
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-11 21:21:27 +08:00
|
|
|
|
private unsafe void PrepareUI(uint* screenData)
|
|
|
|
|
{
|
|
|
|
|
if (GPUTurbo)
|
|
|
|
|
{
|
|
|
|
|
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-12-25 01:03:00 +08:00
|
|
|
|
rt_gpu = RenderTexture.GetTemporary(wrapTex_gpu.width, wrapTex_gpu.height, 0);
|
|
|
|
|
rt_gpu.filterMode = FilterMode.Point;
|
|
|
|
|
rt_gpu.anisoLevel = 0;
|
|
|
|
|
rt_gpu.antiAliasing = 1;
|
2024-12-11 21:21:27 +08:00
|
|
|
|
|
|
|
|
|
TexBufferSize_gpu = wrapTex_gpu.width * wrapTex_gpu.height * 4;
|
|
|
|
|
}
|
|
|
|
|
|
2024-12-25 01:03:00 +08:00
|
|
|
|
if (Image.texture != rt_gpu) Image.texture = rt_gpu;
|
2024-12-11 21:21:27 +08:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (wrapTex_cpu == null)
|
|
|
|
|
{
|
|
|
|
|
wrapTex_cpu = new Texture2D(PPU.SCREEN_WIDTH - 16, PPU.SCREEN_HEIGHT, TextureFormat.RGBA32, false);
|
|
|
|
|
wrapTex_cpu.filterMode = FilterMode.Point;
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
2024-12-25 01:03:00 +08:00
|
|
|
|
|
2024-12-11 21:21:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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++)
|
|
|
|
|
{
|
|
|
|
|
uint colorRaw = palRaw[i];
|
|
|
|
|
var argbColor = BitConverter.GetBytes(colorRaw);
|
|
|
|
|
Color temp = Color.white;
|
|
|
|
|
temp.r = argbColor[2] / 255f;
|
|
|
|
|
temp.g = argbColor[1] / 255f;
|
|
|
|
|
temp.b = argbColor[0] / 255f;
|
|
|
|
|
temp.a = 1;
|
|
|
|
|
pPal_gpu.SetPixel(i, 0, temp);
|
|
|
|
|
}
|
|
|
|
|
pPal_gpu.Apply();
|
|
|
|
|
GPUTurboMat_gpu.SetTexture("_PalTex", pPal_gpu);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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];
|
|
|
|
|
|
|
|
|
|
for (int line = 0; line < PPU.SCREEN_HEIGHT; line++)
|
|
|
|
|
{
|
|
|
|
|
//PS<50><53><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>CPU<50><55><EFBFBD>㣬<EFBFBD><E3A3AC><EFBFBD>ȼ<EFBFBD><C8BC><EFBFBD>16<31>IJ<EFBFBD><C4B2><EFBFBD>Ҫ<EFBFBD><D2AA><EFBFBD>֣<EFBFBD><D6A3><EFBFBD><EFBFBD>ܶ<EFBFBD><DCB6><EFBFBD>
|
|
|
|
|
width = PPU.SCREEN_WIDTH - 16;
|
|
|
|
|
|
|
|
|
|
while (width > 0)
|
|
|
|
|
{
|
|
|
|
|
var edx = screenData[pScn + 8];
|
|
|
|
|
|
|
|
|
|
uint index = edx & 0xFF;
|
|
|
|
|
//<2F><><EFBFBD>±<EFBFBD><C2B1><EFBFBD>ɫ<EFBFBD><C9AB><EFBFBD>ұ<EFBFBD><D2B1><EFBFBD><EFBFBD><EFBFBD>ʵ<EFBFBD><CAB5>ɫ
|
|
|
|
|
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>
|
|
|
|
|
|
|
|
|
|
//<2F><>rgb<67><62><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ɫ<EFBFBD><C9AB><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>unity <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD><EFBFBD>rgb <20><><EFBFBD><EFBFBD>System.Drawing.Color <20><><EFBFBD><EFBFBD><EFBFBD><EFBFBD>ɫ<EFBFBD><C9AB><EFBFBD><EFBFBD><EFBFBD><EFBFBD>
|
|
|
|
|
uint abgr = 0xFF000000 | (b << 16) | (g << 8) | (r << 0);
|
|
|
|
|
|
|
|
|
|
//<2F>Ž<EFBFBD><C5BD><EFBFBD>ɫ<EFBFBD><C9AB><EFBFBD><EFBFBD>
|
|
|
|
|
Dst[pDst] = abgr;
|
|
|
|
|
|
|
|
|
|
pScn += 1;
|
|
|
|
|
pDst += 1;
|
|
|
|
|
width -= 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pScn += 16;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|