Blame src/ModMap.c

Packit 5bd3a9
/*
Packit 5bd3a9
Packit 5bd3a9
Copyright 1986, 1998  The Open Group
Packit 5bd3a9
Packit 5bd3a9
Permission to use, copy, modify, distribute, and sell this software and its
Packit 5bd3a9
documentation for any purpose is hereby granted without fee, provided that
Packit 5bd3a9
the above copyright notice appear in all copies and that both that
Packit 5bd3a9
copyright notice and this permission notice appear in supporting
Packit 5bd3a9
documentation.
Packit 5bd3a9
Packit 5bd3a9
The above copyright notice and this permission notice shall be included in
Packit 5bd3a9
all copies or substantial portions of the Software.
Packit 5bd3a9
Packit 5bd3a9
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit 5bd3a9
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit 5bd3a9
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
Packit 5bd3a9
OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
Packit 5bd3a9
AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
Packit 5bd3a9
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Packit 5bd3a9
Packit 5bd3a9
Except as contained in this notice, the name of The Open Group shall not be
Packit 5bd3a9
used in advertising or otherwise to promote the sale, use or other dealings
Packit 5bd3a9
in this Software without prior written authorization from The Open Group.
Packit 5bd3a9
Packit 5bd3a9
*/
Packit 5bd3a9
Packit 5bd3a9
#ifdef HAVE_CONFIG_H
Packit 5bd3a9
#include <config.h>
Packit 5bd3a9
#endif
Packit 5bd3a9
#include "Xlibint.h"
Packit 5bd3a9
#include "reallocarray.h"
Packit 5bd3a9
#include <limits.h>
Packit 5bd3a9
Packit 5bd3a9
XModifierKeymap *
Packit 5bd3a9
XGetModifierMapping(register Display *dpy)
Packit 5bd3a9
{
Packit 5bd3a9
    xGetModifierMappingReply rep;
Packit 5bd3a9
    _X_UNUSED register xReq *req;
Packit 5bd3a9
    unsigned long nbytes;
Packit 5bd3a9
    XModifierKeymap *res;
Packit 5bd3a9
Packit 5bd3a9
    LockDisplay(dpy);
Packit 5bd3a9
    GetEmptyReq(GetModifierMapping, req);
Packit 5bd3a9
    (void) _XReply (dpy, (xReply *)&rep, 0, xFalse);
Packit 5bd3a9
Packit 5bd3a9
    if (rep.length < (INT_MAX >> 2) &&
Packit 5bd3a9
	(rep.length >> 1) == rep.numKeyPerModifier) {
Packit 5bd3a9
	nbytes = (unsigned long)rep.length << 2;
Packit 5bd3a9
	res = Xmalloc(sizeof (XModifierKeymap));
Packit 5bd3a9
	if (res)
Packit 5bd3a9
	    res->modifiermap = Xmalloc (nbytes);
Packit 5bd3a9
    } else
Packit 5bd3a9
	res = NULL;
Packit 5bd3a9
    if ((! res) || (! res->modifiermap)) {
Packit 5bd3a9
	Xfree(res);
Packit 5bd3a9
	res = (XModifierKeymap *) NULL;
Packit 5bd3a9
	_XEatDataWords(dpy, rep.length);
Packit 5bd3a9
    } else {
Packit 5bd3a9
	_XReadPad(dpy, (char *) res->modifiermap, (long) nbytes);
Packit 5bd3a9
	res->max_keypermod = rep.numKeyPerModifier;
Packit 5bd3a9
    }
Packit 5bd3a9
Packit 5bd3a9
    UnlockDisplay(dpy);
Packit 5bd3a9
    SyncHandle();
Packit 5bd3a9
    return (res);
Packit 5bd3a9
}
Packit 5bd3a9
Packit 5bd3a9
/*
Packit 5bd3a9
 *	Returns:
Packit 5bd3a9
 *	MappingSuccess (0)	Success
Packit 5bd3a9
 *	MappingBusy (1) 	Busy - one or more old or new modifiers are down
Packit 5bd3a9
 *	MappingFailed (2)	Failed - one or more new modifiers unacceptable
Packit 5bd3a9
 */
Packit 5bd3a9
int
Packit 5bd3a9
XSetModifierMapping(
Packit 5bd3a9
    register Display *dpy,
Packit 5bd3a9
    register XModifierKeymap *modifier_map)
Packit 5bd3a9
{
Packit 5bd3a9
    register xSetModifierMappingReq *req;
Packit 5bd3a9
    xSetModifierMappingReply rep;
Packit 5bd3a9
    int         mapSize = modifier_map->max_keypermod << 3;	/* 8 modifiers */
Packit 5bd3a9
Packit 5bd3a9
    LockDisplay(dpy);
Packit 5bd3a9
    GetReq(SetModifierMapping, req);
Packit 5bd3a9
    req->length += mapSize >> 2;
Packit 5bd3a9
    req->numKeyPerModifier = modifier_map->max_keypermod;
Packit 5bd3a9
Packit 5bd3a9
    Data(dpy, (const char *)modifier_map->modifiermap, mapSize);
Packit 5bd3a9
Packit 5bd3a9
    (void) _XReply(dpy, (xReply *) & rep,
Packit 5bd3a9
	(SIZEOF(xSetModifierMappingReply) - SIZEOF(xReply)) >> 2, xTrue);
Packit 5bd3a9
    UnlockDisplay(dpy);
Packit 5bd3a9
    SyncHandle();
Packit 5bd3a9
    return (rep.success);
Packit 5bd3a9
}
Packit 5bd3a9
Packit 5bd3a9
XModifierKeymap *
Packit 5bd3a9
XNewModifiermap(int keyspermodifier)
Packit 5bd3a9
{
Packit 5bd3a9
    XModifierKeymap *res = Xmalloc((sizeof (XModifierKeymap)));
Packit 5bd3a9
    if (res) {
Packit 5bd3a9
	res->max_keypermod = keyspermodifier;
Packit 5bd3a9
	res->modifiermap = (keyspermodifier > 0 ?
Packit 5bd3a9
			    Xmallocarray(keyspermodifier, 8)
Packit 5bd3a9
			    : (KeyCode *) NULL);
Packit 5bd3a9
	if (keyspermodifier && (res->modifiermap == NULL)) {
Packit 5bd3a9
	    Xfree(res);
Packit 5bd3a9
	    return (XModifierKeymap *) NULL;
Packit 5bd3a9
	}
Packit 5bd3a9
    }
Packit 5bd3a9
    return (res);
Packit 5bd3a9
}
Packit 5bd3a9
Packit 5bd3a9
Packit 5bd3a9
int
Packit 5bd3a9
XFreeModifiermap(XModifierKeymap *map)
Packit 5bd3a9
{
Packit 5bd3a9
    if (map) {
Packit 5bd3a9
        Xfree(map->modifiermap);
Packit 5bd3a9
	Xfree(map);
Packit 5bd3a9
    }
Packit 5bd3a9
    return 1;
Packit 5bd3a9
}
Packit 5bd3a9
Packit 5bd3a9
XModifierKeymap *
Packit 5bd3a9
XInsertModifiermapEntry(XModifierKeymap *map,
Packit 5bd3a9
#if NeedWidePrototypes
Packit 5bd3a9
			unsigned int keycode,
Packit 5bd3a9
#else
Packit 5bd3a9
			KeyCode keycode,
Packit 5bd3a9
#endif
Packit 5bd3a9
			int modifier)
Packit 5bd3a9
{
Packit 5bd3a9
    XModifierKeymap *newmap;
Packit 5bd3a9
    int i,
Packit 5bd3a9
	row = modifier * map->max_keypermod,
Packit 5bd3a9
	newrow,
Packit 5bd3a9
	lastrow;
Packit 5bd3a9
Packit 5bd3a9
    for (i=0; i<map->max_keypermod; i++) {
Packit 5bd3a9
        if (map->modifiermap[ row+i ] == keycode)
Packit 5bd3a9
	    return(map); /* already in the map */
Packit 5bd3a9
        if (map->modifiermap[ row+i ] == 0) {
Packit 5bd3a9
            map->modifiermap[ row+i ] = keycode;
Packit 5bd3a9
	    return(map); /* we added it without stretching the map */
Packit 5bd3a9
	}
Packit 5bd3a9
    }
Packit 5bd3a9
Packit 5bd3a9
    /* stretch the map */
Packit 5bd3a9
    if ((newmap = XNewModifiermap(map->max_keypermod+1)) == NULL)
Packit 5bd3a9
	return (XModifierKeymap *) NULL;
Packit 5bd3a9
    newrow = row = 0;
Packit 5bd3a9
    lastrow = newmap->max_keypermod * 8;
Packit 5bd3a9
    while (newrow < lastrow) {
Packit 5bd3a9
	for (i=0; i<map->max_keypermod; i++)
Packit 5bd3a9
	    newmap->modifiermap[ newrow+i ] = map->modifiermap[ row+i ];
Packit 5bd3a9
	newmap->modifiermap[ newrow+i ] = 0;
Packit 5bd3a9
	row += map->max_keypermod;
Packit 5bd3a9
	newrow += newmap->max_keypermod;
Packit 5bd3a9
    }
Packit 5bd3a9
    (void) XFreeModifiermap(map);
Packit 5bd3a9
    newrow = newmap->max_keypermod * modifier + newmap->max_keypermod - 1;
Packit 5bd3a9
    newmap->modifiermap[ newrow ] = keycode;
Packit 5bd3a9
    return(newmap);
Packit 5bd3a9
}
Packit 5bd3a9
Packit 5bd3a9
XModifierKeymap *
Packit 5bd3a9
XDeleteModifiermapEntry(XModifierKeymap *map,
Packit 5bd3a9
#if NeedWidePrototypes
Packit 5bd3a9
			unsigned int keycode,
Packit 5bd3a9
#else
Packit 5bd3a9
			KeyCode keycode,
Packit 5bd3a9
#endif
Packit 5bd3a9
			int modifier)
Packit 5bd3a9
{
Packit 5bd3a9
    int i,
Packit 5bd3a9
	row = modifier * map->max_keypermod;
Packit 5bd3a9
Packit 5bd3a9
    for (i=0; i<map->max_keypermod; i++) {
Packit 5bd3a9
        if (map->modifiermap[ row+i ] == keycode)
Packit 5bd3a9
            map->modifiermap[ row+i ] = 0;
Packit 5bd3a9
    }
Packit 5bd3a9
    /* should we shrink the map?? */
Packit 5bd3a9
    return (map);
Packit 5bd3a9
}