/*
* Copyright © 2009 Red Hat, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a
* copy of this software and associated documentation files (the "Software"),
* to deal in the Software without restriction, including without limitation
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
* and/or sell copies of the Software, and to permit persons to whom the
* Software is furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice (including the next
* paragraph) shall be included in all copies or substantial portions of the
* Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
* DEALINGS IN THE SOFTWARE.
*
*/
#ifdef HAVE_CONFIG_H
#include <config.h>
#endif
#include <stdint.h>
#include <X11/Xlibint.h>
#include <X11/extensions/XI2proto.h>
#include <X11/extensions/XInput2.h>
#include <X11/extensions/extutil.h>
#include <limits.h>
#include "XIint.h"
static int
_XIPassiveGrabDevice(Display* dpy, int deviceid, int grabtype, int detail,
Window grab_window, Cursor cursor,
int grab_mode, int paired_device_mode,
Bool owner_events, XIEventMask *mask,
int num_modifiers, XIGrabModifiers *modifiers_inout,
Time time)
{
xXIPassiveGrabDeviceReq *req;
xXIPassiveGrabDeviceReply reply;
xXIGrabModifierInfo *failed_mods;
int len = 0, i;
int ret = -1;
char *buff;
XExtDisplayInfo *extinfo = XInput_find_display(dpy);
LockDisplay(dpy);
if (_XiCheckExtInit(dpy, XInput_2_0, extinfo) == -1)
return -1;
if (mask->mask_len > INT_MAX - 3 ||
(mask->mask_len + 3)/4 >= 0xffff)
goto out;
buff = calloc(4, (mask->mask_len + 3)/4);
if (!buff)
goto out;
GetReq(XIPassiveGrabDevice, req);
req->reqType = extinfo->codes->major_opcode;
req->ReqType = X_XIPassiveGrabDevice;
req->deviceid = deviceid;
req->grab_mode = grab_mode;
req->paired_device_mode = paired_device_mode;
req->owner_events = owner_events;
req->grab_window = grab_window;
req->cursor = cursor;
req->detail = detail;
req->num_modifiers = num_modifiers;
req->mask_len = (mask->mask_len + 3)/4;
req->grab_type = grabtype;
req->time = time;
len = req->mask_len + num_modifiers;
SetReqLen(req, len, len);
memcpy(buff, mask->mask, mask->mask_len);
Data(dpy, buff, req->mask_len * 4);
for (i = 0; i < num_modifiers; i++)
Data(dpy, (char*)&modifiers_inout[i].modifiers, 4);
free(buff);
if (!_XReply(dpy, (xReply *)&reply, 0, xFalse))
goto out;
failed_mods = calloc(reply.num_modifiers, sizeof(xXIGrabModifierInfo));
if (!failed_mods)
goto out;
_XRead(dpy, (char*)failed_mods, reply.num_modifiers * sizeof(xXIGrabModifierInfo));
for (i = 0; i < reply.num_modifiers && i < num_modifiers; i++)
{
modifiers_inout[i].status = failed_mods[i].status;
modifiers_inout[i].modifiers = failed_mods[i].modifiers;
}
free(failed_mods);
ret = reply.num_modifiers;
out:
UnlockDisplay(dpy);
SyncHandle();
return ret;
}
int
XIGrabButton(Display* dpy, int deviceid, int button,
Window grab_window, Cursor cursor,
int grab_mode, int paired_device_mode,
Bool owner_events, XIEventMask *mask,
int num_modifiers, XIGrabModifiers *modifiers_inout)
{
return _XIPassiveGrabDevice(dpy, deviceid, XIGrabtypeButton, button,
grab_window, cursor, grab_mode,
paired_device_mode, owner_events, mask,
num_modifiers, modifiers_inout, CurrentTime);
}
int
XIGrabKeycode(Display* dpy, int deviceid, int keycode,
Window grab_window, int grab_mode, int paired_device_mode,
Bool owner_events, XIEventMask *mask,
int num_modifiers, XIGrabModifiers *modifiers_inout)
{
return _XIPassiveGrabDevice(dpy, deviceid, XIGrabtypeKeycode, keycode,
grab_window, None, grab_mode, paired_device_mode,
owner_events, mask, num_modifiers,
modifiers_inout, CurrentTime);
}
int
XIGrabEnter(Display *dpy, int deviceid, Window grab_window, Cursor cursor,
int grab_mode, int paired_device_mode, Bool owner_events,
XIEventMask *mask, int num_modifiers,
XIGrabModifiers *modifiers_inout)
{
return _XIPassiveGrabDevice(dpy, deviceid, XIGrabtypeEnter, 0,
grab_window, cursor, grab_mode, paired_device_mode,
owner_events, mask, num_modifiers,
modifiers_inout, CurrentTime);
}
int
XIGrabFocusIn(Display *dpy, int deviceid, Window grab_window, int grab_mode,
int paired_device_mode, Bool owner_events, XIEventMask *mask,
int num_modifiers, XIGrabModifiers *modifiers_inout)
{
return _XIPassiveGrabDevice(dpy, deviceid, XIGrabtypeFocusIn, 0,
grab_window, None, grab_mode, paired_device_mode,
owner_events, mask, num_modifiers,
modifiers_inout, CurrentTime);
}
int
XIGrabTouchBegin(Display *dpy, int deviceid, Window grab_window,
Bool owner_events, XIEventMask *mask,
int num_modifiers, XIGrabModifiers *modifiers_inout)
{
XExtDisplayInfo *extinfo = XInput_find_display(dpy);
LockDisplay(dpy);
if (_XiCheckExtInit(dpy, XInput_2_2, extinfo) == -1)
return -1;
UnlockDisplay(dpy);
/* FIXME: allow selection of GrabMode for paired devices? */
return _XIPassiveGrabDevice(dpy, deviceid, XIGrabtypeTouchBegin, 0,
grab_window, None, XIGrabModeTouch,
GrabModeAsync, owner_events, mask,
num_modifiers, modifiers_inout, CurrentTime);
}
int XIGrabPinchGestureBegin(Display* dpy, int deviceid, Window grab_window, int grab_mode,
int paired_device_mode, int owner_events,
XIEventMask *mask, int num_modifiers, XIGrabModifiers *modifiers_inout)
{
XExtDisplayInfo *extinfo = XInput_find_display(dpy);
LockDisplay(dpy);
if (_XiCheckExtInit(dpy, XInput_2_4, extinfo) == -1)
return -1;
UnlockDisplay(dpy);
return _XIPassiveGrabDevice(dpy, deviceid, XIGrabtypeGesturePinchBegin, 0,
grab_window, None, grab_mode, paired_device_mode,
owner_events, mask, num_modifiers, modifiers_inout, CurrentTime);
}
int XIGrabSwipeGestureBegin(Display* dpy, int deviceid, Window grab_window, int grab_mode,
int paired_device_mode, int owner_events,
XIEventMask *mask, int num_modifiers, XIGrabModifiers *modifiers_inout)
{
XExtDisplayInfo *extinfo = XInput_find_display(dpy);
LockDisplay(dpy);
if (_XiCheckExtInit(dpy, XInput_2_4, extinfo) == -1)
return -1;
UnlockDisplay(dpy);
return _XIPassiveGrabDevice(dpy, deviceid, XIGrabtypeGestureSwipeBegin, 0,
grab_window, None, grab_mode, paired_device_mode,
owner_events, mask, num_modifiers, modifiers_inout, CurrentTime);
}
static int
_XIPassiveUngrabDevice(Display* dpy, int deviceid, int grabtype, int detail,
Window grab_window, int num_modifiers, XIGrabModifiers *modifiers)
{
xXIPassiveUngrabDeviceReq *req;
int i;
XExtDisplayInfo *extinfo = XInput_find_display(dpy);
LockDisplay(dpy);
if (_XiCheckExtInit(dpy, XInput_2_0, extinfo) == -1)
return -1;
GetReq(XIPassiveUngrabDevice, req);
req->reqType = extinfo->codes->major_opcode;
req->ReqType = X_XIPassiveUngrabDevice;
req->deviceid = deviceid;
req->grab_window = grab_window;
req->detail = detail;
req->num_modifiers = num_modifiers;
req->grab_type = grabtype;
SetReqLen(req, num_modifiers, num_modifiers);
for (i = 0; i < num_modifiers; i++)
Data32(dpy, &modifiers[i].modifiers, 4);
UnlockDisplay(dpy);
SyncHandle();
return Success;
}
int
XIUngrabButton(Display* display, int deviceid, int button,Window grab_window,
int num_modifiers, XIGrabModifiers *modifiers)
{
return _XIPassiveUngrabDevice(display, deviceid, XIGrabtypeButton, button,
grab_window, num_modifiers, modifiers);
}
int
XIUngrabKeycode(Display* display, int deviceid, int keycode, Window grab_window,
int num_modifiers, XIGrabModifiers *modifiers)
{
return _XIPassiveUngrabDevice(display, deviceid, XIGrabtypeKeycode, keycode,
grab_window, num_modifiers, modifiers);
}
int
XIUngrabEnter(Display* display, int deviceid, Window grab_window,
int num_modifiers, XIGrabModifiers *modifiers)
{
return _XIPassiveUngrabDevice(display, deviceid, XIGrabtypeEnter, 0,
grab_window, num_modifiers, modifiers);
}
int
XIUngrabFocusIn(Display* display, int deviceid, Window grab_window,
int num_modifiers, XIGrabModifiers *modifiers)
{
return _XIPassiveUngrabDevice(display, deviceid, XIGrabtypeFocusIn, 0,
grab_window, num_modifiers, modifiers);
}
int
XIUngrabTouchBegin(Display* display, int deviceid, Window grab_window,
int num_modifiers, XIGrabModifiers *modifiers)
{
XExtDisplayInfo *extinfo = XInput_find_display(display);
LockDisplay(display);
if (_XiCheckExtInit(display, XInput_2_2, extinfo) == -1)
return -1;
UnlockDisplay(display);
return _XIPassiveUngrabDevice(display, deviceid, XIGrabtypeTouchBegin, 0,
grab_window, num_modifiers, modifiers);
}
int XIUngrabPinchGestureBegin(Display* display, int deviceid, Window grab_window,
int num_modifiers, XIGrabModifiers *modifiers)
{
XExtDisplayInfo *extinfo = XInput_find_display(display);
LockDisplay(display);
if (_XiCheckExtInit(display, XInput_2_4, extinfo) == -1)
return -1;
UnlockDisplay(display);
return _XIPassiveUngrabDevice(display, deviceid, XIGrabtypeGesturePinchBegin, 0,
grab_window, num_modifiers, modifiers);
}
int XIUngrabSwipeGestureBegin(Display* display, int deviceid, Window grab_window,
int num_modifiers, XIGrabModifiers *modifiers)
{
XExtDisplayInfo *extinfo = XInput_find_display(display);
LockDisplay(display);
if (_XiCheckExtInit(display, XInput_2_4, extinfo) == -1)
return -1;
UnlockDisplay(display);
return _XIPassiveUngrabDevice(display, deviceid, XIGrabtypeGestureSwipeBegin, 0,
grab_window, num_modifiers, modifiers);
}