chgfctl.c revision 05b261ec
105b261ecSmrg/************************************************************
205b261ecSmrg
305b261ecSmrgCopyright 1989, 1998  The Open Group
405b261ecSmrg
505b261ecSmrgPermission to use, copy, modify, distribute, and sell this software and its
605b261ecSmrgdocumentation for any purpose is hereby granted without fee, provided that
705b261ecSmrgthe above copyright notice appear in all copies and that both that
805b261ecSmrgcopyright notice and this permission notice appear in supporting
905b261ecSmrgdocumentation.
1005b261ecSmrg
1105b261ecSmrgThe above copyright notice and this permission notice shall be included in
1205b261ecSmrgall copies or substantial portions of the Software.
1305b261ecSmrg
1405b261ecSmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1505b261ecSmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1605b261ecSmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
1705b261ecSmrgOPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
1805b261ecSmrgAN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
1905b261ecSmrgCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
2005b261ecSmrg
2105b261ecSmrgExcept as contained in this notice, the name of The Open Group shall not be
2205b261ecSmrgused in advertising or otherwise to promote the sale, use or other dealings
2305b261ecSmrgin this Software without prior written authorization from The Open Group.
2405b261ecSmrg
2505b261ecSmrgCopyright 1989 by Hewlett-Packard Company, Palo Alto, California.
2605b261ecSmrg
2705b261ecSmrg			All Rights Reserved
2805b261ecSmrg
2905b261ecSmrgPermission to use, copy, modify, and distribute this software and its
3005b261ecSmrgdocumentation for any purpose and without fee is hereby granted,
3105b261ecSmrgprovided that the above copyright notice appear in all copies and that
3205b261ecSmrgboth that copyright notice and this permission notice appear in
3305b261ecSmrgsupporting documentation, and that the name of Hewlett-Packard not be
3405b261ecSmrgused in advertising or publicity pertaining to distribution of the
3505b261ecSmrgsoftware without specific, written prior permission.
3605b261ecSmrg
3705b261ecSmrgHEWLETT-PACKARD DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
3805b261ecSmrgALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
3905b261ecSmrgHEWLETT-PACKARD BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
4005b261ecSmrgANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
4105b261ecSmrgWHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
4205b261ecSmrgARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
4305b261ecSmrgSOFTWARE.
4405b261ecSmrg
4505b261ecSmrg********************************************************/
4605b261ecSmrg
4705b261ecSmrg/********************************************************************
4805b261ecSmrg *
4905b261ecSmrg *  Change feedback control attributes for an extension device.
5005b261ecSmrg *
5105b261ecSmrg */
5205b261ecSmrg
5305b261ecSmrg#define	 NEED_EVENTS	/* for inputstr.h    */
5405b261ecSmrg#define	 NEED_REPLIES
5505b261ecSmrg#ifdef HAVE_DIX_CONFIG_H
5605b261ecSmrg#include <dix-config.h>
5705b261ecSmrg#endif
5805b261ecSmrg
5905b261ecSmrg#include <X11/X.h>	/* for inputstr.h    */
6005b261ecSmrg#include <X11/Xproto.h>	/* Request macro     */
6105b261ecSmrg#include "inputstr.h"	/* DeviceIntPtr      */
6205b261ecSmrg#include <X11/extensions/XI.h>
6305b261ecSmrg#include <X11/extensions/XIproto.h>	/* control constants */
6405b261ecSmrg
6505b261ecSmrg#include "extnsionst.h"
6605b261ecSmrg#include "extinit.h"	/* LookupDeviceIntRec */
6705b261ecSmrg#include "exglobals.h"
6805b261ecSmrg
6905b261ecSmrg#include "chgfctl.h"
7005b261ecSmrg
7105b261ecSmrg#define DO_ALL    (-1)
7205b261ecSmrg
7305b261ecSmrg/***********************************************************************
7405b261ecSmrg *
7505b261ecSmrg * This procedure changes the control attributes for an extension device,
7605b261ecSmrg * for clients on machines with a different byte ordering than the server.
7705b261ecSmrg *
7805b261ecSmrg */
7905b261ecSmrg
8005b261ecSmrgint
8105b261ecSmrgSProcXChangeFeedbackControl(ClientPtr client)
8205b261ecSmrg{
8305b261ecSmrg    char n;
8405b261ecSmrg
8505b261ecSmrg    REQUEST(xChangeFeedbackControlReq);
8605b261ecSmrg    swaps(&stuff->length, n);
8705b261ecSmrg    REQUEST_AT_LEAST_SIZE(xChangeFeedbackControlReq);
8805b261ecSmrg    swapl(&stuff->mask, n);
8905b261ecSmrg    return (ProcXChangeFeedbackControl(client));
9005b261ecSmrg}
9105b261ecSmrg
9205b261ecSmrg/******************************************************************************
9305b261ecSmrg *
9405b261ecSmrg * This procedure changes KbdFeedbackClass data.
9505b261ecSmrg *
9605b261ecSmrg */
9705b261ecSmrg
9805b261ecSmrgstatic int
9905b261ecSmrgChangeKbdFeedback(ClientPtr client, DeviceIntPtr dev, long unsigned int mask,
10005b261ecSmrg		  KbdFeedbackPtr k, xKbdFeedbackCtl * f)
10105b261ecSmrg{
10205b261ecSmrg    char n;
10305b261ecSmrg    KeybdCtrl kctrl;
10405b261ecSmrg    int t;
10505b261ecSmrg    int key = DO_ALL;
10605b261ecSmrg
10705b261ecSmrg    if (client->swapped) {
10805b261ecSmrg	swaps(&f->length, n);
10905b261ecSmrg	swaps(&f->pitch, n);
11005b261ecSmrg	swaps(&f->duration, n);
11105b261ecSmrg	swapl(&f->led_mask, n);
11205b261ecSmrg	swapl(&f->led_values, n);
11305b261ecSmrg    }
11405b261ecSmrg
11505b261ecSmrg    kctrl = k->ctrl;
11605b261ecSmrg    if (mask & DvKeyClickPercent) {
11705b261ecSmrg	t = f->click;
11805b261ecSmrg	if (t == -1)
11905b261ecSmrg	    t = defaultKeyboardControl.click;
12005b261ecSmrg	else if (t < 0 || t > 100) {
12105b261ecSmrg	    client->errorValue = t;
12205b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
12305b261ecSmrg			      BadValue);
12405b261ecSmrg	    return Success;
12505b261ecSmrg	}
12605b261ecSmrg	kctrl.click = t;
12705b261ecSmrg    }
12805b261ecSmrg
12905b261ecSmrg    if (mask & DvPercent) {
13005b261ecSmrg	t = f->percent;
13105b261ecSmrg	if (t == -1)
13205b261ecSmrg	    t = defaultKeyboardControl.bell;
13305b261ecSmrg	else if (t < 0 || t > 100) {
13405b261ecSmrg	    client->errorValue = t;
13505b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
13605b261ecSmrg			      BadValue);
13705b261ecSmrg	    return Success;
13805b261ecSmrg	}
13905b261ecSmrg	kctrl.bell = t;
14005b261ecSmrg    }
14105b261ecSmrg
14205b261ecSmrg    if (mask & DvPitch) {
14305b261ecSmrg	t = f->pitch;
14405b261ecSmrg	if (t == -1)
14505b261ecSmrg	    t = defaultKeyboardControl.bell_pitch;
14605b261ecSmrg	else if (t < 0) {
14705b261ecSmrg	    client->errorValue = t;
14805b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
14905b261ecSmrg			      BadValue);
15005b261ecSmrg	    return Success;
15105b261ecSmrg	}
15205b261ecSmrg	kctrl.bell_pitch = t;
15305b261ecSmrg    }
15405b261ecSmrg
15505b261ecSmrg    if (mask & DvDuration) {
15605b261ecSmrg	t = f->duration;
15705b261ecSmrg	if (t == -1)
15805b261ecSmrg	    t = defaultKeyboardControl.bell_duration;
15905b261ecSmrg	else if (t < 0) {
16005b261ecSmrg	    client->errorValue = t;
16105b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
16205b261ecSmrg			      BadValue);
16305b261ecSmrg	    return Success;
16405b261ecSmrg	}
16505b261ecSmrg	kctrl.bell_duration = t;
16605b261ecSmrg    }
16705b261ecSmrg
16805b261ecSmrg    if (mask & DvLed) {
16905b261ecSmrg	kctrl.leds &= ~(f->led_mask);
17005b261ecSmrg	kctrl.leds |= (f->led_mask & f->led_values);
17105b261ecSmrg    }
17205b261ecSmrg
17305b261ecSmrg    if (mask & DvKey) {
17405b261ecSmrg	key = (KeyCode) f->key;
17505b261ecSmrg	if (key < 8 || key > 255) {
17605b261ecSmrg	    client->errorValue = key;
17705b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
17805b261ecSmrg			      BadValue);
17905b261ecSmrg	    return Success;
18005b261ecSmrg	}
18105b261ecSmrg	if (!(mask & DvAutoRepeatMode)) {
18205b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
18305b261ecSmrg			      BadMatch);
18405b261ecSmrg	    return Success;
18505b261ecSmrg	}
18605b261ecSmrg    }
18705b261ecSmrg
18805b261ecSmrg    if (mask & DvAutoRepeatMode) {
18905b261ecSmrg	int inx = (key >> 3);
19005b261ecSmrg	int kmask = (1 << (key & 7));
19105b261ecSmrg
19205b261ecSmrg	t = (CARD8) f->auto_repeat_mode;
19305b261ecSmrg	if (t == AutoRepeatModeOff) {
19405b261ecSmrg	    if (key == DO_ALL)
19505b261ecSmrg		kctrl.autoRepeat = FALSE;
19605b261ecSmrg	    else
19705b261ecSmrg		kctrl.autoRepeats[inx] &= ~kmask;
19805b261ecSmrg	} else if (t == AutoRepeatModeOn) {
19905b261ecSmrg	    if (key == DO_ALL)
20005b261ecSmrg		kctrl.autoRepeat = TRUE;
20105b261ecSmrg	    else
20205b261ecSmrg		kctrl.autoRepeats[inx] |= kmask;
20305b261ecSmrg	} else if (t == AutoRepeatModeDefault) {
20405b261ecSmrg	    if (key == DO_ALL)
20505b261ecSmrg		kctrl.autoRepeat = defaultKeyboardControl.autoRepeat;
20605b261ecSmrg	    else
20705b261ecSmrg		kctrl.autoRepeats[inx] &= ~kmask;
20805b261ecSmrg	    kctrl.autoRepeats[inx] =
20905b261ecSmrg		(kctrl.autoRepeats[inx] & ~kmask) |
21005b261ecSmrg		(defaultKeyboardControl.autoRepeats[inx] & kmask);
21105b261ecSmrg	} else {
21205b261ecSmrg	    client->errorValue = t;
21305b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
21405b261ecSmrg			      BadValue);
21505b261ecSmrg	    return Success;
21605b261ecSmrg	}
21705b261ecSmrg    }
21805b261ecSmrg
21905b261ecSmrg    k->ctrl = kctrl;
22005b261ecSmrg    (*k->CtrlProc) (dev, &k->ctrl);
22105b261ecSmrg    return Success;
22205b261ecSmrg}
22305b261ecSmrg
22405b261ecSmrg/******************************************************************************
22505b261ecSmrg *
22605b261ecSmrg * This procedure changes PtrFeedbackClass data.
22705b261ecSmrg *
22805b261ecSmrg */
22905b261ecSmrg
23005b261ecSmrgstatic int
23105b261ecSmrgChangePtrFeedback(ClientPtr client, DeviceIntPtr dev, long unsigned int mask,
23205b261ecSmrg		  PtrFeedbackPtr p, xPtrFeedbackCtl * f)
23305b261ecSmrg{
23405b261ecSmrg    char n;
23505b261ecSmrg    PtrCtrl pctrl;	/* might get BadValue part way through */
23605b261ecSmrg
23705b261ecSmrg    if (client->swapped) {
23805b261ecSmrg	swaps(&f->length, n);
23905b261ecSmrg	swaps(&f->num, n);
24005b261ecSmrg	swaps(&f->denom, n);
24105b261ecSmrg	swaps(&f->thresh, n);
24205b261ecSmrg    }
24305b261ecSmrg
24405b261ecSmrg    pctrl = p->ctrl;
24505b261ecSmrg    if (mask & DvAccelNum) {
24605b261ecSmrg	int accelNum;
24705b261ecSmrg
24805b261ecSmrg	accelNum = f->num;
24905b261ecSmrg	if (accelNum == -1)
25005b261ecSmrg	    pctrl.num = defaultPointerControl.num;
25105b261ecSmrg	else if (accelNum < 0) {
25205b261ecSmrg	    client->errorValue = accelNum;
25305b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
25405b261ecSmrg			      BadValue);
25505b261ecSmrg	    return Success;
25605b261ecSmrg	} else
25705b261ecSmrg	    pctrl.num = accelNum;
25805b261ecSmrg    }
25905b261ecSmrg
26005b261ecSmrg    if (mask & DvAccelDenom) {
26105b261ecSmrg	int accelDenom;
26205b261ecSmrg
26305b261ecSmrg	accelDenom = f->denom;
26405b261ecSmrg	if (accelDenom == -1)
26505b261ecSmrg	    pctrl.den = defaultPointerControl.den;
26605b261ecSmrg	else if (accelDenom <= 0) {
26705b261ecSmrg	    client->errorValue = accelDenom;
26805b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
26905b261ecSmrg			      BadValue);
27005b261ecSmrg	    return Success;
27105b261ecSmrg	} else
27205b261ecSmrg	    pctrl.den = accelDenom;
27305b261ecSmrg    }
27405b261ecSmrg
27505b261ecSmrg    if (mask & DvThreshold) {
27605b261ecSmrg	int threshold;
27705b261ecSmrg
27805b261ecSmrg	threshold = f->thresh;
27905b261ecSmrg	if (threshold == -1)
28005b261ecSmrg	    pctrl.threshold = defaultPointerControl.threshold;
28105b261ecSmrg	else if (threshold < 0) {
28205b261ecSmrg	    client->errorValue = threshold;
28305b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
28405b261ecSmrg			      BadValue);
28505b261ecSmrg	    return Success;
28605b261ecSmrg	} else
28705b261ecSmrg	    pctrl.threshold = threshold;
28805b261ecSmrg    }
28905b261ecSmrg
29005b261ecSmrg    p->ctrl = pctrl;
29105b261ecSmrg    (*p->CtrlProc) (dev, &p->ctrl);
29205b261ecSmrg    return Success;
29305b261ecSmrg}
29405b261ecSmrg
29505b261ecSmrg/******************************************************************************
29605b261ecSmrg *
29705b261ecSmrg * This procedure changes IntegerFeedbackClass data.
29805b261ecSmrg *
29905b261ecSmrg */
30005b261ecSmrg
30105b261ecSmrgstatic int
30205b261ecSmrgChangeIntegerFeedback(ClientPtr client, DeviceIntPtr dev,
30305b261ecSmrg		      long unsigned int mask, IntegerFeedbackPtr i,
30405b261ecSmrg		      xIntegerFeedbackCtl * f)
30505b261ecSmrg{
30605b261ecSmrg    char n;
30705b261ecSmrg
30805b261ecSmrg    if (client->swapped) {
30905b261ecSmrg	swaps(&f->length, n);
31005b261ecSmrg	swapl(&f->int_to_display, n);
31105b261ecSmrg    }
31205b261ecSmrg
31305b261ecSmrg    i->ctrl.integer_displayed = f->int_to_display;
31405b261ecSmrg    (*i->CtrlProc) (dev, &i->ctrl);
31505b261ecSmrg    return Success;
31605b261ecSmrg}
31705b261ecSmrg
31805b261ecSmrg/******************************************************************************
31905b261ecSmrg *
32005b261ecSmrg * This procedure changes StringFeedbackClass data.
32105b261ecSmrg *
32205b261ecSmrg */
32305b261ecSmrg
32405b261ecSmrgstatic int
32505b261ecSmrgChangeStringFeedback(ClientPtr client, DeviceIntPtr dev,
32605b261ecSmrg		     long unsigned int mask, StringFeedbackPtr s,
32705b261ecSmrg		     xStringFeedbackCtl * f)
32805b261ecSmrg{
32905b261ecSmrg    char n;
33005b261ecSmrg    int i, j;
33105b261ecSmrg    KeySym *syms, *sup_syms;
33205b261ecSmrg
33305b261ecSmrg    syms = (KeySym *) (f + 1);
33405b261ecSmrg    if (client->swapped) {
33505b261ecSmrg	swaps(&f->length, n);	/* swapped num_keysyms in calling proc */
33605b261ecSmrg	SwapLongs((CARD32 *) syms, f->num_keysyms);
33705b261ecSmrg    }
33805b261ecSmrg
33905b261ecSmrg    if (f->num_keysyms > s->ctrl.max_symbols) {
34005b261ecSmrg	SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
34105b261ecSmrg			  BadValue);
34205b261ecSmrg	return Success;
34305b261ecSmrg    }
34405b261ecSmrg    sup_syms = s->ctrl.symbols_supported;
34505b261ecSmrg    for (i = 0; i < f->num_keysyms; i++) {
34605b261ecSmrg	for (j = 0; j < s->ctrl.num_symbols_supported; j++)
34705b261ecSmrg	    if (*(syms + i) == *(sup_syms + j))
34805b261ecSmrg		break;
34905b261ecSmrg	if (j == s->ctrl.num_symbols_supported) {
35005b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
35105b261ecSmrg			      BadMatch);
35205b261ecSmrg	    return Success;
35305b261ecSmrg	}
35405b261ecSmrg    }
35505b261ecSmrg
35605b261ecSmrg    s->ctrl.num_symbols_displayed = f->num_keysyms;
35705b261ecSmrg    for (i = 0; i < f->num_keysyms; i++)
35805b261ecSmrg	*(s->ctrl.symbols_displayed + i) = *(syms + i);
35905b261ecSmrg    (*s->CtrlProc) (dev, &s->ctrl);
36005b261ecSmrg    return Success;
36105b261ecSmrg}
36205b261ecSmrg
36305b261ecSmrg/******************************************************************************
36405b261ecSmrg *
36505b261ecSmrg * This procedure changes BellFeedbackClass data.
36605b261ecSmrg *
36705b261ecSmrg */
36805b261ecSmrg
36905b261ecSmrgstatic int
37005b261ecSmrgChangeBellFeedback(ClientPtr client, DeviceIntPtr dev,
37105b261ecSmrg		   long unsigned int mask, BellFeedbackPtr b,
37205b261ecSmrg		   xBellFeedbackCtl * f)
37305b261ecSmrg{
37405b261ecSmrg    char n;
37505b261ecSmrg    int t;
37605b261ecSmrg    BellCtrl bctrl;	/* might get BadValue part way through */
37705b261ecSmrg
37805b261ecSmrg    if (client->swapped) {
37905b261ecSmrg	swaps(&f->length, n);
38005b261ecSmrg	swaps(&f->pitch, n);
38105b261ecSmrg	swaps(&f->duration, n);
38205b261ecSmrg    }
38305b261ecSmrg
38405b261ecSmrg    bctrl = b->ctrl;
38505b261ecSmrg    if (mask & DvPercent) {
38605b261ecSmrg	t = f->percent;
38705b261ecSmrg	if (t == -1)
38805b261ecSmrg	    t = defaultKeyboardControl.bell;
38905b261ecSmrg	else if (t < 0 || t > 100) {
39005b261ecSmrg	    client->errorValue = t;
39105b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
39205b261ecSmrg			      BadValue);
39305b261ecSmrg	    return Success;
39405b261ecSmrg	}
39505b261ecSmrg	bctrl.percent = t;
39605b261ecSmrg    }
39705b261ecSmrg
39805b261ecSmrg    if (mask & DvPitch) {
39905b261ecSmrg	t = f->pitch;
40005b261ecSmrg	if (t == -1)
40105b261ecSmrg	    t = defaultKeyboardControl.bell_pitch;
40205b261ecSmrg	else if (t < 0) {
40305b261ecSmrg	    client->errorValue = t;
40405b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
40505b261ecSmrg			      BadValue);
40605b261ecSmrg	    return Success;
40705b261ecSmrg	}
40805b261ecSmrg	bctrl.pitch = t;
40905b261ecSmrg    }
41005b261ecSmrg
41105b261ecSmrg    if (mask & DvDuration) {
41205b261ecSmrg	t = f->duration;
41305b261ecSmrg	if (t == -1)
41405b261ecSmrg	    t = defaultKeyboardControl.bell_duration;
41505b261ecSmrg	else if (t < 0) {
41605b261ecSmrg	    client->errorValue = t;
41705b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
41805b261ecSmrg			      BadValue);
41905b261ecSmrg	    return Success;
42005b261ecSmrg	}
42105b261ecSmrg	bctrl.duration = t;
42205b261ecSmrg    }
42305b261ecSmrg    b->ctrl = bctrl;
42405b261ecSmrg    (*b->CtrlProc) (dev, &b->ctrl);
42505b261ecSmrg    return Success;
42605b261ecSmrg}
42705b261ecSmrg
42805b261ecSmrg/******************************************************************************
42905b261ecSmrg *
43005b261ecSmrg * This procedure changes LedFeedbackClass data.
43105b261ecSmrg *
43205b261ecSmrg */
43305b261ecSmrg
43405b261ecSmrgstatic int
43505b261ecSmrgChangeLedFeedback(ClientPtr client, DeviceIntPtr dev, long unsigned int mask,
43605b261ecSmrg		  LedFeedbackPtr l, xLedFeedbackCtl * f)
43705b261ecSmrg{
43805b261ecSmrg    char n;
43905b261ecSmrg    LedCtrl lctrl;	/* might get BadValue part way through */
44005b261ecSmrg
44105b261ecSmrg    if (client->swapped) {
44205b261ecSmrg	swaps(&f->length, n);
44305b261ecSmrg	swapl(&f->led_values, n);
44405b261ecSmrg	swapl(&f->led_mask, n);
44505b261ecSmrg    }
44605b261ecSmrg
44705b261ecSmrg    f->led_mask &= l->ctrl.led_mask;	/* set only supported leds */
44805b261ecSmrg    f->led_values &= l->ctrl.led_mask;	/* set only supported leds */
44905b261ecSmrg    if (mask & DvLed) {
45005b261ecSmrg	lctrl.led_mask = f->led_mask;
45105b261ecSmrg	lctrl.led_values = f->led_values;
45205b261ecSmrg	(*l->CtrlProc) (dev, &lctrl);
45305b261ecSmrg	l->ctrl.led_values &= ~(f->led_mask);	/* zero changed leds */
45405b261ecSmrg	l->ctrl.led_values |= (f->led_mask & f->led_values);	/* OR in set leds */
45505b261ecSmrg    }
45605b261ecSmrg
45705b261ecSmrg    return Success;
45805b261ecSmrg}
45905b261ecSmrg
46005b261ecSmrg/***********************************************************************
46105b261ecSmrg *
46205b261ecSmrg * Change the control attributes.
46305b261ecSmrg *
46405b261ecSmrg */
46505b261ecSmrg
46605b261ecSmrgint
46705b261ecSmrgProcXChangeFeedbackControl(ClientPtr client)
46805b261ecSmrg{
46905b261ecSmrg    unsigned len;
47005b261ecSmrg    DeviceIntPtr dev;
47105b261ecSmrg    KbdFeedbackPtr k;
47205b261ecSmrg    PtrFeedbackPtr p;
47305b261ecSmrg    IntegerFeedbackPtr i;
47405b261ecSmrg    StringFeedbackPtr s;
47505b261ecSmrg    BellFeedbackPtr b;
47605b261ecSmrg    LedFeedbackPtr l;
47705b261ecSmrg
47805b261ecSmrg    REQUEST(xChangeFeedbackControlReq);
47905b261ecSmrg    REQUEST_AT_LEAST_SIZE(xChangeFeedbackControlReq);
48005b261ecSmrg
48105b261ecSmrg    len = stuff->length - (sizeof(xChangeFeedbackControlReq) >> 2);
48205b261ecSmrg    dev = LookupDeviceIntRec(stuff->deviceid);
48305b261ecSmrg    if (dev == NULL) {
48405b261ecSmrg	SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0,
48505b261ecSmrg			  BadDevice);
48605b261ecSmrg	return Success;
48705b261ecSmrg    }
48805b261ecSmrg
48905b261ecSmrg    switch (stuff->feedbackid) {
49005b261ecSmrg    case KbdFeedbackClass:
49105b261ecSmrg	if (len != (sizeof(xKbdFeedbackCtl) >> 2)) {
49205b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl,
49305b261ecSmrg			      0, BadLength);
49405b261ecSmrg	    return Success;
49505b261ecSmrg	}
49605b261ecSmrg	for (k = dev->kbdfeed; k; k = k->next)
49705b261ecSmrg	    if (k->ctrl.id == ((xKbdFeedbackCtl *) & stuff[1])->id) {
49805b261ecSmrg		ChangeKbdFeedback(client, dev, stuff->mask, k,
49905b261ecSmrg				  (xKbdFeedbackCtl *) & stuff[1]);
50005b261ecSmrg		return Success;
50105b261ecSmrg	    }
50205b261ecSmrg	break;
50305b261ecSmrg    case PtrFeedbackClass:
50405b261ecSmrg	if (len != (sizeof(xPtrFeedbackCtl) >> 2)) {
50505b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl,
50605b261ecSmrg			      0, BadLength);
50705b261ecSmrg	    return Success;
50805b261ecSmrg	}
50905b261ecSmrg	for (p = dev->ptrfeed; p; p = p->next)
51005b261ecSmrg	    if (p->ctrl.id == ((xPtrFeedbackCtl *) & stuff[1])->id) {
51105b261ecSmrg		ChangePtrFeedback(client, dev, stuff->mask, p,
51205b261ecSmrg				  (xPtrFeedbackCtl *) & stuff[1]);
51305b261ecSmrg		return Success;
51405b261ecSmrg	    }
51505b261ecSmrg	break;
51605b261ecSmrg    case StringFeedbackClass:
51705b261ecSmrg    {
51805b261ecSmrg	char n;
51905b261ecSmrg	xStringFeedbackCtl *f = ((xStringFeedbackCtl *) & stuff[1]);
52005b261ecSmrg
52105b261ecSmrg	if (client->swapped) {
52205b261ecSmrg	    swaps(&f->num_keysyms, n);
52305b261ecSmrg	}
52405b261ecSmrg	if (len != ((sizeof(xStringFeedbackCtl) >> 2) + f->num_keysyms)) {
52505b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl,
52605b261ecSmrg			      0, BadLength);
52705b261ecSmrg	    return Success;
52805b261ecSmrg	}
52905b261ecSmrg	for (s = dev->stringfeed; s; s = s->next)
53005b261ecSmrg	    if (s->ctrl.id == ((xStringFeedbackCtl *) & stuff[1])->id) {
53105b261ecSmrg		ChangeStringFeedback(client, dev, stuff->mask, s,
53205b261ecSmrg				     (xStringFeedbackCtl *) & stuff[1]);
53305b261ecSmrg		return Success;
53405b261ecSmrg	    }
53505b261ecSmrg	break;
53605b261ecSmrg    }
53705b261ecSmrg    case IntegerFeedbackClass:
53805b261ecSmrg	if (len != (sizeof(xIntegerFeedbackCtl) >> 2)) {
53905b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl,
54005b261ecSmrg			      0, BadLength);
54105b261ecSmrg	    return Success;
54205b261ecSmrg	}
54305b261ecSmrg	for (i = dev->intfeed; i; i = i->next)
54405b261ecSmrg	    if (i->ctrl.id == ((xIntegerFeedbackCtl *) & stuff[1])->id) {
54505b261ecSmrg		ChangeIntegerFeedback(client, dev, stuff->mask, i,
54605b261ecSmrg				      (xIntegerFeedbackCtl *) & stuff[1]);
54705b261ecSmrg		return Success;
54805b261ecSmrg	    }
54905b261ecSmrg	break;
55005b261ecSmrg    case LedFeedbackClass:
55105b261ecSmrg	if (len != (sizeof(xLedFeedbackCtl) >> 2)) {
55205b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl,
55305b261ecSmrg			      0, BadLength);
55405b261ecSmrg	    return Success;
55505b261ecSmrg	}
55605b261ecSmrg	for (l = dev->leds; l; l = l->next)
55705b261ecSmrg	    if (l->ctrl.id == ((xLedFeedbackCtl *) & stuff[1])->id) {
55805b261ecSmrg		ChangeLedFeedback(client, dev, stuff->mask, l,
55905b261ecSmrg				  (xLedFeedbackCtl *) & stuff[1]);
56005b261ecSmrg		return Success;
56105b261ecSmrg	    }
56205b261ecSmrg	break;
56305b261ecSmrg    case BellFeedbackClass:
56405b261ecSmrg	if (len != (sizeof(xBellFeedbackCtl) >> 2)) {
56505b261ecSmrg	    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl,
56605b261ecSmrg			      0, BadLength);
56705b261ecSmrg	    return Success;
56805b261ecSmrg	}
56905b261ecSmrg	for (b = dev->bell; b; b = b->next)
57005b261ecSmrg	    if (b->ctrl.id == ((xBellFeedbackCtl *) & stuff[1])->id) {
57105b261ecSmrg		ChangeBellFeedback(client, dev, stuff->mask, b,
57205b261ecSmrg				   (xBellFeedbackCtl *) & stuff[1]);
57305b261ecSmrg		return Success;
57405b261ecSmrg	    }
57505b261ecSmrg	break;
57605b261ecSmrg    default:
57705b261ecSmrg	break;
57805b261ecSmrg    }
57905b261ecSmrg
58005b261ecSmrg    SendErrorToClient(client, IReqCode, X_ChangeFeedbackControl, 0, BadMatch);
58105b261ecSmrg    return Success;
58205b261ecSmrg}
58305b261ecSmrg
584