2024-05-22 12:16:29 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
|
2024-09-01 12:08:31 +00:00
|
|
|
// SPDX-License-Identifier: PolyForm-Strict-1.0.0
|
2022-12-04 11:03:45 +00:00
|
|
|
|
2021-04-02 14:55:09 +00:00
|
|
|
#include "window_info.h"
|
2023-08-13 03:42:02 +00:00
|
|
|
|
|
|
|
#include "common/assert.h"
|
2024-05-22 12:16:29 +00:00
|
|
|
#include "common/error.h"
|
|
|
|
#include "common/heap_array.h"
|
|
|
|
#include "common/log.h"
|
|
|
|
#include "common/scoped_guard.h"
|
|
|
|
|
|
|
|
Log_SetChannel(WindowInfo);
|
2021-04-02 14:55:09 +00:00
|
|
|
|
2022-11-23 09:13:28 +00:00
|
|
|
void WindowInfo::SetSurfaceless()
|
|
|
|
{
|
|
|
|
type = Type::Surfaceless;
|
|
|
|
window_handle = nullptr;
|
|
|
|
surface_width = 0;
|
|
|
|
surface_height = 0;
|
|
|
|
surface_refresh_rate = 0.0f;
|
|
|
|
surface_scale = 1.0f;
|
2023-08-13 03:42:02 +00:00
|
|
|
surface_format = GPUTexture::Format::Unknown;
|
2022-11-23 09:13:28 +00:00
|
|
|
|
|
|
|
#ifdef __APPLE__
|
|
|
|
surface_handle = nullptr;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2022-08-22 09:55:38 +00:00
|
|
|
#if defined(_WIN32)
|
2021-04-02 14:55:09 +00:00
|
|
|
|
|
|
|
#include "common/windows_headers.h"
|
|
|
|
#include <dwmapi.h>
|
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
static std::optional<float> GetRefreshRateFromDisplayConfig(HWND hwnd)
|
|
|
|
{
|
|
|
|
// Partially based on Chromium ui/display/win/display_config_helper.cc.
|
|
|
|
const HMONITOR monitor = MonitorFromWindow(hwnd, 0);
|
|
|
|
if (!monitor) [[unlikely]]
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("{}() failed: {}", "MonitorFromWindow", Error::CreateWin32(GetLastError()).GetDescription());
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
MONITORINFOEXW mi = {};
|
|
|
|
mi.cbSize = sizeof(mi);
|
|
|
|
if (!GetMonitorInfoW(monitor, &mi))
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("{}() failed: {}", "GetMonitorInfoW", Error::CreateWin32(GetLastError()).GetDescription());
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
DynamicHeapArray<DISPLAYCONFIG_PATH_INFO> path_info;
|
|
|
|
DynamicHeapArray<DISPLAYCONFIG_MODE_INFO> mode_info;
|
|
|
|
|
|
|
|
// I guess this could fail if it changes inbetween two calls... unlikely.
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
UINT32 path_size = 0, mode_size = 0;
|
|
|
|
LONG res = GetDisplayConfigBufferSizes(QDC_ONLY_ACTIVE_PATHS, &path_size, &mode_size);
|
|
|
|
if (res != ERROR_SUCCESS)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("{}() failed: {}", "GetDisplayConfigBufferSizes", Error::CreateWin32(res).GetDescription());
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
path_info.resize(path_size);
|
|
|
|
mode_info.resize(mode_size);
|
|
|
|
res =
|
|
|
|
QueryDisplayConfig(QDC_ONLY_ACTIVE_PATHS, &path_size, path_info.data(), &mode_size, mode_info.data(), nullptr);
|
|
|
|
if (res == ERROR_SUCCESS)
|
|
|
|
break;
|
|
|
|
if (res != ERROR_INSUFFICIENT_BUFFER)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("{}() failed: {}", "QueryDisplayConfig", Error::CreateWin32(res).GetDescription());
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (const DISPLAYCONFIG_PATH_INFO& pi : path_info)
|
|
|
|
{
|
|
|
|
DISPLAYCONFIG_SOURCE_DEVICE_NAME sdn = {.header = {.type = DISPLAYCONFIG_DEVICE_INFO_GET_SOURCE_NAME,
|
|
|
|
.size = sizeof(DISPLAYCONFIG_SOURCE_DEVICE_NAME),
|
|
|
|
.adapterId = pi.sourceInfo.adapterId,
|
|
|
|
.id = pi.sourceInfo.id}};
|
|
|
|
LONG res = DisplayConfigGetDeviceInfo(&sdn.header);
|
|
|
|
if (res != ERROR_SUCCESS)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("{}() failed: {}", "DisplayConfigGetDeviceInfo", Error::CreateWin32(res).GetDescription());
|
2024-05-22 12:16:29 +00:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (std::wcscmp(sdn.viewGdiDeviceName, mi.szDevice) == 0)
|
|
|
|
{
|
|
|
|
// Found the monitor!
|
|
|
|
return static_cast<float>(static_cast<double>(pi.targetInfo.refreshRate.Numerator) /
|
|
|
|
static_cast<double>(pi.targetInfo.refreshRate.Denominator));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::optional<float> GetRefreshRateFromDWM(HWND hwnd)
|
2021-04-02 14:55:09 +00:00
|
|
|
{
|
|
|
|
BOOL composition_enabled;
|
2023-08-13 03:42:02 +00:00
|
|
|
if (FAILED(DwmIsCompositionEnabled(&composition_enabled)))
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
|
|
|
|
DWM_TIMING_INFO ti = {};
|
|
|
|
ti.cbSize = sizeof(ti);
|
2023-08-13 03:42:02 +00:00
|
|
|
HRESULT hr = DwmGetCompositionTimingInfo(nullptr, &ti);
|
2021-04-02 14:55:09 +00:00
|
|
|
if (SUCCEEDED(hr))
|
|
|
|
{
|
|
|
|
if (ti.rateRefresh.uiNumerator == 0 || ti.rateRefresh.uiDenominator == 0)
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
return static_cast<float>(ti.rateRefresh.uiNumerator) / static_cast<float>(ti.rateRefresh.uiDenominator);
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
static std::optional<float> GetRefreshRateFromMonitor(HWND hwnd)
|
2021-04-02 14:55:09 +00:00
|
|
|
{
|
|
|
|
HMONITOR mon = MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST);
|
|
|
|
if (!mon)
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
|
|
|
|
MONITORINFOEXW mi = {};
|
|
|
|
mi.cbSize = sizeof(mi);
|
|
|
|
if (GetMonitorInfoW(mon, &mi))
|
|
|
|
{
|
|
|
|
DEVMODEW dm = {};
|
|
|
|
dm.dmSize = sizeof(dm);
|
|
|
|
|
|
|
|
// 0/1 are reserved for "defaults".
|
|
|
|
if (EnumDisplaySettingsW(mi.szDevice, ENUM_CURRENT_SETTINGS, &dm) && dm.dmDisplayFrequency > 1)
|
2024-05-22 12:16:29 +00:00
|
|
|
return static_cast<float>(dm.dmDisplayFrequency);
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
std::optional<float> WindowInfo::QueryRefreshRateForWindow(const WindowInfo& wi)
|
2021-04-02 14:55:09 +00:00
|
|
|
{
|
2024-05-22 12:16:29 +00:00
|
|
|
std::optional<float> ret;
|
2021-04-02 14:55:09 +00:00
|
|
|
if (wi.type != Type::Win32 || !wi.window_handle)
|
2024-05-22 12:16:29 +00:00
|
|
|
return ret;
|
2021-04-02 14:55:09 +00:00
|
|
|
|
|
|
|
// Try DWM first, then fall back to integer values.
|
|
|
|
const HWND hwnd = static_cast<HWND>(wi.window_handle);
|
2024-05-22 12:16:29 +00:00
|
|
|
ret = GetRefreshRateFromDisplayConfig(hwnd);
|
|
|
|
if (!ret.has_value())
|
|
|
|
{
|
|
|
|
ret = GetRefreshRateFromDWM(hwnd);
|
|
|
|
if (!ret.has_value())
|
|
|
|
ret = GetRefreshRateFromMonitor(hwnd);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2024-05-23 15:59:35 +00:00
|
|
|
#elif defined(__APPLE__)
|
|
|
|
|
|
|
|
#include "util/platform_misc.h"
|
|
|
|
|
|
|
|
std::optional<float> WindowInfo::QueryRefreshRateForWindow(const WindowInfo& wi)
|
|
|
|
{
|
|
|
|
if (wi.type == WindowInfo::Type::MacOS)
|
|
|
|
return CocoaTools::GetViewRefreshRate(wi);
|
|
|
|
|
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2021-04-02 17:03:42 +00:00
|
|
|
#else
|
2021-04-02 14:55:09 +00:00
|
|
|
|
2023-09-17 02:28:11 +00:00
|
|
|
#ifdef ENABLE_X11
|
2021-04-02 14:55:09 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
#include <X11/Xlib.h>
|
|
|
|
#include <X11/Xutil.h>
|
2021-04-02 14:55:09 +00:00
|
|
|
#include <X11/extensions/Xrandr.h>
|
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
// Helper class for managing X errors
|
|
|
|
namespace {
|
|
|
|
class X11InhibitErrors;
|
|
|
|
|
|
|
|
static X11InhibitErrors* s_current_error_inhibiter;
|
|
|
|
|
|
|
|
class X11InhibitErrors
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
X11InhibitErrors()
|
|
|
|
{
|
|
|
|
Assert(!s_current_error_inhibiter);
|
|
|
|
m_old_handler = XSetErrorHandler(ErrorHandler);
|
|
|
|
s_current_error_inhibiter = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
~X11InhibitErrors()
|
|
|
|
{
|
|
|
|
Assert(s_current_error_inhibiter == this);
|
|
|
|
s_current_error_inhibiter = nullptr;
|
|
|
|
XSetErrorHandler(m_old_handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
ALWAYS_INLINE bool HadError() const { return m_had_error; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
static int ErrorHandler(Display* display, XErrorEvent* ee)
|
|
|
|
{
|
|
|
|
char error_string[256] = {};
|
|
|
|
XGetErrorText(display, ee->error_code, error_string, sizeof(error_string));
|
2024-05-23 10:55:28 +00:00
|
|
|
WARNING_LOG("X11 Error: {} (Error {} Minor {} Request {})", error_string, ee->error_code, ee->minor_code,
|
|
|
|
ee->request_code);
|
2023-08-13 03:42:02 +00:00
|
|
|
|
|
|
|
s_current_error_inhibiter->m_had_error = true;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
XErrorHandler m_old_handler = {};
|
|
|
|
bool m_had_error = false;
|
|
|
|
};
|
|
|
|
} // namespace
|
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
static std::optional<float> GetRefreshRateFromXRandR(const WindowInfo& wi)
|
2021-04-02 14:55:09 +00:00
|
|
|
{
|
|
|
|
Display* display = static_cast<Display*>(wi.display_connection);
|
|
|
|
Window window = static_cast<Window>(reinterpret_cast<uintptr_t>(wi.window_handle));
|
|
|
|
if (!display || !window)
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
|
2023-08-13 03:42:02 +00:00
|
|
|
X11InhibitErrors inhibiter;
|
2021-04-02 14:55:09 +00:00
|
|
|
|
|
|
|
XRRScreenResources* res = XRRGetScreenResources(display, window);
|
|
|
|
if (!res)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("XRRGetScreenResources() failed");
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2022-07-26 10:10:19 +00:00
|
|
|
ScopedGuard res_guard([res]() { XRRFreeScreenResources(res); });
|
2021-04-02 14:55:09 +00:00
|
|
|
|
|
|
|
int num_monitors;
|
|
|
|
XRRMonitorInfo* mi = XRRGetMonitors(display, window, True, &num_monitors);
|
|
|
|
if (num_monitors < 0)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("XRRGetMonitors() failed");
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
else if (num_monitors > 1)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
WARNING_LOG("XRRGetMonitors() returned {} monitors, using first", num_monitors);
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2022-07-26 10:10:19 +00:00
|
|
|
ScopedGuard mi_guard([mi]() { XRRFreeMonitors(mi); });
|
2021-04-02 14:55:09 +00:00
|
|
|
if (mi->noutput <= 0)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("Monitor has no outputs");
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
else if (mi->noutput > 1)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
WARNING_LOG("Monitor has {} outputs, using first", mi->noutput);
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XRROutputInfo* oi = XRRGetOutputInfo(display, res, mi->outputs[0]);
|
|
|
|
if (!oi)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("XRRGetOutputInfo() failed");
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2022-07-26 10:10:19 +00:00
|
|
|
ScopedGuard oi_guard([oi]() { XRRFreeOutputInfo(oi); });
|
2021-04-02 14:55:09 +00:00
|
|
|
|
|
|
|
XRRCrtcInfo* ci = XRRGetCrtcInfo(display, res, oi->crtc);
|
|
|
|
if (!ci)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("XRRGetCrtcInfo() failed");
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2022-07-26 10:10:19 +00:00
|
|
|
ScopedGuard ci_guard([ci]() { XRRFreeCrtcInfo(ci); });
|
2021-04-02 14:55:09 +00:00
|
|
|
|
|
|
|
XRRModeInfo* mode = nullptr;
|
|
|
|
for (int i = 0; i < res->nmode; i++)
|
|
|
|
{
|
|
|
|
if (res->modes[i].id == ci->mode)
|
|
|
|
{
|
|
|
|
mode = &res->modes[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!mode)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("Failed to look up mode {} (of {})", static_cast<int>(ci->mode), res->nmode);
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mode->dotClock == 0 || mode->hTotal == 0 || mode->vTotal == 0)
|
|
|
|
{
|
2024-05-23 10:55:28 +00:00
|
|
|
ERROR_LOG("Modeline is invalid: {}/{}/{}", mode->dotClock, mode->hTotal, mode->vTotal);
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
return static_cast<float>(static_cast<double>(mode->dotClock) /
|
|
|
|
(static_cast<double>(mode->hTotal) * static_cast<double>(mode->vTotal)));
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
2023-09-17 02:28:11 +00:00
|
|
|
#endif // ENABLE_X11
|
2021-04-02 14:55:09 +00:00
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
std::optional<float> WindowInfo::QueryRefreshRateForWindow(const WindowInfo& wi)
|
2021-04-02 14:55:09 +00:00
|
|
|
{
|
2023-09-17 02:28:11 +00:00
|
|
|
#if defined(ENABLE_X11)
|
2021-04-02 14:55:09 +00:00
|
|
|
if (wi.type == WindowInfo::Type::X11)
|
2024-05-22 12:16:29 +00:00
|
|
|
return GetRefreshRateFromXRandR(wi);
|
2021-04-02 14:55:09 +00:00
|
|
|
#endif
|
|
|
|
|
2024-05-22 12:16:29 +00:00
|
|
|
return std::nullopt;
|
2021-04-02 14:55:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|