xiquerypointer.c revision 6747b715
16747b715Smrg/* 26747b715Smrg * Copyright 2007-2008 Peter Hutterer 36747b715Smrg * 46747b715Smrg * Permission is hereby granted, free of charge, to any person obtaining a 56747b715Smrg * copy of this software and associated documentation files (the "Software"), 66747b715Smrg * to deal in the Software without restriction, including without limitation 76747b715Smrg * the rights to use, copy, modify, merge, publish, distribute, sublicense, 86747b715Smrg * and/or sell copies of the Software, and to permit persons to whom the 96747b715Smrg * Software is furnished to do so, subject to the following conditions: 106747b715Smrg * 116747b715Smrg * The above copyright notice and this permission notice (including the next 126747b715Smrg * paragraph) shall be included in all copies or substantial portions of the 136747b715Smrg * Software. 146747b715Smrg * 156747b715Smrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 166747b715Smrg * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 176747b715Smrg * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL 186747b715Smrg * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 196747b715Smrg * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 206747b715Smrg * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER 216747b715Smrg * DEALINGS IN THE SOFTWARE. 226747b715Smrg * 236747b715Smrg * Author: Peter Hutterer, University of South Australia, NICTA 246747b715Smrg */ 256747b715Smrg 266747b715Smrg/*********************************************************************** 276747b715Smrg * 286747b715Smrg * Request to query the pointer location of an extension input device. 296747b715Smrg * 306747b715Smrg */ 316747b715Smrg 326747b715Smrg#ifdef HAVE_DIX_CONFIG_H 336747b715Smrg#include <dix-config.h> 346747b715Smrg#endif 356747b715Smrg 366747b715Smrg#include <X11/X.h> /* for inputstr.h */ 376747b715Smrg#include <X11/Xproto.h> /* Request macro */ 386747b715Smrg#include "inputstr.h" /* DeviceIntPtr */ 396747b715Smrg#include "windowstr.h" /* window structure */ 406747b715Smrg#include <X11/extensions/XI.h> 416747b715Smrg#include <X11/extensions/XI2proto.h> 426747b715Smrg#include "extnsionst.h" 436747b715Smrg#include "exevents.h" 446747b715Smrg#include "exglobals.h" 456747b715Smrg#include "eventconvert.h" 466747b715Smrg#include "scrnintstr.h" 476747b715Smrg#include "xkbsrv.h" 486747b715Smrg 496747b715Smrg#ifdef PANORAMIX 506747b715Smrg#include "panoramiXsrv.h" 516747b715Smrg#endif 526747b715Smrg 536747b715Smrg#include "xiquerypointer.h" 546747b715Smrg 556747b715Smrg/*********************************************************************** 566747b715Smrg * 576747b715Smrg * This procedure allows a client to query the pointer of a device. 586747b715Smrg * 596747b715Smrg */ 606747b715Smrg 616747b715Smrgint 626747b715SmrgSProcXIQueryPointer(ClientPtr client) 636747b715Smrg{ 646747b715Smrg char n; 656747b715Smrg 666747b715Smrg REQUEST(xXIQueryPointerReq); 676747b715Smrg swaps(&stuff->length, n); 686747b715Smrg swaps(&stuff->deviceid, n); 696747b715Smrg swapl(&stuff->win, n); 706747b715Smrg return (ProcXIQueryPointer(client)); 716747b715Smrg} 726747b715Smrg 736747b715Smrgint 746747b715SmrgProcXIQueryPointer(ClientPtr client) 756747b715Smrg{ 766747b715Smrg int rc; 776747b715Smrg xXIQueryPointerReply rep; 786747b715Smrg DeviceIntPtr pDev, kbd; 796747b715Smrg WindowPtr pWin, t; 806747b715Smrg SpritePtr pSprite; 816747b715Smrg XkbStatePtr state; 826747b715Smrg char *buttons = NULL; 836747b715Smrg int buttons_size = 0; /* size of buttons array */ 846747b715Smrg 856747b715Smrg REQUEST(xXIQueryPointerReq); 866747b715Smrg REQUEST_SIZE_MATCH(xXIQueryPointerReq); 876747b715Smrg 886747b715Smrg rc = dixLookupDevice(&pDev, stuff->deviceid, client, DixReadAccess); 896747b715Smrg if (rc != Success) 906747b715Smrg { 916747b715Smrg client->errorValue = stuff->deviceid; 926747b715Smrg return rc; 936747b715Smrg } 946747b715Smrg 956747b715Smrg if (pDev->valuator == NULL || IsKeyboardDevice(pDev) || 966747b715Smrg (!IsMaster(pDev) && pDev->u.master)) /* no attached devices */ 976747b715Smrg { 986747b715Smrg client->errorValue = stuff->deviceid; 996747b715Smrg return BadDevice; 1006747b715Smrg } 1016747b715Smrg 1026747b715Smrg rc = dixLookupWindow(&pWin, stuff->win, client, DixGetAttrAccess); 1036747b715Smrg if (rc != Success) 1046747b715Smrg { 1056747b715Smrg SendErrorToClient(client, IReqCode, X_XIQueryPointer, 1066747b715Smrg stuff->win, rc); 1076747b715Smrg return Success; 1086747b715Smrg } 1096747b715Smrg 1106747b715Smrg if (pDev->valuator->motionHintWindow) 1116747b715Smrg MaybeStopHint(pDev, client); 1126747b715Smrg 1136747b715Smrg if (IsMaster(pDev)) 1146747b715Smrg kbd = GetPairedDevice(pDev); 1156747b715Smrg else 1166747b715Smrg kbd = (pDev->key) ? pDev : NULL; 1176747b715Smrg 1186747b715Smrg pSprite = pDev->spriteInfo->sprite; 1196747b715Smrg 1206747b715Smrg memset(&rep, 0, sizeof(rep)); 1216747b715Smrg rep.repType = X_Reply; 1226747b715Smrg rep.RepType = X_XIQueryPointer; 1236747b715Smrg rep.length = 6; 1246747b715Smrg rep.sequenceNumber = client->sequence; 1256747b715Smrg rep.root = (GetCurrentRootWindow(pDev))->drawable.id; 1266747b715Smrg rep.root_x = FP1616(pSprite->hot.x, 0); 1276747b715Smrg rep.root_y = FP1616(pSprite->hot.y, 0); 1286747b715Smrg rep.child = None; 1296747b715Smrg 1306747b715Smrg if (kbd) 1316747b715Smrg { 1326747b715Smrg state = &kbd->key->xkbInfo->prev_state; 1336747b715Smrg rep.mods.base_mods = state->base_mods; 1346747b715Smrg rep.mods.latched_mods = state->latched_mods; 1356747b715Smrg rep.mods.locked_mods = state->locked_mods; 1366747b715Smrg 1376747b715Smrg rep.group.base_group = state->base_group; 1386747b715Smrg rep.group.latched_group = state->latched_group; 1396747b715Smrg rep.group.locked_group = state->locked_group; 1406747b715Smrg } 1416747b715Smrg 1426747b715Smrg if (pDev->button) 1436747b715Smrg { 1446747b715Smrg int i, down; 1456747b715Smrg rep.buttons_len = bytes_to_int32(bits_to_bytes(pDev->button->numButtons)); 1466747b715Smrg rep.length += rep.buttons_len; 1476747b715Smrg buttons_size = rep.buttons_len * 4; 1486747b715Smrg buttons = calloc(1, buttons_size); 1496747b715Smrg if (!buttons) 1506747b715Smrg return BadAlloc; 1516747b715Smrg 1526747b715Smrg down = pDev->button->buttonsDown; 1536747b715Smrg 1546747b715Smrg for (i = 0; i < pDev->button->numButtons && down; i++) 1556747b715Smrg { 1566747b715Smrg if (BitIsOn(pDev->button->down, i)) 1576747b715Smrg { 1586747b715Smrg SetBit(buttons, i); 1596747b715Smrg down--; 1606747b715Smrg } 1616747b715Smrg } 1626747b715Smrg } else 1636747b715Smrg rep.buttons_len = 0; 1646747b715Smrg 1656747b715Smrg if (pSprite->hot.pScreen == pWin->drawable.pScreen) 1666747b715Smrg { 1676747b715Smrg rep.same_screen = xTrue; 1686747b715Smrg rep.win_x = FP1616(pSprite->hot.x - pWin->drawable.x, 0); 1696747b715Smrg rep.win_y = FP1616(pSprite->hot.y - pWin->drawable.y, 0); 1706747b715Smrg for (t = pSprite->win; t; t = t->parent) 1716747b715Smrg if (t->parent == pWin) 1726747b715Smrg { 1736747b715Smrg rep.child = t->drawable.id; 1746747b715Smrg break; 1756747b715Smrg } 1766747b715Smrg } else 1776747b715Smrg { 1786747b715Smrg rep.same_screen = xFalse; 1796747b715Smrg rep.win_x = 0; 1806747b715Smrg rep.win_y = 0; 1816747b715Smrg } 1826747b715Smrg 1836747b715Smrg#ifdef PANORAMIX 1846747b715Smrg if(!noPanoramiXExtension) { 1856747b715Smrg rep.root_x += FP1616(screenInfo.screens[0]->x, 0); 1866747b715Smrg rep.root_y += FP1616(screenInfo.screens[0]->y, 0); 1876747b715Smrg if (stuff->win == rep.root) 1886747b715Smrg { 1896747b715Smrg rep.win_x += FP1616(screenInfo.screens[0]->x, 0); 1906747b715Smrg rep.win_y += FP1616(screenInfo.screens[0]->y, 0); 1916747b715Smrg } 1926747b715Smrg } 1936747b715Smrg#endif 1946747b715Smrg 1956747b715Smrg WriteReplyToClient(client, sizeof(xXIQueryPointerReply), &rep); 1966747b715Smrg if (buttons) 1976747b715Smrg WriteToClient(client, buttons_size, buttons); 1986747b715Smrg 1996747b715Smrg free(buttons); 2006747b715Smrg 2016747b715Smrg return Success; 2026747b715Smrg} 2036747b715Smrg 2046747b715Smrg/*********************************************************************** 2056747b715Smrg * 2066747b715Smrg * This procedure writes the reply for the XIQueryPointer function, 2076747b715Smrg * if the client and server have a different byte ordering. 2086747b715Smrg * 2096747b715Smrg */ 2106747b715Smrg 2116747b715Smrgvoid 2126747b715SmrgSRepXIQueryPointer(ClientPtr client, int size, 2136747b715Smrg xXIQueryPointerReply * rep) 2146747b715Smrg{ 2156747b715Smrg char n; 2166747b715Smrg 2176747b715Smrg swaps(&rep->sequenceNumber, n); 2186747b715Smrg swapl(&rep->length, n); 2196747b715Smrg swapl(&rep->root, n); 2206747b715Smrg swapl(&rep->child, n); 2216747b715Smrg swapl(&rep->root_x, n); 2226747b715Smrg swapl(&rep->root_y, n); 2236747b715Smrg swapl(&rep->win_x, n); 2246747b715Smrg swapl(&rep->win_y, n); 2256747b715Smrg swaps(&rep->buttons_len, n); 2266747b715Smrg 2276747b715Smrg WriteToClient(client, size, (char *)rep); 2286747b715Smrg} 2296747b715Smrg 230