GBA.Unity/Assets/Iris/Iris.GBA/Video.cs
2024-08-14 16:02:39 +08:00

1347 lines
47 KiB
C#

using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
namespace Iris.GBA
{
internal sealed class Video : IDisposable
{
internal enum Register
{
DISPCNT,
DISPSTAT,
VCOUNT,
BG0CNT,
BG1CNT,
BG2CNT,
BG3CNT,
BG0HOFS,
BG0VOFS,
BG1HOFS,
BG1VOFS,
BG2HOFS,
BG2VOFS,
BG3HOFS,
BG3VOFS,
BG2PA,
BG2PB,
BG2PC,
BG2PD,
BG2X_L,
BG2X_H,
BG2Y_L,
BG2Y_H,
BG3PA,
BG3PB,
BG3PC,
BG3PD,
BG3X_L,
BG3X_H,
BG3Y_L,
BG3Y_H,
WIN0H,
WIN1H,
WIN0V,
WIN1V,
WININ,
WINOUT,
MOSAIC,
BLDCNT,
BLDALPHA,
BLDY
}
private const int KB = 1024;
private const int PaletteRAM_Size = 1 * KB;
private const int VRAM_Size = 96 * KB;
private const int OAM_Size = 1 * KB;
private readonly IntPtr _paletteRAM = Marshal.AllocHGlobal(PaletteRAM_Size);
private readonly IntPtr _vram = Marshal.AllocHGlobal(VRAM_Size);
private readonly IntPtr _oam = Marshal.AllocHGlobal(OAM_Size);
private const UInt32 PaletteRAM_StartAddress = 0x0500_0000;
private const UInt32 PaletteRAM_EndAddress = 0x0600_0000;
private const UInt32 VRAM_StartAddress = 0x0600_0000;
private const UInt32 VRAM_EndAddress = 0x0700_0000;
private const UInt32 OAM_StartAddress = 0x0700_0000;
private const UInt32 OAM_EndAddress = 0x0800_0000;
private UInt16 _DISPCNT;
private UInt16 _DISPSTAT;
private UInt16 _VCOUNT;
private UInt16 _BG0CNT;
private UInt16 _BG1CNT;
private UInt16 _BG2CNT;
private UInt16 _BG3CNT;
private UInt16 _BG0HOFS;
private UInt16 _BG0VOFS;
private UInt16 _BG1HOFS;
private UInt16 _BG1VOFS;
private UInt16 _BG2HOFS;
private UInt16 _BG2VOFS;
private UInt16 _BG3HOFS;
private UInt16 _BG3VOFS;
private UInt16 _BG2PA;
private UInt16 _BG2PB;
private UInt16 _BG2PC;
private UInt16 _BG2PD;
private UInt16 _BG2X_L;
private UInt16 _BG2X_H;
private UInt16 _BG2Y_L;
private UInt16 _BG2Y_H;
private UInt16 _BG3PA;
private UInt16 _BG3PB;
private UInt16 _BG3PC;
private UInt16 _BG3PD;
private UInt16 _BG3X_L;
private UInt16 _BG3X_H;
private UInt16 _BG3Y_L;
private UInt16 _BG3Y_H;
private UInt16 _WIN0H;
private UInt16 _WIN1H;
private UInt16 _WIN0V;
private UInt16 _WIN1V;
private UInt16 _WININ;
private UInt16 _WINOUT;
private UInt16 _MOSAIC;
private UInt16 _BLDCNT;
private UInt16 _BLDALPHA;
private UInt16 _BLDY;
private const int DisplayScreenWidth = 240;
private const int DisplayScreenHeight = 160;
private const int DisplayScreenSize = DisplayScreenWidth * DisplayScreenHeight;
private const int CharacterWidth = 8;
private const int CharacterHeight = 8;
private const UInt64 HDrawCycleCount = 1006;
private const UInt64 HBlankCycleCount = 226;
private readonly Common.Scheduler _scheduler;
private readonly Common.System.PresentFrame_Delegate _presentFrameCallback;
private DMA _dma;
private InterruptControl _interruptControl;
private bool _disposed;
private readonly UInt16[] _displayFrameBuffer = new UInt16[DisplayScreenSize];
private Int32 _currentBG2X;
private Int32 _currentBG2Y;
private Int32 _currentBG3X;
private Int32 _currentBG3Y;
internal Video(Common.Scheduler scheduler, Common.System.PresentFrame_Delegate presentFrameCallback)
{
_scheduler = scheduler;
_presentFrameCallback = presentFrameCallback;
_scheduler.RegisterTask((int)GBA_System.TaskId.StartHBlank, StartHBlank);
_scheduler.RegisterTask((int)GBA_System.TaskId.StartScanline, StartScanline);
}
~Video()
{
Dispose();
}
public void Dispose()
{
if (_disposed)
return;
Marshal.FreeHGlobal(_paletteRAM);
Marshal.FreeHGlobal(_vram);
Marshal.FreeHGlobal(_oam);
GC.SuppressFinalize(this);
_disposed = true;
}
internal void Initialize(DMA dma, InterruptControl interruptControl, Memory memory)
{
_dma = dma;
_interruptControl = interruptControl;
const Memory.Flag flags = Memory.Flag.All & ~Memory.Flag.Write8;
memory.Map(_paletteRAM, PaletteRAM_Size, PaletteRAM_StartAddress, PaletteRAM_EndAddress, flags);
memory.Map(_vram, VRAM_Size, VRAM_StartAddress, VRAM_EndAddress, flags);
memory.Map(_oam, OAM_Size, OAM_StartAddress, OAM_EndAddress, flags);
}
internal void ResetState()
{
byte[] paletteRamData = new byte[PaletteRAM_Size];
byte[] vramData = new byte[VRAM_Size];
byte[] oamData = new byte[OAM_Size];
Marshal.Copy(paletteRamData, 0, _paletteRAM, PaletteRAM_Size);
Marshal.Copy(vramData, 0, _vram, VRAM_Size);
Marshal.Copy(oamData, 0, _oam, OAM_Size);
_DISPCNT = 0;
_DISPSTAT = 0;
_VCOUNT = 0;
_BG0CNT = 0;
_BG1CNT = 0;
_BG2CNT = 0;
_BG3CNT = 0;
_BG0HOFS = 0;
_BG0VOFS = 0;
_BG1HOFS = 0;
_BG1VOFS = 0;
_BG2HOFS = 0;
_BG2VOFS = 0;
_BG3HOFS = 0;
_BG3VOFS = 0;
_BG2PA = 0;
_BG2PB = 0;
_BG2PC = 0;
_BG2PD = 0;
_BG2X_L = 0;
_BG2X_H = 0;
_BG2Y_L = 0;
_BG2Y_H = 0;
_BG3PA = 0;
_BG3PB = 0;
_BG3PC = 0;
_BG3PD = 0;
_BG3X_L = 0;
_BG3X_H = 0;
_BG3Y_L = 0;
_BG3Y_H = 0;
_WIN0H = 0;
_WIN1H = 0;
_WIN0V = 0;
_WIN1V = 0;
_WININ = 0;
_WINOUT = 0;
_MOSAIC = 0;
_BLDCNT = 0;
_BLDALPHA = 0;
_BLDY = 0;
_scheduler.ScheduleTask((int)GBA_System.TaskId.StartHBlank, HDrawCycleCount);
Array.Clear(_displayFrameBuffer);
_currentBG2X = 0;
_currentBG2Y = 0;
_currentBG3X = 0;
_currentBG3Y = 0;
}
internal void LoadState(BinaryReader reader)
{
byte[] paletteRamData = reader.ReadBytes(PaletteRAM_Size);
byte[] vramData = reader.ReadBytes(VRAM_Size);
byte[] oamData = reader.ReadBytes(OAM_Size);
Marshal.Copy(paletteRamData, 0, _paletteRAM, PaletteRAM_Size);
Marshal.Copy(vramData, 0, _vram, VRAM_Size);
Marshal.Copy(oamData, 0, _oam, OAM_Size);
_DISPCNT = reader.ReadUInt16();
_DISPSTAT = reader.ReadUInt16();
_VCOUNT = reader.ReadUInt16();
_BG0CNT = reader.ReadUInt16();
_BG1CNT = reader.ReadUInt16();
_BG2CNT = reader.ReadUInt16();
_BG3CNT = reader.ReadUInt16();
_BG0HOFS = reader.ReadUInt16();
_BG0VOFS = reader.ReadUInt16();
_BG1HOFS = reader.ReadUInt16();
_BG1VOFS = reader.ReadUInt16();
_BG2HOFS = reader.ReadUInt16();
_BG2VOFS = reader.ReadUInt16();
_BG3HOFS = reader.ReadUInt16();
_BG3VOFS = reader.ReadUInt16();
_BG2PA = reader.ReadUInt16();
_BG2PB = reader.ReadUInt16();
_BG2PC = reader.ReadUInt16();
_BG2PD = reader.ReadUInt16();
_BG2X_L = reader.ReadUInt16();
_BG2X_H = reader.ReadUInt16();
_BG2Y_L = reader.ReadUInt16();
_BG2Y_H = reader.ReadUInt16();
_BG3PA = reader.ReadUInt16();
_BG3PB = reader.ReadUInt16();
_BG3PC = reader.ReadUInt16();
_BG3PD = reader.ReadUInt16();
_BG3X_L = reader.ReadUInt16();
_BG3X_H = reader.ReadUInt16();
_BG3Y_L = reader.ReadUInt16();
_BG3Y_H = reader.ReadUInt16();
_WIN0H = reader.ReadUInt16();
_WIN1H = reader.ReadUInt16();
_WIN0V = reader.ReadUInt16();
_WIN1V = reader.ReadUInt16();
_WININ = reader.ReadUInt16();
_WINOUT = reader.ReadUInt16();
_MOSAIC = reader.ReadUInt16();
_BLDCNT = reader.ReadUInt16();
_BLDALPHA = reader.ReadUInt16();
_BLDY = reader.ReadUInt16();
foreach (ref UInt16 color in _displayFrameBuffer.AsSpan())
color = reader.ReadUInt16();
_currentBG2X = reader.ReadInt32();
_currentBG2Y = reader.ReadInt32();
_currentBG3X = reader.ReadInt32();
_currentBG3Y = reader.ReadInt32();
}
internal void SaveState(BinaryWriter writer)
{
byte[] paletteRamData = new byte[PaletteRAM_Size];
byte[] vramData = new byte[VRAM_Size];
byte[] oamData = new byte[OAM_Size];
Marshal.Copy(_paletteRAM, paletteRamData, 0, PaletteRAM_Size);
Marshal.Copy(_vram, vramData, 0, VRAM_Size);
Marshal.Copy(_oam, oamData, 0, OAM_Size);
writer.Write(paletteRamData);
writer.Write(vramData);
writer.Write(oamData);
writer.Write(_DISPCNT);
writer.Write(_DISPSTAT);
writer.Write(_VCOUNT);
writer.Write(_BG0CNT);
writer.Write(_BG1CNT);
writer.Write(_BG2CNT);
writer.Write(_BG3CNT);
writer.Write(_BG0HOFS);
writer.Write(_BG0VOFS);
writer.Write(_BG1HOFS);
writer.Write(_BG1VOFS);
writer.Write(_BG2HOFS);
writer.Write(_BG2VOFS);
writer.Write(_BG3HOFS);
writer.Write(_BG3VOFS);
writer.Write(_BG2PA);
writer.Write(_BG2PB);
writer.Write(_BG2PC);
writer.Write(_BG2PD);
writer.Write(_BG2X_L);
writer.Write(_BG2X_H);
writer.Write(_BG2Y_L);
writer.Write(_BG2Y_H);
writer.Write(_BG3PA);
writer.Write(_BG3PB);
writer.Write(_BG3PC);
writer.Write(_BG3PD);
writer.Write(_BG3X_L);
writer.Write(_BG3X_H);
writer.Write(_BG3Y_L);
writer.Write(_BG3Y_H);
writer.Write(_WIN0H);
writer.Write(_WIN1H);
writer.Write(_WIN0V);
writer.Write(_WIN1V);
writer.Write(_WININ);
writer.Write(_WINOUT);
writer.Write(_MOSAIC);
writer.Write(_BLDCNT);
writer.Write(_BLDALPHA);
writer.Write(_BLDY);
foreach (UInt16 color in _displayFrameBuffer)
writer.Write(color);
writer.Write(_currentBG2X);
writer.Write(_currentBG2Y);
writer.Write(_currentBG3X);
writer.Write(_currentBG3Y);
}
internal UInt16 ReadRegister(Register register)
{
return register switch
{
Register.DISPCNT => _DISPCNT,
Register.DISPSTAT => _DISPSTAT,
Register.VCOUNT => _VCOUNT,
Register.BG0CNT => _BG0CNT,
Register.BG1CNT => _BG1CNT,
Register.BG2CNT => _BG2CNT,
Register.BG3CNT => _BG3CNT,
Register.WININ => _WININ,
Register.WINOUT => _WINOUT,
Register.BLDCNT => _BLDCNT,
Register.BLDALPHA => _BLDALPHA,
// should never happen
_ => throw new Exception("Iris.GBA.Video: Register read error"),
};
}
internal void WriteRegister(Register register, UInt16 value, Memory.RegisterWriteMode mode)
{
switch (register)
{
case Register.DISPCNT:
Memory.WriteRegisterHelper(ref _DISPCNT, value, mode);
break;
case Register.DISPSTAT:
Memory.WriteRegisterHelper(ref _DISPSTAT, value, mode);
break;
case Register.BG0CNT:
Memory.WriteRegisterHelper(ref _BG0CNT, value, mode);
break;
case Register.BG1CNT:
Memory.WriteRegisterHelper(ref _BG1CNT, value, mode);
break;
case Register.BG2CNT:
Memory.WriteRegisterHelper(ref _BG2CNT, value, mode);
break;
case Register.BG3CNT:
Memory.WriteRegisterHelper(ref _BG3CNT, value, mode);
break;
case Register.BG0HOFS:
Memory.WriteRegisterHelper(ref _BG0HOFS, value, mode);
break;
case Register.BG0VOFS:
Memory.WriteRegisterHelper(ref _BG0VOFS, value, mode);
break;
case Register.BG1HOFS:
Memory.WriteRegisterHelper(ref _BG1HOFS, value, mode);
break;
case Register.BG1VOFS:
Memory.WriteRegisterHelper(ref _BG1VOFS, value, mode);
break;
case Register.BG2HOFS:
Memory.WriteRegisterHelper(ref _BG2HOFS, value, mode);
break;
case Register.BG2VOFS:
Memory.WriteRegisterHelper(ref _BG2VOFS, value, mode);
break;
case Register.BG3HOFS:
Memory.WriteRegisterHelper(ref _BG3HOFS, value, mode);
break;
case Register.BG3VOFS:
Memory.WriteRegisterHelper(ref _BG3VOFS, value, mode);
break;
case Register.BG2PA:
Memory.WriteRegisterHelper(ref _BG2PA, value, mode);
break;
case Register.BG2PB:
Memory.WriteRegisterHelper(ref _BG2PB, value, mode);
break;
case Register.BG2PC:
Memory.WriteRegisterHelper(ref _BG2PC, value, mode);
break;
case Register.BG2PD:
Memory.WriteRegisterHelper(ref _BG2PD, value, mode);
break;
case Register.BG2X_L:
Memory.WriteRegisterHelper(ref _BG2X_L, value, mode);
_currentBG2X = ((_BG2X_H << 20) | (_BG2X_L << 4)) >> 4;
break;
case Register.BG2X_H:
Memory.WriteRegisterHelper(ref _BG2X_H, value, mode);
_currentBG2X = ((_BG2X_H << 20) | (_BG2X_L << 4)) >> 4;
break;
case Register.BG2Y_L:
Memory.WriteRegisterHelper(ref _BG2Y_L, value, mode);
_currentBG2Y = ((_BG2Y_H << 20) | (_BG2Y_L << 4)) >> 4;
break;
case Register.BG2Y_H:
Memory.WriteRegisterHelper(ref _BG2Y_H, value, mode);
_currentBG2Y = ((_BG2Y_H << 20) | (_BG2Y_L << 4)) >> 4;
break;
case Register.BG3PA:
Memory.WriteRegisterHelper(ref _BG3PA, value, mode);
break;
case Register.BG3PB:
Memory.WriteRegisterHelper(ref _BG3PB, value, mode);
break;
case Register.BG3PC:
Memory.WriteRegisterHelper(ref _BG3PC, value, mode);
break;
case Register.BG3PD:
Memory.WriteRegisterHelper(ref _BG3PD, value, mode);
break;
case Register.BG3X_L:
Memory.WriteRegisterHelper(ref _BG3X_L, value, mode);
_currentBG3X = ((_BG3X_H << 20) | (_BG3X_L << 4)) >> 4;
break;
case Register.BG3X_H:
Memory.WriteRegisterHelper(ref _BG3X_H, value, mode);
_currentBG3X = ((_BG3X_H << 20) | (_BG3X_L << 4)) >> 4;
break;
case Register.BG3Y_L:
Memory.WriteRegisterHelper(ref _BG3Y_L, value, mode);
_currentBG3Y = ((_BG3Y_H << 20) | (_BG3Y_L << 4)) >> 4;
break;
case Register.BG3Y_H:
Memory.WriteRegisterHelper(ref _BG3Y_H, value, mode);
_currentBG3Y = ((_BG3Y_H << 20) | (_BG3Y_L << 4)) >> 4;
break;
case Register.WIN0H:
Memory.WriteRegisterHelper(ref _WIN0H, value, mode);
break;
case Register.WIN1H:
Memory.WriteRegisterHelper(ref _WIN1H, value, mode);
break;
case Register.WIN0V:
Memory.WriteRegisterHelper(ref _WIN0V, value, mode);
break;
case Register.WIN1V:
Memory.WriteRegisterHelper(ref _WIN1V, value, mode);
break;
case Register.WININ:
Memory.WriteRegisterHelper(ref _WININ, value, mode);
break;
case Register.WINOUT:
Memory.WriteRegisterHelper(ref _WINOUT, value, mode);
break;
case Register.MOSAIC:
Memory.WriteRegisterHelper(ref _MOSAIC, value, mode);
break;
case Register.BLDCNT:
Memory.WriteRegisterHelper(ref _BLDCNT, value, mode);
break;
case Register.BLDALPHA:
Memory.WriteRegisterHelper(ref _BLDALPHA, value, mode);
break;
case Register.BLDY:
Memory.WriteRegisterHelper(ref _BLDY, value, mode);
break;
// should never happen
default:
throw new Exception("Iris.GBA.Video: Register write error");
}
}
internal void Write8_PaletteRAM(UInt32 address, Byte value)
{
UInt32 offset = (UInt32)((address - PaletteRAM_StartAddress) & ~1) % PaletteRAM_Size;
unsafe
{
Unsafe.Write<Byte>((Byte*)_paletteRAM + offset, value);
Unsafe.Write<Byte>((Byte*)_paletteRAM + offset + 1, value);
}
}
internal void Write8_VRAM(UInt32 address, Byte value)
{
UInt32 offset = (UInt32)((address - VRAM_StartAddress) & ~1) % VRAM_Size;
unsafe
{
Unsafe.Write<Byte>((Byte*)_vram + offset, value);
Unsafe.Write<Byte>((Byte*)_vram + offset + 1, value);
}
}
private void StartHBlank(UInt64 cycleCountDelay)
{
_DISPSTAT |= 0x0002; // set HBlank status
if ((_DISPSTAT & 0x0010) == 0x0010)
_interruptControl.RequestInterrupt(InterruptControl.Interrupt.HBlank);
if (_VCOUNT < DisplayScreenHeight)
_dma.PerformHBlankTransfers();
_scheduler.ScheduleTask((int)GBA_System.TaskId.StartScanline, HBlankCycleCount - cycleCountDelay);
}
private void StartScanline(UInt64 cycleCountDelay)
{
_DISPSTAT = (UInt16)(_DISPSTAT & ~0x0002); // clear HBlank status
switch (_VCOUNT)
{
// rendering
case 0:
_VCOUNT = 1;
Render();
break;
// rendering
// video transfer DMA
case > 0 and < 159:
++_VCOUNT;
_dma.PerformVideoTransfer(false);
Render();
break;
// end rendering
// start vblank
// video transfer DMA
case 159:
_VCOUNT = 160;
_DISPSTAT |= 0x0001; // set VBlank status
if ((_DISPSTAT & 0x0008) == 0x0008)
_interruptControl.RequestInterrupt(InterruptControl.Interrupt.VBlank);
_dma.PerformVBlankTransfers();
_dma.PerformVideoTransfer(false);
_presentFrameCallback(_displayFrameBuffer);
Array.Clear(_displayFrameBuffer);
_currentBG2X = ((_BG2X_H << 20) | (_BG2X_L << 4)) >> 4;
_currentBG2Y = ((_BG2Y_H << 20) | (_BG2Y_L << 4)) >> 4;
_currentBG3X = ((_BG3X_H << 20) | (_BG3X_L << 4)) >> 4;
_currentBG3Y = ((_BG3Y_H << 20) | (_BG3Y_L << 4)) >> 4;
break;
// vblank
// end video transfer DMA
case 160:
_VCOUNT = 161;
_dma.PerformVideoTransfer(true);
break;
// vblank
case > 160 and < 226:
++_VCOUNT;
break;
// end vblank
case 226:
_VCOUNT = 227;
_DISPSTAT = (UInt16)(_DISPSTAT & ~0x0001); // clear VBlank status
break;
// start rendering
case 227:
_VCOUNT = 0;
Render();
break;
}
if (_VCOUNT == ((_DISPSTAT >> 8) & 0xff))
{
_DISPSTAT |= 0x0004; // set VCountMatch status
if ((_DISPSTAT & 0x0020) == 0x0020)
_interruptControl.RequestInterrupt(InterruptControl.Interrupt.VCountMatch);
}
else
{
_DISPSTAT = (UInt16)(_DISPSTAT & ~0x0004); // clear VCountMatch status
}
_scheduler.ScheduleTask((int)GBA_System.TaskId.StartHBlank, HDrawCycleCount - cycleCountDelay);
}
private void Render()
{
UInt16 bgMode = (UInt16)(_DISPCNT & 0b111);
switch (bgMode)
{
case 0b000:
RenderBackgroundMode0();
break;
case 0b001:
RenderBackgroundMode1();
break;
case 0b010:
RenderBackgroundMode2();
break;
case 0b011:
RenderBackgroundMode3();
break;
case 0b100:
RenderBackgroundMode4();
break;
case 0b101:
RenderBackgroundMode5();
break;
// TODO: verify
case 0b110:
case 0b111:
throw new Exception($"Iris.GBA.Video: Unknown background mode {bgMode}");
}
_currentBG2X += (Int16)_BG2PB;
_currentBG2Y += (Int16)_BG2PD;
_currentBG3X += (Int16)_BG3PB;
_currentBG3Y += (Int16)_BG3PD;
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private void RenderBackgroundMode0()
{
bool isFirst = true;
for (int bgPriority = 3; bgPriority >= 0; --bgPriority)
{
if (((_DISPCNT & 0x0800) == 0x0800) && ((_BG3CNT & 0b11) == bgPriority))
{
RenderTextBackground(_BG3CNT, _BG3HOFS, _BG3VOFS, isFirst);
isFirst = false;
}
if (((_DISPCNT & 0x0400) == 0x0400) && ((_BG2CNT & 0b11) == bgPriority))
{
RenderTextBackground(_BG2CNT, _BG2HOFS, _BG2VOFS, isFirst);
isFirst = false;
}
if (((_DISPCNT & 0x0200) == 0x0200) && ((_BG1CNT & 0b11) == bgPriority))
{
RenderTextBackground(_BG1CNT, _BG1HOFS, _BG1VOFS, isFirst);
isFirst = false;
}
if (((_DISPCNT & 0x0100) == 0x0100) && ((_BG0CNT & 0b11) == bgPriority))
{
RenderTextBackground(_BG0CNT, _BG0HOFS, _BG0VOFS, isFirst);
isFirst = false;
}
if ((_DISPCNT & 0x1000) == 0x1000)
RenderObjects((UInt16)bgPriority);
}
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private void RenderBackgroundMode1()
{
bool isFirst = true;
for (int bgPriority = 3; bgPriority >= 0; --bgPriority)
{
if (((_DISPCNT & 0x0400) == 0x0400) && ((_BG2CNT & 0b11) == bgPriority))
{
RenderRotationScalingBackground(_BG2CNT, _currentBG2X, _currentBG2Y, _BG2PA, _BG2PC, isFirst);
isFirst = false;
}
if (((_DISPCNT & 0x0200) == 0x0200) && ((_BG1CNT & 0b11) == bgPriority))
{
RenderTextBackground(_BG1CNT, _BG1HOFS, _BG1VOFS, isFirst);
isFirst = false;
}
if (((_DISPCNT & 0x0100) == 0x0100) && ((_BG0CNT & 0b11) == bgPriority))
{
RenderTextBackground(_BG0CNT, _BG0HOFS, _BG0VOFS, isFirst);
isFirst = false;
}
if ((_DISPCNT & 0x1000) == 0x1000)
RenderObjects((UInt16)bgPriority);
}
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private void RenderBackgroundMode2()
{
bool isFirst = true;
for (int bgPriority = 3; bgPriority >= 0; --bgPriority)
{
if (((_DISPCNT & 0x0800) == 0x0800) && ((_BG3CNT & 0b11) == bgPriority))
{
RenderRotationScalingBackground(_BG3CNT, _currentBG3X, _currentBG3Y, _BG3PA, _BG3PC, isFirst);
isFirst = false;
}
if (((_DISPCNT & 0x0400) == 0x0400) && ((_BG2CNT & 0b11) == bgPriority))
{
RenderRotationScalingBackground(_BG2CNT, _currentBG2X, _currentBG2Y, _BG2PA, _BG2PC, isFirst);
isFirst = false;
}
if ((_DISPCNT & 0x1000) == 0x1000)
RenderObjects((UInt16)bgPriority);
}
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private void RenderBackgroundMode3()
{
if ((_DISPCNT & 0x0400) == 0x0400)
{
ref UInt16 displayFrameBufferDataRef = ref MemoryMarshal.GetArrayDataReference(_displayFrameBuffer);
int pixelNumberBegin = _VCOUNT * DisplayScreenWidth;
int pixelNumberEnd = pixelNumberBegin + DisplayScreenWidth;
for (int pixelNumber = pixelNumberBegin; pixelNumber < pixelNumberEnd; ++pixelNumber)
{
unsafe
{
UInt16 color = Unsafe.Read<UInt16>((UInt16*)_vram + pixelNumber);
Unsafe.Add(ref displayFrameBufferDataRef, pixelNumber) = color;
}
}
}
if ((_DISPCNT & 0x1000) == 0x1000)
{
for (int bgPriority = 3; bgPriority >= 0; --bgPriority)
RenderObjects((UInt16)bgPriority);
}
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private void RenderBackgroundMode4()
{
if ((_DISPCNT & 0x0400) == 0x0400)
{
ref UInt16 displayFrameBufferDataRef = ref MemoryMarshal.GetArrayDataReference(_displayFrameBuffer);
int pixelNumberBegin = _VCOUNT * DisplayScreenWidth;
int pixelNumberEnd = pixelNumberBegin + DisplayScreenWidth;
UInt32 vramFrameBufferOffset = ((_DISPCNT & 0x0010) == 0) ? 0x0_0000u : 0x0_a000u;
for (int pixelNumber = pixelNumberBegin; pixelNumber < pixelNumberEnd; ++pixelNumber)
{
unsafe
{
Byte colorNumber = Unsafe.Read<Byte>((Byte*)_vram + vramFrameBufferOffset + pixelNumber);
UInt16 color = Unsafe.Read<UInt16>((UInt16*)_paletteRAM + colorNumber);
Unsafe.Add(ref displayFrameBufferDataRef, pixelNumber) = color;
}
}
}
if ((_DISPCNT & 0x1000) == 0x1000)
{
for (int bgPriority = 3; bgPriority >= 0; --bgPriority)
RenderObjects((UInt16)bgPriority);
}
}
[MethodImpl(MethodImplOptions.AggressiveInlining)]
private void RenderBackgroundMode5()
{
if ((_DISPCNT & 0x0400) == 0x0400)
{
const int VRAM_FrameBufferWidth = 160;
const int VRAM_FrameBufferHeight = 128;
if (_VCOUNT < VRAM_FrameBufferHeight)
{
ref UInt16 displayFrameBufferDataRef = ref MemoryMarshal.GetArrayDataReference(_displayFrameBuffer);
int vramPixelNumberBegin = _VCOUNT * VRAM_FrameBufferWidth;
int vramPixelNumberEnd = vramPixelNumberBegin + VRAM_FrameBufferWidth;
int displayPixelNumberBegin = _VCOUNT * DisplayScreenWidth;
UInt32 vramFrameBufferOffset = ((_DISPCNT & 0x0010) == 0) ? 0x0_0000u : 0x0_a000u;
for (int vramPixelNumber = vramPixelNumberBegin, displayPixelNumber = displayPixelNumberBegin; vramPixelNumber < vramPixelNumberEnd; ++vramPixelNumber, ++displayPixelNumber)
{
unsafe
{
UInt16 color = Unsafe.Read<UInt16>((Byte*)_vram + vramFrameBufferOffset + (vramPixelNumber * 2));
Unsafe.Add(ref displayFrameBufferDataRef, displayPixelNumber) = color;
}
}
}
}
if ((_DISPCNT & 0x1000) == 0x1000)
{
for (int bgPriority = 3; bgPriority >= 0; --bgPriority)
RenderObjects((UInt16)bgPriority);
}
}
private void RenderTextBackground(UInt16 cnt, UInt16 hofs, UInt16 vofs, bool isFirst)
{
ref UInt16 displayFrameBufferDataRef = ref MemoryMarshal.GetArrayDataReference(_displayFrameBuffer);
int displayPixelNumberBegin = _VCOUNT * DisplayScreenWidth;
UInt16 virtualScreenSize = (UInt16)((cnt >> 14) & 0b11);
UInt16 screenBaseBlock = (UInt16)((cnt >> 8) & 0b1_1111);
UInt16 colorMode = (UInt16)((cnt >> 7) & 1);
UInt16 characterBaseBlock = (UInt16)((cnt >> 2) & 0b11);
int virtualScreenWidth = ((virtualScreenSize & 0b01) == 0) ? 256 : 512;
int virtualScreenHeight = ((virtualScreenSize & 0b10) == 0) ? 256 : 512;
UInt32 screenBaseBlockOffset = (UInt32)(screenBaseBlock * 2 * KB);
UInt32 characterBaseBlockOffset = (UInt32)(characterBaseBlock * 16 * KB);
int v = (_VCOUNT + vofs) % virtualScreenHeight;
const int SC_Width = 256;
const int SC_Height = 256;
const int SC_CharacterCount = (SC_Width / CharacterWidth) * (SC_Height / CharacterHeight);
int scV = v / SC_Height;
int scPixelV = v % SC_Height;
int scCharacterV = scPixelV / CharacterHeight;
int characterPixelV = scPixelV % CharacterHeight;
int scNumberBegin = scV * (virtualScreenWidth / SC_Width);
int characterNumberBegin = scCharacterV * (SC_Width / CharacterWidth);
int characterPixelNumberBegin = characterPixelV * CharacterWidth;
int characterPixelNumberBegin_VerticalFlip = (CharacterHeight - 1 - characterPixelV) * CharacterWidth;
for (int hcount = 0; hcount < DisplayScreenWidth; ++hcount)
{
int h = (hcount + hofs) % virtualScreenWidth;
int scH = h / SC_Width;
int scPixelH = h % SC_Width;
int scCharacterH = scPixelH / CharacterWidth;
int characterPixelH = scPixelH % CharacterWidth;
int scNumber = scNumberBegin + scH;
int characterNumber = characterNumberBegin + scCharacterH;
unsafe
{
UInt16 screenData = Unsafe.Read<UInt16>((Byte*)_vram + screenBaseBlockOffset + (scNumber * SC_CharacterCount * 2) + (characterNumber * 2));
UInt16 colorPalette = (UInt16)((screenData >> 12) & 0b1111);
UInt16 verticalFlipFlag = (UInt16)((screenData >> 11) & 1);
UInt16 horizontalFlipFlag = (UInt16)((screenData >> 10) & 1);
UInt16 characterName = (UInt16)(screenData & 0x3ff);
int characterPixelNumber;
if (verticalFlipFlag == 0)
characterPixelNumber = characterPixelNumberBegin;
else
characterPixelNumber = characterPixelNumberBegin_VerticalFlip;
if (horizontalFlipFlag == 0)
characterPixelNumber += characterPixelH;
else
characterPixelNumber += CharacterWidth - 1 - characterPixelH;
const UInt32 ObjectCharacterDataOffset = 0x1_0000;
UInt16 color;
// 16 colors x 16 palettes
if (colorMode == 0)
{
const int CharacterSize = 32;
UInt32 characterOffset = (UInt32)(characterBaseBlockOffset + (characterName * CharacterSize));
if (characterOffset >= ObjectCharacterDataOffset)
continue;
Byte colorNumber = Unsafe.Read<Byte>((Byte*)_vram + characterOffset + (characterPixelNumber / 2));
if ((characterPixelNumber % 2) == 0)
colorNumber &= 0b1111;
else
colorNumber >>= 4;
if (!isFirst && (colorNumber == 0))
continue;
color = Unsafe.Read<UInt16>((UInt16*)_paletteRAM + (colorPalette * 16) + colorNumber);
}
// 256 colors x 1 palette
else
{
const int CharacterSize = 64;
UInt32 characterOffset = (UInt32)(characterBaseBlockOffset + (characterName * CharacterSize));
if (characterOffset >= ObjectCharacterDataOffset)
continue;
Byte colorNumber = Unsafe.Read<Byte>((Byte*)_vram + characterOffset + characterPixelNumber);
if (!isFirst && (colorNumber == 0))
continue;
color = Unsafe.Read<UInt16>((UInt16*)_paletteRAM + colorNumber);
}
int displayPixelNumber = displayPixelNumberBegin + hcount;
Unsafe.Add(ref displayFrameBufferDataRef, displayPixelNumber) = color;
}
}
}
private void RenderRotationScalingBackground(UInt16 cnt, Int32 x, Int32 y, UInt16 pa, UInt16 pc, bool isFirst)
{
ref UInt16 displayFrameBufferDataRef = ref MemoryMarshal.GetArrayDataReference(_displayFrameBuffer);
int displayPixelNumberBegin = _VCOUNT * DisplayScreenWidth;
UInt16 virtualScreenSize = (UInt16)((cnt >> 14) & 0b11);
UInt16 areaOverflow = (UInt16)((cnt >> 13) & 1);
UInt16 screenBaseBlock = (UInt16)((cnt >> 8) & 0b1_1111);
UInt16 characterBaseBlock = (UInt16)((cnt >> 2) & 0b11);
int virtualScreenWidth = 128 << virtualScreenSize;
int virtualScreenHeight = 128 << virtualScreenSize;
UInt32 screenBaseBlockOffset = (UInt32)(screenBaseBlock * 2 * KB);
UInt32 characterBaseBlockOffset = (UInt32)(characterBaseBlock * 16 * KB);
for (int hcount = 0; hcount < DisplayScreenWidth; ++hcount, x += (Int16)pa, y += (Int16)pc)
{
int h = x >> 8;
int v = y >> 8;
if (areaOverflow == 0)
{
if ((h < 0) || (h >= virtualScreenWidth))
continue;
if ((v < 0) || (v >= virtualScreenHeight))
continue;
}
else
{
h %= virtualScreenWidth;
v %= virtualScreenHeight;
if (h < 0)
h += virtualScreenWidth;
if (v < 0)
v += virtualScreenHeight;
}
int characterH = h / CharacterWidth;
int characterPixelH = h % CharacterWidth;
int characterV = v / CharacterHeight;
int characterPixelV = v % CharacterHeight;
int characterNumber = (characterV * (virtualScreenWidth / CharacterWidth)) + characterH;
int characterPixelNumber = (characterPixelV * CharacterWidth) + characterPixelH;
unsafe
{
Byte characterName = Unsafe.Read<Byte>((Byte*)_vram + screenBaseBlockOffset + characterNumber);
const int CharacterSize = 64;
UInt32 characterOffset = (UInt32)(characterBaseBlockOffset + (characterName * CharacterSize));
const UInt32 ObjectCharacterDataOffset = 0x1_0000;
if (characterOffset >= ObjectCharacterDataOffset)
continue;
Byte colorNumber = Unsafe.Read<Byte>((Byte*)_vram + characterOffset + characterPixelNumber);
if (!isFirst && (colorNumber == 0))
continue;
UInt16 color = Unsafe.Read<UInt16>((UInt16*)_paletteRAM + colorNumber);
int displayPixelNumber = displayPixelNumberBegin + hcount;
Unsafe.Add(ref displayFrameBufferDataRef, displayPixelNumber) = color;
}
}
}
private void RenderObjects(UInt16 bgPriority)
{
ref UInt16 displayFrameBufferDataRef = ref MemoryMarshal.GetArrayDataReference(_displayFrameBuffer);
int displayPixelNumberBegin = _VCOUNT * DisplayScreenWidth;
UInt16 mappingFormat = (UInt16)((_DISPCNT >> 6) & 1);
for (int objNumber = 127; objNumber >= 0; --objNumber)
{
unsafe
{
UInt16 attribute0 = Unsafe.Read<UInt16>((UInt16*)_oam + (objNumber * 4));
UInt16 attribute1 = Unsafe.Read<UInt16>((UInt16*)_oam + (objNumber * 4) + 1);
UInt16 attribute2 = Unsafe.Read<UInt16>((UInt16*)_oam + (objNumber * 4) + 2);
UInt16 shape = (UInt16)((attribute0 >> 14) & 0b11);
UInt16 colorMode = (UInt16)((attribute0 >> 13) & 1);
UInt16 rotationScalingFlag = (UInt16)((attribute0 >> 8) & 1);
UInt16 yCoordinate = (UInt16)(attribute0 & 0xff);
UInt16 objSize = (UInt16)((attribute1 >> 14) & 0b11);
UInt16 verticalFlipFlag = (UInt16)((attribute1 >> 13) & 1);
UInt16 horizontalFlipFlag = (UInt16)((attribute1 >> 12) & 1);
UInt16 xCoordinate = (UInt16)(attribute1 & 0x1ff);
UInt16 colorPalette = (UInt16)((attribute2 >> 12) & 0b1111);
UInt16 objPriority = (UInt16)((attribute2 >> 10) & 0b11);
UInt16 characterName = (UInt16)(attribute2 & 0x3ff);
if (objPriority != bgPriority)
continue;
(int objWidth, int objHeight) = (shape, objSize) switch
{
// square
(0b00, 0b00) => (8, 8),
(0b00, 0b01) => (16, 16),
(0b00, 0b10) => (32, 32),
(0b00, 0b11) => (64, 64),
// horizontal rectangle
(0b01, 0b00) => (16, 8),
(0b01, 0b01) => (32, 8),
(0b01, 0b10) => (32, 16),
(0b01, 0b11) => (64, 32),
// vertical rectangle
(0b10, 0b00) => (8, 16),
(0b10, 0b01) => (8, 32),
(0b10, 0b10) => (16, 32),
(0b10, 0b11) => (32, 64),
// prohibited
// TODO: verify
_ => (0, 0)
};
const int VirtualScreenWidth = 512;
const int VirtualScreenHeight = 256;
int left = xCoordinate;
int right = (xCoordinate + objWidth) % VirtualScreenWidth;
int top = yCoordinate;
int bottom = (yCoordinate + objHeight) % VirtualScreenHeight;
bool leftHidden = left >= DisplayScreenWidth;
bool rightHidden = right >= DisplayScreenWidth;
bool topHidden = top >= DisplayScreenHeight;
bool bottomHidden = bottom >= DisplayScreenHeight;
if ((leftHidden && rightHidden) || (topHidden && bottomHidden))
continue;
int hBegin;
int vBegin;
if (leftHidden)
{
hBegin = VirtualScreenWidth - left;
left = 0;
}
else if (rightHidden)
{
hBegin = 0;
right = DisplayScreenWidth;
}
else
{
hBegin = 0;
}
if (topHidden)
{
vBegin = VirtualScreenHeight - top;
top = 0;
}
else if (bottomHidden)
{
vBegin = 0;
bottom = DisplayScreenHeight;
}
else
{
vBegin = 0;
}
if ((top > _VCOUNT) || (bottom <= _VCOUNT))
continue;
int v = _VCOUNT - top + vBegin;
int characterV = v / CharacterHeight;
int characterPixelV = v % CharacterHeight;
int characterNumberBegin;
int characterPixelNumberBegin = CharacterWidth;
if (verticalFlipFlag == 0)
{
characterNumberBegin = characterV;
characterPixelNumberBegin *= characterPixelV;
}
else
{
characterNumberBegin = (objHeight / CharacterHeight) - 1 - characterV;
characterPixelNumberBegin *= CharacterHeight - 1 - characterPixelV;
}
// 2D mapping
if (mappingFormat == 0)
characterNumberBegin *= (colorMode == 0) ? 32 : 16;
// 1D mapping
else
characterNumberBegin *= objWidth / CharacterWidth;
for (int hcount = left; hcount < right; ++hcount)
{
int h = hcount - left + hBegin;
int characterH = h / CharacterWidth;
int characterPixelH = h % CharacterWidth;
int characterNumber = characterNumberBegin;
int characterPixelNumber = characterPixelNumberBegin;
if (horizontalFlipFlag == 0)
{
characterNumber += characterH;
characterPixelNumber += characterPixelH;
}
else
{
characterNumber += (objWidth / CharacterWidth) - 1 - characterH;
characterPixelNumber += CharacterWidth - 1 - characterPixelH;
}
const UInt32 CharacterDataOffset = 0x1_0000;
const UInt32 PaletteOffset = 0x200;
UInt16 color;
// 16 colors x 16 palettes
if (colorMode == 0)
{
const int CharacterSize = 32;
Byte colorNumber = Unsafe.Read<Byte>((Byte*)_vram + CharacterDataOffset + (characterName * 32) + (characterNumber * CharacterSize) + (characterPixelNumber / 2));
if ((characterPixelNumber % 2) == 0)
colorNumber &= 0b1111;
else
colorNumber >>= 4;
if (colorNumber == 0)
continue;
color = Unsafe.Read<UInt16>((Byte*)_paletteRAM + PaletteOffset + (colorPalette * 16 * 2) + (colorNumber * 2));
}
// 256 colors x 1 palette
else
{
const int CharacterSize = 64;
Byte colorNumber = Unsafe.Read<Byte>((Byte*)_vram + CharacterDataOffset + (characterName * 32) + (characterNumber * CharacterSize) + characterPixelNumber);
if (colorNumber == 0)
continue;
color = Unsafe.Read<UInt16>((Byte*)_paletteRAM + PaletteOffset + (colorNumber * 2));
}
int displayPixelNumber = displayPixelNumberBegin + hcount;
Unsafe.Add(ref displayFrameBufferDataRef, displayPixelNumber) = color;
}
}
}
}
}
}