/** * Looking Glass * Copyright © 2017-2024 The Looking Glass Authors * https://looking-glass.io * * This program is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License as published by the Free * Software Foundation; either version 2 of the License, or (at your option) * any later version. * * This program is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for * more details. * * You should have received a copy of the GNU General Public License along * with this program; if not, write to the Free Software Foundation, Inc., 59 * Temple Place, Suite 330, Boston, MA 02111-1307 USA */ #include "CIndirectMonitorContext.h" #include "Direct3DDevice.h" #include "Debug.h" CIndirectMonitorContext::CIndirectMonitorContext(_In_ IDDCX_MONITOR monitor, CIndirectDeviceContext * device) : m_monitor(monitor), m_devContext(device) { m_terminateEvent .Attach(CreateEvent(nullptr, FALSE, FALSE, nullptr)); m_cursorDataEvent.Attach(CreateEvent(nullptr, FALSE, FALSE, nullptr)); m_thread.Attach(CreateThread(nullptr, 0, _CursorThread, this, 0, nullptr)); m_shapeBuffer = new BYTE[512 * 512 * 4]; } CIndirectMonitorContext::~CIndirectMonitorContext() { m_swapChain.reset(); SetEvent(m_terminateEvent.Get()); delete[] m_shapeBuffer; } void CIndirectMonitorContext::AssignSwapChain(IDDCX_SWAPCHAIN swapChain, LUID renderAdapter, HANDLE newFrameEvent) { m_swapChain.reset(); auto device = std::make_shared(renderAdapter); if (FAILED(device->Init())) { WdfObjectDelete(swapChain); return; } m_swapChain.reset(new CSwapChainProcessor(m_devContext, swapChain, device, newFrameEvent)); IDARG_IN_SETUP_HWCURSOR c = {}; c.CursorInfo.Size = sizeof(c.CursorInfo); c.CursorInfo.AlphaCursorSupport = TRUE; c.CursorInfo.ColorXorCursorSupport = IDDCX_XOR_CURSOR_SUPPORT_FULL; c.CursorInfo.MaxX = 512; c.CursorInfo.MaxY = 512; c.hNewCursorDataAvailable = m_cursorDataEvent.Get(); NTSTATUS status = IddCxMonitorSetupHardwareCursor(m_monitor, &c); if (!NT_SUCCESS(status)) DBGPRINT("IddCxMonitorSetupHardwareCursor Failed: %08x", status); } void CIndirectMonitorContext::UnassignSwapChain() { m_swapChain.reset(); } DWORD CALLBACK CIndirectMonitorContext::_CursorThread(LPVOID arg) { reinterpret_cast(arg)->CursorThread(); return 0; } void CIndirectMonitorContext::CursorThread() { HRESULT hr = 0; for (;;) { HANDLE waitHandles[] = { m_cursorDataEvent.Get(), m_terminateEvent.Get() }; DWORD waitResult = WaitForMultipleObjects(ARRAYSIZE(waitHandles), waitHandles, FALSE, 100); if (waitResult == WAIT_TIMEOUT) continue; else if (waitResult == WAIT_OBJECT_0 + 1) break; else if (waitResult != WAIT_OBJECT_0) { hr = HRESULT_FROM_WIN32(waitResult); DBGPRINT("WaitForMultipleObjects: %08", hr); return; } IDARG_IN_QUERY_HWCURSOR in = {}; in.LastShapeId = m_lastShapeId; in.pShapeBuffer = m_shapeBuffer; in.ShapeBufferSizeInBytes = 512 * 512 * 4; IDARG_OUT_QUERY_HWCURSOR out = {}; NTSTATUS status = IddCxMonitorQueryHardwareCursor(m_monitor, &in, &out); if (FAILED(status)) { DBGPRINT("IddCxMonitorQueryHardwareCursor failed: %08x", status); return; } m_devContext->SendCursor(out, m_shapeBuffer); } }