From 96da0880b4a3bfb7f5c3ea05681b6c6a1198e3d4 Mon Sep 17 00:00:00 2001 From: zeromus Date: Fri, 14 Feb 2014 00:55:18 +0000 Subject: [PATCH] work towards generalizing lua display layers. "emu" and "native" surfaces now work and are accessible from lua. --- .../DisplayManager/DisplayManager.cs | 88 ++++++++++++++----- .../tools/Lua/Libraries/EmuLuaLibrary.Gui.cs | 35 ++------ .../tools/Lua/LuaConsole.cs | 6 +- BizHawk.Client.EmuHawk/tools/ToolManager.cs | 1 - 4 files changed, 77 insertions(+), 53 deletions(-) diff --git a/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs b/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs index faf1f576db..7cd86f86b8 100644 --- a/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs +++ b/BizHawk.Client.EmuHawk/DisplayManager/DisplayManager.cs @@ -39,9 +39,9 @@ namespace BizHawk.Client.EmuHawk currEmuWidth = currEmuHeight = 1; Renderer = new GuiRenderer(GL); - VideoTextureFrugalizer = new TextureFrugalizer(GL); - LuaEmuTextureFrugalizer = new TextureFrugalizer(GL); + Video2xFrugalizer = new RenderTargetFrugalizer(GL); + VideoTextureFrugalizer = new TextureFrugalizer(GL); ShaderChainFrugalizers = new RenderTargetFrugalizer[16]; //hacky hardcoded limit.. need some other way to manage these for (int i = 0; i < 16; i++) @@ -61,6 +61,11 @@ namespace BizHawk.Client.EmuHawk if (fiScanlines.Exists) using (var stream = fiScanlines.OpenRead()) ShaderChain_scanlines = new RetroShaderChain(GL,new RetroShaderPreset(stream), System.IO.Path.Combine(PathManager.GetExeDirectoryAbsolute(),"Shaders/BizHawk")); + + LuaSurfaceSets["emu"] = new SwappableDisplaySurfaceSet(); + LuaSurfaceSets["native"] = new SwappableDisplaySurfaceSet(); + LuaSurfaceFrugalizers["emu"] = new TextureFrugalizer(GL); + LuaSurfaceFrugalizers["native"] = new TextureFrugalizer(GL); } public bool Disposed { get; private set; } @@ -70,37 +75,33 @@ namespace BizHawk.Client.EmuHawk if (Disposed) return; Disposed = true; VideoTextureFrugalizer.Dispose(); - LuaEmuTextureFrugalizer.Dispose(); + foreach (var f in LuaSurfaceFrugalizers.Values) + f.Dispose(); foreach (var f in ShaderChainFrugalizers) if (f != null) f.Dispose(); } + //dont know what to do about this yet + public bool NeedsToPaint { get; set; } + //rendering resources: IGL GL; StringRenderer TheOneFont; GuiRenderer Renderer; - //layer resources - DisplaySurface luaEmuSurface = null; PresentationPanel presentationPanel; //well, its the final layer's target, at least GraphicsControl GraphicsControl; //well, its the final layer's target, at least - public bool NeedsToPaint { get; set; } - - public void PreFrameUpdateLuaSource() - { - luaEmuSurface = luaEmuSurfaceSet.GetCurrent(); - } - /// /// these variables will track the dimensions of the last frame's (or the next frame? this is confusing) emulator native output size /// int currEmuWidth, currEmuHeight; - TextureFrugalizer VideoTextureFrugalizer, LuaEmuTextureFrugalizer; + TextureFrugalizer VideoTextureFrugalizer; + Dictionary LuaSurfaceFrugalizers = new Dictionary(); RenderTargetFrugalizer Video2xFrugalizer; RenderTargetFrugalizer[] ShaderChainFrugalizers; RetroShaderChain ShaderChain_hq2x, ShaderChain_scanlines; @@ -121,10 +122,16 @@ namespace BizHawk.Client.EmuHawk //now, acquire the data sent from the videoProvider into a texture var videoTexture = VideoTextureFrugalizer.Get(bb); - //acquire the lua emu surface as a texture + //acquire the lua surfaces as textures Texture2d luaEmuTexture = null; + var luaEmuSurface = LuaSurfaceSets["emu"].GetCurrent(); if (luaEmuSurface != null) - luaEmuTexture = LuaEmuTextureFrugalizer.Get(luaEmuSurface); + luaEmuTexture = LuaSurfaceFrugalizers["emu"].Get(luaEmuSurface); + + Texture2d luaNativeTexture = null; + var luaNativeSurface = LuaSurfaceSets["native"].GetCurrent(); + if (luaNativeSurface != null) + luaNativeTexture = LuaSurfaceFrugalizers["native"].Get(luaNativeSurface); //select shader chain RetroShaderChain selectedChain = null; @@ -200,9 +207,11 @@ namespace BizHawk.Client.EmuHawk if(luaEmuTexture != null) Renderer.Draw(luaEmuTexture); Renderer.Modelview.Pop(); - //(should we draw native layer lua here? thats broken right now) + //5a. draw the native layer content + //4.b draw the "lua emu surface" which is designed for art matching up exactly with the emulator output + if (luaNativeTexture != null) Renderer.Draw(luaNativeTexture); - //5. draw the native layer OSD + //5b. draw the native layer OSD MyBlitter myBlitter = new MyBlitter(this); myBlitter.ClipBounds = new Rectangle(0, 0, GraphicsControl.Width, GraphicsControl.Height); GlobalWin.OSD.Begin(myBlitter); @@ -225,20 +234,53 @@ namespace BizHawk.Client.EmuHawk NeedsToPaint = false; //?? } + Dictionary MapNameToLuaSurface = new Dictionary(); + Dictionary MapLuaSurfaceToName = new Dictionary(); + Dictionary LuaSurfaceSets = new Dictionary(); SwappableDisplaySurfaceSet luaNativeSurfaceSet = new SwappableDisplaySurfaceSet(); public void SetLuaSurfaceNativePreOSD(DisplaySurface surface) { luaNativeSurfaceSet.SetPending(surface); } - public DisplaySurface GetLuaSurfaceNative() + + /// + /// Locks the requested lua surface name + /// + public DisplaySurface LockLuaSurface(string name) { + if (MapNameToLuaSurface.ContainsKey(name)) + throw new InvalidOperationException("Lua surface is already locked: " + name); + + SwappableDisplaySurfaceSet sdss; + if (!LuaSurfaceSets.TryGetValue(name, out sdss)) + { + sdss = new SwappableDisplaySurfaceSet(); + LuaSurfaceSets.Add(name, sdss); + } + + //placeholder logic for more abstracted surface definitions from filter chain int currNativeWidth = presentationPanel.NativeSize.Width; int currNativeHeight = presentationPanel.NativeSize.Height; - return luaNativeSurfaceSet.AllocateSurface(currNativeWidth, currNativeHeight); + + int width,height; + if(name == "emu") { width = currEmuWidth; height = currEmuHeight; } + else if(name == "native") { width = currNativeWidth; height = currNativeHeight; } + else throw new InvalidOperationException("Unknown lua surface name: " +name); + + DisplaySurface ret = sdss.AllocateSurface(width, height); + MapNameToLuaSurface[name] = ret; + MapLuaSurfaceToName[ret] = name; + return ret; } - SwappableDisplaySurfaceSet luaEmuSurfaceSet = new SwappableDisplaySurfaceSet(); - public void SetLuaSurfaceEmu(DisplaySurface surface) { luaEmuSurfaceSet.SetPending(surface); } - public DisplaySurface GetLuaEmuSurfaceEmu() + /// + /// Unlocks this DisplaySurface which had better have been locked as a lua surface + /// + public void UnlockLuaSurface(DisplaySurface surface) { - return luaEmuSurfaceSet.AllocateSurface(currEmuWidth, currEmuHeight); + if (!MapLuaSurfaceToName.ContainsKey(surface)) + throw new InvalidOperationException("Surface was not locked as a lua surface"); + string name = MapLuaSurfaceToName[surface]; + MapLuaSurfaceToName.Remove(surface); + MapNameToLuaSurface.Remove(name); + LuaSurfaceSets[name].SetPending(surface); } //helper classes: diff --git a/BizHawk.Client.EmuHawk/tools/Lua/Libraries/EmuLuaLibrary.Gui.cs b/BizHawk.Client.EmuHawk/tools/Lua/Libraries/EmuLuaLibrary.Gui.cs index c145ea2af9..cd877091b3 100644 --- a/BizHawk.Client.EmuHawk/tools/Lua/Libraries/EmuLuaLibrary.Gui.cs +++ b/BizHawk.Client.EmuHawk/tools/Lua/Libraries/EmuLuaLibrary.Gui.cs @@ -35,37 +35,20 @@ namespace BizHawk.Client.EmuHawk } } - /// - /// sets the current drawing context to a new surface. - /// you COULD pass these back to lua to use as a target in rendering jobs, instead of setting it as current here. - /// could be more powerful. - /// performance test may reveal that repeatedly calling GetGraphics could be slow. - /// we may want to make a class here in LuaImplementation that wraps a DisplaySurface and a Graphics which would be created once - /// - public void DrawNew() + [LuaMethodAttributes( + "DrawNew", + "Changes drawing target to the specified lua surface name. This may clobber any previous drawing to this surface." + )] + public void DrawNew(string name) { - _luaSurface = GlobalWin.DisplayManager.GetLuaSurfaceNative(); + DrawFinish(); + _luaSurface = GlobalWin.DisplayManager.LockLuaSurface(name); } - public void DrawNewEmu() - { - _luaSurface = GlobalWin.DisplayManager.GetLuaEmuSurfaceEmu(); - } - - /// - /// finishes the current drawing and submits it to the display manager (at native [host] resolution pre-osd) - /// you would probably want some way to specify which surface to set it to, when there are other surfaces. - /// most notably, the client output [emulated] resolution - /// public void DrawFinish() { - GlobalWin.DisplayManager.SetLuaSurfaceNativePreOSD(_luaSurface); - _luaSurface = null; - } - - public void DrawFinishEmu() - { - GlobalWin.DisplayManager.SetLuaSurfaceEmu(_luaSurface); + if(_luaSurface != null) + GlobalWin.DisplayManager.UnlockLuaSurface(_luaSurface); _luaSurface = null; } diff --git a/BizHawk.Client.EmuHawk/tools/Lua/LuaConsole.cs b/BizHawk.Client.EmuHawk/tools/Lua/LuaConsole.cs index 356c929a31..06a5a7986a 100644 --- a/BizHawk.Client.EmuHawk/tools/Lua/LuaConsole.cs +++ b/BizHawk.Client.EmuHawk/tools/Lua/LuaConsole.cs @@ -339,7 +339,7 @@ namespace BizHawk.Client.EmuHawk { if (LuaImp.GuiLibrary.SurfaceIsNull) { - LuaImp.GuiLibrary.DrawNewEmu(); + LuaImp.GuiLibrary.DrawNew("emu"); } foreach (var lf in _luaList) @@ -396,7 +396,7 @@ namespace BizHawk.Client.EmuHawk { if (_luaList.Any() && LuaImp.GuiLibrary.SurfaceIsNull) { - LuaImp.GuiLibrary.DrawNewEmu(); + LuaImp.GuiLibrary.DrawNew("emu"); } } @@ -404,7 +404,7 @@ namespace BizHawk.Client.EmuHawk { if (_luaList.Any()) { - LuaImp.GuiLibrary.DrawFinishEmu(); + LuaImp.GuiLibrary.DrawFinish(); } } diff --git a/BizHawk.Client.EmuHawk/tools/ToolManager.cs b/BizHawk.Client.EmuHawk/tools/ToolManager.cs index a9067b6405..9902c73b8e 100644 --- a/BizHawk.Client.EmuHawk/tools/ToolManager.cs +++ b/BizHawk.Client.EmuHawk/tools/ToolManager.cs @@ -232,7 +232,6 @@ namespace BizHawk.Client.EmuHawk LuaConsole.LuaImp.CallFrameAfterEvent(); if (!fromLua) { - GlobalWin.DisplayManager.PreFrameUpdateLuaSource(); LuaConsole.EndLuaDrawing(); } }