diff --git a/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs b/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs new file mode 100644 index 0000000000..339a274f57 --- /dev/null +++ b/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs @@ -0,0 +1,145 @@ +using System; +using System.Diagnostics; +using System.Drawing; + +using BizHawk.Bizware.BizwareGL; +using BizHawk.Bizware.DirectX; +using BizHawk.Client.Common; +using BizHawk.Emulation.Common; + +namespace BizHawk.Client.EmuHawk +{ + public class DisplayManager : DisplayManagerBase + { + private readonly Func _getIsSecondaryThrottlingDisabled; + + private bool? _lastVsyncSetting; + + private GraphicsControl _lastVsyncSettingGraphicsControl; + + // rendering resources: + + private readonly GLManager _glManager; + + // layer resources + + private readonly PresentationPanel _presentationPanel; // well, its the final layer's target, at least + + private readonly GLManager.ContextRef _crGraphicsControl; + + private GraphicsControl _graphicsControl => _presentationPanel.GraphicsControl; + + public DisplayManager( + Config config, + IEmulator emulator, + InputManager inputManager, + IMovieSession movieSession, + IGL gl, + PresentationPanel presentationPanel, + Func getIsSecondaryThrottlingDisabled) + : base(config, emulator, inputManager, movieSession, gl.DispMethodEnum(), gl, gl.CreateRenderer()) + { + _presentationPanel = presentationPanel; + _getIsSecondaryThrottlingDisabled = getIsSecondaryThrottlingDisabled; + + // setup the GL context manager, needed for coping with multiple opengl cores vs opengl display method + // but is it tho? --yoshi + _glManager = GLManager.Instance; + _crGraphicsControl = _glManager.GetContextForGraphicsControl(_graphicsControl); + } + + protected override void ActivateGLContext() => _glManager.Activate(_crGraphicsControl); + + protected override Size GetGraphicsControlSize() => _graphicsControl.Size; + + public override Size GetPanelNativeSize() => _presentationPanel.NativeSize; + + protected override Point GraphicsControlPointToClient(Point p) => _graphicsControl.PointToClient(p); + + protected override void SwapBuffersOfGraphicsControl() => _graphicsControl.SwapBuffers(); + + protected override void UpdateSourceDrawingWork(JobInfo job) + { + bool alternateVsync = false; + + // only used by alternate vsync + IGL_SlimDX9 dx9 = null; + + if (!job.Offscreen) + { + //apply the vsync setting (should probably try to avoid repeating this) + var vsync = GlobalConfig.VSyncThrottle || GlobalConfig.VSync; + + //ok, now this is a bit undesirable. + //maybe the user wants vsync, but not vsync throttle. + //this makes sense... but we don't have the infrastructure to support it now (we'd have to enable triple buffering or something like that) + //so what we're gonna do is disable vsync no matter what if throttling is off, and maybe nobody will notice. + //update 26-mar-2016: this upsets me. When fast-forwarding and skipping frames, vsync should still work. But I'm not changing it yet + if (_getIsSecondaryThrottlingDisabled()) + vsync = false; + + //for now, it's assumed that the presentation panel is the main window, but that may not always be true + if (vsync && GlobalConfig.DispAlternateVsync && GlobalConfig.VSyncThrottle) + { + dx9 = _gl as IGL_SlimDX9; + if (dx9 != null) + { + alternateVsync = true; + //unset normal vsync if we've chosen the alternate vsync + vsync = false; + } + } + + //TODO - whats so hard about triple buffering anyway? just enable it always, and change api to SetVsync(enable,throttle) + //maybe even SetVsync(enable,throttlemethod) or just SetVsync(enable,throttle,advanced) + + if (_lastVsyncSetting != vsync || _lastVsyncSettingGraphicsControl != _graphicsControl) + { + if (_lastVsyncSetting == null && vsync) + { + // Workaround for vsync not taking effect at startup (Intel graphics related?) + _graphicsControl.SetVsync(false); + } + _graphicsControl.SetVsync(vsync); + _lastVsyncSettingGraphicsControl = _graphicsControl; + _lastVsyncSetting = vsync; + } + } + + // begin rendering on this context + // should this have been done earlier? + // do i need to check this on an intel video card to see if running excessively is a problem? (it used to be in the FinalTarget command below, shouldn't be a problem) + //GraphicsControl.Begin(); // CRITICAL POINT for yabause+GL + + //TODO - auto-create and age these (and dispose when old) + int rtCounter = 0; + + _currentFilterProgram.RenderTargetProvider = new DisplayManagerRenderTargetProvider(size => _shaderChainFrugalizers[rtCounter++].Get(size)); + + _gl.BeginScene(); + RunFilterChainSteps(ref rtCounter, out var rtCurr, out var inFinalTarget); + _gl.EndScene(); + + if (job.Offscreen) + { + job.OffscreenBb = rtCurr.Texture2d.Resolve(); + job.OffscreenBb.DiscardAlpha(); + return; + } + + Debug.Assert(inFinalTarget); + + // wait for vsync to begin + if (alternateVsync) dx9.AlternateVsyncPass(0); + + // present and conclude drawing + _graphicsControl.SwapBuffers(); + + // wait for vsync to end + if (alternateVsync) dx9.AlternateVsyncPass(1); + + // nope. don't do this. workaround for slow context switching on intel GPUs. just switch to another context when necessary before doing anything + // presentationPanel.GraphicsControl.End(); + } + } +} diff --git a/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManagerBase.cs b/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManagerBase.cs index 2b7db04de2..1f647c754a 100644 --- a/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManagerBase.cs +++ b/src/BizHawk.Client.EmuHawk/DisplayManager/DisplayManagerBase.cs @@ -2,25 +2,22 @@ // we could flag textures as 'actually' render targets (keep a reference to the render target?) which could allow us to convert between them more quickly in some cases using System; -using System.IO; -using System.Linq; using System.Collections.Generic; -using System.Diagnostics; using System.Drawing; using System.Drawing.Text; +using System.IO; +using System.Linq; using System.Runtime.InteropServices; using BizHawk.Bizware.BizwareGL; using BizHawk.Bizware.BizwareGL.DrawingExtensions; -using BizHawk.Bizware.DirectX; -using BizHawk.Bizware.OpenTK3; using BizHawk.Client.Common; -using BizHawk.Client.Common.Filters; using BizHawk.Client.Common.FilterManager; +using BizHawk.Client.Common.Filters; using BizHawk.Common.PathExtensions; using BizHawk.Emulation.Common; -using BizHawk.Emulation.Cores.Sony.PSX; using BizHawk.Emulation.Cores.Consoles.Nintendo.NDS; +using BizHawk.Emulation.Cores.Sony.PSX; namespace BizHawk.Client.EmuHawk { @@ -33,7 +30,7 @@ namespace BizHawk.Client.EmuHawk { private static DisplaySurface CreateDisplaySurface(int w, int h) => new(w, h); - private class DisplayManagerRenderTargetProvider : IRenderTargetProvider + protected class DisplayManagerRenderTargetProvider : IRenderTargetProvider { private readonly Func _callback; @@ -48,34 +45,30 @@ namespace BizHawk.Client.EmuHawk } } - private readonly Func _getIsSecondaryThrottlingDisabled; - public OSDManager OSD { get; } - private Config GlobalConfig; + protected Config GlobalConfig; private IEmulator GlobalEmulator; - public DisplayManagerBase(Config config, IEmulator emulator, InputManager inputManager, IMovieSession movieSession, IGL gl, PresentationPanel presentationPanel, Func getIsSecondaryThrottlingDisabled) + public DisplayManagerBase( + Config config, + IEmulator emulator, + InputManager inputManager, + IMovieSession movieSession, + EDispMethod dispMethod, + IGL gl, + IGuiRenderer renderer) { GlobalConfig = config; GlobalEmulator = emulator; OSD = new OSDManager(config, emulator, inputManager, movieSession); - _getIsSecondaryThrottlingDisabled = getIsSecondaryThrottlingDisabled; _gl = gl; - - // setup the GL context manager, needed for coping with multiple opengl cores vs opengl display method - // but is it tho? --yoshi - _glManager = GLManager.Instance; - - this._presentationPanel = presentationPanel; - _crGraphicsControl = _glManager.GetContextForGraphicsControl(_graphicsControl); + _renderer = renderer; // it's sort of important for these to be initialized to something nonzero _currEmuWidth = _currEmuHeight = 1; - _renderer = _gl.CreateRenderer(); - _videoTextureFrugalizer = new TextureFrugalizer(_gl); _shaderChainFrugalizers = new RenderTargetFrugalizer[16]; // hacky hardcoded limit.. need some other way to manage these @@ -85,16 +78,16 @@ namespace BizHawk.Client.EmuHawk } { - using var xml = Client.Common.ReflectionCache.EmbeddedResourceStream("Resources.courier16px.fnt"); - using var tex = Client.Common.ReflectionCache.EmbeddedResourceStream("Resources.courier16px_0.png"); + using var xml = ReflectionCache.EmbeddedResourceStream("Resources.courier16px.fnt"); + using var tex = ReflectionCache.EmbeddedResourceStream("Resources.courier16px_0.png"); _theOneFont = new StringRenderer(_gl, xml, tex); - using var gens = Client.Common.ReflectionCache.EmbeddedResourceStream("Resources.gens.ttf"); + using var gens = ReflectionCache.EmbeddedResourceStream("Resources.gens.ttf"); LoadCustomFont(gens); - using var fceux = Client.Common.ReflectionCache.EmbeddedResourceStream("Resources.fceux.ttf"); + using var fceux = ReflectionCache.EmbeddedResourceStream("Resources.fceux.ttf"); LoadCustomFont(fceux); } - if (_gl is IGL_TK || _gl is IGL_SlimDX9) + if (dispMethod == EDispMethod.OpenGL || dispMethod == EDispMethod.SlimDX9) { var fiHq2x = new FileInfo(Path.Combine(PathUtils.ExeDirectoryPath, "Shaders/BizHawk/hq2x.cgp")); if (fiHq2x.Exists) @@ -108,12 +101,7 @@ namespace BizHawk.Client.EmuHawk using var stream = fiScanlines.OpenRead(); _shaderChainScanlines = new RetroShaderChain(_gl, new RetroShaderPreset(stream), Path.Combine(PathUtils.ExeDirectoryPath, "Shaders/BizHawk")); } - - string bicubicPath = "Shaders/BizHawk/bicubic-fast.cgp"; - if (_gl is IGL_SlimDX9) - { - bicubicPath = "Shaders/BizHawk/bicubic-normal.cgp"; - } + var bicubicPath = dispMethod == EDispMethod.SlimDX9 ? "Shaders/BizHawk/bicubic-normal.cgp" : "Shaders/BizHawk/bicubic-fast.cgp"; var fiBicubic = new FileInfo(Path.Combine(PathUtils.ExeDirectoryPath, bicubicPath)); if (fiBicubic.Exists) { @@ -164,17 +152,14 @@ namespace BizHawk.Client.EmuHawk } // rendering resources: - private readonly IGL _gl; - private readonly GLManager _glManager; + protected readonly IGL _gl; + private readonly StringRenderer _theOneFont; + private readonly IGuiRenderer _renderer; // layer resources - private readonly PresentationPanel _presentationPanel; // well, its the final layer's target, at least - private readonly GLManager.ContextRef _crGraphicsControl; - private FilterProgram _currentFilterProgram; - - private GraphicsControl _graphicsControl => _presentationPanel.GraphicsControl; + protected FilterProgram _currentFilterProgram; /// /// these variables will track the dimensions of the last frame's (or the next frame? this is confusing) emulator native output size @@ -198,14 +183,22 @@ namespace BizHawk.Client.EmuHawk public PrivateFontCollection CustomFonts { get; } = new PrivateFontCollection(); private readonly TextureFrugalizer _videoTextureFrugalizer; + private readonly Dictionary _apiHawkSurfaceFrugalizers = new(); - private readonly RenderTargetFrugalizer[] _shaderChainFrugalizers; - private readonly RetroShaderChain _shaderChainHq2X, _shaderChainScanlines, _shaderChainBicubic; + + protected readonly RenderTargetFrugalizer[] _shaderChainFrugalizers; + + private RetroShaderChain _shaderChainHq2X; + + private RetroShaderChain _shaderChainScanlines; + + private RetroShaderChain _shaderChainBicubic; + private RetroShaderChain _shaderChainUser; - private void ActivateGLContext() => _glManager.Activate(_crGraphicsControl); + protected virtual void ActivateGLContext() => throw new NotImplementedException(); - private void SwapBuffersOfGraphicsControl() => _graphicsControl.SwapBuffers(); + protected virtual void SwapBuffersOfGraphicsControl() => throw new NotImplementedException(); public void RefreshUserShader() { @@ -424,7 +417,7 @@ namespace BizHawk.Client.EmuHawk chain.AddFilter(fLuaLayer, surfaceID.GetName()); } - private Point GraphicsControlPointToClient(Point p) => _graphicsControl.PointToClient(p); + protected virtual Point GraphicsControlPointToClient(Point p) => throw new NotImplementedException(); /// /// Using the current filter program, turn a mouse coordinate from window space to the original emulator screen space. @@ -469,9 +462,9 @@ namespace BizHawk.Client.EmuHawk return new Point((int)v.X, (int)v.Y); } - public Size GetPanelNativeSize() => _presentationPanel.NativeSize; + public virtual Size GetPanelNativeSize() => throw new NotImplementedException(); - private Size GetGraphicsControlSize() => _graphicsControl.Size; + protected virtual Size GetGraphicsControlSize() => throw new NotImplementedException(); /// /// This will receive an emulated output frame from an IVideoProvider and run it through the complete frame processing pipeline @@ -769,7 +762,7 @@ namespace BizHawk.Client.EmuHawk return size; } - private class JobInfo + protected class JobInfo { public IVideoProvider VideoProvider; public bool Simulate; @@ -953,53 +946,9 @@ namespace BizHawk.Client.EmuHawk SwapBuffersOfGraphicsControl(); } - private void UpdateSourceDrawingWork(JobInfo job) + protected virtual void UpdateSourceDrawingWork(JobInfo job) { - bool alternateVsync = false; - - // only used by alternate vsync - IGL_SlimDX9 dx9 = null; - - if (!job.Offscreen) - { - //apply the vsync setting (should probably try to avoid repeating this) - var vsync = GlobalConfig.VSyncThrottle || GlobalConfig.VSync; - - //ok, now this is a bit undesirable. - //maybe the user wants vsync, but not vsync throttle. - //this makes sense... but we don't have the infrastructure to support it now (we'd have to enable triple buffering or something like that) - //so what we're gonna do is disable vsync no matter what if throttling is off, and maybe nobody will notice. - //update 26-mar-2016: this upsets me. When fast-forwarding and skipping frames, vsync should still work. But I'm not changing it yet - if (_getIsSecondaryThrottlingDisabled()) - vsync = false; - - //for now, it's assumed that the presentation panel is the main window, but that may not always be true - if (vsync && GlobalConfig.DispAlternateVsync && GlobalConfig.VSyncThrottle) - { - dx9 = _gl as IGL_SlimDX9; - if (dx9 != null) - { - alternateVsync = true; - //unset normal vsync if we've chosen the alternate vsync - vsync = false; - } - } - - //TODO - whats so hard about triple buffering anyway? just enable it always, and change api to SetVsync(enable,throttle) - //maybe even SetVsync(enable,throttlemethod) or just SetVsync(enable,throttle,advanced) - - if (_lastVsyncSetting != vsync || _lastVsyncSettingGraphicsControl != _graphicsControl) - { - if (_lastVsyncSetting == null && vsync) - { - // Workaround for vsync not taking effect at startup (Intel graphics related?) - _graphicsControl.SetVsync(false); - } - _graphicsControl.SetVsync(vsync); - _lastVsyncSettingGraphicsControl = _graphicsControl; - _lastVsyncSetting = vsync; - } - } + if (!job.Offscreen) throw new InvalidOperationException(); // begin rendering on this context // should this have been done earlier? @@ -1015,29 +964,11 @@ namespace BizHawk.Client.EmuHawk RunFilterChainSteps(ref rtCounter, out var rtCurr, out var inFinalTarget); _gl.EndScene(); - if (job.Offscreen) - { - job.OffscreenBb = rtCurr.Texture2d.Resolve(); - job.OffscreenBb.DiscardAlpha(); - return; - } - - Debug.Assert(inFinalTarget); - - // wait for vsync to begin - if (alternateVsync) dx9.AlternateVsyncPass(0); - - // present and conclude drawing - _graphicsControl.SwapBuffers(); - - // wait for vsync to end - if (alternateVsync) dx9.AlternateVsyncPass(1); - - // nope. don't do this. workaround for slow context switching on intel GPUs. just switch to another context when necessary before doing anything - // presentationPanel.GraphicsControl.End(); + job.OffscreenBb = rtCurr.Texture2d.Resolve(); + job.OffscreenBb.DiscardAlpha(); } - private void RunFilterChainSteps(ref int rtCounter, out RenderTarget rtCurr, out bool inFinalTarget) + protected void RunFilterChainSteps(ref int rtCounter, out RenderTarget rtCurr, out bool inFinalTarget) { Texture2d texCurr = null; rtCurr = null; @@ -1079,9 +1010,6 @@ namespace BizHawk.Client.EmuHawk Marshal.FreeCoTaskMem(data); } - private bool? _lastVsyncSetting; - private GraphicsControl _lastVsyncSettingGraphicsControl; - private readonly Dictionary _apiHawkIDToSurface = new(); /// Can't this just be a prop of ? --yoshi diff --git a/src/BizHawk.Client.EmuHawk/GraphicsImplementations/IGLExtensions.cs b/src/BizHawk.Client.EmuHawk/GraphicsImplementations/IGLExtensions.cs index 26d32fb562..5be89a97a1 100644 --- a/src/BizHawk.Client.EmuHawk/GraphicsImplementations/IGLExtensions.cs +++ b/src/BizHawk.Client.EmuHawk/GraphicsImplementations/IGLExtensions.cs @@ -3,6 +3,7 @@ using System; using BizHawk.Bizware.BizwareGL; using BizHawk.Bizware.DirectX; using BizHawk.Bizware.OpenTK3; +using BizHawk.Client.Common; namespace BizHawk.Client.EmuHawk { @@ -15,5 +16,13 @@ namespace BizHawk.Client.EmuHawk IGL_TK => new GuiRenderer(gl), _ => throw new NotSupportedException() }; + + public static EDispMethod DispMethodEnum(this IGL gl) => gl switch + { + IGL_GdiPlus => EDispMethod.GdiPlus, + IGL_SlimDX9 => EDispMethod.SlimDX9, + IGL_TK => EDispMethod.OpenGL, + _ => throw new ArgumentException("unknown GL impl", nameof(gl)) + }; } } diff --git a/src/BizHawk.Client.EmuHawk/MainForm.cs b/src/BizHawk.Client.EmuHawk/MainForm.cs index 8b3335094c..b6c0dd02d8 100644 --- a/src/BizHawk.Client.EmuHawk/MainForm.cs +++ b/src/BizHawk.Client.EmuHawk/MainForm.cs @@ -384,7 +384,7 @@ namespace BizHawk.Client.EmuHawk { GraphicsControl = { MainWindow = true } }; - DisplayManager = new DisplayManagerBase(Config, Emulator, InputManager, MovieSession, GL, _presentationPanel, () => DisableSecondaryThrottling); + DisplayManager = new DisplayManager(Config, Emulator, InputManager, MovieSession, GL, _presentationPanel, () => DisableSecondaryThrottling); Controls.Add(_presentationPanel); Controls.SetChildIndex(_presentationPanel, 0); @@ -929,7 +929,7 @@ namespace BizHawk.Client.EmuHawk public readonly ToolManager Tools; - private DisplayManagerBase DisplayManager; + private DisplayManager DisplayManager; private OSDManager OSD => DisplayManager.OSD; diff --git a/src/BizHawk.Client.EmuHawk/tools/ToolFormBase.cs b/src/BizHawk.Client.EmuHawk/tools/ToolFormBase.cs index 870c08cb38..0f1ae3c230 100644 --- a/src/BizHawk.Client.EmuHawk/tools/ToolFormBase.cs +++ b/src/BizHawk.Client.EmuHawk/tools/ToolFormBase.cs @@ -13,7 +13,7 @@ namespace BizHawk.Client.EmuHawk { public ToolManager Tools { protected get; set; } - public DisplayManagerBase DisplayManager { protected get; set; } + public DisplayManager DisplayManager { protected get; set; } public InputManager InputManager { protected get; set; } diff --git a/src/BizHawk.Client.EmuHawk/tools/ToolManager.cs b/src/BizHawk.Client.EmuHawk/tools/ToolManager.cs index fdb035c259..26deba23c4 100644 --- a/src/BizHawk.Client.EmuHawk/tools/ToolManager.cs +++ b/src/BizHawk.Client.EmuHawk/tools/ToolManager.cs @@ -20,7 +20,7 @@ namespace BizHawk.Client.EmuHawk { private readonly MainForm _owner; private Config _config; - private readonly DisplayManagerBase _displayManager; + private readonly DisplayManager _displayManager; private readonly InputManager _inputManager; private IExternalApiProvider _apiProvider; private IEmulator _emulator; @@ -44,7 +44,7 @@ namespace BizHawk.Client.EmuHawk public ToolManager( MainForm owner, Config config, - DisplayManagerBase displayManager, + DisplayManager displayManager, InputManager inputManager, IEmulator emulator, IMovieSession movieSession,