xiquerydevice.c revision 0b0d8713
16747b715Smrg/*
26747b715Smrg * Copyright © 2009 Red Hat, Inc.
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 * Authors: Peter Hutterer
246747b715Smrg *
256747b715Smrg */
266747b715Smrg
276747b715Smrg/**
286747b715Smrg * @file Protocol handling for the XIQueryDevice request/reply.
296747b715Smrg */
306747b715Smrg
316747b715Smrg#ifdef HAVE_DIX_CONFIG_H
326747b715Smrg#include <dix-config.h>
336747b715Smrg#endif
346747b715Smrg
356747b715Smrg#include "inputstr.h"
366747b715Smrg#include <X11/X.h>
376747b715Smrg#include <X11/Xatom.h>
386747b715Smrg#include <X11/extensions/XI2proto.h>
396747b715Smrg#include "xkbstr.h"
406747b715Smrg#include "xkbsrv.h"
416747b715Smrg#include "xserver-properties.h"
426747b715Smrg#include "exevents.h"
436747b715Smrg#include "xace.h"
446747b715Smrg
456747b715Smrg#include "xiquerydevice.h"
466747b715Smrg
476747b715Smrgstatic Bool ShouldSkipDevice(ClientPtr client, int deviceid, DeviceIntPtr d);
486747b715Smrgstatic int
496747b715SmrgListDeviceInfo(ClientPtr client, DeviceIntPtr dev, xXIDeviceInfo* info);
506747b715Smrgstatic int SizeDeviceInfo(DeviceIntPtr dev);
516747b715Smrgstatic void SwapDeviceInfo(DeviceIntPtr dev, xXIDeviceInfo* info);
526747b715Smrgint
536747b715SmrgSProcXIQueryDevice(ClientPtr client)
546747b715Smrg{
556747b715Smrg    char n;
566747b715Smrg
576747b715Smrg    REQUEST(xXIQueryDeviceReq);
580b0d8713Smrg    REQUEST_SIZE_MATCH(xXIQueryDeviceReq);
596747b715Smrg
606747b715Smrg    swaps(&stuff->length, n);
616747b715Smrg    swaps(&stuff->deviceid, n);
626747b715Smrg
636747b715Smrg    return ProcXIQueryDevice(client);
646747b715Smrg}
656747b715Smrg
666747b715Smrgint
676747b715SmrgProcXIQueryDevice(ClientPtr client)
686747b715Smrg{
696747b715Smrg    xXIQueryDeviceReply rep;
706747b715Smrg    DeviceIntPtr dev = NULL;
716747b715Smrg    int rc = Success;
726747b715Smrg    int i = 0, len = 0;
736747b715Smrg    char *info, *ptr;
746747b715Smrg    Bool *skip = NULL;
756747b715Smrg
766747b715Smrg    REQUEST(xXIQueryDeviceReq);
776747b715Smrg    REQUEST_SIZE_MATCH(xXIQueryDeviceReq);
786747b715Smrg
796747b715Smrg    if (stuff->deviceid != XIAllDevices && stuff->deviceid != XIAllMasterDevices)
806747b715Smrg    {
816747b715Smrg        rc = dixLookupDevice(&dev, stuff->deviceid, client, DixGetAttrAccess);
826747b715Smrg        if (rc != Success)
836747b715Smrg        {
846747b715Smrg            client->errorValue = stuff->deviceid;
856747b715Smrg            return rc;
866747b715Smrg        }
876747b715Smrg        len += SizeDeviceInfo(dev);
886747b715Smrg    }
896747b715Smrg    else
906747b715Smrg    {
916747b715Smrg        skip = calloc(sizeof(Bool), inputInfo.numDevices);
926747b715Smrg        if (!skip)
936747b715Smrg            return BadAlloc;
946747b715Smrg
956747b715Smrg        for (dev = inputInfo.devices; dev; dev = dev->next, i++)
966747b715Smrg        {
976747b715Smrg            skip[i] = ShouldSkipDevice(client, stuff->deviceid, dev);
986747b715Smrg            if (!skip[i])
996747b715Smrg                len += SizeDeviceInfo(dev);
1006747b715Smrg        }
1016747b715Smrg
1026747b715Smrg        for (dev = inputInfo.off_devices; dev; dev = dev->next, i++)
1036747b715Smrg        {
1046747b715Smrg            skip[i] = ShouldSkipDevice(client, stuff->deviceid, dev);
1056747b715Smrg            if (!skip[i])
1066747b715Smrg                len += SizeDeviceInfo(dev);
1076747b715Smrg        }
1086747b715Smrg    }
1096747b715Smrg
1106747b715Smrg    info = calloc(1, len);
1119ace9065Smrg    if (!info) {
1129ace9065Smrg        free(skip);
1136747b715Smrg        return BadAlloc;
1149ace9065Smrg    }
1156747b715Smrg
1166747b715Smrg    memset(&rep, 0, sizeof(xXIQueryDeviceReply));
1176747b715Smrg    rep.repType = X_Reply;
1186747b715Smrg    rep.RepType = X_XIQueryDevice;
1196747b715Smrg    rep.sequenceNumber = client->sequence;
1206747b715Smrg    rep.length = len/4;
1216747b715Smrg    rep.num_devices = 0;
1226747b715Smrg
1236747b715Smrg    ptr = info;
1246747b715Smrg    if (dev)
1256747b715Smrg    {
1266747b715Smrg        len = ListDeviceInfo(client, dev, (xXIDeviceInfo*)info);
1276747b715Smrg        if (client->swapped)
1286747b715Smrg            SwapDeviceInfo(dev, (xXIDeviceInfo*)info);
1296747b715Smrg        info += len;
1306747b715Smrg        rep.num_devices = 1;
1316747b715Smrg    } else
1326747b715Smrg    {
1336747b715Smrg        i = 0;
1346747b715Smrg        for (dev = inputInfo.devices; dev; dev = dev->next, i++)
1356747b715Smrg        {
1366747b715Smrg            if (!skip[i])
1376747b715Smrg            {
1386747b715Smrg                len = ListDeviceInfo(client, dev, (xXIDeviceInfo*)info);
1396747b715Smrg                if (client->swapped)
1406747b715Smrg                    SwapDeviceInfo(dev, (xXIDeviceInfo*)info);
1416747b715Smrg                info += len;
1426747b715Smrg                rep.num_devices++;
1436747b715Smrg            }
1446747b715Smrg        }
1456747b715Smrg
1466747b715Smrg        for (dev = inputInfo.off_devices; dev; dev = dev->next, i++)
1476747b715Smrg        {
1486747b715Smrg            if (!skip[i])
1496747b715Smrg            {
1506747b715Smrg                len = ListDeviceInfo(client, dev, (xXIDeviceInfo*)info);
1516747b715Smrg                if (client->swapped)
1526747b715Smrg                    SwapDeviceInfo(dev, (xXIDeviceInfo*)info);
1536747b715Smrg                info += len;
1546747b715Smrg                rep.num_devices++;
1556747b715Smrg            }
1566747b715Smrg        }
1576747b715Smrg    }
1586747b715Smrg
1599ace9065Smrg    len = rep.length * 4;
1606747b715Smrg    WriteReplyToClient(client, sizeof(xXIQueryDeviceReply), &rep);
1619ace9065Smrg    WriteToClient(client, len, ptr);
1626747b715Smrg    free(ptr);
1636747b715Smrg    free(skip);
1646747b715Smrg    return rc;
1656747b715Smrg}
1666747b715Smrg
1676747b715Smrgvoid
1686747b715SmrgSRepXIQueryDevice(ClientPtr client, int size, xXIQueryDeviceReply *rep)
1696747b715Smrg{
1706747b715Smrg    char n;
1716747b715Smrg
1726747b715Smrg    swaps(&rep->sequenceNumber, n);
1736747b715Smrg    swapl(&rep->length, n);
1746747b715Smrg    swaps(&rep->num_devices, n);
1756747b715Smrg
1766747b715Smrg    /* Device info is already swapped, see ProcXIQueryDevice */
1776747b715Smrg
1786747b715Smrg    WriteToClient(client, size, (char *)rep);
1796747b715Smrg}
1806747b715Smrg
1816747b715Smrg
1826747b715Smrg/**
1836747b715Smrg * @return Whether the device should be included in the returned list.
1846747b715Smrg */
1856747b715Smrgstatic Bool
1866747b715SmrgShouldSkipDevice(ClientPtr client, int deviceid, DeviceIntPtr dev)
1876747b715Smrg{
1886747b715Smrg    /* if all devices are not being queried, only master devices are */
1896747b715Smrg    if (deviceid == XIAllDevices || IsMaster(dev))
1906747b715Smrg    {
1916747b715Smrg        int rc = XaceHook(XACE_DEVICE_ACCESS, client, dev, DixGetAttrAccess);
1926747b715Smrg        if (rc == Success)
1936747b715Smrg            return FALSE;
1946747b715Smrg    }
1956747b715Smrg    return TRUE;
1966747b715Smrg}
1976747b715Smrg
1986747b715Smrg/**
1996747b715Smrg * @return The number of bytes needed to store this device's xXIDeviceInfo
2006747b715Smrg * (and its classes).
2016747b715Smrg */
2026747b715Smrgstatic int
2036747b715SmrgSizeDeviceInfo(DeviceIntPtr dev)
2046747b715Smrg{
2056747b715Smrg    int len = sizeof(xXIDeviceInfo);
2066747b715Smrg
2076747b715Smrg    /* 4-padded name */
2086747b715Smrg    len += pad_to_int32(strlen(dev->name));
2096747b715Smrg
2106747b715Smrg    return len + SizeDeviceClasses(dev);
2116747b715Smrg
2126747b715Smrg}
2136747b715Smrg
2146747b715Smrg/*
2156747b715Smrg * @return The number of bytes needed to store this device's classes.
2166747b715Smrg */
2176747b715Smrgint
2186747b715SmrgSizeDeviceClasses(DeviceIntPtr dev)
2196747b715Smrg{
2206747b715Smrg    int len = 0;
2216747b715Smrg
2226747b715Smrg    if (dev->button)
2236747b715Smrg    {
2246747b715Smrg        len += sizeof(xXIButtonInfo);
2256747b715Smrg        len += dev->button->numButtons * sizeof(Atom);
2266747b715Smrg        len += pad_to_int32(bits_to_bytes(dev->button->numButtons));
2276747b715Smrg    }
2286747b715Smrg
2296747b715Smrg    if (dev->key)
2306747b715Smrg    {
2316747b715Smrg        XkbDescPtr xkb = dev->key->xkbInfo->desc;
2326747b715Smrg        len += sizeof(xXIKeyInfo);
2336747b715Smrg        len += (xkb->max_key_code - xkb->min_key_code + 1) * sizeof(uint32_t);
2346747b715Smrg    }
2356747b715Smrg
2366747b715Smrg    if (dev->valuator)
2376747b715Smrg        len += sizeof(xXIValuatorInfo) * dev->valuator->numAxes;
2386747b715Smrg
2396747b715Smrg    return len;
2406747b715Smrg}
2416747b715Smrg
2426747b715Smrg
2436747b715Smrg/**
2446747b715Smrg * Write button information into info.
2456747b715Smrg * @return Number of bytes written into info.
2466747b715Smrg */
2476747b715Smrgint
2486747b715SmrgListButtonInfo(DeviceIntPtr dev, xXIButtonInfo* info, Bool reportState)
2496747b715Smrg{
2506747b715Smrg    unsigned char *bits;
2516747b715Smrg    int mask_len;
2526747b715Smrg    int i;
2536747b715Smrg
2546747b715Smrg    if (!dev || !dev->button)
2556747b715Smrg	return 0;
2566747b715Smrg
2576747b715Smrg    mask_len = bytes_to_int32(bits_to_bytes(dev->button->numButtons));
2586747b715Smrg
2596747b715Smrg    info->type = ButtonClass;
2606747b715Smrg    info->num_buttons = dev->button->numButtons;
2616747b715Smrg    info->length = bytes_to_int32(sizeof(xXIButtonInfo)) +
2626747b715Smrg                   info->num_buttons + mask_len;
2636747b715Smrg    info->sourceid = dev->button->sourceid;
2646747b715Smrg
2656747b715Smrg    bits = (unsigned char*)&info[1];
2666747b715Smrg    memset(bits, 0, mask_len * 4);
2676747b715Smrg
2686747b715Smrg    if (reportState)
2696747b715Smrg	for (i = 0; i < dev->button->numButtons; i++)
2706747b715Smrg	    if (BitIsOn(dev->button->down, i))
2716747b715Smrg		SetBit(bits, i);
2726747b715Smrg
2736747b715Smrg    bits += mask_len * 4;
2746747b715Smrg    memcpy(bits, dev->button->labels, dev->button->numButtons * sizeof(Atom));
2756747b715Smrg
2766747b715Smrg    return info->length * 4;
2776747b715Smrg}
2786747b715Smrg
2796747b715Smrgstatic void
2806747b715SmrgSwapButtonInfo(DeviceIntPtr dev, xXIButtonInfo* info)
2816747b715Smrg{
2826747b715Smrg    char n;
2836747b715Smrg    Atom *btn;
2846747b715Smrg    int i;
2856747b715Smrg    swaps(&info->type, n);
2866747b715Smrg    swaps(&info->length, n);
2876747b715Smrg    swaps(&info->sourceid, n);
2886747b715Smrg
2896747b715Smrg    for (i = 0, btn = (Atom*)&info[1]; i < info->num_buttons; i++, btn++)
2906747b715Smrg        swaps(btn, n);
2916747b715Smrg
2926747b715Smrg    swaps(&info->num_buttons, n);
2936747b715Smrg}
2946747b715Smrg
2956747b715Smrg/**
2966747b715Smrg * Write key information into info.
2976747b715Smrg * @return Number of bytes written into info.
2986747b715Smrg */
2996747b715Smrgint
3006747b715SmrgListKeyInfo(DeviceIntPtr dev, xXIKeyInfo* info)
3016747b715Smrg{
3026747b715Smrg    int i;
3036747b715Smrg    XkbDescPtr xkb = dev->key->xkbInfo->desc;
3046747b715Smrg    uint32_t *kc;
3056747b715Smrg
3066747b715Smrg    info->type = KeyClass;
3076747b715Smrg    info->num_keycodes = xkb->max_key_code - xkb->min_key_code + 1;
3086747b715Smrg    info->length = sizeof(xXIKeyInfo)/4 + info->num_keycodes;
3096747b715Smrg    info->sourceid = dev->key->sourceid;
3106747b715Smrg
3116747b715Smrg    kc = (uint32_t*)&info[1];
3126747b715Smrg    for (i = xkb->min_key_code; i <= xkb->max_key_code; i++, kc++)
3136747b715Smrg        *kc = i;
3146747b715Smrg
3156747b715Smrg    return info->length * 4;
3166747b715Smrg}
3176747b715Smrg
3186747b715Smrgstatic void
3196747b715SmrgSwapKeyInfo(DeviceIntPtr dev, xXIKeyInfo* info)
3206747b715Smrg{
3216747b715Smrg    char n;
3226747b715Smrg    uint32_t *key;
3236747b715Smrg    int i;
3246747b715Smrg    swaps(&info->type, n);
3256747b715Smrg    swaps(&info->length, n);
3266747b715Smrg    swaps(&info->sourceid, n);
3276747b715Smrg
3286747b715Smrg    for (i = 0, key = (uint32_t*)&info[1]; i < info->num_keycodes; i++, key++)
3296747b715Smrg        swapl(key, n);
3306747b715Smrg
3316747b715Smrg    swaps(&info->num_keycodes, n);
3326747b715Smrg}
3336747b715Smrg
3346747b715Smrg/**
3356747b715Smrg * List axis information for the given axis.
3366747b715Smrg *
3376747b715Smrg * @return The number of bytes written into info.
3386747b715Smrg */
3396747b715Smrgint
3406747b715SmrgListValuatorInfo(DeviceIntPtr dev, xXIValuatorInfo* info, int axisnumber,
3416747b715Smrg		 Bool reportState)
3426747b715Smrg{
3436747b715Smrg    ValuatorClassPtr v = dev->valuator;
3446747b715Smrg
3456747b715Smrg    info->type = ValuatorClass;
3466747b715Smrg    info->length = sizeof(xXIValuatorInfo)/4;
3476747b715Smrg    info->label = v->axes[axisnumber].label;
3486747b715Smrg    info->min.integral = v->axes[axisnumber].min_value;
3496747b715Smrg    info->min.frac = 0;
3506747b715Smrg    info->max.integral = v->axes[axisnumber].max_value;
3516747b715Smrg    info->max.frac = 0;
3526747b715Smrg    info->value.integral = (int)v->axisVal[axisnumber];
3536747b715Smrg    info->value.frac = (int)(v->axisVal[axisnumber] * (1 << 16) * (1 << 16));
3546747b715Smrg    info->resolution = v->axes[axisnumber].resolution;
3556747b715Smrg    info->number = axisnumber;
3569ace9065Smrg    info->mode = valuator_get_mode(dev, axisnumber);
3576747b715Smrg    info->sourceid = v->sourceid;
3586747b715Smrg
3596747b715Smrg    if (!reportState)
3606747b715Smrg	info->value = info->min;
3616747b715Smrg
3626747b715Smrg    return info->length * 4;
3636747b715Smrg}
3646747b715Smrg
3656747b715Smrgstatic void
3666747b715SmrgSwapValuatorInfo(DeviceIntPtr dev, xXIValuatorInfo* info)
3676747b715Smrg{
3686747b715Smrg    char n;
3696747b715Smrg    swaps(&info->type, n);
3706747b715Smrg    swaps(&info->length, n);
3716747b715Smrg    swapl(&info->label, n);
3726747b715Smrg    swapl(&info->min.integral, n);
3736747b715Smrg    swapl(&info->min.frac, n);
3746747b715Smrg    swapl(&info->max.integral, n);
3756747b715Smrg    swapl(&info->max.frac, n);
3766747b715Smrg    swaps(&info->number, n);
3776747b715Smrg    swaps(&info->sourceid, n);
3786747b715Smrg}
3796747b715Smrg
3806747b715Smrgint GetDeviceUse(DeviceIntPtr dev, uint16_t *attachment)
3816747b715Smrg{
3826747b715Smrg    DeviceIntPtr master = dev->u.master;
3836747b715Smrg    int use;
3846747b715Smrg
3856747b715Smrg    if (IsMaster(dev))
3866747b715Smrg    {
3876747b715Smrg        DeviceIntPtr paired = GetPairedDevice(dev);
3886747b715Smrg        use = IsPointerDevice(dev) ? XIMasterPointer : XIMasterKeyboard;
3896747b715Smrg        *attachment = (paired ? paired->id : 0);
3906747b715Smrg    } else if (master)
3916747b715Smrg    {
3926747b715Smrg        use = IsPointerDevice(master) ? XISlavePointer : XISlaveKeyboard;
3936747b715Smrg        *attachment = master->id;
3946747b715Smrg    } else
3956747b715Smrg        use = XIFloatingSlave;
3966747b715Smrg
3976747b715Smrg    return use;
3986747b715Smrg}
3996747b715Smrg
4006747b715Smrg/**
4016747b715Smrg * Write the info for device dev into the buffer pointed to by info.
4026747b715Smrg *
4036747b715Smrg * @return The number of bytes used.
4046747b715Smrg */
4056747b715Smrgstatic int
4066747b715SmrgListDeviceInfo(ClientPtr client, DeviceIntPtr dev, xXIDeviceInfo* info)
4076747b715Smrg{
4086747b715Smrg    char *any = (char*)&info[1];
4096747b715Smrg    int len = 0, total_len = 0;
4106747b715Smrg
4116747b715Smrg    info->deviceid = dev->id;
4126747b715Smrg    info->use = GetDeviceUse(dev, &info->attachment);
4136747b715Smrg    info->num_classes = 0;
4146747b715Smrg    info->name_len = strlen(dev->name);
4156747b715Smrg    info->enabled = dev->enabled;
4166747b715Smrg    total_len = sizeof(xXIDeviceInfo);
4176747b715Smrg
4186747b715Smrg    len = pad_to_int32(info->name_len);
4196747b715Smrg    memset(any, 0, len);
4206747b715Smrg    strncpy(any, dev->name, info->name_len);
4216747b715Smrg    any += len;
4226747b715Smrg    total_len += len;
4236747b715Smrg
4246747b715Smrg    total_len += ListDeviceClasses(client, dev, any, &info->num_classes);
4256747b715Smrg    return total_len;
4266747b715Smrg}
4276747b715Smrg
4286747b715Smrg/**
4296747b715Smrg * Write the class info of the device into the memory pointed to by any, set
4306747b715Smrg * nclasses to the number of classes in total and return the number of bytes
4316747b715Smrg * written.
4326747b715Smrg */
4336747b715Smrgint
4346747b715SmrgListDeviceClasses(ClientPtr client, DeviceIntPtr dev,
4356747b715Smrg		  char *any, uint16_t *nclasses)
4366747b715Smrg{
4376747b715Smrg    int total_len = 0;
4386747b715Smrg    int len;
4396747b715Smrg    int i;
4406747b715Smrg    int rc;
4416747b715Smrg
4426747b715Smrg    /* Check if the current device state should be suppressed */
4436747b715Smrg    rc = XaceHook(XACE_DEVICE_ACCESS, client, dev, DixReadAccess);
4446747b715Smrg
4456747b715Smrg    if (dev->button)
4466747b715Smrg    {
4476747b715Smrg        (*nclasses)++;
4486747b715Smrg        len = ListButtonInfo(dev, (xXIButtonInfo*)any, rc == Success);
4496747b715Smrg        any += len;
4506747b715Smrg        total_len += len;
4516747b715Smrg    }
4526747b715Smrg
4536747b715Smrg    if (dev->key)
4546747b715Smrg    {
4556747b715Smrg        (*nclasses)++;
4566747b715Smrg        len = ListKeyInfo(dev, (xXIKeyInfo*)any);
4576747b715Smrg        any += len;
4586747b715Smrg        total_len += len;
4596747b715Smrg    }
4606747b715Smrg
4616747b715Smrg    for (i = 0; dev->valuator && i < dev->valuator->numAxes; i++)
4626747b715Smrg    {
4636747b715Smrg        (*nclasses)++;
4646747b715Smrg        len = ListValuatorInfo(dev, (xXIValuatorInfo*)any, i, rc == Success);
4656747b715Smrg        any += len;
4666747b715Smrg        total_len += len;
4676747b715Smrg    }
4686747b715Smrg
4696747b715Smrg    return total_len;
4706747b715Smrg}
4716747b715Smrg
4726747b715Smrgstatic void
4736747b715SmrgSwapDeviceInfo(DeviceIntPtr dev, xXIDeviceInfo* info)
4746747b715Smrg{
4756747b715Smrg    char n;
4766747b715Smrg    char *any = (char*)&info[1];
4776747b715Smrg    int i;
4786747b715Smrg
4796747b715Smrg    /* Skip over name */
4806747b715Smrg    any += pad_to_int32(info->name_len);
4816747b715Smrg
4826747b715Smrg    for (i = 0; i < info->num_classes; i++)
4836747b715Smrg    {
4846747b715Smrg        int len = ((xXIAnyInfo*)any)->length;
4856747b715Smrg        switch(((xXIAnyInfo*)any)->type)
4866747b715Smrg        {
4876747b715Smrg            case XIButtonClass:
4886747b715Smrg                SwapButtonInfo(dev, (xXIButtonInfo*)any);
4896747b715Smrg                break;
4906747b715Smrg            case XIKeyClass:
4916747b715Smrg                SwapKeyInfo(dev, (xXIKeyInfo*)any);
4926747b715Smrg                break;
4936747b715Smrg            case XIValuatorClass:
4946747b715Smrg                SwapValuatorInfo(dev, (xXIValuatorInfo*)any);
4956747b715Smrg                break;
4966747b715Smrg        }
4976747b715Smrg
4986747b715Smrg        any += len * 4;
4996747b715Smrg    }
5006747b715Smrg
5016747b715Smrg    swaps(&info->deviceid, n);
5026747b715Smrg    swaps(&info->use, n);
5036747b715Smrg    swaps(&info->attachment, n);
5046747b715Smrg    swaps(&info->num_classes, n);
5056747b715Smrg    swaps(&info->name_len, n);
5066747b715Smrg
5076747b715Smrg}
508