Blame src/dmx.c

Packit ec660a
/*
Packit ec660a
 * Copyright 2002-2004 Red Hat Inc., Durham, North Carolina.
Packit ec660a
 *
Packit ec660a
 * All Rights Reserved.
Packit ec660a
 *
Packit ec660a
 * Permission is hereby granted, free of charge, to any person obtaining
Packit ec660a
 * a copy of this software and associated documentation files (the
Packit ec660a
 * "Software"), to deal in the Software without restriction, including
Packit ec660a
 * without limitation on the rights to use, copy, modify, merge,
Packit ec660a
 * publish, distribute, sublicense, and/or sell copies of the Software,
Packit ec660a
 * and to permit persons to whom the Software is furnished to do so,
Packit ec660a
 * subject to the following conditions:
Packit ec660a
 *
Packit ec660a
 * The above copyright notice and this permission notice (including the
Packit ec660a
 * next paragraph) shall be included in all copies or substantial
Packit ec660a
 * portions of the Software.
Packit ec660a
 *
Packit ec660a
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
Packit ec660a
 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
Packit ec660a
 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
Packit ec660a
 * NON-INFRINGEMENT.  IN NO EVENT SHALL RED HAT AND/OR THEIR SUPPLIERS
Packit ec660a
 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
Packit ec660a
 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
Packit ec660a
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
Packit ec660a
 * SOFTWARE.
Packit ec660a
 */
Packit ec660a
Packit ec660a
/*
Packit ec660a
 * Authors:
Packit ec660a
 *   Rickard E. (Rik) Faith <faith@redhat.com>
Packit ec660a
 *
Packit ec660a
 */
Packit ec660a
Packit ec660a
/* THIS IS NOT AN X CONSORTIUM STANDARD */
Packit ec660a
Packit ec660a
/** \file
Packit ec660a
 * This file implements the client-side part of the DMX protocol.  It
Packit ec660a
 * can be included in client applications by linking with the libdmx.a
Packit ec660a
 * library. */
Packit ec660a
Packit ec660a
#ifdef HAVE_CONFIG_H
Packit ec660a
# include "config.h"
Packit ec660a
#endif
Packit ec660a
#include <X11/Xlibint.h>
Packit ec660a
#include <X11/extensions/Xext.h>
Packit ec660a
#define EXTENSION_PROC_ARGS void *
Packit ec660a
#include <X11/extensions/extutil.h>
Packit ec660a
#include <X11/extensions/dmxproto.h>
Packit ec660a
#include <X11/extensions/dmxext.h>
Packit ec660a
#include <limits.h>
Packit ec660a
Packit ec660a
static XExtensionInfo dmx_extension_info_data;
Packit ec660a
static XExtensionInfo *dmx_extension_info = &dmx_extension_info_data;
Packit ec660a
static const char     *dmx_extension_name = DMX_EXTENSION_NAME;
Packit ec660a
Packit ec660a
#define DMXCheckExtension(dpy,i,val) \
Packit ec660a
  XextCheckExtension(dpy, i, dmx_extension_name, val)
Packit ec660a
#define DMXSimpleCheckExtension(dpy,i) \
Packit ec660a
  XextSimpleCheckExtension(dpy, i, dmx_extension_name)
Packit ec660a
Packit ec660a
/*****************************************************************************
Packit ec660a
 *                                                                           *
Packit ec660a
 *                         private utility routines                          *
Packit ec660a
 *                                                                           *
Packit ec660a
 *****************************************************************************/
Packit ec660a
Packit ec660a
static int close_display(Display *dpy, XExtCodes *extCodes);
Packit ec660a
static /* const */ XExtensionHooks dmx_extension_hooks = {
Packit ec660a
    NULL,				/* create_gc */
Packit ec660a
    NULL,				/* copy_gc */
Packit ec660a
    NULL,				/* flush_gc */
Packit ec660a
    NULL,				/* free_gc */
Packit ec660a
    NULL,				/* create_font */
Packit ec660a
    NULL,				/* free_font */
Packit ec660a
    close_display,			/* close_display */
Packit ec660a
    NULL,				/* wire_to_event */
Packit ec660a
    NULL,				/* event_to_wire */
Packit ec660a
    NULL,				/* error */
Packit ec660a
    NULL,				/* error_string */
Packit ec660a
};
Packit ec660a
Packit ec660a
static XEXT_GENERATE_FIND_DISPLAY(find_display, dmx_extension_info,
Packit ec660a
                                  dmx_extension_name,
Packit ec660a
                                  &dmx_extension_hooks,
Packit ec660a
                                  0, NULL)
Packit ec660a
Packit ec660a
static XEXT_GENERATE_CLOSE_DISPLAY(close_display, dmx_extension_info)
Packit ec660a
Packit ec660a
Packit ec660a
/*****************************************************************************
Packit ec660a
 *                                                                           *
Packit ec660a
 *                  public DMX Extension routines                            *
Packit ec660a
 *                                                                           *
Packit ec660a
 *****************************************************************************/
Packit ec660a
Packit ec660a
/** If the server has the DMX extension, the event and error bases will
Packit ec660a
 * be placed in \a event_basep and \a error_basep, and True will be
Packit ec660a
 * returned.  Otherwise, False will be returned.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 1.0 */
Packit ec660a
Bool DMXQueryExtension(Display *dpy, int *event_basep, int *error_basep)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo *info = find_display(dpy);
Packit ec660a
Packit ec660a
    if (XextHasExtension(info)) {
Packit ec660a
	*event_basep = info->codes->first_event;
Packit ec660a
	*error_basep = info->codes->first_error;
Packit ec660a
	return True;
Packit ec660a
    } else {
Packit ec660a
	return False;
Packit ec660a
    }
Packit ec660a
}
Packit ec660a
Packit ec660a
/** If the DMXQueryVersion protocol request returns version information
Packit ec660a
 * from the server, \a majorVersion, \a minorVersion, and \a
Packit ec660a
 * patchVersion are filled in with the appropriate information and True
Packit ec660a
 * is returned.  Otherwise, False will be returned.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 1.0 */
Packit ec660a
Bool DMXQueryVersion(Display *dpy,
Packit ec660a
                     int *majorVersion, int *minorVersion, int *patchVersion)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo       *info = find_display(dpy);
Packit ec660a
    xDMXQueryVersionReply rep;
Packit ec660a
    xDMXQueryVersionReq   *req;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXQueryVersion, req);
Packit ec660a
    req->reqType     = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType  = X_DMXQueryVersion;
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xTrue)) {
Packit ec660a
	UnlockDisplay(dpy);
Packit ec660a
	SyncHandle();
Packit ec660a
	return False;
Packit ec660a
    }
Packit ec660a
    *majorVersion = rep.majorVersion;
Packit ec660a
    *minorVersion = rep.minorVersion;
Packit ec660a
    *patchVersion = rep.patchVersion;
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return True;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** Flush all pending dmxSync requests in DMX server.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 1.5 */
Packit ec660a
Bool DMXSync(Display *dpy)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo *info = find_display(dpy);
Packit ec660a
    xDMXSyncReply   rep;
Packit ec660a
    xDMXSyncReq     *req;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXSync, req);
Packit ec660a
    req->reqType    = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType = X_DMXSync;
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xTrue)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return rep.status == Success ? True : False;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** The creation of the specified \a window will be forced.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 1.2
Packit ec660a
 * Reply added in DMX Protocol Version 2.0 */
Packit ec660a
Bool DMXForceWindowCreation(Display *dpy, Window window)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo              *info = find_display(dpy);
Packit ec660a
    xDMXForceWindowCreationReq   *req;
Packit ec660a
    xDMXForceWindowCreationReply rep;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXForceWindowCreation, req);
Packit ec660a
    req->reqType    = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType = X_DMXForceWindowCreation;
Packit ec660a
    req->window     = window;
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xTrue)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return rep.status == Success ? True : False;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** If the DMXGetScreenCount protocol request returns the screen count,
Packit ec660a
 * the value will be placed in \a screen_count, and True will be
Packit ec660a
 * returned.  Otherwise, False will be returned.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 1.0 */
Packit ec660a
Bool DMXGetScreenCount(Display *dpy, int *screen_count)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo         *info = find_display(dpy);
Packit ec660a
    xDMXGetScreenCountReply rep;
Packit ec660a
    xDMXGetScreenCountReq   *req;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXGetScreenCount, req);
Packit ec660a
    req->reqType    = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType = X_DMXGetScreenCount;
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xTrue)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
    *screen_count = rep.screenCount;
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return True;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** If the DMXGetScreenAttributes protocol request returns information
Packit ec660a
 * for the specified \a physical_screen, information about the screen
Packit ec660a
 * will be placed in \a attr, and True will be returned.  Otherwise,
Packit ec660a
 * False will be returned.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 1.0; Modified in Version 2.0 */
Packit ec660a
Bool DMXGetScreenAttributes(Display *dpy, int physical_screen,
Packit ec660a
                            DMXScreenAttributes *attr)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo              *info = find_display(dpy);
Packit ec660a
    xDMXGetScreenAttributesReply rep;
Packit ec660a
    xDMXGetScreenAttributesReq   *req;
Packit ec660a
    Bool                          ret = False;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXGetScreenAttributes, req);
Packit ec660a
    req->reqType        = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType     = X_DMXGetScreenAttributes;
Packit ec660a
    req->physicalScreen = physical_screen;
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep,
Packit ec660a
                 (SIZEOF(xDMXGetScreenAttributesReply) - 32) >> 2, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
Packit ec660a
    if (rep.displayNameLength < 1024)
Packit ec660a
        attr->displayName = Xmalloc(rep.displayNameLength + 1 + 4 /* for pad */);
Packit ec660a
    else
Packit ec660a
        attr->displayName = NULL;  /* name length is unbelievable, reject */
Packit ec660a
    if (attr->displayName == NULL) {
Packit ec660a
        _XEatDataWords(dpy, rep.length);
Packit ec660a
        goto end;
Packit ec660a
    }
Packit ec660a
    _XReadPad(dpy, attr->displayName, rep.displayNameLength);
Packit ec660a
    attr->displayName[rep.displayNameLength] = '\0';
Packit ec660a
    attr->logicalScreen       = rep.logicalScreen;
Packit ec660a
Packit ec660a
    attr->screenWindowWidth   = rep.screenWindowWidth;
Packit ec660a
    attr->screenWindowHeight  = rep.screenWindowHeight;
Packit ec660a
    attr->screenWindowXoffset = rep.screenWindowXoffset;
Packit ec660a
    attr->screenWindowYoffset = rep.screenWindowYoffset;
Packit ec660a
Packit ec660a
    attr->rootWindowWidth     = rep.rootWindowWidth;
Packit ec660a
    attr->rootWindowHeight    = rep.rootWindowHeight;
Packit ec660a
    attr->rootWindowXoffset   = rep.rootWindowXoffset;
Packit ec660a
    attr->rootWindowYoffset   = rep.rootWindowYoffset;
Packit ec660a
    attr->rootWindowXorigin   = rep.rootWindowXorigin;
Packit ec660a
    attr->rootWindowYorigin   = rep.rootWindowYorigin;
Packit ec660a
Packit ec660a
    ret = True;
Packit ec660a
Packit ec660a
  end:
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return ret;
Packit ec660a
}
Packit ec660a
Packit ec660a
static CARD32 _DMXGetScreenAttribute(int bit, DMXScreenAttributes *attr)
Packit ec660a
{
Packit ec660a
    switch (1 << bit) {
Packit ec660a
    case DMXScreenWindowWidth:   return attr->screenWindowWidth;
Packit ec660a
    case DMXScreenWindowHeight:  return attr->screenWindowHeight;
Packit ec660a
    case DMXScreenWindowXoffset: return attr->screenWindowXoffset;
Packit ec660a
    case DMXScreenWindowYoffset: return attr->screenWindowYoffset;
Packit ec660a
    case DMXRootWindowWidth:     return attr->rootWindowWidth;
Packit ec660a
    case DMXRootWindowHeight:    return attr->rootWindowHeight;
Packit ec660a
    case DMXRootWindowXoffset:   return attr->rootWindowXoffset;
Packit ec660a
    case DMXRootWindowYoffset:   return attr->rootWindowYoffset;
Packit ec660a
    case DMXRootWindowXorigin:   return attr->rootWindowXorigin;
Packit ec660a
    case DMXRootWindowYorigin:   return attr->rootWindowYorigin;
Packit ec660a
    default:                     return 0;
Packit ec660a
    }
Packit ec660a
}
Packit ec660a
Packit ec660a
static int _DMXDumpScreenAttributes(Display *dpy,
Packit ec660a
                                    unsigned long mask,
Packit ec660a
                                    DMXScreenAttributes *attr)
Packit ec660a
{
Packit ec660a
    int           i;
Packit ec660a
    unsigned long value_list[32];
Packit ec660a
    unsigned long *value = value_list;
Packit ec660a
    int           count  = 0;
Packit ec660a
Packit ec660a
    for (i = 0; i < 32; i++) {
Packit ec660a
        if (mask & (1 << i)) {
Packit ec660a
            *value++ = _DMXGetScreenAttribute(i, attr);
Packit ec660a
            ++count;
Packit ec660a
        }
Packit ec660a
    }
Packit ec660a
    Data32(dpy, value_list, count * sizeof(CARD32));
Packit ec660a
    return count;
Packit ec660a
}
Packit ec660a
Packit ec660a
static CARD32 _DMXGetInputAttribute(int bit, DMXInputAttributes *attr)
Packit ec660a
{
Packit ec660a
    switch (1 << bit) {
Packit ec660a
    case DMXInputType:
Packit ec660a
        switch (attr->inputType) {
Packit ec660a
        case DMXLocalInputType:   return 0;
Packit ec660a
        case DMXConsoleInputType: return 1;
Packit ec660a
        case DMXBackendInputType: return 2;
Packit ec660a
        }
Packit ec660a
        return attr->inputType;
Packit ec660a
    case DMXInputPhysicalScreen: return attr->physicalScreen;
Packit ec660a
    case DMXInputSendsCore:      return attr->sendsCore;
Packit ec660a
    default:                     return 0;
Packit ec660a
    }
Packit ec660a
}
Packit ec660a
Packit ec660a
static int _DMXDumpInputAttributes(Display *dpy,
Packit ec660a
                                   unsigned long mask,
Packit ec660a
                                   DMXInputAttributes *attr)
Packit ec660a
{
Packit ec660a
    int           i;
Packit ec660a
    unsigned long value_list[32];
Packit ec660a
    unsigned long *value = value_list;
Packit ec660a
    int           count = 0;
Packit ec660a
Packit ec660a
    for (i = 0; i < 32; i++) {
Packit ec660a
        if (mask & (1 << i)) {
Packit ec660a
            *value++ = _DMXGetInputAttribute(i, attr);
Packit ec660a
            ++count;
Packit ec660a
        }
Packit ec660a
    }
Packit ec660a
    Data32(dpy, value_list, count * sizeof(CARD32));
Packit ec660a
    return count;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** Change geometries and positions of the DMX screen and root windows
Packit ec660a
 * on the back-end X server. */
Packit ec660a
int DMXChangeScreensAttributes(Display *dpy,
Packit ec660a
                               int screen_count,
Packit ec660a
                               int *screens,
Packit ec660a
                               int mask_count,
Packit ec660a
                               unsigned int *masks,
Packit ec660a
                               DMXScreenAttributes *attrs, /* vector */
Packit ec660a
                               int *error_screen)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo                  *info = find_display(dpy);
Packit ec660a
    xDMXChangeScreensAttributesReply rep;
Packit ec660a
    xDMXChangeScreensAttributesReq   *req;
Packit ec660a
    int                              i;
Packit ec660a
    unsigned int                     mask  = 0;
Packit ec660a
    CARD32                           *screen_list;
Packit ec660a
    CARD32                           *mask_list;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    if (screen_count < 1 || mask_count < 1) return DmxBadValue;
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXChangeScreensAttributes, req);
Packit ec660a
    req->reqType      = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType   = X_DMXChangeScreensAttributes;
Packit ec660a
    req->screenCount  = screen_count;
Packit ec660a
    req->maskCount    = mask_count;
Packit ec660a
    req->length      += screen_count + mask_count;
Packit ec660a
Packit ec660a
    screen_list = (CARD32 *)Xmalloc(sizeof(*screen_list) * screen_count);
Packit ec660a
    for (i = 0; i < screen_count; i++) screen_list[i] = screens[i];
Packit ec660a
    Data32(dpy, screen_list, screen_count * sizeof(CARD32));
Packit ec660a
    Xfree(screen_list);
Packit ec660a
Packit ec660a
    mask_list = (CARD32 *)Xmalloc(sizeof(*mask_list) * mask_count);
Packit ec660a
    for (i = 0; i < mask_count;   i++) mask_list[i]   = masks[i];
Packit ec660a
    Data32(dpy, mask_list, mask_count * sizeof(CARD32));
Packit ec660a
    Xfree(mask_list);
Packit ec660a
Packit ec660a
    for (i = 0; i < screen_count; i++) {
Packit ec660a
        if (i < mask_count) mask = masks[i];
Packit ec660a
        req->length += _DMXDumpScreenAttributes(dpy, mask, attrs + i);
Packit ec660a
    }
Packit ec660a
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return DmxBadReply;
Packit ec660a
    }
Packit ec660a
    if (error_screen) *error_screen = rep.errorScreen;
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return rep.status;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** Add a screen. */
Packit ec660a
Bool DMXAddScreen(Display *dpy, const char *displayName, unsigned int mask,
Packit ec660a
                  DMXScreenAttributes *attr, int *screen)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo    *info = find_display(dpy);
Packit ec660a
    xDMXAddScreenReply rep;
Packit ec660a
    xDMXAddScreenReq   *req;
Packit ec660a
    int                length;
Packit ec660a
    int                paddedLength;
Packit ec660a
Packit ec660a
    if (!screen)
Packit ec660a
	return False;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXAddScreen, req);
Packit ec660a
    length                 = displayName ? strlen(displayName) : 0;
Packit ec660a
    paddedLength           = (length + 3) & ~3;
Packit ec660a
    req->reqType           = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType        = X_DMXAddScreen;
Packit ec660a
    req->displayNameLength = length;
Packit ec660a
    req->physicalScreen    = *screen;
Packit ec660a
    req->valueMask         = mask;
Packit ec660a
    req->length           += paddedLength/4;
Packit ec660a
    req->length           += _DMXDumpScreenAttributes(dpy, mask, attr);
Packit ec660a
Packit ec660a
    if (length) {
Packit ec660a
        char *buffer       = Xcalloc(paddedLength, 1);
Packit ec660a
        memcpy(buffer, displayName, length);
Packit ec660a
        Data32(dpy, buffer, paddedLength);
Packit ec660a
        Xfree(buffer);
Packit ec660a
    }
Packit ec660a
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
    if (screen) *screen = rep.physicalScreen;
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return rep.status == Success ? True : False;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** Remove a screen. */
Packit ec660a
Bool DMXRemoveScreen(Display *dpy, int screen)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo       *info = find_display(dpy);
Packit ec660a
    xDMXRemoveScreenReply rep;
Packit ec660a
    xDMXRemoveScreenReq   *req;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXRemoveScreen, req);
Packit ec660a
    req->reqType           = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType        = X_DMXRemoveScreen;
Packit ec660a
    req->physicalScreen    = screen;
Packit ec660a
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return rep.status == Success ? True : False;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** If the DMXGetWindowAttributes protocol request returns information
Packit ec660a
 * about the specified \a window, the number of screens for which
Packit ec660a
 * information is available will be returned in \a screen_count and
Packit ec660a
 * information about the first \a available_count of those screens will
Packit ec660a
 * be placed in \a inf.  Because this call transports a great deal of
Packit ec660a
 * information over the wire, please call #DMXGetScreenCount first, and
Packit ec660a
 * make sure \a inf is that large.
Packit ec660a
 *
Packit ec660a
 * Note that if the specified \a window has not yet been mapped when
Packit ec660a
 * #DMXGetWindowAttributes is called, then a subsequent XMapWindow call
Packit ec660a
 * might be buffered in xlib while requests directly to the back-end X
Packit ec660a
 * servers are processed.  This race condition can be solved by calling
Packit ec660a
 * #DMXSync before talking directly to the back-end X servers.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 1.0, but not working correctly
Packit ec660a
 * until DMX Protocol Version 1.4 */
Packit ec660a
Bool DMXGetWindowAttributes(Display *dpy, Window window,
Packit ec660a
                            int *screen_count, int available_count,
Packit ec660a
                            DMXWindowAttributes *inf)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo              *info = find_display(dpy);
Packit ec660a
    xDMXGetWindowAttributesReply rep;
Packit ec660a
    xDMXGetWindowAttributesReq   *req;
Packit ec660a
    unsigned long                current;
Packit ec660a
    CARD32                       *screens; /* Must match protocol size */
Packit ec660a
    CARD32                       *windows; /* Must match protocol size */
Packit ec660a
    XRectangle                   *pos;     /* Must match protocol size */
Packit ec660a
    XRectangle                   *vis;     /* Must match protocol size */
Packit ec660a
    Bool                          ret = False;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXGetWindowAttributes, req);
Packit ec660a
    req->reqType    = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType = X_DMXGetWindowAttributes;
Packit ec660a
    req->window     = window;
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
Packit ec660a
    /*
Packit ec660a
     * rep.screenCount is a CARD32 so could be as large as 2^32
Packit ec660a
     * The X11 protocol limits the total screen size to 64k x 64k,
Packit ec660a
     * and no screen can be smaller than a pixel.  While technically
Packit ec660a
     * that means we could theoretically reach 2^32 screens, and that's
Packit ec660a
     * not even taking overlap into account, 64k seems far larger than
Packit ec660a
     * any reasonable configuration, so we limit to that to prevent both
Packit ec660a
     * integer overflow in the size calculations, and bad X server
Packit ec660a
     * responses causing massive memory allocation.
Packit ec660a
     */
Packit ec660a
    if (rep.screenCount < 65536) {
Packit ec660a
        screens    = Xmalloc(rep.screenCount * sizeof(*screens));
Packit ec660a
        windows    = Xmalloc(rep.screenCount * sizeof(*windows));
Packit ec660a
        pos        = Xmalloc(rep.screenCount * sizeof(*pos));
Packit ec660a
        vis        = Xmalloc(rep.screenCount * sizeof(*vis));
Packit ec660a
    } else {
Packit ec660a
        screens = windows = NULL;
Packit ec660a
        pos = vis = NULL;
Packit ec660a
    }
Packit ec660a
Packit ec660a
    if (!screens || !windows || !pos || !vis) {
Packit ec660a
        _XEatDataWords(dpy, rep.length);
Packit ec660a
        goto end;
Packit ec660a
    }
Packit ec660a
Packit ec660a
    _XRead(dpy, (char *)screens, rep.screenCount * sizeof(*screens));
Packit ec660a
    _XRead(dpy, (char *)windows, rep.screenCount * sizeof(*windows));
Packit ec660a
    _XRead(dpy, (char *)pos,     rep.screenCount * sizeof(*pos));
Packit ec660a
    _XRead(dpy, (char *)vis,     rep.screenCount * sizeof(*vis));
Packit ec660a
Packit ec660a
    *screen_count = rep.screenCount;
Packit ec660a
    for (current = 0;
Packit ec660a
         current < rep.screenCount && current < (unsigned)available_count;
Packit ec660a
         current++, inf++) {
Packit ec660a
        inf->screen    = screens[current];
Packit ec660a
        inf->window    = windows[current];
Packit ec660a
        inf->pos       = pos[current];
Packit ec660a
        inf->vis       = vis[current];
Packit ec660a
    }
Packit ec660a
    ret = True;
Packit ec660a
Packit ec660a
  end:
Packit ec660a
    Xfree(vis);
Packit ec660a
    Xfree(pos);
Packit ec660a
    Xfree(windows);
Packit ec660a
    Xfree(screens);
Packit ec660a
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return ret;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** If the DMXGetDesktopAttributes protocol request returns information
Packit ec660a
 * correctly, the information will be placed in \a attr, and True will
Packit ec660a
 * be returned.  Otherwise, False will be returned.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 2.0 */
Packit ec660a
Bool DMXGetDesktopAttributes(Display *dpy, DMXDesktopAttributes *attr)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo               *info = find_display(dpy);
Packit ec660a
    xDMXGetDesktopAttributesReply rep;
Packit ec660a
    xDMXGetDesktopAttributesReq   *req;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXGetDesktopAttributes, req);
Packit ec660a
    req->reqType        = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType     = X_DMXGetDesktopAttributes;
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
    attr->width  = rep.width;
Packit ec660a
    attr->height = rep.height;
Packit ec660a
    attr->shiftX = rep.shiftX;
Packit ec660a
    attr->shiftY = rep.shiftY;
Packit ec660a
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return True;
Packit ec660a
}
Packit ec660a
Packit ec660a
static CARD32 _DMXGetDesktopAttribute(int bit, DMXDesktopAttributes *attr)
Packit ec660a
{
Packit ec660a
    switch (1 << bit) {
Packit ec660a
    case DMXDesktopWidth:  return attr->width;
Packit ec660a
    case DMXDesktopHeight: return attr->height;
Packit ec660a
    case DMXDesktopShiftX: return attr->shiftX;
Packit ec660a
    case DMXDesktopShiftY: return attr->shiftY;
Packit ec660a
    default:               return 0;
Packit ec660a
    }
Packit ec660a
}
Packit ec660a
Packit ec660a
static int _DMXDumpDesktopAttributes(Display *dpy,
Packit ec660a
                                     unsigned long mask,
Packit ec660a
                                     DMXDesktopAttributes *attr)
Packit ec660a
{
Packit ec660a
    int           i;
Packit ec660a
    unsigned long value_list[32];
Packit ec660a
    unsigned long *value = value_list;
Packit ec660a
    int           count  = 0;
Packit ec660a
Packit ec660a
    for (i = 0; i < 32; i++) {
Packit ec660a
        if (mask & (1 << i)) {
Packit ec660a
            *value++ = _DMXGetDesktopAttribute(i, attr);
Packit ec660a
            ++count;
Packit ec660a
        }
Packit ec660a
    }
Packit ec660a
    Data32(dpy, value_list, count * sizeof(CARD32));
Packit ec660a
    return count;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** Change the global bounding box and origin offset.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 2.0 */
Packit ec660a
int DMXChangeDesktopAttributes(Display *dpy,
Packit ec660a
                               unsigned int mask,
Packit ec660a
                               DMXDesktopAttributes *attr)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo                  *info = find_display(dpy);
Packit ec660a
    xDMXChangeDesktopAttributesReply rep;
Packit ec660a
    xDMXChangeDesktopAttributesReq   *req;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXChangeDesktopAttributes, req);
Packit ec660a
    req->reqType      = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType   = X_DMXChangeDesktopAttributes;
Packit ec660a
    req->valueMask    = mask;
Packit ec660a
    req->length      +=_DMXDumpDesktopAttributes(dpy, mask, attr);
Packit ec660a
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return DmxBadReply;
Packit ec660a
    }
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return rep.status;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** If the DMXGetInputCount protocol request returns the input count,
Packit ec660a
 * the value will be placed in \a input_count, and True will be
Packit ec660a
 * returned.  Otherwise, False will be returned.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 1.1 */
Packit ec660a
Bool DMXGetInputCount(Display *dpy, int *input_count)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo         *info = find_display(dpy);
Packit ec660a
    xDMXGetInputCountReply  rep;
Packit ec660a
    xDMXGetInputCountReq    *req;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXGetInputCount, req);
Packit ec660a
    req->reqType    = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType = X_DMXGetInputCount;
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xTrue)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
    *input_count = rep.inputCount;
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return True;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** If the DMXGetInputAttributes protocol request returns information
Packit ec660a
 * about the input device with the specified \a id, information about
Packit ec660a
 * the input device will be placed in \a inf, and True will be returned.
Packit ec660a
 * Otherwise, False will be returned.
Packit ec660a
 *
Packit ec660a
 * Available in DMX Protocol Version 1.1 */
Packit ec660a
Bool DMXGetInputAttributes(Display *dpy, int id, DMXInputAttributes *inf)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo             *info = find_display(dpy);
Packit ec660a
    xDMXGetInputAttributesReply rep;
Packit ec660a
    xDMXGetInputAttributesReq   *req;
Packit ec660a
    char                        *buffer;
Packit ec660a
    Bool                         ret = False;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXGetInputAttributes, req);
Packit ec660a
    req->reqType    = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType = X_DMXGetInputAttributes;
Packit ec660a
    req->deviceId   = id;
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
Packit ec660a
    if (rep.nameLength < 1024)
Packit ec660a
        buffer      = Xmalloc(rep.nameLength + 1 + 4 /* for pad */);
Packit ec660a
    else
Packit ec660a
        buffer      = NULL;  /* name length is unbelievable, reject */
Packit ec660a
Packit ec660a
    if (buffer == NULL) {
Packit ec660a
        _XEatDataWords(dpy, rep.length);
Packit ec660a
        goto end;
Packit ec660a
    }
Packit ec660a
Packit ec660a
    switch (rep.inputType) {
Packit ec660a
    case 0: inf->inputType = DMXLocalInputType;   break;
Packit ec660a
    case 1: inf->inputType = DMXConsoleInputType; break;
Packit ec660a
    case 2: inf->inputType = DMXBackendInputType; break;
Packit ec660a
    }
Packit ec660a
Packit ec660a
    inf->physicalScreen = rep.physicalScreen;
Packit ec660a
    inf->physicalId     = rep.physicalId;
Packit ec660a
    inf->isCore         = rep.isCore;
Packit ec660a
    inf->sendsCore      = rep.sendsCore;
Packit ec660a
    inf->detached       = rep.detached;
Packit ec660a
    _XReadPad(dpy, buffer, rep.nameLength);
Packit ec660a
    buffer[rep.nameLength] = '\0';
Packit ec660a
    inf->name           = buffer;
Packit ec660a
    ret = True;
Packit ec660a
  end:
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return ret;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** Add input. */
Packit ec660a
Bool DMXAddInput(Display *dpy, unsigned int mask, DMXInputAttributes *attr,
Packit ec660a
                 int *id)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo         *info = find_display(dpy);
Packit ec660a
    xDMXAddInputReply       rep;
Packit ec660a
    xDMXAddInputReq         *req;
Packit ec660a
    int                     length;
Packit ec660a
    int                     paddedLength;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXAddInput, req);
Packit ec660a
    length                 = attr->name ? strlen(attr->name) : 0;
Packit ec660a
    paddedLength           = (length + 3) & ~3;
Packit ec660a
    req->reqType           = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType        = X_DMXAddInput;
Packit ec660a
    req->displayNameLength = length;
Packit ec660a
    req->valueMask         = mask;
Packit ec660a
    req->length           += paddedLength/4;
Packit ec660a
    req->length           += _DMXDumpInputAttributes(dpy, mask, attr);
Packit ec660a
Packit ec660a
    if (length) {
Packit ec660a
        char *buffer       = Xcalloc(paddedLength, 1);
Packit ec660a
        memcpy(buffer, attr->name, paddedLength);
Packit ec660a
        Data32(dpy, buffer, paddedLength);
Packit ec660a
        Xfree(buffer);
Packit ec660a
    }
Packit ec660a
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
    if (id) *id = rep.physicalId;
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return rep.status == Success ? True : False;
Packit ec660a
}
Packit ec660a
Packit ec660a
/** Add backend input (a helper function that calls #DMXAddInput). */
Packit ec660a
Bool DMXAddBackendInput(Display *dpy, int screen, int sendsCore, int *newId)
Packit ec660a
{
Packit ec660a
    DMXInputAttributes attr;
Packit ec660a
    unsigned int       mask = (DMXInputType
Packit ec660a
                               | DMXInputPhysicalScreen
Packit ec660a
                               | DMXInputSendsCore);
Packit ec660a
Packit ec660a
    attr.inputType        = DMXBackendInputType;
Packit ec660a
    attr.physicalScreen   = screen;
Packit ec660a
    attr.sendsCore        = sendsCore;
Packit ec660a
    attr.name             = NULL;
Packit ec660a
    return DMXAddInput(dpy, mask, &attr, newId);
Packit ec660a
}
Packit ec660a
Packit ec660a
/** Add console input (a helper function that calls #DMXAddInput). */
Packit ec660a
Bool DMXAddConsoleInput(Display *dpy, const char *name, int sendsCore,
Packit ec660a
                        int *newId)
Packit ec660a
{
Packit ec660a
    DMXInputAttributes attr;
Packit ec660a
    unsigned int       mask = (DMXInputType
Packit ec660a
                               | DMXInputSendsCore);
Packit ec660a
Packit ec660a
    attr.inputType        = DMXConsoleInputType;
Packit ec660a
    attr.physicalScreen   = 0;
Packit ec660a
    attr.sendsCore        = sendsCore;
Packit ec660a
    attr.name             = name;
Packit ec660a
    return DMXAddInput(dpy, mask, &attr, newId);
Packit ec660a
}
Packit ec660a
Packit ec660a
/** Remove an input. */
Packit ec660a
Bool DMXRemoveInput(Display *dpy, int id)
Packit ec660a
{
Packit ec660a
    XExtDisplayInfo      *info = find_display(dpy);
Packit ec660a
    xDMXRemoveInputReply rep;
Packit ec660a
    xDMXRemoveInputReq   *req;
Packit ec660a
Packit ec660a
    DMXCheckExtension(dpy, info, False);
Packit ec660a
Packit ec660a
    LockDisplay(dpy);
Packit ec660a
    GetReq(DMXRemoveInput, req);
Packit ec660a
    req->reqType          = info->codes->major_opcode;
Packit ec660a
    req->dmxReqType       = X_DMXRemoveInput;
Packit ec660a
    req->physicalId       = id;
Packit ec660a
Packit ec660a
    if (!_XReply(dpy, (xReply *)&rep, 0, xFalse)) {
Packit ec660a
        UnlockDisplay(dpy);
Packit ec660a
        SyncHandle();
Packit ec660a
        return False;
Packit ec660a
    }
Packit ec660a
    UnlockDisplay(dpy);
Packit ec660a
    SyncHandle();
Packit ec660a
    return rep.status == Success ? True : False;
Packit ec660a
}