diff --git a/src/renderer/base/RenderEngineBase.cpp b/src/renderer/base/RenderEngineBase.cpp index f6baafd4cc8..66988d0c5c3 100644 --- a/src/renderer/base/RenderEngineBase.cpp +++ b/src/renderer/base/RenderEngineBase.cpp @@ -40,3 +40,10 @@ HRESULT RenderEngineBase::PrepareRenderInfo(const RenderFrameInfo& /*info*/) noe { return S_FALSE; } + +// Method Description: +// - Blocks until the engine is able to render without blocking. +void RenderEngineBase::WaitUntilCanRender() noexcept +{ + // do nothing by default +} diff --git a/src/renderer/base/renderer.cpp b/src/renderer/base/renderer.cpp index 3caaf4b836b..79e7ab3be3e 100644 --- a/src/renderer/base/renderer.cpp +++ b/src/renderer/base/renderer.cpp @@ -1194,3 +1194,13 @@ void Renderer::ResetErrorStateAndResume() // because we're not stateful (we could be in the future), all we want to do is reenable painting. EnablePainting(); } + +// Method Description: +// - Blocks until the engines are able to render without blocking. +void Renderer::WaitUntilCanRender() +{ + for (const auto pEngine : _rgpEngines) + { + pEngine->WaitUntilCanRender(); + } +} diff --git a/src/renderer/base/renderer.hpp b/src/renderer/base/renderer.hpp index 37a72c56925..a59ff366044 100644 --- a/src/renderer/base/renderer.hpp +++ b/src/renderer/base/renderer.hpp @@ -73,6 +73,7 @@ namespace Microsoft::Console::Render void EnablePainting() override; void WaitForPaintCompletionAndDisable(const DWORD dwTimeoutMs) override; + void WaitUntilCanRender() override; void AddRenderEngine(_In_ IRenderEngine* const pEngine) override; diff --git a/src/renderer/base/thread.cpp b/src/renderer/base/thread.cpp index 1af1bd34249..c4d4bc36c7d 100644 --- a/src/renderer/base/thread.cpp +++ b/src/renderer/base/thread.cpp @@ -201,6 +201,7 @@ DWORD WINAPI RenderThread::_ThreadProc() ResetEvent(_hPaintCompletedEvent); + _pRenderer->WaitUntilCanRender(); LOG_IF_FAILED(_pRenderer->PaintFrame()); SetEvent(_hPaintCompletedEvent); diff --git a/src/renderer/dx/DxRenderer.cpp b/src/renderer/dx/DxRenderer.cpp index 5e95cbe31a7..8966ff18e54 100644 --- a/src/renderer/dx/DxRenderer.cpp +++ b/src/renderer/dx/DxRenderer.cpp @@ -83,6 +83,7 @@ DxEngine::DxEngine() : _glyphCell{}, _boxDrawingEffect{}, _haveDeviceResources{ false }, + _swapChainFrameLatencyWaitableObject{ INVALID_HANDLE_VALUE }, _retroTerminalEffects{ false }, _forceFullRepaintRendering{ false }, _softwareRendering{ false }, @@ -425,6 +426,11 @@ try if (createSwapChain) { + _swapChainFlags = 0; + + // requires DXGI 1.3 which was introduced in Windows 8.1 + WI_SetFlagIf(_swapChainFlags, DXGI_SWAP_CHAIN_FLAG_FRAME_LATENCY_WAITABLE_OBJECT, IsWindows8Point1OrGreater()); + DXGI_SWAP_CHAIN_DESC1 SwapChainDesc = { 0 }; SwapChainDesc.Format = DXGI_FORMAT_B8G8R8A8_UNORM; SwapChainDesc.BufferUsage = DXGI_USAGE_RENDER_TARGET_OUTPUT; @@ -433,6 +439,7 @@ try SwapChainDesc.SampleDesc.Count = 1; SwapChainDesc.AlphaMode = DXGI_ALPHA_MODE_UNSPECIFIED; SwapChainDesc.Scaling = DXGI_SCALING_NONE; + SwapChainDesc.Flags = _swapChainFlags; switch (_chainMode) { @@ -487,6 +494,20 @@ try THROW_HR(E_NOTIMPL); } + if (IsWindows8Point1OrGreater()) + { + ::Microsoft::WRL::ComPtr swapChain2; + const HRESULT asResult = _dxgiSwapChain.As(&swapChain2); + if (SUCCEEDED(asResult)) + { + _swapChainFrameLatencyWaitableObject = wil::unique_handle{ swapChain2->GetFrameLatencyWaitableObject() }; + } + else + { + LOG_HR_MSG(asResult, "Failed to obtain IDXGISwapChain2 from swap chain"); + } + } + if (_retroTerminalEffects) { const HRESULT hr = _SetupTerminalEffects(); @@ -612,6 +633,7 @@ void DxEngine::_ReleaseDeviceResources() noexcept _dxgiSurface.Reset(); _dxgiSwapChain.Reset(); + _swapChainFrameLatencyWaitableObject.reset(); if (nullptr != _d3dDeviceContext.Get()) { @@ -960,7 +982,7 @@ try _d2dRenderTarget.Reset(); // Change the buffer size and recreate the render target (and surface) - RETURN_IF_FAILED(_dxgiSwapChain->ResizeBuffers(2, clientSize.width(), clientSize.height(), DXGI_FORMAT_B8G8R8A8_UNORM, 0)); + RETURN_IF_FAILED(_dxgiSwapChain->ResizeBuffers(2, clientSize.width(), clientSize.height(), DXGI_FORMAT_B8G8R8A8_UNORM, _swapChainFlags)); RETURN_IF_FAILED(_PrepareRenderTarget()); // OK we made it past the parts that can cause errors. We can release our failure handler. @@ -1085,6 +1107,26 @@ CATCH_RETURN() return S_OK; } +// Method Description: +// - Blocks until the engine is able to render without blocking. +// - See https://docs.microsoft.com/en-us/windows/uwp/gaming/reduce-latency-with-dxgi-1-3-swap-chains. +void DxEngine::WaitUntilCanRender() noexcept +{ + if (!_swapChainFrameLatencyWaitableObject) + { + return; + } + + const auto ret = WaitForSingleObjectEx( + _swapChainFrameLatencyWaitableObject.get(), + 1000, // 1 second timeout (shouldn't ever occur) + true); + if (ret != WAIT_OBJECT_0) + { + LOG_WIN32_MSG(ret, "Waiting for swap chain frame latency waitable object returned error or timeout."); + } +} + // Routine Description: // - Takes queued drawing information and presents it to the screen. // - This is separated out so it can be done outside the lock as it's expensive. diff --git a/src/renderer/dx/DxRenderer.hpp b/src/renderer/dx/DxRenderer.hpp index 06e869b4d00..dc66ae1f668 100644 --- a/src/renderer/dx/DxRenderer.hpp +++ b/src/renderer/dx/DxRenderer.hpp @@ -9,6 +9,7 @@ #include #include +#include #include #include @@ -73,6 +74,8 @@ namespace Microsoft::Console::Render [[nodiscard]] HRESULT StartPaint() noexcept override; [[nodiscard]] HRESULT EndPaint() noexcept override; + + void WaitUntilCanRender() noexcept override; [[nodiscard]] HRESULT Present() noexcept override; [[nodiscard]] HRESULT ScrollFrame() noexcept override; @@ -180,7 +183,9 @@ namespace Microsoft::Console::Render ::Microsoft::WRL::ComPtr _d2dRenderTarget; ::Microsoft::WRL::ComPtr _d2dBrushForeground; ::Microsoft::WRL::ComPtr _d2dBrushBackground; + UINT _swapChainFlags; ::Microsoft::WRL::ComPtr _dxgiSwapChain; + wil::unique_handle _swapChainFrameLatencyWaitableObject; // Terminal effects resources. bool _retroTerminalEffects; diff --git a/src/renderer/inc/IRenderEngine.hpp b/src/renderer/inc/IRenderEngine.hpp index ad9b6e49559..96929dd83f7 100644 --- a/src/renderer/inc/IRenderEngine.hpp +++ b/src/renderer/inc/IRenderEngine.hpp @@ -49,6 +49,8 @@ namespace Microsoft::Console::Render public: [[nodiscard]] virtual HRESULT StartPaint() noexcept = 0; [[nodiscard]] virtual HRESULT EndPaint() noexcept = 0; + + virtual void WaitUntilCanRender() noexcept = 0; [[nodiscard]] virtual HRESULT Present() noexcept = 0; [[nodiscard]] virtual HRESULT PrepareForTeardown(_Out_ bool* const pForcePaint) noexcept = 0; diff --git a/src/renderer/inc/IRenderer.hpp b/src/renderer/inc/IRenderer.hpp index 80838eb327d..ce9dbec18ac 100644 --- a/src/renderer/inc/IRenderer.hpp +++ b/src/renderer/inc/IRenderer.hpp @@ -58,6 +58,7 @@ namespace Microsoft::Console::Render virtual void EnablePainting() = 0; virtual void WaitForPaintCompletionAndDisable(const DWORD dwTimeoutMs) = 0; + virtual void WaitUntilCanRender() = 0; virtual void AddRenderEngine(_In_ IRenderEngine* const pEngine) = 0; diff --git a/src/renderer/inc/RenderEngineBase.hpp b/src/renderer/inc/RenderEngineBase.hpp index 4d6298a1d97..e65cfd6476d 100644 --- a/src/renderer/inc/RenderEngineBase.hpp +++ b/src/renderer/inc/RenderEngineBase.hpp @@ -40,6 +40,8 @@ namespace Microsoft::Console::Render [[nodiscard]] HRESULT PrepareRenderInfo(const RenderFrameInfo& info) noexcept override; + void WaitUntilCanRender() noexcept override; + protected: [[nodiscard]] virtual HRESULT _DoUpdateTitle(const std::wstring& newTitle) noexcept = 0;