Reimplement ScreenArranger, refactor MelonDS.GetVideoBuffer
This commit is contained in:
parent
a574fab07f
commit
e0e90a5f33
|
@ -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;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Provides a way to arrange displays inside a frame buffer.
|
||||
/// </summary>
|
||||
/// <summary>Provides a way to arrange displays inside a frame buffer.</summary>
|
||||
public static class ScreenArranger
|
||||
{
|
||||
// TODO: pass in int[] to reuse buffer
|
||||
public static unsafe int[] Stack(VideoScreen screen1, VideoScreen screen2, int screenGap)
|
||||
/// <remarks>this is taken as an assumption to allow for simpler algorithms; in the future this may need to be rethought (e.g. for 3DS)</remarks>
|
||||
[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();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Simply populates a buffer with a single screen
|
||||
/// </summary>
|
||||
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;
|
||||
}
|
||||
|
||||
/// <summary>Simply populates a buffer with a single screen</summary>
|
||||
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;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
|
@ -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);
|
||||
|
|
Loading…
Reference in New Issue