Compare commits

...

2 Commits

Author SHA1 Message Date
Enrico Weigelt, metux IT consult a09884578a modesetting: use new VRR mode signaling DDX/driver API
Instead of highjacking core request handlers, use the recently introduced
DDX/driver API.

Signed-off-by: Enrico Weigelt, metux IT consult <info@metux.net>
2024-10-10 18:20:45 +00:00
Enrico Weigelt, metux IT consult b0f305b106 dix: add generic Xinerama capable VRR infrastructure
We don't have a standard protocol for enabling VRR yet, but some time ago an
ad-hoc had been made in the amdgpu driver (later also copied to modsetting),
which works by client setting the _VARIABLE_REFRESH window property.

The way it's currently done - driver is highjacking the X_ChangeProperty and
X_DeleteProperty request handlers - is pretty fragile, and is also a violation
of layers: drivers never should be twisted with core protocol details. (And in
the future, this should be done by some suitable extension).

Another problem is Xinerama: when it's enabled, this only works on the first
screen - the others won't ever see this signal, no matter on which one(s) the
Window is physically placed (for the wire protocol, all windows are on screen 0,
unless the client explicitly creates them on another one)

This commit adds a generic Screen proc for telling the DDX, whether the VRR mode
shall be changed (for now, it's only DISABLED and ENABLED). Drivers can hook into
here in order to receive this signal, w/o having to highjack any core request
handlers. Catching the property change is now entirely done in the DIX.

The (non-standard) status qou of (ab)using window properties is kept, but it's
now also easy to add a new signaling mechanism, in case a standard is agreed on.

Yet a quite naive implementation (eg. not acting on moving windows between screens),
but enough to fix the most pressing problems supporting extra screens in general,
as well as stopping the highjacking of core request handlers by drivers.

Signed-off-by: Enrico Weigelt, metux IT consult <info@metux.net>
2024-10-10 18:20:45 +00:00
4 changed files with 66 additions and 128 deletions

View File

@ -59,6 +59,11 @@ SOFTWARE.
#include "swaprep.h" #include "swaprep.h"
#include "xace.h" #include "xace.h"
#ifdef PANORAMIX
#include "Xext/panoramiX.h"
#include "Xext/panoramiXsrv.h"
#endif
/***************************************************************** /*****************************************************************
* Property Stuff * Property Stuff
* *
@ -107,6 +112,46 @@ dixLookupProperty(PropertyPtr *result, WindowPtr pWin, Atom propertyName,
return rc; return rc;
} }
static void
setVRRMode(WindowPtr pWin, WindowVRRMode mode)
{
SetWindowVRRModeProcPtr proc = pWin->drawable.pScreen->SetWindowVRRMode;
if (proc != NULL)
proc(pWin, mode);
}
static void
notifyVRRMode(ClientPtr pClient, WindowPtr pWindow, int state, PropertyPtr pProp)
{
const char *pName = NameForAtom(pProp->propertyName);
if (pName == NULL || strcmp(pName, "_VARIABLE_REFRESH") || pProp->format != 32 || pProp->size != 1)
return;
WindowVRRMode mode = (WindowVRRMode)(state == PropertyNewValue ? (*((uint32_t*)pProp->data)) : 0);
#ifdef PANORAMIX
if (!noPanoramiXExtension) {
PanoramiXRes *win;
int rc, j;
rc = dixLookupResourceByType((void **) &win, pWindow->drawable.id, XRT_WINDOW,
pClient, DixWriteAccess);
if (rc != Success)
goto no_panoramix;
FOR_NSCREENS_BACKWARD(j) {
WindowPtr pWin;
rc = dixLookupWindow(&pWin, win->info[j].id, pClient, DixSetPropAccess);
if (rc == Success)
setVRRMode(pWin, mode);
}
}
return;
no_panoramix:
#endif
setVRRMode(pWindow, mode);
}
CallbackListPtr PropertyStateCallback; CallbackListPtr PropertyStateCallback;
static void static void
@ -187,6 +232,7 @@ ProcRotateProperties(ClientPtr client)
for (i = 0; i < stuff->nAtoms; i++) { for (i = 0; i < stuff->nAtoms; i++) {
j = (i + delta) % stuff->nAtoms; j = (i + delta) % stuff->nAtoms;
deliverPropertyNotifyEvent(pWin, PropertyNewValue, props[i]); deliverPropertyNotifyEvent(pWin, PropertyNewValue, props[i]);
notifyVRRMode(client, pWin, PropertyNewValue, props[i]);
/* Preserve name and devPrivates */ /* Preserve name and devPrivates */
props[j]->type = saved[i].type; props[j]->type = saved[i].type;
@ -366,8 +412,10 @@ dixChangeWindowProperty(ClientPtr pClient, WindowPtr pWin, Atom property,
else else
return rc; return rc;
if (sendevent) if (sendevent) {
deliverPropertyNotifyEvent(pWin, PropertyNewValue, pProp); deliverPropertyNotifyEvent(pWin, PropertyNewValue, pProp);
notifyVRRMode(pClient, pWin, PropertyNewValue, pProp);
}
return Success; return Success;
} }
@ -397,6 +445,7 @@ DeleteProperty(ClientPtr client, WindowPtr pWin, Atom propName)
} }
deliverPropertyNotifyEvent(pWin, PropertyDelete, pProp); deliverPropertyNotifyEvent(pWin, PropertyDelete, pProp);
notifyVRRMode(client, pWin, PropertyDelete, pProp);
free(pProp->data); free(pProp->data);
dixFreeObjectWithPrivates(pProp, PRIVATE_PROPERTY); dixFreeObjectWithPrivates(pProp, PRIVATE_PROPERTY);
} }
@ -533,7 +582,10 @@ ProcGetProperty(ClientPtr client)
}; };
if (stuff->delete && (reply.bytesAfter == 0)) if (stuff->delete && (reply.bytesAfter == 0))
{
deliverPropertyNotifyEvent(pWin, PropertyDelete, pProp); deliverPropertyNotifyEvent(pWin, PropertyDelete, pProp);
notifyVRRMode(client, pWin, PropertyDelete, pProp);
}
WriteReplyToClient(client, sizeof(xGenericReply), &reply); WriteReplyToClient(client, sizeof(xGenericReply), &reply);
if (len) { if (len) {

View File

@ -85,14 +85,6 @@ static Bool ms_pci_probe(DriverPtr driver,
intptr_t match_data); intptr_t match_data);
static Bool ms_driver_func(ScrnInfoPtr scrn, xorgDriverFuncOp op, void *data); static Bool ms_driver_func(ScrnInfoPtr scrn, xorgDriverFuncOp op, void *data);
/* window wrapper functions used to get the notification when
* the window property changes */
static Atom vrr_atom;
static Bool property_vectors_wrapped;
static Bool restore_property_vector;
static int (*saved_change_property) (ClientPtr client);
static int (*saved_delete_property) (ClientPtr client);
#ifdef XSERVER_LIBPCIACCESS #ifdef XSERVER_LIBPCIACCESS
static const struct pci_id_match ms_device_match[] = { static const struct pci_id_match ms_device_match[] = {
{ {
@ -903,8 +895,9 @@ ms_window_has_variable_refresh(modesettingPtr ms, WindowPtr win) {
} }
static void static void
ms_vrr_property_update(WindowPtr window, Bool variable_refresh) msSetWindowVRRMode(WindowPtr window, WindowVRRMode mode)
{ {
Bool variable_refresh = (mode == WINDOW_VRR_ENABLED);
ScrnInfoPtr scrn = xf86ScreenToScrn(window->drawable.pScreen); ScrnInfoPtr scrn = xf86ScreenToScrn(window->drawable.pScreen);
modesettingPtr ms = modesettingPTR(scrn); modesettingPtr ms = modesettingPTR(scrn);
@ -916,112 +909,6 @@ ms_vrr_property_update(WindowPtr window, Bool variable_refresh)
ms_present_set_screen_vrr(scrn, variable_refresh); ms_present_set_screen_vrr(scrn, variable_refresh);
} }
/* Wrapper for xserver/dix/property.c:ProcChangeProperty */
static int
ms_change_property(ClientPtr client)
{
WindowPtr window = NULL;
int ret = 0;
REQUEST(xChangePropertyReq);
client->requestVector[X_ChangeProperty] = saved_change_property;
ret = saved_change_property(client);
if (restore_property_vector)
return ret;
client->requestVector[X_ChangeProperty] = ms_change_property;
if (ret != Success)
return ret;
ret = dixLookupWindow(&window, stuff->window, client, DixSetPropAccess);
if (ret != Success)
return ret;
// Checking for the VRR property change on the window
if (stuff->property == vrr_atom &&
xf86ScreenToScrn(window->drawable.pScreen)->PreInit == PreInit &&
stuff->format == 32 && stuff->nUnits == 1) {
uint32_t *value = (uint32_t *)(stuff + 1);
ms_vrr_property_update(window, *value != 0);
}
return ret;
}
/* Wrapper for xserver/dix/property.c:ProcDeleteProperty */
static int
ms_delete_property(ClientPtr client)
{
WindowPtr window;
int ret;
REQUEST(xDeletePropertyReq);
client->requestVector[X_DeleteProperty] = saved_delete_property;
ret = saved_delete_property(client);
if (restore_property_vector)
return ret;
client->requestVector[X_DeleteProperty] = ms_delete_property;
if (ret != Success)
return ret;
ret = dixLookupWindow(&window, stuff->window, client, DixSetPropAccess);
if (ret != Success)
return ret;
if (stuff->property == vrr_atom &&
xf86ScreenToScrn(window->drawable.pScreen)->PreInit == PreInit)
ms_vrr_property_update(window, FALSE);
return ret;
}
static void
ms_unwrap_property_requests(ScrnInfoPtr scrn)
{
int i;
if (!property_vectors_wrapped)
return;
if (ProcVector[X_ChangeProperty] == ms_change_property)
ProcVector[X_ChangeProperty] = saved_change_property;
else
restore_property_vector = TRUE;
if (ProcVector[X_DeleteProperty] == ms_delete_property)
ProcVector[X_DeleteProperty] = saved_delete_property;
else
restore_property_vector = TRUE;
for (i = 0; i < currentMaxClients; i++) {
if (clients[i]->requestVector[X_ChangeProperty] == ms_change_property) {
clients[i]->requestVector[X_ChangeProperty] = saved_change_property;
} else {
restore_property_vector = TRUE;
}
if (clients[i]->requestVector[X_DeleteProperty] == ms_delete_property) {
clients[i]->requestVector[X_DeleteProperty] = saved_delete_property;
} else {
restore_property_vector = TRUE;
}
}
if (restore_property_vector) {
xf86DrvMsg(scrn->scrnIndex, X_WARNING,
"Couldn't unwrap some window property request vectors\n");
}
property_vectors_wrapped = FALSE;
}
static void static void
FreeScreen(ScrnInfoPtr pScrn) FreeScreen(ScrnInfoPtr pScrn)
{ {
@ -1041,7 +928,6 @@ FreeScreen(ScrnInfoPtr pScrn)
ms_ent = ms_ent_priv(pScrn); ms_ent = ms_ent_priv(pScrn);
ms_ent->fd_ref--; ms_ent->fd_ref--;
if (!ms_ent->fd_ref) { if (!ms_ent->fd_ref) {
ms_unwrap_property_requests(pScrn);
if (ms->pEnt->location.type == BUS_PCI) if (ms->pEnt->location.type == BUS_PCI)
ret = drmClose(ms->fd); ret = drmClose(ms->fd);
else else
@ -2191,17 +2077,8 @@ ScreenInit(ScreenPtr pScreen, int argc, char **argv)
pScrn->vtSema = TRUE; pScrn->vtSema = TRUE;
if (ms->vrr_support) { if (ms->vrr_support)
if (!property_vectors_wrapped) { pScreen->SetWindowVRRMode = msSetWindowVRRMode;
saved_change_property = ProcVector[X_ChangeProperty];
ProcVector[X_ChangeProperty] = ms_change_property;
saved_delete_property = ProcVector[X_DeleteProperty];
ProcVector[X_DeleteProperty] = ms_delete_property;
property_vectors_wrapped = TRUE;
}
vrr_atom = MakeAtom("_VARIABLE_REFRESH",
strlen("_VARIABLE_REFRESH"), TRUE);
}
return TRUE; return TRUE;
} }

View File

@ -324,6 +324,7 @@ conf_data.set('SYSCONS_SUPPORT', supports_syscons ? '1' : false)
conf_data.set('WSCONS_SUPPORT', supports_wscons ? '1' : false) conf_data.set('WSCONS_SUPPORT', supports_wscons ? '1' : false)
conf_data.set('XSERVER_LIBPCIACCESS', get_option('pciaccess') ? '1' : false) conf_data.set('XSERVER_LIBPCIACCESS', get_option('pciaccess') ? '1' : false)
conf_data.set('XSERVER_PLATFORM_BUS', build_udev_kms ? '1' : false) conf_data.set('XSERVER_PLATFORM_BUS', build_udev_kms ? '1' : false)
conf_data.set('XSERVER_SCREEN_VRR', '1')
configure_file(output : 'dix-config.h', configure_file(output : 'dix-config.h',
configuration : conf_data) configuration : conf_data)

View File

@ -90,6 +90,11 @@ typedef struct _ScreenSaverStuff {
Bool /*force */ ); Bool /*force */ );
} ScreenSaverStuffRec; } ScreenSaverStuffRec;
typedef enum {
WINDOW_VRR_DISABLED = 0,
WINDOW_VRR_ENABLED,
} WindowVRRMode;
/* /*
* There is a typedef for each screen function pointer so that code that * There is a typedef for each screen function pointer so that code that
* needs to declare a screen function pointer (e.g. in a screen private * needs to declare a screen function pointer (e.g. in a screen private
@ -177,6 +182,8 @@ typedef void (*ClipNotifyProcPtr) (WindowPtr /*pWindow */ ,
int /*dx */ , int /*dx */ ,
int /*dy */ ); int /*dy */ );
typedef void (*SetWindowVRRModeProcPtr) (WindowPtr pWindow, WindowVRRMode mode);
/* pixmap will exist only for the duration of the current rendering operation */ /* pixmap will exist only for the duration of the current rendering operation */
#define CREATE_PIXMAP_USAGE_SCRATCH 1 #define CREATE_PIXMAP_USAGE_SCRATCH 1
/* pixmap will be the backing pixmap for a redirected window */ /* pixmap will be the backing pixmap for a redirected window */
@ -545,6 +552,7 @@ typedef struct _Screen {
ClipNotifyProcPtr ClipNotify; ClipNotifyProcPtr ClipNotify;
RestackWindowProcPtr RestackWindow; RestackWindowProcPtr RestackWindow;
PaintWindowProcPtr PaintWindow; PaintWindowProcPtr PaintWindow;
SetWindowVRRModeProcPtr SetWindowVRRMode;
/* Pixmap procedures */ /* Pixmap procedures */