xiquerypointer.c revision 0b0d8713
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); 670b0d8713Smrg REQUEST_SIZE_MATCH(xXIQueryPointerReq); 680b0d8713Smrg 696747b715Smrg swaps(&stuff->length, n); 706747b715Smrg swaps(&stuff->deviceid, n); 716747b715Smrg swapl(&stuff->win, n); 726747b715Smrg return (ProcXIQueryPointer(client)); 736747b715Smrg} 746747b715Smrg 756747b715Smrgint 766747b715SmrgProcXIQueryPointer(ClientPtr client) 776747b715Smrg{ 786747b715Smrg int rc; 796747b715Smrg xXIQueryPointerReply rep; 806747b715Smrg DeviceIntPtr pDev, kbd; 816747b715Smrg WindowPtr pWin, t; 826747b715Smrg SpritePtr pSprite; 836747b715Smrg XkbStatePtr state; 846747b715Smrg char *buttons = NULL; 856747b715Smrg int buttons_size = 0; /* size of buttons array */ 866747b715Smrg 876747b715Smrg REQUEST(xXIQueryPointerReq); 886747b715Smrg REQUEST_SIZE_MATCH(xXIQueryPointerReq); 896747b715Smrg 906747b715Smrg rc = dixLookupDevice(&pDev, stuff->deviceid, client, DixReadAccess); 916747b715Smrg if (rc != Success) 926747b715Smrg { 936747b715Smrg client->errorValue = stuff->deviceid; 946747b715Smrg return rc; 956747b715Smrg } 966747b715Smrg 976747b715Smrg if (pDev->valuator == NULL || IsKeyboardDevice(pDev) || 986747b715Smrg (!IsMaster(pDev) && pDev->u.master)) /* no attached devices */ 996747b715Smrg { 1006747b715Smrg client->errorValue = stuff->deviceid; 1016747b715Smrg return BadDevice; 1026747b715Smrg } 1036747b715Smrg 1046747b715Smrg rc = dixLookupWindow(&pWin, stuff->win, client, DixGetAttrAccess); 1056747b715Smrg if (rc != Success) 1066747b715Smrg { 1076747b715Smrg SendErrorToClient(client, IReqCode, X_XIQueryPointer, 1086747b715Smrg stuff->win, rc); 1096747b715Smrg return Success; 1106747b715Smrg } 1116747b715Smrg 1126747b715Smrg if (pDev->valuator->motionHintWindow) 1136747b715Smrg MaybeStopHint(pDev, client); 1146747b715Smrg 1156747b715Smrg if (IsMaster(pDev)) 1166747b715Smrg kbd = GetPairedDevice(pDev); 1176747b715Smrg else 1186747b715Smrg kbd = (pDev->key) ? pDev : NULL; 1196747b715Smrg 1206747b715Smrg pSprite = pDev->spriteInfo->sprite; 1216747b715Smrg 1226747b715Smrg memset(&rep, 0, sizeof(rep)); 1236747b715Smrg rep.repType = X_Reply; 1246747b715Smrg rep.RepType = X_XIQueryPointer; 1256747b715Smrg rep.length = 6; 1266747b715Smrg rep.sequenceNumber = client->sequence; 1276747b715Smrg rep.root = (GetCurrentRootWindow(pDev))->drawable.id; 1286747b715Smrg rep.root_x = FP1616(pSprite->hot.x, 0); 1296747b715Smrg rep.root_y = FP1616(pSprite->hot.y, 0); 1306747b715Smrg rep.child = None; 1316747b715Smrg 1326747b715Smrg if (kbd) 1336747b715Smrg { 1349ace9065Smrg state = &kbd->key->xkbInfo->state; 1356747b715Smrg rep.mods.base_mods = state->base_mods; 1366747b715Smrg rep.mods.latched_mods = state->latched_mods; 1376747b715Smrg rep.mods.locked_mods = state->locked_mods; 1386747b715Smrg 1396747b715Smrg rep.group.base_group = state->base_group; 1406747b715Smrg rep.group.latched_group = state->latched_group; 1416747b715Smrg rep.group.locked_group = state->locked_group; 1426747b715Smrg } 1436747b715Smrg 1446747b715Smrg if (pDev->button) 1456747b715Smrg { 1466747b715Smrg int i, down; 1476747b715Smrg rep.buttons_len = bytes_to_int32(bits_to_bytes(pDev->button->numButtons)); 1486747b715Smrg rep.length += rep.buttons_len; 1496747b715Smrg buttons_size = rep.buttons_len * 4; 1506747b715Smrg buttons = calloc(1, buttons_size); 1516747b715Smrg if (!buttons) 1526747b715Smrg return BadAlloc; 1536747b715Smrg 1546747b715Smrg down = pDev->button->buttonsDown; 1556747b715Smrg 1566747b715Smrg for (i = 0; i < pDev->button->numButtons && down; i++) 1576747b715Smrg { 1586747b715Smrg if (BitIsOn(pDev->button->down, i)) 1596747b715Smrg { 1606747b715Smrg SetBit(buttons, i); 1616747b715Smrg down--; 1626747b715Smrg } 1636747b715Smrg } 1646747b715Smrg } else 1656747b715Smrg rep.buttons_len = 0; 1666747b715Smrg 1676747b715Smrg if (pSprite->hot.pScreen == pWin->drawable.pScreen) 1686747b715Smrg { 1696747b715Smrg rep.same_screen = xTrue; 1706747b715Smrg rep.win_x = FP1616(pSprite->hot.x - pWin->drawable.x, 0); 1716747b715Smrg rep.win_y = FP1616(pSprite->hot.y - pWin->drawable.y, 0); 1726747b715Smrg for (t = pSprite->win; t; t = t->parent) 1736747b715Smrg if (t->parent == pWin) 1746747b715Smrg { 1756747b715Smrg rep.child = t->drawable.id; 1766747b715Smrg break; 1776747b715Smrg } 1786747b715Smrg } else 1796747b715Smrg { 1806747b715Smrg rep.same_screen = xFalse; 1816747b715Smrg rep.win_x = 0; 1826747b715Smrg rep.win_y = 0; 1836747b715Smrg } 1846747b715Smrg 1856747b715Smrg#ifdef PANORAMIX 1866747b715Smrg if(!noPanoramiXExtension) { 1876747b715Smrg rep.root_x += FP1616(screenInfo.screens[0]->x, 0); 1886747b715Smrg rep.root_y += FP1616(screenInfo.screens[0]->y, 0); 1896747b715Smrg if (stuff->win == rep.root) 1906747b715Smrg { 1916747b715Smrg rep.win_x += FP1616(screenInfo.screens[0]->x, 0); 1926747b715Smrg rep.win_y += FP1616(screenInfo.screens[0]->y, 0); 1936747b715Smrg } 1946747b715Smrg } 1956747b715Smrg#endif 1966747b715Smrg 1976747b715Smrg WriteReplyToClient(client, sizeof(xXIQueryPointerReply), &rep); 1986747b715Smrg if (buttons) 1996747b715Smrg WriteToClient(client, buttons_size, buttons); 2006747b715Smrg 2016747b715Smrg free(buttons); 2026747b715Smrg 2036747b715Smrg return Success; 2046747b715Smrg} 2056747b715Smrg 2066747b715Smrg/*********************************************************************** 2076747b715Smrg * 2086747b715Smrg * This procedure writes the reply for the XIQueryPointer function, 2096747b715Smrg * if the client and server have a different byte ordering. 2106747b715Smrg * 2116747b715Smrg */ 2126747b715Smrg 2136747b715Smrgvoid 2146747b715SmrgSRepXIQueryPointer(ClientPtr client, int size, 2156747b715Smrg xXIQueryPointerReply * rep) 2166747b715Smrg{ 2176747b715Smrg char n; 2186747b715Smrg 2196747b715Smrg swaps(&rep->sequenceNumber, n); 2206747b715Smrg swapl(&rep->length, n); 2216747b715Smrg swapl(&rep->root, n); 2226747b715Smrg swapl(&rep->child, n); 2236747b715Smrg swapl(&rep->root_x, n); 2246747b715Smrg swapl(&rep->root_y, n); 2256747b715Smrg swapl(&rep->win_x, n); 2266747b715Smrg swapl(&rep->win_y, n); 2276747b715Smrg swaps(&rep->buttons_len, n); 2286747b715Smrg 2296747b715Smrg WriteToClient(client, size, (char *)rep); 2306747b715Smrg} 2316747b715Smrg 232