[client] fix the warp logic to account for still pending warps to finish

As X11 is a server/client protocol, issuing commands such as
XWarpPointer do not happen immediately, as such we need to identify when
the warp is complete to know to null out the movement. To do this we
track each warp issued and look for it's completion in the event filter.
As some events come in via XInput2 we need to also make use of this
instead of just relying on MotionNotify, as such the support has been
implemented for XI_Motion events.
This commit is contained in:
Geoffrey McRae 2021-01-05 11:47:17 +11:00
parent 18e84c88a0
commit 6f99280fe3
2 changed files with 154 additions and 89 deletions

View file

@ -37,7 +37,6 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#include <stdatomic.h>
#if SDL_VIDEO_DRIVER_X11_XINPUT2
// because SDL2 sucks and we need to turn it off
#include <X11/extensions/XInput2.h>
#endif
@ -71,12 +70,20 @@ static LGThread *t_cursor = NULL;
static LGThread *t_frame = NULL;
static SDL_Cursor *cursor = NULL;
static int g_XInputOp; // XInput Opcode
struct AppState g_state;
struct CursorState g_cursor;
// this structure is initialized in config.c
struct AppParams params = { 0 };
union WarpInfo
{
unsigned long serial;
SDL_Point p;
};
static void handleMouseMoveEvent(int ex, int ey);
static void lgInit()
@ -773,51 +780,36 @@ void spiceClipboardRequest(const SpiceDataType type)
g_state.lgc->request(spice_type_to_clipboard_type(type));
}
static void warpMouse(int x, int y)
static void warpMouse(int x, int y, bool disable)
{
if (g_cursor.warpState == WARP_STATE_OFF)
return;
if (g_cursor.warpState == WARP_STATE_WIN_EXIT)
{
if (g_state.wminfo.subsystem == SDL_SYSWM_X11)
{
XWarpPointer(
g_state.wminfo.info.x11.display,
None,
g_state.wminfo.info.x11.window,
0, 0, 0, 0,
x, y);
}
else
SDL_WarpMouseInWindow(g_state.window, x, y);
g_cursor.warpState = WARP_STATE_OFF;
if (ll_count(g_cursor.warpList) > 0 && !disable)
return;
}
if (g_cursor.warpState == WARP_STATE_ON)
if (disable)
g_cursor.warpState = WARP_STATE_OFF;
union WarpInfo * warp = malloc(sizeof(union WarpInfo));
if (g_state.wminfo.subsystem == SDL_SYSWM_X11)
{
g_cursor.warpState = WARP_STATE_ACTIVE;
if (g_state.wminfo.subsystem == SDL_SYSWM_X11)
{
/* with X11 we can be far more elegent and use the warp serial number to
* determine when the warp is complete instead of trying to match the
* target x/y coordinates */
g_cursor.warpSerial = NextRequest(g_state.wminfo.info.x11.display);
XWarpPointer(
g_state.wminfo.info.x11.display,
None,
g_state.wminfo.info.x11.window,
0, 0, 0, 0,
x, y);
}
else
{
g_cursor.warpTo.x = x;
g_cursor.warpTo.y = y;
SDL_WarpMouseInWindow(g_state.window, x, y);
}
warp->serial = NextRequest(g_state.wminfo.info.x11.display);
ll_push(g_cursor.warpList, warp);
XWarpPointer(
g_state.wminfo.info.x11.display,
None,
g_state.wminfo.info.x11.window,
0, 0, 0, 0,
x, y);
}
else
{
warp->p.x = x;
warp->p.y = y;
ll_push(g_cursor.warpList, warp);
SDL_WarpMouseInWindow(g_state.window, x, y);
}
}
@ -840,16 +832,6 @@ static void handleMouseMoveEvent(int ex, int ey)
if (!params.useSpiceInput)
return;
/* check if there is a warp in progress, and if it was completed */
if (g_cursor.warpState == WARP_STATE_ACTIVE &&
ex == g_cursor.warpTo.x && ey == g_cursor.warpTo.y)
{
g_cursor.last.x = ex;
g_cursor.last.y = ey;
g_cursor.warpState = WARP_STATE_ON;
return;
}
if (!g_cursor.inWindow || g_state.ignoreInput)
return;
@ -874,7 +856,7 @@ static void handleMouseMoveEvent(int ex, int ey)
spice_mouse_motion(delta.x, delta.y);
if (ex < g_state.windowCX - 25 || ex > g_state.windowCX + 25 ||
ey < g_state.windowCY - 25 || ey > g_state.windowCY + 25)
warpMouse(g_state.windowCX, g_state.windowCY);
warpMouse(g_state.windowCX, g_state.windowCY, false);
}
return;
@ -926,7 +908,7 @@ static void handleMouseMoveEvent(int ex, int ey)
/* stop the mouse from runing into the edges of the window */
if (ex < g_state.windowCX - 25 || ex > g_state.windowCX + 25 ||
ey < g_state.windowCY - 25 || ey > g_state.windowCY + 25)
warpMouse(g_state.windowCX, g_state.windowCY);
warpMouse(g_state.windowCX, g_state.windowCY, false);
}
else
{
@ -983,10 +965,10 @@ static void handleMouseMoveEvent(int ex, int ey)
if (isValidCursorLocation(nx, ny))
{
/* put the mouse where it should be and disable warp */
g_cursor.warpState = WARP_STATE_WIN_EXIT;
warpMouse(
g_state.dstRect.x + newPos.x,
g_state.dstRect.y + newPos.y
g_state.dstRect.y + newPos.y,
true
);
SDL_ShowCursor(SDL_ENABLE);
return;
@ -1033,9 +1015,7 @@ static void handleWindowEnter()
{
g_cursor.inWindow = true;
if (g_cursor.warpState == WARP_STATE_OFF)
g_cursor.warpState = WARP_STATE_ON;
g_cursor.warpState = WARP_STATE_ON;
if (!params.useSpiceInput)
return;
@ -1134,6 +1114,7 @@ int eventFilter(void * userdata, SDL_Event * event)
if (g_state.wminfo.subsystem == SDL_SYSWM_X11)
{
XEvent xe = event->syswm.msg->msg.x11.event;
switch(xe.type)
{
case ConfigureNotify:
@ -1151,27 +1132,86 @@ int eventFilter(void * userdata, SDL_Event * event)
break;
}
case MotionNotify:
/* detect and filter out the warp event */
if (g_cursor.warpState == WARP_STATE_ACTIVE &&
xe.xmotion.serial == g_cursor.warpSerial)
#if SDL_VIDEO_DRIVER_X11_XINPUT2
/* support movements via XInput2 */
case GenericEvent:
{
XGenericEventCookie *cookie = (XGenericEventCookie*)&xe.xcookie;
if (cookie->extension != g_XInputOp)
break;
XIDeviceEvent *device = cookie->data;
if (device->evtype != XI_Motion)
break;
const int x = round(device->event_x);
const int y = round(device->event_y);
/* detect and filter out warp events */
union WarpInfo * warp;
if (ll_peek_head(g_cursor.warpList, (void **)&warp) &&
xe.xany.serial == warp->serial)
{
g_cursor.warpState = WARP_STATE_ON;
g_cursor.last.x = xe.xmotion.x;
g_cursor.last.y = xe.xmotion.y;
ll_shift(g_cursor.warpList, NULL);
free(warp);
g_cursor.last.x = x;
g_cursor.last.y = y;
break;
}
handleMouseMoveEvent(x, y);
break;
}
#endif
/* even if using XInput2 we still need this otherwise we dont get
* motion events when a button is held */
case MotionNotify:
{
/* detect and filter out warp events */
union WarpInfo * warp;
if (ll_peek_head(g_cursor.warpList, (void **)&warp) &&
xe.xany.serial == warp->serial)
{
ll_shift(g_cursor.warpList, NULL);
free(warp);
g_cursor.last.x = xe.xmotion.x;
g_cursor.last.y = xe.xmotion.y;
break;
}
handleMouseMoveEvent(xe.xmotion.x, xe.xmotion.y);
break;
}
case EnterNotify:
{
union WarpInfo * warp;
if (ll_peek_head(g_cursor.warpList, (void **)&warp) &&
xe.xany.serial == warp->serial)
{
ll_shift(g_cursor.warpList, NULL);
free(warp);
}
g_cursor.last.x = xe.xcrossing.x;
g_cursor.last.y = xe.xcrossing.y;
handleWindowEnter();
break;
}
case LeaveNotify:
{
union WarpInfo * warp;
if (ll_peek_head(g_cursor.warpList, (void **)&warp) &&
xe.xany.serial == warp->serial)
{
ll_shift(g_cursor.warpList, NULL);
free(warp);
}
if (xe.xcrossing.mode != NotifyNormal)
break;
@ -1179,6 +1219,7 @@ int eventFilter(void * userdata, SDL_Event * event)
g_cursor.last.y = xe.xcrossing.y;
handleWindowLeave();
break;
}
case FocusIn:
if (!params.useSpiceInput)
@ -1197,6 +1238,18 @@ int eventFilter(void * userdata, SDL_Event * event)
xe.xfocus.mode == NotifyWhileGrabbed)
keyboardUngrab();
break;
default:
{
union WarpInfo * warp;
if (ll_peek_head(g_cursor.warpList, (void **)&warp) &&
xe.xany.serial == warp->serial)
{
DEBUG_INFO("bug: %d", xe.type);
ll_shift(g_cursor.warpList, NULL);
free(warp);
}
}
}
}
@ -1206,9 +1259,27 @@ int eventFilter(void * userdata, SDL_Event * event)
}
case SDL_MOUSEMOTION:
if (g_state.wminfo.subsystem != SDL_SYSWM_X11)
handleMouseMoveEvent(event->motion.x, event->motion.y);
{
if (g_state.wminfo.subsystem == SDL_SYSWM_X11)
break;
/* detect and filter out warp events */
union WarpInfo * warp;
if (ll_peek_head(g_cursor.warpList, (void **)&warp) &&
warp->p.x == event->motion.x &&
warp->p.y == event->motion.y)
{
ll_shift(g_cursor.warpList, NULL);
free(warp);
g_cursor.last.x = event->motion.x;
g_cursor.last.y = event->motion.y;
break;
}
handleMouseMoveEvent(event->motion.x, event->motion.y);
break;
}
case SDL_KEYDOWN:
{
@ -1573,6 +1644,8 @@ static void initSDLCursor()
static int lg_run()
{
memset(&g_state, 0, sizeof(g_state));
g_cursor.warpList = ll_new();
lgInit();
g_cursor.sens = params.mouseSens;
@ -1742,27 +1815,13 @@ static int lg_run()
{
if (g_state.wminfo.subsystem == SDL_SYSWM_X11)
{
int event, error;
// enable X11 events to work around SDL2 bugs
SDL_EventState(SDL_SYSWMEVENT, SDL_ENABLE);
#if SDL_VIDEO_DRIVER_X11_XINPUT2
// SDL2 bug, using xinput2 disables all motion notify events
// we really don't care about touch, so turn it off and go back
// to the default behaiovur.
XIEventMask xinputmask =
{
.deviceid = XIAllMasterDevices,
.mask = 0,
.mask_len = 0
};
XISelectEvents(
g_state.wminfo.info.x11.display,
g_state.wminfo.info.x11.window,
&xinputmask,
1
);
#endif
XQueryExtension(g_state.wminfo.info.x11.display, "XInputExtension",
&g_XInputOp, &event, &error);
Atom NETWM_BYPASS_COMPOSITOR = XInternAtom(
g_state.wminfo.info.x11.display,
@ -2036,6 +2095,15 @@ static void lg_shutdown()
ivshmemClose(&g_state.shm);
release_key_binds();
if (g_cursor.warpList)
{
union WarpInfo * warp;
while(ll_shift(g_cursor.warpList, (void **)&warp))
free(warp);
ll_free(g_cursor.warpList);
}
SDL_Quit();
}

View file

@ -25,6 +25,7 @@ Place, Suite 330, Boston, MA 02111-1307 USA
#include "dynamic/renderers.h"
#include "dynamic/clipboards.h"
#include "common/ivshmem.h"
#include "ll.h"
#include "spice/spice.h"
#include <lgmp/client.h>
@ -156,8 +157,6 @@ struct KeybindHandle
enum WarpState
{
WARP_STATE_ON,
WARP_STATE_ACTIVE,
WARP_STATE_WIN_EXIT,
WARP_STATE_OFF
};
@ -215,11 +214,9 @@ struct CursorState
int sens;
float sensX, sensY;
/* the mouse warp state and target */
/* the mouse warp state and queue */
enum WarpState warpState;
bool warpExit;
unsigned long warpSerial;
SDL_Point warpTo;
struct ll * warpList;
/* the guest's cursor position */
struct CursorInfo guest;