From 8aac335d9b4e9748d9ed14c98b4bcb86a19e1129 Mon Sep 17 00:00:00 2001 From: CasualPokePlayer <50538166+CasualPokePlayer@users.noreply.github.com> Date: Sat, 18 May 2024 01:26:37 -0700 Subject: [PATCH] Change RenderTarget to IRenderTarget, implement classes accordingly Changes GDI+ handling quite a bit, hopefully didn't break anything --- .../Controls/GDIPlusControl.cs | 25 +++--- .../D3D11/D3D11RenderTarget.cs | 38 +++++++++ .../D3D11/D3D11Resources.cs | 25 +++++- .../D3D11/D3D11Texture2D.cs | 19 ++--- .../D3D11/IGL_D3D11.cs | 79 +++---------------- .../GDIPlus/GDIPlusControlRenderTarget.cs | 37 +++++++++ .../GDIPlus/GDIPlusRenderTarget.cs | 65 ++++----------- .../GDIPlus/GDIPlusTexture2D.cs | 4 +- .../GDIPlus/IGL_GDIPlus.cs | 64 ++++----------- .../Interfaces/IGL.cs | 45 +++++------ .../Interfaces/IRenderTarget.cs | 15 ++++ .../OpenGL/IGL_OpenGL.cs | 50 ++---------- .../OpenGL/OpenGLRenderTarget.cs | 52 ++++++++++++ .../OpenGL/OpenGLTexture2D.cs | 2 +- src/BizHawk.Bizware.Graphics/RenderTarget.cs | 38 --------- .../DisplayManager/DisplayManagerBase.cs | 14 ++-- .../DisplayManager/FilterManager.cs | 6 +- .../DisplayManager/Filters/BaseFilter.cs | 2 +- .../DisplayManager/Filters/Utils.cs | 2 +- .../DisplayManager/RenderTargetFrugalizer.cs | 12 +-- .../DisplayManager/DisplayManager.cs | 2 +- .../RetainedGraphicsControl.cs | 10 +-- 22 files changed, 281 insertions(+), 325 deletions(-) create mode 100644 src/BizHawk.Bizware.Graphics/D3D11/D3D11RenderTarget.cs create mode 100644 src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusControlRenderTarget.cs create mode 100644 src/BizHawk.Bizware.Graphics/Interfaces/IRenderTarget.cs create mode 100644 src/BizHawk.Bizware.Graphics/OpenGL/OpenGLRenderTarget.cs delete mode 100644 src/BizHawk.Bizware.Graphics/RenderTarget.cs diff --git a/src/BizHawk.Bizware.Graphics.Controls/Controls/GDIPlusControl.cs b/src/BizHawk.Bizware.Graphics.Controls/Controls/GDIPlusControl.cs index 89952ed7c2..1788abd5e4 100644 --- a/src/BizHawk.Bizware.Graphics.Controls/Controls/GDIPlusControl.cs +++ b/src/BizHawk.Bizware.Graphics.Controls/Controls/GDIPlusControl.cs @@ -9,15 +9,19 @@ namespace BizHawk.Bizware.Graphics.Controls { internal sealed class GDIPlusControl : GraphicsControl { - public GDIPlusControl(Func, GDIPlusRenderTarget> createControlRenderTarget) + /// + /// The render target for rendering to this control + /// + private readonly GDIPlusControlRenderTarget _renderTarget; + + public GDIPlusControl(Func, GDIPlusControlRenderTarget> createControlRenderTarget) { - RenderTarget = createControlRenderTarget(GetControlRenderContext); + _renderTarget = createControlRenderTarget(GetControlRenderContext); SetStyle(ControlStyles.UserPaint, true); SetStyle(ControlStyles.Opaque, true); SetStyle(ControlStyles.UserMouse, true); DoubleBuffered = true; - BackColor = Color.Black; } private (SDGraphics Graphics, Rectangle Rectangle) GetControlRenderContext() @@ -28,11 +32,6 @@ namespace BizHawk.Bizware.Graphics.Controls return (graphics, ClientRectangle); } - /// - /// The render target for rendering to this control - /// - private GDIPlusRenderTarget RenderTarget { get; } - public override void AllowTearing(bool state) { // not controllable @@ -54,22 +53,22 @@ namespace BizHawk.Bizware.Graphics.Controls protected override void OnHandleCreated(EventArgs e) { base.OnHandleCreated(e); - RenderTarget.CreateGraphics(); + _renderTarget.CreateGraphics(); } protected override void OnHandleDestroyed(EventArgs e) { base.OnHandleDestroyed(e); - RenderTarget.Dispose(); + _renderTarget.Dispose(); } protected override void OnResize(EventArgs e) { base.OnResize(e); - RenderTarget.CreateGraphics(); + _renderTarget.CreateGraphics(); } public override void SwapBuffers() - => RenderTarget.BufferedGraphics?.Render(RenderTarget.CurGraphics); + => _renderTarget.BufferedGraphics?.Render(_renderTarget.ControlGraphics); } -} \ No newline at end of file +} diff --git a/src/BizHawk.Bizware.Graphics/D3D11/D3D11RenderTarget.cs b/src/BizHawk.Bizware.Graphics/D3D11/D3D11RenderTarget.cs new file mode 100644 index 0000000000..fef5755181 --- /dev/null +++ b/src/BizHawk.Bizware.Graphics/D3D11/D3D11RenderTarget.cs @@ -0,0 +1,38 @@ +using Vortice.Direct3D11; +using Vortice.DXGI; + +namespace BizHawk.Bizware.Graphics +{ + internal sealed class D3D11RenderTarget : D3D11Texture2D, IRenderTarget + { + public ID3D11RenderTargetView RTV; + + public D3D11RenderTarget(D3D11Resources resources, int width, int height) + : base(resources, BindFlags.ShaderResource | BindFlags.RenderTarget, ResourceUsage.Default, CpuAccessFlags.None, width, height) + { + } + + public override void CreateTexture() + { + base.CreateTexture(); + var rtvd = new RenderTargetViewDescription(RenderTargetViewDimension.Texture2D, Format.B8G8R8A8_UNorm); + RTV = Device.CreateRenderTargetView(Texture, rtvd); + } + + public override void DestroyTexture() + { + RTV?.Dispose(); + RTV = null; + base.DestroyTexture(); + } + + public void Bind() + { + Context.OMSetRenderTargets(RTV); + _resources.CurRenderTarget = this; + } + + public override string ToString() + => $"D3D11 RenderTarget: {Width}x{Height}"; + } +} diff --git a/src/BizHawk.Bizware.Graphics/D3D11/D3D11Resources.cs b/src/BizHawk.Bizware.Graphics/D3D11/D3D11Resources.cs index ec6dc88dfd..8e29a06070 100644 --- a/src/BizHawk.Bizware.Graphics/D3D11/D3D11Resources.cs +++ b/src/BizHawk.Bizware.Graphics/D3D11/D3D11Resources.cs @@ -26,8 +26,9 @@ namespace BizHawk.Bizware.Graphics public FeatureLevel DeviceFeatureLevel; - public readonly HashSet RenderTargets = new(); - public readonly HashSet ShaderTextures = new(); + public D3D11RenderTarget CurRenderTarget; + + public readonly HashSet Textures = new(); public readonly HashSet VertexShaders = new(); public readonly HashSet PixelShaders = new(); public readonly HashSet Pipelines = new(); @@ -94,6 +95,11 @@ namespace BizHawk.Bizware.Graphics RasterizerState = Device.CreateRasterizerState(rd); Context.IASetPrimitiveTopology(PrimitiveTopology.TriangleStrip); + + foreach (var tex2d in Textures) + { + tex2d.CreateTexture(); + } } catch { @@ -102,8 +108,15 @@ namespace BizHawk.Bizware.Graphics } } - public void Dispose() + public void DestroyResources() { + foreach (var tex2d in Textures) + { + tex2d.DestroyTexture(); + } + + CurRenderTarget = null; + LinearSamplerState?.Dispose(); LinearSamplerState = null; PointSamplerState?.Dispose(); @@ -128,5 +141,11 @@ namespace BizHawk.Bizware.Graphics Factory1?.Dispose(); Factory1 = null; } + + public void Dispose() + { + DestroyResources(); + Textures.Clear(); + } } } diff --git a/src/BizHawk.Bizware.Graphics/D3D11/D3D11Texture2D.cs b/src/BizHawk.Bizware.Graphics/D3D11/D3D11Texture2D.cs index 860f4ff4df..130e5e61cd 100644 --- a/src/BizHawk.Bizware.Graphics/D3D11/D3D11Texture2D.cs +++ b/src/BizHawk.Bizware.Graphics/D3D11/D3D11Texture2D.cs @@ -10,18 +10,18 @@ namespace BizHawk.Bizware.Graphics { internal class D3D11Texture2D : ITexture2D { - private readonly D3D11Resources _resources; + protected readonly D3D11Resources _resources; private readonly BindFlags _bindFlags; private readonly ResourceUsage _usage; private readonly CpuAccessFlags _cpuAccessFlags; - private ID3D11Device Device => _resources.Device; - private ID3D11DeviceContext Context => _resources.Context; - private HashSet ShaderTextures => _resources.ShaderTextures; + protected ID3D11Device Device => _resources.Device; + protected ID3D11DeviceContext Context => _resources.Context; + private HashSet Textures => _resources.Textures; private ID3D11Texture2D StagingTexture; - public ID3D11Texture2D Texture; + protected ID3D11Texture2D Texture; public ID3D11ShaderResourceView SRV; public bool LinearFiltering; @@ -37,17 +37,18 @@ namespace BizHawk.Bizware.Graphics _cpuAccessFlags = cpuAccessFlags; Width = width; Height = height; + // ReSharper disable once VirtualMemberCallInConstructor CreateTexture(); - ShaderTextures.Add(this); + Textures.Add(this); } public void Dispose() { DestroyTexture(); - ShaderTextures.Remove(this); + Textures.Remove(this); } - public void CreateTexture() + public virtual void CreateTexture() { Texture = Device.CreateTexture2D( Format.B8G8R8A8_UNorm, @@ -62,7 +63,7 @@ namespace BizHawk.Bizware.Graphics SRV = Device.CreateShaderResourceView(Texture, srvd); } - public void DestroyTexture() + public virtual void DestroyTexture() { SRV?.Dispose(); SRV = null; diff --git a/src/BizHawk.Bizware.Graphics/D3D11/IGL_D3D11.cs b/src/BizHawk.Bizware.Graphics/D3D11/IGL_D3D11.cs index 5d9263f9f0..b25d729dbf 100644 --- a/src/BizHawk.Bizware.Graphics/D3D11/IGL_D3D11.cs +++ b/src/BizHawk.Bizware.Graphics/D3D11/IGL_D3D11.cs @@ -38,10 +38,10 @@ namespace BizHawk.Bizware.Graphics private ID3D11RasterizerState RasterizerState => _resources.RasterizerState; private FeatureLevel DeviceFeatureLevel => _resources.DeviceFeatureLevel; + private D3D11RenderTarget CurRenderTarget => _resources.CurRenderTarget; // rendering state private Pipeline _curPipeline; - private RenderTarget _curRenderTarget; private D3D11SwapChain.SwapChainResources _controlSwapChain; public IGL_D3D11() @@ -151,32 +151,9 @@ namespace BizHawk.Bizware.Graphics sw.PS = null; } - foreach (var rw in _resources.RenderTargets.Select(rt => (RenderTargetWrapper)rt.Opaque)) - { - rw.RTV.Dispose(); - rw.RTV = null; - } - - foreach (var tex2d in _resources.ShaderTextures) - { - tex2d.DestroyTexture(); - } - - _resources.Dispose(); + _resources.DestroyResources(); _resources.CreateResources(); - foreach (var tex2d in _resources.ShaderTextures) - { - tex2d.CreateTexture(); - } - - foreach (var rt in _resources.RenderTargets) - { - var rw = (RenderTargetWrapper)rt.Opaque; - var rtvd = new RenderTargetViewDescription(RenderTargetViewDimension.Texture2D, Format.B8G8R8A8_UNorm); - rw.RTV = Device.CreateRenderTargetView(((D3D11Texture2D)rt.Texture2D).Texture, rtvd); - } - foreach (var sw in _resources.VertexShaders.Select(vertexShader => (ShaderWrapper)vertexShader.Opaque)) { sw.VS = Device.CreateVertexShader(sw.Bytecode.Span); @@ -264,17 +241,7 @@ namespace BizHawk.Bizware.Graphics } public void ClearColor(Color color) - { - if (_curRenderTarget == null) - { - Context.ClearRenderTargetView(_controlSwapChain.RTV, new(color.R, color.B, color.G, color.A)); - } - else - { - var rw = (RenderTargetWrapper)_curRenderTarget.Opaque; - Context.ClearRenderTargetView(rw.RTV, new(color.R, color.B, color.G, color.A)); - } - } + => Context.ClearRenderTargetView(CurRenderTarget?.RTV ?? _controlSwapChain.RTV, new(color.R, color.B, color.G, color.A)); private class ShaderWrapper // Disposable fields cleaned up by Internal_FreeShader { @@ -607,11 +574,6 @@ namespace BizHawk.Bizware.Graphics public ShaderWrapper VertexShader, FragmentShader; } - private class RenderTargetWrapper - { - public ID3D11RenderTargetView RTV; - } - private class VertexLayoutWrapper { public ID3D11InputLayout VertexInputLayout; @@ -875,36 +837,13 @@ namespace BizHawk.Bizware.Graphics Context.RSSetScissorRect(x, y, width, height); } - public void FreeRenderTarget(RenderTarget rt) + public IRenderTarget CreateRenderTarget(int width, int height) + => new D3D11RenderTarget(_resources, width, height); + + public void BindDefaultRenderTarget() { - var rw = (RenderTargetWrapper)rt.Opaque; - rw.RTV.Dispose(); - _resources.RenderTargets.Remove(rt); - rt.Texture2D.Dispose(); - } - - public RenderTarget CreateRenderTarget(int width, int height) - { - var tex = new D3D11Texture2D(_resources, BindFlags.ShaderResource | BindFlags.RenderTarget, ResourceUsage.Default, CpuAccessFlags.None, width, height); - var rtvd = new RenderTargetViewDescription(RenderTargetViewDimension.Texture2D, Format.B8G8R8A8_UNorm); - var rw = new RenderTargetWrapper { RTV = Device.CreateRenderTargetView(tex.Texture, rtvd) }; - var rt = new RenderTarget(this, rw, tex); - _resources.RenderTargets.Add(rt); - return rt; - } - - public void BindRenderTarget(RenderTarget rt) - { - _curRenderTarget = rt; - - if (rt == null) - { - Context.OMSetRenderTargets(_controlSwapChain.RTV); - return; - } - - var rw = (RenderTargetWrapper)rt.Opaque; - Context.OMSetRenderTargets(rw.RTV); + _resources.CurRenderTarget = null; + Context.OMSetRenderTargets(_controlSwapChain.RTV); } public void Draw(IntPtr data, int count) diff --git a/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusControlRenderTarget.cs b/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusControlRenderTarget.cs new file mode 100644 index 0000000000..f56f9d5ab8 --- /dev/null +++ b/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusControlRenderTarget.cs @@ -0,0 +1,37 @@ +using System; +using System.Drawing; + +using SDGraphics = System.Drawing.Graphics; + +namespace BizHawk.Bizware.Graphics +{ + public sealed class GDIPlusControlRenderTarget : IDisposable + { + private readonly Func<(SDGraphics, Rectangle)> _getControlRenderContext; + private BufferedGraphicsContext _bufferedGraphicsContext = new(); + + public SDGraphics ControlGraphics; + public BufferedGraphics BufferedGraphics; + + internal GDIPlusControlRenderTarget(Func<(SDGraphics Graphics, Rectangle Rectangle)> getControlRenderContext) + => _getControlRenderContext = getControlRenderContext; + + public void Dispose() + { + ControlGraphics?.Dispose(); + ControlGraphics = null; + BufferedGraphics?.Dispose(); + BufferedGraphics = null; + _bufferedGraphicsContext?.Dispose(); + _bufferedGraphicsContext = null; + } + + public void CreateGraphics() + { + ControlGraphics?.Dispose(); + BufferedGraphics?.Dispose(); + (ControlGraphics, var r) = _getControlRenderContext(); + BufferedGraphics = _bufferedGraphicsContext.Allocate(ControlGraphics, r); + } + } +} diff --git a/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusRenderTarget.cs b/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusRenderTarget.cs index 38b8c5065f..71476c7371 100644 --- a/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusRenderTarget.cs +++ b/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusRenderTarget.cs @@ -1,63 +1,30 @@ -using System; -using System.Drawing; - using SDGraphics = System.Drawing.Graphics; namespace BizHawk.Bizware.Graphics { - public class GDIPlusRenderTarget : IDisposable + internal sealed class GDIPlusRenderTarget : GDIPlusTexture2D, IRenderTarget { - internal GDIPlusRenderTarget(Func getBufferedGraphicsContext, - Func<(SDGraphics Graphics, Rectangle Rectangle)> getControlRenderContext = null) + private readonly IGL_GDIPlus _gdiPlus; + public SDGraphics TextureGraphics; + + internal GDIPlusRenderTarget(IGL_GDIPlus gdiPlus, int width, int height) + : base(width, height) { - _getBufferedGraphicsContext = getBufferedGraphicsContext; - _getControlRenderContext = getControlRenderContext; + _gdiPlus = gdiPlus; + TextureGraphics = SDGraphics.FromImage(SDBitmap); } - public void Dispose() + public override void Dispose() { - if (_getControlRenderContext != null) - { - CurGraphics?.Dispose(); - CurGraphics = null; - } - - BufferedGraphics?.Dispose(); - BufferedGraphics = null; + TextureGraphics?.Dispose(); + TextureGraphics = null; + base.Dispose(); } - private readonly Func _getBufferedGraphicsContext; + public void Bind() + => _gdiPlus.CurRenderTarget = this; - /// - /// get Graphics and Rectangle from a control, if any - /// - private readonly Func<(SDGraphics, Rectangle)> _getControlRenderContext; - - /// - /// the offscreen render target, if that's what this is representing - /// - public RenderTarget Target; - - public SDGraphics CurGraphics; - public BufferedGraphics BufferedGraphics; - - public void CreateGraphics() - { - Rectangle r; - if (_getControlRenderContext != null) - { - (CurGraphics, r) = _getControlRenderContext(); - } - else - { - var gtex = (GDIPlusTexture2D)Target.Texture2D; - CurGraphics?.Dispose(); - CurGraphics = SDGraphics.FromImage(gtex.SDBitmap); - r = gtex.GetRectangle(); - } - - BufferedGraphics?.Dispose(); - BufferedGraphics = _getBufferedGraphicsContext().Allocate(CurGraphics, r); - } + public override string ToString() + => $"GDI+ RenderTarget: {Width}x{Height}"; } } diff --git a/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusTexture2D.cs b/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusTexture2D.cs index 27bedf58b8..97654fea32 100644 --- a/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusTexture2D.cs +++ b/src/BizHawk.Bizware.Graphics/GDIPlus/GDIPlusTexture2D.cs @@ -3,7 +3,7 @@ using System.Drawing.Imaging; namespace BizHawk.Bizware.Graphics { - public class GDIPlusTexture2D : ITexture2D + internal class GDIPlusTexture2D : ITexture2D { public Bitmap SDBitmap; public bool LinearFiltering; @@ -19,7 +19,7 @@ namespace BizHawk.Bizware.Graphics SDBitmap = new(width, height, PixelFormat.Format32bppArgb); } - public void Dispose() + public virtual void Dispose() { SDBitmap?.Dispose(); SDBitmap = null; diff --git a/src/BizHawk.Bizware.Graphics/GDIPlus/IGL_GDIPlus.cs b/src/BizHawk.Bizware.Graphics/GDIPlus/IGL_GDIPlus.cs index 9145213b5d..2cd54b3f5c 100644 --- a/src/BizHawk.Bizware.Graphics/GDIPlus/IGL_GDIPlus.cs +++ b/src/BizHawk.Bizware.Graphics/GDIPlus/IGL_GDIPlus.cs @@ -5,15 +5,22 @@ using System.Numerics; using SDGraphics = System.Drawing.Graphics; -//TODO - maybe a layer to cache Graphics parameters (notably, filtering) ? namespace BizHawk.Bizware.Graphics { public class IGL_GDIPlus : IGL { + private GDIPlusControlRenderTarget _controlRenderTarget; + + internal GDIPlusRenderTarget CurRenderTarget; + public EDispMethod DispMethodEnum => EDispMethod.GdiPlus; public void Dispose() - => BufferedGraphicsContext.Dispose(); + { + } + + internal SDGraphics GetCurrentGraphics() + => CurRenderTarget?.TextureGraphics ?? _controlRenderTarget.BufferedGraphics.Graphics; public void ClearColor(Color color) => GetCurrentGraphics().Clear(color); @@ -118,62 +125,21 @@ namespace BizHawk.Bizware.Graphics { } - public void FreeRenderTarget(RenderTarget rt) - { - var grt = (GDIPlusRenderTarget)rt.Opaque; - grt.Dispose(); - } + public IRenderTarget CreateRenderTarget(int width, int height) + => new GDIPlusRenderTarget(this, width, height); - public RenderTarget CreateRenderTarget(int width, int height) - { - var tex2d = new GDIPlusTexture2D(width, height); - var grt = new GDIPlusRenderTarget(() => BufferedGraphicsContext); - var rt = new RenderTarget(this, grt, tex2d); - grt.Target = rt; - return rt; - } + public void BindDefaultRenderTarget() + => CurRenderTarget = null; - public void BindRenderTarget(RenderTarget rt) - { - if (_currOffscreenGraphics != null) - { - _currOffscreenGraphics.Dispose(); - _currOffscreenGraphics = null; - } - - if (rt == null) - { - // null means to use the default RT for the current control - CurrentRenderTarget = _controlRenderTarget; - } - else - { - var gtex = (GDIPlusTexture2D)rt.Texture2D; - CurrentRenderTarget = (GDIPlusRenderTarget)rt.Opaque; - _currOffscreenGraphics = SDGraphics.FromImage(gtex.SDBitmap); - } - } - - private GDIPlusRenderTarget _controlRenderTarget; - - public GDIPlusRenderTarget CreateControlRenderTarget(Func<(SDGraphics Graphics, Rectangle Rectangle)> getControlRenderContext) + public GDIPlusControlRenderTarget CreateControlRenderTarget(Func<(SDGraphics Graphics, Rectangle Rectangle)> getControlRenderContext) { if (_controlRenderTarget != null) { throw new InvalidOperationException($"{nameof(IGL_GDIPlus)} can only have one control render target"); } - _controlRenderTarget = new(() => BufferedGraphicsContext, getControlRenderContext); + _controlRenderTarget = new(getControlRenderContext); return _controlRenderTarget; } - - private SDGraphics _currOffscreenGraphics; - - public SDGraphics GetCurrentGraphics() - => _currOffscreenGraphics ?? CurrentRenderTarget.BufferedGraphics.Graphics; - - public GDIPlusRenderTarget CurrentRenderTarget; - - public readonly BufferedGraphicsContext BufferedGraphicsContext = new(); } } diff --git a/src/BizHawk.Bizware.Graphics/Interfaces/IGL.cs b/src/BizHawk.Bizware.Graphics/Interfaces/IGL.cs index 1368729435..a229f279b3 100644 --- a/src/BizHawk.Bizware.Graphics/Interfaces/IGL.cs +++ b/src/BizHawk.Bizware.Graphics/Interfaces/IGL.cs @@ -19,17 +19,17 @@ namespace BizHawk.Bizware.Graphics EDispMethod DispMethodEnum { get; } /// - /// Clears the color buffer with the specified color + /// Clears the currently bound render target with the specified color /// void ClearColor(Color color); /// - /// compile a fragment shader. This is the simplified method. A more complex method may be added later which will accept multiple sources and preprocessor definitions independently + /// Compile a fragment shader. This is the simplified method. A more complex method may be added later which will accept multiple sources and preprocessor definitions independently /// Shader CreateFragmentShader(string source, string entry, bool required); /// - /// compile a vertex shader. This is the simplified method. A more complex method may be added later which will accept multiple sources and preprocessor definitions independently + /// Compile a vertex shader. This is the simplified method. A more complex method may be added later which will accept multiple sources and preprocessor definitions independently /// Shader CreateVertexShader(string source, string entry, bool required); @@ -59,40 +59,40 @@ namespace BizHawk.Bizware.Graphics void SetPipelineUniformMatrix(PipelineUniform uniform, ref Matrix4x4 mat, bool transpose); /// - /// sets a uniform value + /// Sets a uniform value /// void SetPipelineUniform(PipelineUniform uniform, Vector4 value); /// - /// sets a uniform value + /// Sets a uniform value /// void SetPipelineUniform(PipelineUniform uniform, Vector2 value); /// - /// sets a uniform value + /// Sets a uniform value /// void SetPipelineUniform(PipelineUniform uniform, float value); /// - /// sets uniform values + /// Sets uniform values /// void SetPipelineUniform(PipelineUniform uniform, Vector4[] values); /// - /// sets a uniform value + /// Sets a uniform value /// void SetPipelineUniform(PipelineUniform uniform, bool value); /// /// Draws based on the currently set pipeline - /// data contains vertexes based on the pipeline's VertexLayout - /// count is the vertex count + /// Data contains vertexes based on the pipeline's VertexLayout + /// Count is the vertex count /// Vertexes must form triangle strips /// void Draw(IntPtr data, int count); /// - /// creates a vertex layout resource + /// Creates a vertex layout resource /// VertexLayout CreateVertexLayout(); @@ -119,35 +119,32 @@ namespace BizHawk.Bizware.Graphics ITexture2D WrapGLTexture2D(int glTexId, int width, int height); /// - /// sets the viewport (and scissor) according to the provided specifications + /// Sets the viewport (and scissor) according to the provided specifications /// void SetViewport(int x, int y, int width, int height); /// - /// generates a proper 2d othographic projection for the given destination size, suitable for use in a GUI + /// Generates a proper 2D othographic projection for the given destination size, suitable for use in a GUI /// Matrix4x4 CreateGuiProjectionMatrix(int width, int height); /// - /// generates a proper view transform for a standard 2d ortho projection, including half-pixel jitter if necessary and - /// re-establishing of a normal 2d graphics top-left origin. suitable for use in a GUI + /// Generates a proper view transform for a standard 2D othographic projection, including half-pixel jitter if necessary + /// and re-establishing of a normal 2D graphics top-left origin. Suitable for use in a GUI /// Matrix4x4 CreateGuiViewMatrix(int width, int height, bool autoflip = true); /// - /// Creates a render target. Only includes a color buffer. Pixel format control TBD + /// Creates a render target. Only includes a color buffer, and will always be in byte order BGRA (i.e. little endian ARGB) + /// This may unbind a previously bound render target /// - RenderTarget CreateRenderTarget(int width, int height); + IRenderTarget CreateRenderTarget(int width, int height); /// - /// Binds a RenderTarget for current rendering + /// Binds the IGL's default render target (i.e. to the IGL's control) + /// This implicitly unbinds any previously bound IRenderTarget /// - void BindRenderTarget(RenderTarget rt); - - /// - /// Frees the provided render target. Same as disposing the resource. - /// - void FreeRenderTarget(RenderTarget rt); + void BindDefaultRenderTarget(); /// /// Frees the provided pipeline. Same as disposing the resource. diff --git a/src/BizHawk.Bizware.Graphics/Interfaces/IRenderTarget.cs b/src/BizHawk.Bizware.Graphics/Interfaces/IRenderTarget.cs new file mode 100644 index 0000000000..59515f433c --- /dev/null +++ b/src/BizHawk.Bizware.Graphics/Interfaces/IRenderTarget.cs @@ -0,0 +1,15 @@ +namespace BizHawk.Bizware.Graphics +{ + /// + /// A render target, essentially just a 2D texture which can be written as if it was a window + /// As this is effectively a texture, it inherits ITexture2D + /// However, note that semantically the CPU shouldn't be writing to a render target, so LoadFrom might not work! + /// + public interface IRenderTarget : ITexture2D + { + /// + /// Binds this render target + /// + void Bind(); + } +} diff --git a/src/BizHawk.Bizware.Graphics/OpenGL/IGL_OpenGL.cs b/src/BizHawk.Bizware.Graphics/OpenGL/IGL_OpenGL.cs index 2640eec8ac..f1c20ee32d 100644 --- a/src/BizHawk.Bizware.Graphics/OpenGL/IGL_OpenGL.cs +++ b/src/BizHawk.Bizware.Graphics/OpenGL/IGL_OpenGL.cs @@ -34,7 +34,7 @@ namespace BizHawk.Bizware.Graphics // rendering state private Pipeline _currPipeline; - private RenderTarget _currRenderTarget; + internal bool DefaultRenderTargetBound; // this IGL either requires at least OpenGL 3.0 public static bool Available => OpenGLVersion.SupportsVersion(3, 0); @@ -451,50 +451,14 @@ namespace BizHawk.Bizware.Graphics public ITexture2D WrapGLTexture2D(int glTexId, int width, int height) => new OpenGLTexture2D(GL, (uint)glTexId, width, height); - public void FreeRenderTarget(RenderTarget rt) - { - rt.Texture2D.Dispose(); - GL.DeleteFramebuffer((uint)rt.Opaque); - } - /// framebuffer creation unsuccessful - public RenderTarget CreateRenderTarget(int width, int height) + public IRenderTarget CreateRenderTarget(int width, int height) + => new OpenGLRenderTarget(this, GL, width, height); + + public void BindDefaultRenderTarget() { - // create a texture for it - var tex2d = new OpenGLTexture2D(GL, width, height); - - // create the FBO - var fbId = GL.GenFramebuffer(); - GL.BindFramebuffer(FramebufferTarget.Framebuffer, fbId); - - // bind the tex to the FBO - GL.FramebufferTexture2D(FramebufferTarget.Framebuffer, FramebufferAttachment.ColorAttachment0, TextureTarget.Texture2D, tex2d.TexID, 0); - - // do something, I guess say which color buffers are used by the framebuffer - GL.DrawBuffer(DrawBufferMode.ColorAttachment0); - - if ((FramebufferStatus)GL.CheckFramebufferStatus(FramebufferTarget.Framebuffer) != FramebufferStatus.Complete) - { - throw new InvalidOperationException($"Error creating framebuffer (at {nameof(GL.CheckFramebufferStatus)})"); - } - - // since we're done configuring unbind this framebuffer, to return to the default GL.BindFramebuffer(FramebufferTarget.Framebuffer, 0); - - return new(this, fbId, tex2d); - } - - public void BindRenderTarget(RenderTarget rt) - { - _currRenderTarget = rt; - if (rt == null) - { - GL.BindFramebuffer(FramebufferTarget.Framebuffer, 0); - } - else - { - GL.BindFramebuffer(FramebufferTarget.Framebuffer, (uint)rt.Opaque); - } + DefaultRenderTargetBound = true; } public Matrix4x4 CreateGuiProjectionMatrix(int width, int height) @@ -511,7 +475,7 @@ namespace BizHawk.Bizware.Graphics ret.M22 = -1.0f; ret.M41 = width * -0.5f; ret.M42 = height * 0.5f; - if (autoflip && _currRenderTarget is not null) // flip as long as we're not a final render target + if (autoflip && !DefaultRenderTargetBound) // flip as long as we're not a final render target { ret.M22 = 1.0f; ret.M42 *= -1; diff --git a/src/BizHawk.Bizware.Graphics/OpenGL/OpenGLRenderTarget.cs b/src/BizHawk.Bizware.Graphics/OpenGL/OpenGLRenderTarget.cs new file mode 100644 index 0000000000..14bfc3a7f8 --- /dev/null +++ b/src/BizHawk.Bizware.Graphics/OpenGL/OpenGLRenderTarget.cs @@ -0,0 +1,52 @@ +using System; + +using Silk.NET.OpenGL.Legacy; + +namespace BizHawk.Bizware.Graphics +{ + internal sealed class OpenGLRenderTarget : OpenGLTexture2D, IRenderTarget + { + private readonly IGL_OpenGL _openGL; + private readonly GL GL; + + public readonly uint FBO; + + public OpenGLRenderTarget(IGL_OpenGL openGL, GL gl, int width, int height) + : base(gl, width, height) + { + _openGL = openGL; + GL = gl; + + // create the FBO + FBO = GL.GenFramebuffer(); + Bind(); + + // bind the tex to the FBO + GL.FramebufferTexture2D(FramebufferTarget.Framebuffer, FramebufferAttachment.ColorAttachment0, TextureTarget.Texture2D, TexID, 0); + + // do something, I guess say which color buffers are used by the framebuffer + GL.DrawBuffer(DrawBufferMode.ColorAttachment0); + + if ((FramebufferStatus)GL.CheckFramebufferStatus(FramebufferTarget.Framebuffer) != FramebufferStatus.Complete) + { + Dispose(); + throw new InvalidOperationException($"Error creating framebuffer (at {nameof(GL.CheckFramebufferStatus)})"); + } + } + + public override void Dispose() + { + GL.DeleteFramebuffer(FBO); + base.Dispose(); + } + + public void Bind() + { + GL.BindFramebuffer(FramebufferTarget.Framebuffer, FBO); + _openGL.DefaultRenderTargetBound = false; + } + + public override string ToString() + => $"OpenGL RenderTarget: {Width}x{Height}"; + } +} diff --git a/src/BizHawk.Bizware.Graphics/OpenGL/OpenGLTexture2D.cs b/src/BizHawk.Bizware.Graphics/OpenGL/OpenGLTexture2D.cs index 6a879081a4..42782b70b7 100644 --- a/src/BizHawk.Bizware.Graphics/OpenGL/OpenGLTexture2D.cs +++ b/src/BizHawk.Bizware.Graphics/OpenGL/OpenGLTexture2D.cs @@ -44,7 +44,7 @@ namespace BizHawk.Bizware.Graphics IsUpsideDown = true; } - public void Dispose() + public virtual void Dispose() => GL.DeleteTexture(TexID); public unsafe BitmapBuffer Resolve() diff --git a/src/BizHawk.Bizware.Graphics/RenderTarget.cs b/src/BizHawk.Bizware.Graphics/RenderTarget.cs deleted file mode 100644 index 75dc08f7f5..0000000000 --- a/src/BizHawk.Bizware.Graphics/RenderTarget.cs +++ /dev/null @@ -1,38 +0,0 @@ -using System; - -namespace BizHawk.Bizware.Graphics -{ - public class RenderTarget : IDisposable - { - public RenderTarget(IGL owner, object opaque, ITexture2D tex) - { - Owner = owner; - Opaque = opaque; - Texture2D = tex; - } - - public override string ToString() - { - return $"GL RT: {Texture2D.Width}x{Texture2D.Height}"; - } - - public object Opaque { get; } - public IGL Owner { get; } - public ITexture2D Texture2D { get; } - - public void Unbind() - { - Owner.BindRenderTarget(null); - } - - public void Bind() - { - Owner.BindRenderTarget(this); - } - - public void Dispose() - { - Owner.FreeRenderTarget(this); - } - } -} \ No newline at end of file diff --git a/src/BizHawk.Client.Common/DisplayManager/DisplayManagerBase.cs b/src/BizHawk.Client.Common/DisplayManager/DisplayManagerBase.cs index 1b43e93b61..b5e6f53ad8 100644 --- a/src/BizHawk.Client.Common/DisplayManager/DisplayManagerBase.cs +++ b/src/BizHawk.Client.Common/DisplayManager/DisplayManagerBase.cs @@ -32,14 +32,14 @@ namespace BizHawk.Client.Common protected class DisplayManagerRenderTargetProvider : IRenderTargetProvider { - private readonly Func _callback; + private readonly Func _callback; - RenderTarget IRenderTargetProvider.Get(Size size) + public IRenderTarget Get(Size size) { return _callback(size); } - public DisplayManagerRenderTargetProvider(Func callback) + public DisplayManagerRenderTargetProvider(Func callback) { _callback = callback; } @@ -864,7 +864,7 @@ namespace BizHawk.Client.Common public void Blank() { ActivateGraphicsControlContext(); - _gl.BindRenderTarget(null); + _gl.BindDefaultRenderTarget(); _gl.ClearColor(Color.Black); SwapBuffersOfGraphicsControl(); } @@ -880,11 +880,11 @@ namespace BizHawk.Client.Common RunFilterChainSteps(ref rtCounter, out var rtCurr, out _); - job.OffscreenBb = rtCurr.Texture2D.Resolve(); + job.OffscreenBb = rtCurr.Resolve(); job.OffscreenBb.DiscardAlpha(); } - protected void RunFilterChainSteps(ref int rtCounter, out RenderTarget rtCurr, out bool inFinalTarget) + protected void RunFilterChainSteps(ref int rtCounter, out IRenderTarget rtCurr, out bool inFinalTarget) { ITexture2D texCurr = null; rtCurr = null; @@ -907,7 +907,7 @@ namespace BizHawk.Client.Common break; case FilterProgram.ProgramStepType.FinalTarget: _currentFilterProgram.CurrRenderTarget = rtCurr = null; - _gl.BindRenderTarget(rtCurr); + _gl.BindDefaultRenderTarget(); inFinalTarget = true; break; default: diff --git a/src/BizHawk.Client.Common/DisplayManager/FilterManager.cs b/src/BizHawk.Client.Common/DisplayManager/FilterManager.cs index 4624c6a8b8..bc17cc4a99 100644 --- a/src/BizHawk.Client.Common/DisplayManager/FilterManager.cs +++ b/src/BizHawk.Client.Common/DisplayManager/FilterManager.cs @@ -38,7 +38,7 @@ namespace BizHawk.Client.Common.FilterManager public interface IRenderTargetProvider { - RenderTarget Get(Size size); + IRenderTarget Get(Size size); } public class FilterProgram @@ -69,9 +69,9 @@ namespace BizHawk.Client.Common.FilterManager public IGL GL; public IRenderTargetProvider RenderTargetProvider; - public RenderTarget CurrRenderTarget; + public IRenderTarget CurrRenderTarget; - public RenderTarget GetTempTarget(int width, int height) + public IRenderTarget GetTempTarget(int width, int height) { return RenderTargetProvider.Get(new(width, height)); } diff --git a/src/BizHawk.Client.Common/DisplayManager/Filters/BaseFilter.cs b/src/BizHawk.Client.Common/DisplayManager/Filters/BaseFilter.cs index 172714fa79..1aff988f3a 100644 --- a/src/BizHawk.Client.Common/DisplayManager/Filters/BaseFilter.cs +++ b/src/BizHawk.Client.Common/DisplayManager/Filters/BaseFilter.cs @@ -111,7 +111,7 @@ namespace BizHawk.Client.Common.Filters } // TODO - why a different param order than DeclareOutput? - protected RenderTarget GetTempTarget(int width, int height) + protected IRenderTarget GetTempTarget(int width, int height) { return FilterProgram.GetTempTarget(width, height); } diff --git a/src/BizHawk.Client.Common/DisplayManager/Filters/Utils.cs b/src/BizHawk.Client.Common/DisplayManager/Filters/Utils.cs index d8c5d273e9..69ad542ae1 100644 --- a/src/BizHawk.Client.Common/DisplayManager/Filters/Utils.cs +++ b/src/BizHawk.Client.Common/DisplayManager/Filters/Utils.cs @@ -71,7 +71,7 @@ namespace BizHawk.Client.Common.Filters public override void Run() { - YieldOutput(FilterProgram.CurrRenderTarget.Texture2D); + YieldOutput(FilterProgram.CurrRenderTarget); } } } \ No newline at end of file diff --git a/src/BizHawk.Client.Common/DisplayManager/RenderTargetFrugalizer.cs b/src/BizHawk.Client.Common/DisplayManager/RenderTargetFrugalizer.cs index 12b9a5f286..059dae0cb8 100644 --- a/src/BizHawk.Client.Common/DisplayManager/RenderTargetFrugalizer.cs +++ b/src/BizHawk.Client.Common/DisplayManager/RenderTargetFrugalizer.cs @@ -30,26 +30,26 @@ namespace BizHawk.Client.Common private void ResetList() { - _currentRenderTargets = new List { null, null }; + _currentRenderTargets = new List { null, null }; } private readonly IGL _gl; - private List _currentRenderTargets; + private List _currentRenderTargets; - public RenderTarget Get(Size dimensions) + public IRenderTarget Get(Size dimensions) { return Get(dimensions.Width, dimensions.Height); } - public RenderTarget Get(int width, int height) + public IRenderTarget Get(int width, int height) { //get the current entry var currentRenderTarget = _currentRenderTargets[0]; //check if its rotten and needs recreating if (currentRenderTarget == null - || currentRenderTarget.Texture2D.Width != width - || currentRenderTarget.Texture2D.Height != height) + || currentRenderTarget.Width != width + || currentRenderTarget.Height != height) { // needs recreating. be sure to kill the old one... currentRenderTarget?.Dispose(); diff --git a/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs b/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs index f126d48a78..690d167fd3 100644 --- a/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs +++ b/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs @@ -99,7 +99,7 @@ namespace BizHawk.Client.EmuHawk if (job.Offscreen) { - job.OffscreenBb = rtCurr.Texture2D.Resolve(); + job.OffscreenBb = rtCurr.Resolve(); job.OffscreenBb.DiscardAlpha(); return; } diff --git a/src/BizHawk.Client.EmuHawk/GraphicsImplementations/RetainedGraphicsControl.cs b/src/BizHawk.Client.EmuHawk/GraphicsImplementations/RetainedGraphicsControl.cs index c639a54858..d82160431f 100644 --- a/src/BizHawk.Client.EmuHawk/GraphicsImplementations/RetainedGraphicsControl.cs +++ b/src/BizHawk.Client.EmuHawk/GraphicsImplementations/RetainedGraphicsControl.cs @@ -40,7 +40,7 @@ namespace BizHawk.Client.EmuHawk private readonly IGL _gl; private readonly GraphicsControl _graphicsControl; - private RenderTarget _rt; + private IRenderTarget _rt; private readonly GuiRenderer _guiRenderer; protected override void OnPaint(PaintEventArgs e) @@ -75,7 +75,7 @@ namespace BizHawk.Client.EmuHawk { if (_retain) { - _rt.Unbind(); + _gl.BindDefaultRenderTarget(); } _graphicsControl.End(); @@ -91,7 +91,7 @@ namespace BizHawk.Client.EmuHawk } // if we're retaining, then we cant draw until we unbind! its semantically a bit odd, but we expect users to call SwapBuffers() before end, so we cant unbind in End() even thoug hit makes a bit more sense. - _rt.Unbind(); + _gl.BindDefaultRenderTarget(); Draw(); } @@ -104,9 +104,9 @@ namespace BizHawk.Client.EmuHawk _guiRenderer.Begin(Width, Height); _guiRenderer.DisableBlending(); - _guiRenderer.Draw(_rt.Texture2D); + _guiRenderer.Draw(_rt); _guiRenderer.End(); _graphicsControl.SwapBuffers(); } } -} \ No newline at end of file +}