From e0e90a5f336dbd46ba119b5c0bb6a3e3cce6f861 Mon Sep 17 00:00:00 2001 From: YoshiRulz Date: Mon, 30 Mar 2020 03:50:04 +1000 Subject: [PATCH] Reimplement ScreenArranger, refactor MelonDS.GetVideoBuffer --- BizHawk.Emulation.Common/ScreenArranger.cs | 142 +++++++++++------- .../Nintendo/NDS/MelonDS_VideoProvider.cs | 29 ++-- 2 files changed, 100 insertions(+), 71 deletions(-) diff --git a/BizHawk.Emulation.Common/ScreenArranger.cs b/BizHawk.Emulation.Common/ScreenArranger.cs index 6943536809..9a532de61b 100644 --- a/BizHawk.Emulation.Common/ScreenArranger.cs +++ b/BizHawk.Emulation.Common/ScreenArranger.cs @@ -1,4 +1,7 @@ -namespace BizHawk.Emulation.Common +using System; +using System.Diagnostics; + +namespace BizHawk.Emulation.Common { public unsafe class VideoScreen { @@ -16,79 +19,108 @@ public int Length => Width * Height; } - /// - /// Provides a way to arrange displays inside a frame buffer. - /// + /// Provides a way to arrange displays inside a frame buffer. public static class ScreenArranger { - // TODO: pass in int[] to reuse buffer - public static unsafe int[] Stack(VideoScreen screen1, VideoScreen screen2, int screenGap) + /// this is taken as an assumption to allow for simpler algorithms; in the future this may need to be rethought (e.g. for 3DS) + [Conditional("DEBUG")] + private static void DebugAssertScreenDimensionsMatch(int lengthA, int lengthB) { - var ret = new int[screen1.Width * (screen1.Height + screen2.Height + screenGap)]; - for (int i = 0; i < screen1.Length; i++) - { - ret[i] = screen1.Buffer[i]; - } - - for (int i = 0; i < screen2.Length; i++) - { - ret[screen1.Length + i + (screen1.Width * screenGap)] = screen2.Buffer[i]; - } - - return ret; + if (lengthA != lengthB) throw new ArgumentException(); } - /// - /// Simply populates a buffer with a single screen - /// - public static unsafe int[] Copy(VideoScreen screen1) + [Conditional("DEBUG")] + private static void DebugAssertPreallocatedBufferSize(int expected, int preallocLength) { - var ret = new int[screen1.Length]; + if (preallocLength != expected) throw new Exception(); + } - for (int i = 0; i < ret.Length; i++) + public static unsafe int[] UprightStack(VideoScreen forTop, VideoScreen forBottom, int gapLineCount = 0) + { + DebugAssertScreenDimensionsMatch(forTop.Width, forBottom.Width); + var outputWidth = forTop.Width; + + var gapStartOffset = forTop.Length; + var screen2StartOffset = gapStartOffset + gapLineCount * outputWidth; + var bufferLength = screen2StartOffset + forBottom.Height * outputWidth; + var prealloc = new int[bufferLength]; //TODO actually take a `ref int[] prealloc` (or an int* maybe?) + DebugAssertPreallocatedBufferSize(bufferLength, prealloc.Length); + + for (var i = 0; i < gapStartOffset; i++) prealloc[i] = forTop.Buffer[i]; // copy top screen + // don't bother writing into the gap + for (int i = 0, l = forBottom.Length; i < l; i++) prealloc[screen2StartOffset + i] = forBottom.Buffer[i]; // copy bottom screen + return prealloc; + } + + /// Simply populates a buffer with a single screen + public static unsafe int[] Copy(VideoScreen screen) + { + var bufferLength = screen.Length; + var prealloc = new int[bufferLength]; //TODO actually take a `ref int[] prealloc` (or an int* maybe?) + DebugAssertPreallocatedBufferSize(bufferLength, prealloc.Length); + + for (var i = 0; i < bufferLength; i++) prealloc[i] = screen.Buffer[i]; + return prealloc; + } + + public static unsafe int[] UprightSideBySide(VideoScreen forLeft, VideoScreen forRight, int gapLineCount = 0) + { + DebugAssertScreenDimensionsMatch(forLeft.Height, forRight.Height); + var outputHeight = forLeft.Height; + + var rightOffsetHztl = forLeft.Width + gapLineCount; + var outputWidth = rightOffsetHztl + forRight.Width; + var bufferLength = outputHeight * outputWidth; // which = `forLeft.Length + outputHeight * gapLineCount + forRight.Length` + var prealloc = new int[bufferLength]; //TODO actually take a `ref int[] prealloc` (or an int* maybe?) + DebugAssertPreallocatedBufferSize(bufferLength, prealloc.Length); + + for (var y = 0; y < outputHeight; y++) { - ret[i] = screen1.Buffer[i]; + for (int x = 0, w = forLeft.Width; x < w; x++) prealloc[y * outputWidth + x] = forLeft.Buffer[y * w + x]; // copy this row of the left screen + // don't bother writing into the gap + for (int x = 0, w = forRight.Width; x < w; x++) prealloc[y * outputWidth + rightOffsetHztl + x] = forRight.Buffer[y * w + x]; // copy this row of the right screen } - - return ret; + return prealloc; } - // TODO: pass in int[] to reuse buffer - // TODO: there is a simpler algorithm for sure - public static unsafe int[] SideBySide(VideoScreen screen1, VideoScreen screen2) + public static unsafe int[] Rotate90Stack(VideoScreen forLeft, VideoScreen forRight, int gapLineCount = 0) { - int width = screen1.Width + screen2.Width; - int height = screen2.Height; - var ret = new int[width * height]; + DebugAssertScreenDimensionsMatch(forLeft.Width, forRight.Width); + var outputHeight = forLeft.Width; - for (int y = 0; y < height; y++) + var rightOffsetHztl = forLeft.Height + gapLineCount; + var outputWidth = rightOffsetHztl + forRight.Height; + var bufferLength = outputHeight * outputWidth; // which = `forLeft.Length + outputHeight * gapLineCount + forRight.Length` + var prealloc = new int[bufferLength]; //TODO actually take a `ref int[] prealloc` (or an int* maybe?) + DebugAssertPreallocatedBufferSize(bufferLength, prealloc.Length); + + for (var y = 0; y < outputHeight; y++) { - for (int x = 0; x < width; x++) - { - if (x < screen1.Width) - { - ret[(y * width) + x] = screen1.Buffer[(y * width / 2) + x]; - } - else - { - ret[(y * width) + x] = screen2.Buffer[(y * width / 2) + x]; - } - } - + for (int x = 0, w = forLeft.Height; x < w; x++) prealloc[y * outputWidth + x] = forLeft.Buffer[(x + 1) * outputHeight - y]; // copy and rotate this column of the top screen to the left of the output + // don't bother writing into the gap + for (int x = 0, w = forRight.Height; x < w; x++) prealloc[y * outputWidth + rightOffsetHztl + x] = forRight.Buffer[(x + 1) * outputHeight - y]; // copy and rotate this column of the bottom screen to the right of the output } - - return ret; + return prealloc; } - // Yeah... - public static int[] Rotate90(int[] buffer) + public static unsafe int[] Rotate270Stack(VideoScreen forLeft, VideoScreen forRight, int gapLineCount = 0) { - return buffer; - } + DebugAssertScreenDimensionsMatch(forLeft.Width, forRight.Width); + var outputHeight = forLeft.Width; - public static int[] Rotate270(int[] buffer) - { - return buffer; + var rightOffsetHztl = forLeft.Height + gapLineCount; + var outputWidth = rightOffsetHztl + forRight.Height; + var bufferLength = outputHeight * outputWidth; // which = `forLeft.Length + outputHeight * gapLineCount + forRight.Length` + var prealloc = new int[bufferLength]; //TODO actually take a `ref int[] prealloc` (or an int* maybe?) + DebugAssertPreallocatedBufferSize(bufferLength, prealloc.Length); + + for (var y = 0; y < outputHeight; y++) + { + for (int x = 0, w = forLeft.Height; x < w; x++) prealloc[y * outputWidth + x] = forLeft.Buffer[(w - x) * outputHeight + y]; // copy and rotate this column of the bottom screen to the left of the output + // don't bother writing into the gap + for (int x = 0, w = forRight.Height; x < w; x++) prealloc[y * outputWidth + rightOffsetHztl + x] = forRight.Buffer[(w - x) * outputHeight + y]; // copy and rotate this column of the top screen to the right of the output + } + return prealloc; } } } diff --git a/BizHawk.Emulation.Cores/Consoles/Nintendo/NDS/MelonDS_VideoProvider.cs b/BizHawk.Emulation.Cores/Consoles/Nintendo/NDS/MelonDS_VideoProvider.cs index c33b492b32..833a63ca3b 100644 --- a/BizHawk.Emulation.Cores/Consoles/Nintendo/NDS/MelonDS_VideoProvider.cs +++ b/BizHawk.Emulation.Cores/Consoles/Nintendo/NDS/MelonDS_VideoProvider.cs @@ -1,4 +1,5 @@ -using System.Runtime.InteropServices; +using System; +using System.Runtime.InteropServices; using BizHawk.Emulation.Common; @@ -37,22 +38,18 @@ namespace BizHawk.Emulation.Cores.Consoles.Nintendo.NDS private bool _getNewBuffer = true; public int[] GetVideoBuffer() { - if (_getNewBuffer) + if (!_getNewBuffer) return _buffer; + _getNewBuffer = false; + return _buffer = _settings.ScreenOptions switch { - _getNewBuffer = false; - - _buffer = _settings.ScreenOptions switch - { - VideoScreenOptions.TopOnly => ScreenArranger.Copy(TopScreen), - VideoScreenOptions.SideBySideLR => ScreenArranger.SideBySide(TopScreen, BottomScreen), - VideoScreenOptions.SideBySideRL => ScreenArranger.SideBySide(BottomScreen, TopScreen), - VideoScreenOptions.Rotate90 => ScreenArranger.Rotate90(ScreenArranger.Stack(TopScreen, BottomScreen, 0)), - VideoScreenOptions.Rotate270 => ScreenArranger.Rotate270(ScreenArranger.Stack(TopScreen, BottomScreen, 0)), - _ => ScreenArranger.Stack(TopScreen, BottomScreen, _settings.ScreenGap) - }; - } - - return _buffer; + VideoScreenOptions.Default => ScreenArranger.UprightStack(TopScreen, BottomScreen, _settings.ScreenGap), + VideoScreenOptions.TopOnly => ScreenArranger.Copy(TopScreen), + VideoScreenOptions.SideBySideLR => ScreenArranger.UprightSideBySide(TopScreen, BottomScreen, _settings.ScreenGap), + VideoScreenOptions.SideBySideRL => ScreenArranger.UprightSideBySide(BottomScreen, TopScreen, _settings.ScreenGap), + VideoScreenOptions.Rotate90 => ScreenArranger.Rotate90Stack(TopScreen, BottomScreen, _settings.ScreenGap), + VideoScreenOptions.Rotate270 => ScreenArranger.Rotate270Stack(BottomScreen, TopScreen, _settings.ScreenGap), + _ => throw new InvalidOperationException() + }; } private VideoScreen TopScreen => new VideoScreen(GetTopScreenBuffer(), NativeWidth, NativeHeight);