|
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 |
}
|