From e379f707848b4d1cbea87dfa5d73dcb89f722a8b Mon Sep 17 00:00:00 2001 From: Geoffrey McRae Date: Thu, 14 Dec 2017 02:22:41 +1100 Subject: [PATCH] [host] switch to fast polling mode, fixes stuttering issues --- common/KVMFR.h | 1 + host/Service.cpp | 46 +++++++++++++++++++--------------------------- host/Service.h | 2 +- 3 files changed, 21 insertions(+), 28 deletions(-) diff --git a/common/KVMFR.h b/common/KVMFR.h index dce4c68f..d6e58f6f 100644 --- a/common/KVMFR.h +++ b/common/KVMFR.h @@ -69,6 +69,7 @@ KVMFRFrame; #define KVMFR_HEADER_FLAG_FRAME 1 // frame update available #define KVMFR_HEADER_FLAG_CURSOR 2 // cursor update available #define KVMFR_HEADER_FLAG_RESTART 4 // restart signal from client +#define KVMFR_HEADER_FLAG_READY 8 // ready signal from client typedef struct KVMFRHeader { diff --git a/host/Service.cpp b/host/Service.cpp index c636dc9f..b6715ee1 100644 --- a/host/Service.cpp +++ b/host/Service.cpp @@ -30,7 +30,7 @@ Service * Service::m_instance = NULL; Service::Service() : m_initialized(false), m_memory(NULL), - m_readyEvent(INVALID_HANDLE_VALUE), + m_timer(NULL), m_capture(NULL), m_header(NULL), m_frameIndex(0) @@ -73,11 +73,10 @@ bool Service::Initialize(ICapture * captureDevice) if (!InitPointers()) return false; - m_readyEvent = m_ivshmem->CreateVectorEvent(0); - if (m_readyEvent == INVALID_HANDLE_VALUE) + m_timer = CreateWaitableTimer(NULL, TRUE, NULL); + if (!m_timer) { - DEBUG_ERROR("Failed to get event for vector 0"); - DeInitialize(); + DEBUG_ERROR("Failed to create waitable timer"); return false; } @@ -117,10 +116,10 @@ bool Service::InitPointers() void Service::DeInitialize() { - if (m_readyEvent != INVALID_HANDLE_VALUE) + if (m_timer) { - CloseHandle(m_readyEvent); - m_readyEvent = INVALID_HANDLE_VALUE; + CloseHandle(m_timer); + m_timer = NULL; } m_header = NULL; @@ -153,8 +152,7 @@ bool Service::Process() frame.bufferSize = m_frameSize; // wait for the host to notify that is it is ready to proceed - bool eventDone = false; - while (!eventDone) + while (true) { // check if the client has flagged a restart if (m_header->flags & KVMFR_HEADER_FLAG_RESTART) @@ -164,29 +162,23 @@ bool Service::Process() break; } - switch (WaitForSingleObject(m_readyEvent, 200)) + // check if the client has flagged it's ready + if (m_header->flags & KVMFR_HEADER_FLAG_READY) { - case WAIT_ABANDONED: - DEBUG_ERROR("Wait abandoned"); - return false; - - case WAIT_OBJECT_0: - eventDone = true; + InterlockedAnd8((char *)&m_header->flags, ~(KVMFR_HEADER_FLAG_READY)); break; + } - case WAIT_TIMEOUT: - continue; - - case WAIT_FAILED: - DEBUG_ERROR("Wait failed"); - return false; - - default: - DEBUG_ERROR("Unknown error"); + // wait for 100ns before polling again + LARGE_INTEGER timeout; + timeout.QuadPart = -100; + if (!SetWaitableTimer(m_timer, &timeout, 0, NULL, NULL, FALSE)) + { + DEBUG_ERROR("Failed to set waitable timer"); return false; } + WaitForSingleObject(m_timer, INFINITE); } - ResetEvent(m_readyEvent); bool ok = false; bool cursorOnly = false; diff --git a/host/Service.h b/host/Service.h index d24d5a1f..446cef52 100644 --- a/host/Service.h +++ b/host/Service.h @@ -51,7 +51,7 @@ private: bool m_initialized; uint8_t * m_memory; IVSHMEM * m_ivshmem; - HANDLE m_readyEvent; + HANDLE m_timer; ICapture * m_capture; KVMFRHeader * m_header;