mirror of
https://github.com/moonlight-stream/moonlight-qt
synced 2025-01-18 14:03:54 +00:00
132 lines
4.2 KiB
Text
132 lines
4.2 KiB
Text
#include "displaylinkvsyncsource.h"
|
|
|
|
#include <SDL_syswm.h>
|
|
|
|
#include <CoreVideo/CoreVideo.h>
|
|
|
|
#import <Cocoa/Cocoa.h>
|
|
|
|
class DisplayLinkVsyncSource : public IVsyncSource
|
|
{
|
|
public:
|
|
DisplayLinkVsyncSource(Pacer* pacer)
|
|
: m_Pacer(pacer),
|
|
m_DisplayLink(nullptr)
|
|
{
|
|
|
|
}
|
|
|
|
virtual ~DisplayLinkVsyncSource() override
|
|
{
|
|
if (m_DisplayLink != nullptr) {
|
|
CVDisplayLinkStop(m_DisplayLink);
|
|
CVDisplayLinkRelease(m_DisplayLink);
|
|
}
|
|
}
|
|
|
|
static
|
|
CGDirectDisplayID
|
|
getDisplayID(NSScreen* screen)
|
|
{
|
|
NSNumber* screenNumber = [screen deviceDescription][@"NSScreenNumber"];
|
|
return [screenNumber unsignedIntValue];
|
|
}
|
|
|
|
static
|
|
CVReturn
|
|
displayLinkOutputCallback(
|
|
CVDisplayLinkRef displayLink,
|
|
const CVTimeStamp* /* now */,
|
|
const CVTimeStamp* /* vsyncTime */,
|
|
CVOptionFlags,
|
|
CVOptionFlags*,
|
|
void *displayLinkContext)
|
|
{
|
|
auto me = reinterpret_cast<DisplayLinkVsyncSource*>(displayLinkContext);
|
|
|
|
SDL_assert(displayLink == me->m_DisplayLink);
|
|
|
|
// In my testing on macOS 10.13, this callback is invoked about 24 ms
|
|
// prior to the specified v-sync time (now - vsyncTime). Since this is
|
|
// greater than the standard v-sync interval (16 ms = 60 FPS), we will
|
|
// draw using the current host time, rather than the actual v-sync target
|
|
// time. Because the CVDisplayLink is in sync with the actual v-sync
|
|
// interval, even if many ms prior, we can safely use the current host time
|
|
// and get a consistent callback for each v-sync. This reduces video latency
|
|
// by at least 1 frame vs. rendering with the actual vsyncTime.
|
|
me->m_Pacer->vsyncCallback(500 / me->m_DisplayFps);
|
|
|
|
return kCVReturnSuccess;
|
|
}
|
|
|
|
virtual bool initialize(SDL_Window* window, int displayFps) override
|
|
{
|
|
SDL_SysWMinfo info;
|
|
|
|
SDL_VERSION(&info.version);
|
|
|
|
if (!SDL_GetWindowWMInfo(window, &info)) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"SDL_GetWindowWMInfo() failed: %s",
|
|
SDL_GetError());
|
|
return false;
|
|
}
|
|
|
|
SDL_assert(info.subsystem == SDL_SYSWM_COCOA);
|
|
|
|
m_DisplayFps = displayFps;
|
|
|
|
NSScreen* screen = [info.info.cocoa.window screen];
|
|
CVReturn status;
|
|
if (screen == nullptr) {
|
|
// Window not visible on any display, so use a
|
|
// CVDisplayLink that can work with all active displays.
|
|
// When we become visible, we'll recreate ourselves
|
|
// and associate with the new screen.
|
|
SDL_LogWarn(SDL_LOG_CATEGORY_APPLICATION,
|
|
"NSWindow is not visible on any display");
|
|
status = CVDisplayLinkCreateWithActiveCGDisplays(&m_DisplayLink);
|
|
}
|
|
else {
|
|
CGDirectDisplayID displayId;
|
|
displayId = getDisplayID(screen);
|
|
SDL_LogInfo(SDL_LOG_CATEGORY_APPLICATION,
|
|
"NSWindow on display: %x",
|
|
displayId);
|
|
status = CVDisplayLinkCreateWithCGDisplay(displayId, &m_DisplayLink);
|
|
}
|
|
if (status != kCVReturnSuccess) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"Failed to create CVDisplayLink: %d",
|
|
status);
|
|
return false;
|
|
}
|
|
|
|
status = CVDisplayLinkSetOutputCallback(m_DisplayLink, displayLinkOutputCallback, this);
|
|
if (status != kCVReturnSuccess) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"CVDisplayLinkSetOutputCallback() failed: %d",
|
|
status);
|
|
return false;
|
|
}
|
|
|
|
status = CVDisplayLinkStart(m_DisplayLink);
|
|
if (status != kCVReturnSuccess) {
|
|
SDL_LogError(SDL_LOG_CATEGORY_APPLICATION,
|
|
"CVDisplayLinkStart() failed: %d",
|
|
status);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private:
|
|
Pacer* m_Pacer;
|
|
CVDisplayLinkRef m_DisplayLink;
|
|
int m_DisplayFps;
|
|
};
|
|
|
|
IVsyncSource* DisplayLinkVsyncSourceFactory::createVsyncSource(Pacer* pacer) {
|
|
return new DisplayLinkVsyncSource(pacer);
|
|
}
|