966 lines
24 KiB
C
966 lines
24 KiB
C
/***********************************************************
|
|
|
|
Copyright 1987, 1998 The Open Group
|
|
|
|
Permission to use, copy, modify, distribute, and sell this software and its
|
|
documentation for any purpose is hereby granted without fee, provided that
|
|
the above copyright notice appear in all copies and that both that
|
|
copyright notice and this permission notice appear in supporting
|
|
documentation.
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
|
AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
Except as contained in this notice, the name of The Open Group shall not be
|
|
used in advertising or otherwise to promote the sale, use or other dealings
|
|
in this Software without prior written authorization from The Open Group.
|
|
|
|
|
|
Copyright 1987 by Digital Equipment Corporation, Maynard, Massachusetts.
|
|
|
|
All Rights Reserved
|
|
|
|
Permission to use, copy, modify, and distribute this software and its
|
|
documentation for any purpose and without fee is hereby granted,
|
|
provided that the above copyright notice appear in all copies and that
|
|
both that copyright notice and this permission notice appear in
|
|
supporting documentation, and that the name of Digital not be
|
|
used in advertising or publicity pertaining to distribution of the
|
|
software without specific, written prior permission.
|
|
|
|
DIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
|
|
ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
|
|
DIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
|
|
ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
|
|
WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
|
|
ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
|
SOFTWARE.
|
|
|
|
******************************************************************/
|
|
|
|
/*
|
|
|
|
(c)Copyright 1988,1991 Adobe Systems Incorporated. All rights reserved.
|
|
|
|
Permission to use, copy, modify, distribute, and sublicense this software and its
|
|
documentation for any purpose and without fee is hereby granted, provided that
|
|
the above copyright notices appear in all copies and that both those copyright
|
|
notices and this permission notice appear in supporting documentation and that
|
|
the name of Adobe Systems Incorporated not be used in advertising or publicity
|
|
pertaining to distribution of the software without specific, written prior
|
|
permission. No trademark license to use the Adobe trademarks is hereby
|
|
granted. If the Adobe trademark "Display PostScript"(tm) is used to describe
|
|
this software, its functionality or for any other purpose, such use shall be
|
|
limited to a statement that this software works in conjunction with the Display
|
|
PostScript system. Proper trademark attribution to reflect Adobe's ownership
|
|
of the trademark shall be given whenever any such reference to the Display
|
|
PostScript system is made.
|
|
|
|
ADOBE MAKES NO REPRESENTATIONS ABOUT THE SUITABILITY OF THE SOFTWARE FOR ANY
|
|
PURPOSE. IT IS PROVIDED "AS IS" WITHOUT EXPRESS OR IMPLIED WARRANTY. ADOBE
|
|
DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING ALL IMPLIED
|
|
WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-
|
|
INFRINGEMENT OF THIRD PARTY RIGHTS. IN NO EVENT SHALL ADOBE BE LIABLE TO YOU
|
|
OR ANY OTHER PARTY FOR ANY SPECIAL, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY
|
|
DAMAGES WHATSOEVER WHETHER IN AN ACTION OF CONTRACT,NEGLIGENCE, STRICT
|
|
LIABILITY OR ANY OTHER ACTION ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
|
PERFORMANCE OF THIS SOFTWARE. ADOBE WILL NOT PROVIDE ANY TRAINING OR OTHER
|
|
SUPPORT FOR THE SOFTWARE.
|
|
|
|
Adobe, PostScript, and Display PostScript are trademarks of Adobe Systems
|
|
Incorporated which may be registered in certain jurisdictions.
|
|
|
|
Author: Adobe Systems Incorporated
|
|
|
|
*/
|
|
|
|
|
|
#ifdef HAVE_DIX_CONFIG_H
|
|
#include <dix-config.h>
|
|
#endif
|
|
|
|
#include <X11/X.h>
|
|
#include <X11/Xmd.h>
|
|
#include "misc.h"
|
|
#include "windowstr.h"
|
|
#include "dixstruct.h"
|
|
#include "pixmapstr.h"
|
|
#include "scrnintstr.h"
|
|
#define XK_LATIN1
|
|
#include <X11/keysymdef.h>
|
|
#include "xace.h"
|
|
|
|
/*
|
|
* CompareTimeStamps returns -1, 0, or +1 depending on if the first
|
|
* argument is less than, equal to or greater than the second argument.
|
|
*/
|
|
|
|
_X_EXPORT int
|
|
CompareTimeStamps(TimeStamp a, TimeStamp b)
|
|
{
|
|
if (a.months < b.months)
|
|
return EARLIER;
|
|
if (a.months > b.months)
|
|
return LATER;
|
|
if (a.milliseconds < b.milliseconds)
|
|
return EARLIER;
|
|
if (a.milliseconds > b.milliseconds)
|
|
return LATER;
|
|
return SAMETIME;
|
|
}
|
|
|
|
/*
|
|
* convert client times to server TimeStamps
|
|
*/
|
|
|
|
#define HALFMONTH ((unsigned long) 1<<31)
|
|
_X_EXPORT TimeStamp
|
|
ClientTimeToServerTime(CARD32 c)
|
|
{
|
|
TimeStamp ts;
|
|
if (c == CurrentTime)
|
|
return currentTime;
|
|
ts.months = currentTime.months;
|
|
ts.milliseconds = c;
|
|
if (c > currentTime.milliseconds)
|
|
{
|
|
if (((unsigned long) c - currentTime.milliseconds) > HALFMONTH)
|
|
ts.months -= 1;
|
|
}
|
|
else if (c < currentTime.milliseconds)
|
|
{
|
|
if (((unsigned long)currentTime.milliseconds - c) > HALFMONTH)
|
|
ts.months += 1;
|
|
}
|
|
return ts;
|
|
}
|
|
|
|
/*
|
|
* ISO Latin-1 case conversion routine
|
|
*
|
|
* this routine always null-terminates the result, so
|
|
* beware of too-small buffers
|
|
*/
|
|
|
|
static unsigned char
|
|
ISOLatin1ToLower (unsigned char source)
|
|
{
|
|
unsigned char dest;
|
|
if ((source >= XK_A) && (source <= XK_Z))
|
|
dest = source + (XK_a - XK_A);
|
|
else if ((source >= XK_Agrave) && (source <= XK_Odiaeresis))
|
|
dest = source + (XK_agrave - XK_Agrave);
|
|
else if ((source >= XK_Ooblique) && (source <= XK_Thorn))
|
|
dest = source + (XK_oslash - XK_Ooblique);
|
|
else
|
|
dest = source;
|
|
return dest;
|
|
}
|
|
|
|
|
|
_X_EXPORT void
|
|
CopyISOLatin1Lowered(unsigned char *dest, unsigned char *source, int length)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < length; i++, source++, dest++)
|
|
*dest = ISOLatin1ToLower (*source);
|
|
*dest = '\0';
|
|
}
|
|
|
|
int
|
|
CompareISOLatin1Lowered(unsigned char *s1, int s1len,
|
|
unsigned char *s2, int s2len)
|
|
{
|
|
unsigned char c1, c2;
|
|
|
|
for (;;)
|
|
{
|
|
/* note -- compare against zero so that -1 ignores len */
|
|
c1 = s1len-- ? *s1++ : '\0';
|
|
c2 = s2len-- ? *s2++ : '\0';
|
|
if (!c1 ||
|
|
(c1 != c2 &&
|
|
(c1 = ISOLatin1ToLower (c1)) != (c2 = ISOLatin1ToLower (c2))))
|
|
break;
|
|
}
|
|
return (int) c1 - (int) c2;
|
|
}
|
|
|
|
/*
|
|
* dixLookupWindow and dixLookupDrawable:
|
|
* Look up the window/drawable taking into account the client doing the
|
|
* lookup, the type of drawable desired, and the type of access desired.
|
|
* Return Success with *pDraw set if the window/drawable exists and the client
|
|
* is allowed access, else return an error code with *pDraw set to NULL. The
|
|
* access mask values are defined in resource.h. The type mask values are
|
|
* defined in pixmap.h, with zero equivalent to M_DRAWABLE.
|
|
*/
|
|
_X_EXPORT int
|
|
dixLookupDrawable(DrawablePtr *pDraw, XID id, ClientPtr client,
|
|
Mask type, Mask access)
|
|
{
|
|
DrawablePtr pTmp;
|
|
RESTYPE rtype;
|
|
*pDraw = NULL;
|
|
client->errorValue = id;
|
|
|
|
if (id == INVALID)
|
|
return BadDrawable;
|
|
|
|
if (id == client->lastDrawableID) {
|
|
pTmp = client->lastDrawable;
|
|
|
|
/* an access check is required for cached drawables */
|
|
rtype = (type & M_WINDOW) ? RT_WINDOW : RT_PIXMAP;
|
|
if (!XaceHook(XACE_RESOURCE_ACCESS, client, id, rtype, access, pTmp))
|
|
return BadDrawable;
|
|
} else
|
|
pTmp = (DrawablePtr)SecurityLookupIDByClass(client, id, RC_DRAWABLE,
|
|
access);
|
|
if (!pTmp)
|
|
return BadDrawable;
|
|
if (!((1 << pTmp->type) & (type ? type : M_DRAWABLE)))
|
|
return BadMatch;
|
|
|
|
if (type & M_DRAWABLE) {
|
|
client->lastDrawable = pTmp;
|
|
client->lastDrawableID = id;
|
|
client->lastGCID = INVALID;
|
|
client->lastGC = (GCPtr)NULL;
|
|
}
|
|
*pDraw = pTmp;
|
|
return Success;
|
|
}
|
|
|
|
_X_EXPORT int
|
|
dixLookupWindow(WindowPtr *pWin, XID id, ClientPtr client, Mask access)
|
|
{
|
|
int rc;
|
|
rc = dixLookupDrawable((DrawablePtr*)pWin, id, client, M_WINDOW, access);
|
|
return (rc == BadDrawable) ? BadWindow : rc;
|
|
}
|
|
|
|
_X_EXPORT int
|
|
dixLookupGC(GCPtr *pGC, XID id, ClientPtr client, Mask access)
|
|
{
|
|
GCPtr pTmp = (GCPtr)SecurityLookupIDByType(client, id, RT_GC, access);
|
|
if (pTmp) {
|
|
*pGC = pTmp;
|
|
return Success;
|
|
}
|
|
client->errorValue = id;
|
|
*pGC = NULL;
|
|
return BadGC;
|
|
}
|
|
|
|
_X_EXPORT int
|
|
dixLookupClient(ClientPtr *pClient, XID rid, ClientPtr client, Mask access)
|
|
{
|
|
pointer pRes = (pointer)SecurityLookupIDByClass(client, rid, RC_ANY,
|
|
access);
|
|
int clientIndex = CLIENT_ID(rid);
|
|
client->errorValue = rid;
|
|
|
|
if (clientIndex && pRes && clients[clientIndex] && !(rid & SERVER_BIT)) {
|
|
*pClient = clients[clientIndex];
|
|
return Success;
|
|
}
|
|
*pClient = NULL;
|
|
return BadValue;
|
|
}
|
|
|
|
/*
|
|
* These are deprecated compatibility functions and will be removed soon!
|
|
* Please use the new dixLookup*() functions above.
|
|
*/
|
|
_X_EXPORT _X_DEPRECATED WindowPtr
|
|
SecurityLookupWindow(XID id, ClientPtr client, Mask access_mode)
|
|
{
|
|
WindowPtr pWin;
|
|
int i = dixLookupWindow(&pWin, id, client, access_mode);
|
|
static int warn = 1;
|
|
if (warn-- > 0)
|
|
ErrorF("Warning: LookupWindow()/SecurityLookupWindow() "
|
|
"are deprecated. Please convert your driver/module "
|
|
"to use dixLookupWindow().\n");
|
|
return (i == Success) ? pWin : NULL;
|
|
}
|
|
|
|
_X_EXPORT _X_DEPRECATED WindowPtr
|
|
LookupWindow(XID id, ClientPtr client)
|
|
{
|
|
return SecurityLookupWindow(id, client, DixUnknownAccess);
|
|
}
|
|
|
|
_X_EXPORT _X_DEPRECATED pointer
|
|
SecurityLookupDrawable(XID id, ClientPtr client, Mask access_mode)
|
|
{
|
|
DrawablePtr pDraw;
|
|
int i = dixLookupDrawable(&pDraw, id, client, M_DRAWABLE, access_mode);
|
|
static int warn = 1;
|
|
if (warn-- > 0)
|
|
ErrorF("Warning: LookupDrawable()/SecurityLookupDrawable() "
|
|
"are deprecated. Please convert your driver/module "
|
|
"to use dixLookupDrawable().\n");
|
|
return (i == Success) ? pDraw : NULL;
|
|
}
|
|
|
|
_X_EXPORT _X_DEPRECATED pointer
|
|
LookupDrawable(XID id, ClientPtr client)
|
|
{
|
|
return SecurityLookupDrawable(id, client, DixUnknownAccess);
|
|
}
|
|
|
|
_X_EXPORT _X_DEPRECATED ClientPtr
|
|
LookupClient(XID id, ClientPtr client)
|
|
{
|
|
ClientPtr pClient;
|
|
int i = dixLookupClient(&pClient, id, client, DixUnknownAccess);
|
|
static int warn = 1;
|
|
if (warn-- > 0)
|
|
ErrorF("Warning: LookupClient() is deprecated. Please convert your "
|
|
"driver/module to use dixLookupClient().\n");
|
|
return (i == Success) ? pClient : NULL;
|
|
}
|
|
|
|
/* end deprecated functions */
|
|
|
|
int
|
|
AlterSaveSetForClient(ClientPtr client, WindowPtr pWin, unsigned mode,
|
|
Bool toRoot, Bool remap)
|
|
{
|
|
int numnow;
|
|
SaveSetElt *pTmp = NULL;
|
|
int j;
|
|
|
|
numnow = client->numSaved;
|
|
j = 0;
|
|
if (numnow)
|
|
{
|
|
pTmp = client->saveSet;
|
|
while ((j < numnow) && (SaveSetWindow(pTmp[j]) != (pointer)pWin))
|
|
j++;
|
|
}
|
|
if (mode == SetModeInsert)
|
|
{
|
|
if (j < numnow) /* duplicate */
|
|
return(Success);
|
|
numnow++;
|
|
pTmp = (SaveSetElt *)xrealloc(client->saveSet, sizeof(*pTmp) * numnow);
|
|
if (!pTmp)
|
|
return(BadAlloc);
|
|
client->saveSet = pTmp;
|
|
client->numSaved = numnow;
|
|
SaveSetAssignWindow(client->saveSet[numnow - 1], pWin);
|
|
SaveSetAssignToRoot(client->saveSet[numnow - 1], toRoot);
|
|
SaveSetAssignRemap(client->saveSet[numnow - 1], remap);
|
|
return(Success);
|
|
}
|
|
else if ((mode == SetModeDelete) && (j < numnow))
|
|
{
|
|
while (j < numnow-1)
|
|
{
|
|
pTmp[j] = pTmp[j+1];
|
|
j++;
|
|
}
|
|
numnow--;
|
|
if (numnow)
|
|
{
|
|
pTmp = (SaveSetElt *)xrealloc(client->saveSet, sizeof(*pTmp) * numnow);
|
|
if (pTmp)
|
|
client->saveSet = pTmp;
|
|
}
|
|
else
|
|
{
|
|
xfree(client->saveSet);
|
|
client->saveSet = (SaveSetElt *)NULL;
|
|
}
|
|
client->numSaved = numnow;
|
|
return(Success);
|
|
}
|
|
return(Success);
|
|
}
|
|
|
|
void
|
|
DeleteWindowFromAnySaveSet(WindowPtr pWin)
|
|
{
|
|
int i;
|
|
ClientPtr client;
|
|
|
|
for (i = 0; i< currentMaxClients; i++)
|
|
{
|
|
client = clients[i];
|
|
if (client && client->numSaved)
|
|
(void)AlterSaveSetForClient(client, pWin, SetModeDelete, FALSE, TRUE);
|
|
}
|
|
}
|
|
|
|
/* No-op Don't Do Anything : sometimes we need to be able to call a procedure
|
|
* that doesn't do anything. For example, on screen with only static
|
|
* colormaps, if someone calls install colormap, it's easier to have a dummy
|
|
* procedure to call than to check if there's a procedure
|
|
*/
|
|
_X_EXPORT void
|
|
NoopDDA(void)
|
|
{
|
|
}
|
|
|
|
typedef struct _BlockHandler {
|
|
BlockHandlerProcPtr BlockHandler;
|
|
WakeupHandlerProcPtr WakeupHandler;
|
|
pointer blockData;
|
|
Bool deleted;
|
|
} BlockHandlerRec, *BlockHandlerPtr;
|
|
|
|
static BlockHandlerPtr handlers;
|
|
static int numHandlers;
|
|
static int sizeHandlers;
|
|
static Bool inHandler;
|
|
static Bool handlerDeleted;
|
|
|
|
/**
|
|
*
|
|
* \param pTimeout DIX doesn't want to know how OS represents time
|
|
* \param pReadMask nor how it represents the det of descriptors
|
|
*/
|
|
void
|
|
BlockHandler(pointer pTimeout, pointer pReadmask)
|
|
{
|
|
int i, j;
|
|
|
|
++inHandler;
|
|
for (i = 0; i < screenInfo.numScreens; i++)
|
|
(* screenInfo.screens[i]->BlockHandler)(i,
|
|
screenInfo.screens[i]->blockData,
|
|
pTimeout, pReadmask);
|
|
for (i = 0; i < numHandlers; i++)
|
|
(*handlers[i].BlockHandler) (handlers[i].blockData,
|
|
pTimeout, pReadmask);
|
|
if (handlerDeleted)
|
|
{
|
|
for (i = 0; i < numHandlers;)
|
|
if (handlers[i].deleted)
|
|
{
|
|
for (j = i; j < numHandlers - 1; j++)
|
|
handlers[j] = handlers[j+1];
|
|
numHandlers--;
|
|
}
|
|
else
|
|
i++;
|
|
handlerDeleted = FALSE;
|
|
}
|
|
--inHandler;
|
|
}
|
|
|
|
/**
|
|
*
|
|
* \param result 32 bits of undefined result from the wait
|
|
* \param pReadmask the resulting descriptor mask
|
|
*/
|
|
void
|
|
WakeupHandler(int result, pointer pReadmask)
|
|
{
|
|
int i, j;
|
|
|
|
++inHandler;
|
|
for (i = numHandlers - 1; i >= 0; i--)
|
|
(*handlers[i].WakeupHandler) (handlers[i].blockData,
|
|
result, pReadmask);
|
|
for (i = 0; i < screenInfo.numScreens; i++)
|
|
(* screenInfo.screens[i]->WakeupHandler)(i,
|
|
screenInfo.screens[i]->wakeupData,
|
|
result, pReadmask);
|
|
if (handlerDeleted)
|
|
{
|
|
for (i = 0; i < numHandlers;)
|
|
if (handlers[i].deleted)
|
|
{
|
|
for (j = i; j < numHandlers - 1; j++)
|
|
handlers[j] = handlers[j+1];
|
|
numHandlers--;
|
|
}
|
|
else
|
|
i++;
|
|
handlerDeleted = FALSE;
|
|
}
|
|
--inHandler;
|
|
}
|
|
|
|
/**
|
|
* Reentrant with BlockHandler and WakeupHandler, except wakeup won't
|
|
* get called until next time
|
|
*/
|
|
_X_EXPORT Bool
|
|
RegisterBlockAndWakeupHandlers (BlockHandlerProcPtr blockHandler,
|
|
WakeupHandlerProcPtr wakeupHandler,
|
|
pointer blockData)
|
|
{
|
|
BlockHandlerPtr new;
|
|
|
|
if (numHandlers >= sizeHandlers)
|
|
{
|
|
new = (BlockHandlerPtr) xrealloc (handlers, (numHandlers + 1) *
|
|
sizeof (BlockHandlerRec));
|
|
if (!new)
|
|
return FALSE;
|
|
handlers = new;
|
|
sizeHandlers = numHandlers + 1;
|
|
}
|
|
handlers[numHandlers].BlockHandler = blockHandler;
|
|
handlers[numHandlers].WakeupHandler = wakeupHandler;
|
|
handlers[numHandlers].blockData = blockData;
|
|
handlers[numHandlers].deleted = FALSE;
|
|
numHandlers = numHandlers + 1;
|
|
return TRUE;
|
|
}
|
|
|
|
_X_EXPORT void
|
|
RemoveBlockAndWakeupHandlers (BlockHandlerProcPtr blockHandler,
|
|
WakeupHandlerProcPtr wakeupHandler,
|
|
pointer blockData)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < numHandlers; i++)
|
|
if (handlers[i].BlockHandler == blockHandler &&
|
|
handlers[i].WakeupHandler == wakeupHandler &&
|
|
handlers[i].blockData == blockData)
|
|
{
|
|
if (inHandler)
|
|
{
|
|
handlerDeleted = TRUE;
|
|
handlers[i].deleted = TRUE;
|
|
}
|
|
else
|
|
{
|
|
for (; i < numHandlers - 1; i++)
|
|
handlers[i] = handlers[i+1];
|
|
numHandlers--;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
void
|
|
InitBlockAndWakeupHandlers (void)
|
|
{
|
|
xfree (handlers);
|
|
handlers = (BlockHandlerPtr) 0;
|
|
numHandlers = 0;
|
|
sizeHandlers = 0;
|
|
}
|
|
|
|
/*
|
|
* A general work queue. Perform some task before the server
|
|
* sleeps for input.
|
|
*/
|
|
|
|
WorkQueuePtr workQueue;
|
|
static WorkQueuePtr *workQueueLast = &workQueue;
|
|
|
|
void
|
|
ProcessWorkQueue(void)
|
|
{
|
|
WorkQueuePtr q, *p;
|
|
|
|
p = &workQueue;
|
|
/*
|
|
* Scan the work queue once, calling each function. Those
|
|
* which return TRUE are removed from the queue, otherwise
|
|
* they will be called again. This must be reentrant with
|
|
* QueueWorkProc.
|
|
*/
|
|
while ((q = *p))
|
|
{
|
|
if ((*q->function) (q->client, q->closure))
|
|
{
|
|
/* remove q from the list */
|
|
*p = q->next; /* don't fetch until after func called */
|
|
xfree (q);
|
|
}
|
|
else
|
|
{
|
|
p = &q->next; /* don't fetch until after func called */
|
|
}
|
|
}
|
|
workQueueLast = p;
|
|
}
|
|
|
|
void
|
|
ProcessWorkQueueZombies(void)
|
|
{
|
|
WorkQueuePtr q, *p;
|
|
|
|
p = &workQueue;
|
|
while ((q = *p))
|
|
{
|
|
if (q->client && q->client->clientGone)
|
|
{
|
|
(void) (*q->function) (q->client, q->closure);
|
|
/* remove q from the list */
|
|
*p = q->next; /* don't fetch until after func called */
|
|
xfree (q);
|
|
}
|
|
else
|
|
{
|
|
p = &q->next; /* don't fetch until after func called */
|
|
}
|
|
}
|
|
workQueueLast = p;
|
|
}
|
|
|
|
_X_EXPORT Bool
|
|
QueueWorkProc (
|
|
Bool (*function)(ClientPtr /* pClient */, pointer /* closure */),
|
|
ClientPtr client, pointer closure)
|
|
{
|
|
WorkQueuePtr q;
|
|
|
|
q = (WorkQueuePtr) xalloc (sizeof *q);
|
|
if (!q)
|
|
return FALSE;
|
|
q->function = function;
|
|
q->client = client;
|
|
q->closure = closure;
|
|
q->next = NULL;
|
|
*workQueueLast = q;
|
|
workQueueLast = &q->next;
|
|
return TRUE;
|
|
}
|
|
|
|
/*
|
|
* Manage a queue of sleeping clients, awakening them
|
|
* when requested, by using the OS functions IgnoreClient
|
|
* and AttendClient. Note that this *ignores* the troubles
|
|
* with request data interleaving itself with events, but
|
|
* we'll leave that until a later time.
|
|
*/
|
|
|
|
typedef struct _SleepQueue {
|
|
struct _SleepQueue *next;
|
|
ClientPtr client;
|
|
ClientSleepProcPtr function;
|
|
pointer closure;
|
|
} SleepQueueRec, *SleepQueuePtr;
|
|
|
|
static SleepQueuePtr sleepQueue = NULL;
|
|
|
|
_X_EXPORT Bool
|
|
ClientSleep (ClientPtr client, ClientSleepProcPtr function, pointer closure)
|
|
{
|
|
SleepQueuePtr q;
|
|
|
|
q = (SleepQueuePtr) xalloc (sizeof *q);
|
|
if (!q)
|
|
return FALSE;
|
|
|
|
IgnoreClient (client);
|
|
q->next = sleepQueue;
|
|
q->client = client;
|
|
q->function = function;
|
|
q->closure = closure;
|
|
sleepQueue = q;
|
|
return TRUE;
|
|
}
|
|
|
|
Bool
|
|
ClientSignal (ClientPtr client)
|
|
{
|
|
SleepQueuePtr q;
|
|
|
|
for (q = sleepQueue; q; q = q->next)
|
|
if (q->client == client)
|
|
{
|
|
return QueueWorkProc (q->function, q->client, q->closure);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
_X_EXPORT void
|
|
ClientWakeup (ClientPtr client)
|
|
{
|
|
SleepQueuePtr q, *prev;
|
|
|
|
prev = &sleepQueue;
|
|
while ( (q = *prev) )
|
|
{
|
|
if (q->client == client)
|
|
{
|
|
*prev = q->next;
|
|
xfree (q);
|
|
if (client->clientGone)
|
|
/* Oops -- new zombie cleanup code ensures this only
|
|
* happens from inside CloseDownClient; don't want to
|
|
* recurse here...
|
|
*/
|
|
/* CloseDownClient(client) */;
|
|
else
|
|
AttendClient (client);
|
|
break;
|
|
}
|
|
prev = &q->next;
|
|
}
|
|
}
|
|
|
|
Bool
|
|
ClientIsAsleep (ClientPtr client)
|
|
{
|
|
SleepQueuePtr q;
|
|
|
|
for (q = sleepQueue; q; q = q->next)
|
|
if (q->client == client)
|
|
return TRUE;
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
* Generic Callback Manager
|
|
*/
|
|
|
|
/* ===== Private Procedures ===== */
|
|
|
|
static int numCallbackListsToCleanup = 0;
|
|
static CallbackListPtr **listsToCleanup = NULL;
|
|
|
|
static Bool
|
|
_AddCallback(
|
|
CallbackListPtr *pcbl,
|
|
CallbackProcPtr callback,
|
|
pointer data)
|
|
{
|
|
CallbackPtr cbr;
|
|
|
|
cbr = (CallbackPtr) xalloc(sizeof(CallbackRec));
|
|
if (!cbr)
|
|
return FALSE;
|
|
cbr->proc = callback;
|
|
cbr->data = data;
|
|
cbr->next = (*pcbl)->list;
|
|
cbr->deleted = FALSE;
|
|
(*pcbl)->list = cbr;
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
_DeleteCallback(
|
|
CallbackListPtr *pcbl,
|
|
CallbackProcPtr callback,
|
|
pointer data)
|
|
{
|
|
CallbackListPtr cbl = *pcbl;
|
|
CallbackPtr cbr, pcbr;
|
|
|
|
for (pcbr = NULL, cbr = cbl->list;
|
|
cbr != NULL;
|
|
pcbr = cbr, cbr = cbr->next)
|
|
{
|
|
if ((cbr->proc == callback) && (cbr->data == data))
|
|
break;
|
|
}
|
|
if (cbr != NULL)
|
|
{
|
|
if (cbl->inCallback)
|
|
{
|
|
++(cbl->numDeleted);
|
|
cbr->deleted = TRUE;
|
|
}
|
|
else
|
|
{
|
|
if (pcbr == NULL)
|
|
cbl->list = cbr->next;
|
|
else
|
|
pcbr->next = cbr->next;
|
|
xfree(cbr);
|
|
}
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
_CallCallbacks(
|
|
CallbackListPtr *pcbl,
|
|
pointer call_data)
|
|
{
|
|
CallbackListPtr cbl = *pcbl;
|
|
CallbackPtr cbr, pcbr;
|
|
|
|
++(cbl->inCallback);
|
|
for (cbr = cbl->list; cbr != NULL; cbr = cbr->next)
|
|
{
|
|
(*(cbr->proc)) (pcbl, cbr->data, call_data);
|
|
}
|
|
--(cbl->inCallback);
|
|
|
|
if (cbl->inCallback) return;
|
|
|
|
/* Was the entire list marked for deletion? */
|
|
|
|
if (cbl->deleted)
|
|
{
|
|
DeleteCallbackList(pcbl);
|
|
return;
|
|
}
|
|
|
|
/* Were some individual callbacks on the list marked for deletion?
|
|
* If so, do the deletions.
|
|
*/
|
|
|
|
if (cbl->numDeleted)
|
|
{
|
|
for (pcbr = NULL, cbr = cbl->list; (cbr != NULL) && cbl->numDeleted; )
|
|
{
|
|
if (cbr->deleted)
|
|
{
|
|
if (pcbr)
|
|
{
|
|
cbr = cbr->next;
|
|
xfree(pcbr->next);
|
|
pcbr->next = cbr;
|
|
} else
|
|
{
|
|
cbr = cbr->next;
|
|
xfree(cbl->list);
|
|
cbl->list = cbr;
|
|
}
|
|
cbl->numDeleted--;
|
|
}
|
|
else /* this one wasn't deleted */
|
|
{
|
|
pcbr = cbr;
|
|
cbr = cbr->next;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
_DeleteCallbackList(
|
|
CallbackListPtr *pcbl)
|
|
{
|
|
CallbackListPtr cbl = *pcbl;
|
|
CallbackPtr cbr, nextcbr;
|
|
int i;
|
|
|
|
if (cbl->inCallback)
|
|
{
|
|
cbl->deleted = TRUE;
|
|
return;
|
|
}
|
|
|
|
for (i = 0; i < numCallbackListsToCleanup; i++)
|
|
{
|
|
if ((listsToCleanup[i] = pcbl) != 0)
|
|
{
|
|
listsToCleanup[i] = NULL;
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (cbr = cbl->list; cbr != NULL; cbr = nextcbr)
|
|
{
|
|
nextcbr = cbr->next;
|
|
xfree(cbr);
|
|
}
|
|
xfree(cbl);
|
|
*pcbl = NULL;
|
|
}
|
|
|
|
static CallbackFuncsRec default_cbfuncs =
|
|
{
|
|
_AddCallback,
|
|
_DeleteCallback,
|
|
_CallCallbacks,
|
|
_DeleteCallbackList
|
|
};
|
|
|
|
static Bool
|
|
CreateCallbackList(CallbackListPtr *pcbl, CallbackFuncsPtr cbfuncs)
|
|
{
|
|
CallbackListPtr cbl;
|
|
int i;
|
|
|
|
if (!pcbl) return FALSE;
|
|
cbl = (CallbackListPtr) xalloc(sizeof(CallbackListRec));
|
|
if (!cbl) return FALSE;
|
|
cbl->funcs = cbfuncs ? *cbfuncs : default_cbfuncs;
|
|
cbl->inCallback = 0;
|
|
cbl->deleted = FALSE;
|
|
cbl->numDeleted = 0;
|
|
cbl->list = NULL;
|
|
*pcbl = cbl;
|
|
|
|
for (i = 0; i < numCallbackListsToCleanup; i++)
|
|
{
|
|
if (!listsToCleanup[i])
|
|
{
|
|
listsToCleanup[i] = pcbl;
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
listsToCleanup = (CallbackListPtr **)xnfrealloc(listsToCleanup,
|
|
sizeof(CallbackListPtr *) * (numCallbackListsToCleanup+1));
|
|
listsToCleanup[numCallbackListsToCleanup] = pcbl;
|
|
numCallbackListsToCleanup++;
|
|
return TRUE;
|
|
}
|
|
|
|
/* ===== Public Procedures ===== */
|
|
|
|
_X_EXPORT Bool
|
|
AddCallback(CallbackListPtr *pcbl, CallbackProcPtr callback, pointer data)
|
|
{
|
|
if (!pcbl) return FALSE;
|
|
if (!*pcbl)
|
|
{ /* list hasn't been created yet; go create it */
|
|
if (!CreateCallbackList(pcbl, (CallbackFuncsPtr)NULL))
|
|
return FALSE;
|
|
}
|
|
return ((*(*pcbl)->funcs.AddCallback) (pcbl, callback, data));
|
|
}
|
|
|
|
_X_EXPORT Bool
|
|
DeleteCallback(CallbackListPtr *pcbl, CallbackProcPtr callback, pointer data)
|
|
{
|
|
if (!pcbl || !*pcbl) return FALSE;
|
|
return ((*(*pcbl)->funcs.DeleteCallback) (pcbl, callback, data));
|
|
}
|
|
|
|
void
|
|
CallCallbacks(CallbackListPtr *pcbl, pointer call_data)
|
|
{
|
|
if (!pcbl || !*pcbl) return;
|
|
(*(*pcbl)->funcs.CallCallbacks) (pcbl, call_data);
|
|
}
|
|
|
|
void
|
|
DeleteCallbackList(CallbackListPtr *pcbl)
|
|
{
|
|
if (!pcbl || !*pcbl) return;
|
|
(*(*pcbl)->funcs.DeleteCallbackList) (pcbl);
|
|
}
|
|
|
|
void
|
|
InitCallbackManager(void)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < numCallbackListsToCleanup; i++)
|
|
{
|
|
DeleteCallbackList(listsToCleanup[i]);
|
|
}
|
|
if (listsToCleanup) xfree(listsToCleanup);
|
|
|
|
numCallbackListsToCleanup = 0;
|
|
listsToCleanup = NULL;
|
|
}
|