/*
 * SGI FREE SOFTWARE LICENSE B (Version 2.0, Sept. 18, 2008)
 * Copyright (C) 1991-2000 Silicon Graphics, Inc. All Rights Reserved.
 *
 * Permission is hereby granted, free of charge, to any person obtaining a
 * copy of this software and associated documentation files (the "Software"),
 * to deal in the Software without restriction, including without limitation
 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
 * and/or sell copies of the Software, and to permit persons to whom the
 * Software is furnished to do so, subject to the following conditions:
 *
 * The above copyright notice including the dates of first publication and
 * either this permission notice or a reference to
 * http://oss.sgi.com/projects/FreeB/
 * 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
 * SILICON GRAPHICS, INC. 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 Silicon Graphics, Inc.
 * shall not be used in advertising or otherwise to promote the sale, use or
 * other dealings in this Software without prior written authorization from
 * Silicon Graphics, Inc.
 */

#ifdef HAVE_DMX_CONFIG_H
#include <dmx-config.h>
#endif

#include "dmx.h"
#include "dmxwindow.h"
#include "dmxpixmap.h"
#include "dmxfont.h"
#include "dmxsync.h"

#undef Xmalloc
#undef Xcalloc
#undef Xrealloc
#undef Xfree

#include "glxserver.h"
#include <GL/glxtokens.h>
#include "g_disptab.h"
#include <pixmapstr.h>
#include <windowstr.h>
#include "glxutil.h"
#include "glxext.h"
#include "unpack.h"

#include "GL/glxproto.h"
#include "glxvendor.h"
#include "glxvisuals.h"
#include "glxswap.h"

#ifdef PANORAMIX
#include "panoramiXsrv.h"
#endif

extern __GLXFBConfig **__glXFBConfigs;
extern int            __glXNumFBConfigs;

extern __GLXFBConfig *glxLookupFBConfig( GLXFBConfigID id );
extern __GLXFBConfig *glxLookupFBConfigByVID( VisualID vid );
extern __GLXFBConfig *glxLookupBackEndFBConfig( GLXFBConfigID id, int screen );
extern int glxIsExtensionSupported( char *ext );
extern int __glXGetFBConfigsSGIX(__GLXclientState *cl, GLbyte *pc);

#define BE_TO_CLIENT_ERROR(x) \
           ( (x) >= __glXerrorBase ? \
             (x) - dmxScreen->glxErrorBase + __glXerrorBase \
	     : (x) )

Display *GetBackEndDisplay( __GLXclientState *cl, int s )
{
   if (! cl->be_displays[s] ) {
      cl->be_displays[s] = XOpenDisplay( DisplayString(dmxScreens[s].beDisplay) );
   }
   return( cl->be_displays[s] );
}

/*
** Create a GL context with the given properties.
*/
static int CreateContext(__GLXclientState *cl, 
                         GLXContextID gcId,
			 VisualID vid, GLXFBConfigID fbconfigId,
			 int screen,
			 GLXContextID shareList,
			 int isDirect )
{
    ClientPtr client = cl->client;
    xGLXCreateContextReq *be_req;
    xGLXCreateNewContextReq *be_new_req;
    VisualPtr pVisual;
    ScreenPtr pScreen;
    __GLXcontext *glxc, *shareglxc;
    __GLXvisualConfig *pGlxVisual;
    __GLXscreenInfo *pGlxScreen;
    VisualID visual = vid;
    GLint i;
    int from_screen = screen;
    int to_screen = screen;
    DMXScreenInfo *dmxScreen;
    VisualID be_vid;
    GLXFBConfigID be_fbconfigId;
    int num_be_screens;
    Display *dpy;
    
    /*
    ** Check if screen exists.
    */
    if (screen >= screenInfo.numScreens) {
	client->errorValue = screen;
	return BadValue;
    }


#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    /*
     ** Find the display list space that we want to share.  
     **
     */
    if (shareList == None) {
       shareglxc = NULL;
    } else {
       shareglxc = (__GLXcontext *) LookupIDByType(shareList, __glXContextRes);
       if (!shareglxc) {
	  client->errorValue = shareList;
	  return __glXBadContext;
       }
    }

    /*
    ** Allocate memory for the new context
    */
    glxc = __glXCalloc(1, sizeof(__GLXcontext));
    if (!glxc) {
	return BadAlloc;
    }

    pScreen = screenInfo.screens[screen];
    pGlxScreen = &__glXActiveScreens[screen];

    if (fbconfigId != None) {
       glxc->pFBConfig = glxLookupFBConfig( fbconfigId );
       if (!glxc->pFBConfig) {
	  client->errorValue = fbconfigId;
	  __glXFree( glxc );
	  return BadValue;
       }
       visual = glxc->pFBConfig->associatedVisualId;
    }
    else {
       glxc->pFBConfig = NULL;
    }

    if (visual != None) {
       /*
	** Check if the visual ID is valid for this screen.
	*/
       pVisual = pScreen->visuals;
       for (i = 0; i < pScreen->numVisuals; i++, pVisual++) {
	  if (pVisual->vid == visual) {
	     break;
	  }
       }
       if (i == pScreen->numVisuals) {
	  client->errorValue = visual;
	  __glXFree( glxc );
	  return BadValue;
       }

       pGlxVisual = pGlxScreen->pGlxVisual;
       for (i = 0; i < pGlxScreen->numVisuals; i++, pGlxVisual++) {
	  if (pGlxVisual->vid == visual) {
	     break;
	  }
       }
       if (i == pGlxScreen->numVisuals) {
	  /*
	   ** Visual not support on this screen by this OpenGL implementation.
	   */
	  client->errorValue = visual;
	  __glXFree( glxc );
	  return BadValue;
       }

       if ( glxc->pFBConfig == NULL ) {
	   glxc->pFBConfig = glxLookupFBConfigByVID( visual );

	   if ( glxc->pFBConfig == NULL ) {
	      /*
               * visual does not have an FBConfig ???
	      client->errorValue = visual;
	      __glXFree( glxc );
	      return BadValue;
	       */
	   }
       }
    }
    else {
       pVisual = NULL;
       pGlxVisual = NULL;
    }

    glxc->pScreen = pScreen;
    glxc->pGlxScreen = pGlxScreen;
    glxc->pVisual = pVisual;
    glxc->pGlxVisual = pGlxVisual;

    /*
     * allocate memory for back-end servers info
     */
    num_be_screens = to_screen - from_screen + 1;
    glxc->real_ids = (XID *)__glXMalloc(sizeof(XID) * num_be_screens);
    if (!glxc->real_ids) {
	return BadAlloc;
    }
    glxc->real_vids = (XID *)__glXMalloc(sizeof(XID) * num_be_screens);
    if (!glxc->real_vids) {
	return BadAlloc;
    }

    for (screen = from_screen; screen <= to_screen; screen++) {
       int sent = 0;
       pScreen = screenInfo.screens[screen];
       pGlxScreen = &__glXActiveScreens[screen];
       dmxScreen = &dmxScreens[screen];

       if (glxc->pFBConfig) {
	  __GLXFBConfig *beFBConfig = glxLookupBackEndFBConfig( glxc->pFBConfig->id, 
		                                                screen );
	  be_fbconfigId = beFBConfig->id;
       }

       if (pGlxVisual) {

	  be_vid = glxMatchGLXVisualInConfigList( pGlxVisual,
	                                    dmxScreen->glxVisuals,
					    dmxScreen->numGlxVisuals  );

	  if (!be_vid) {
	     /* visual is not supported on the back-end server */
	     __glXFree( glxc->real_ids );
	     __glXFree( glxc->real_vids );
	     __glXFree( glxc );
	     return BadValue;
	  }
       }

       glxc->real_ids[screen-from_screen] = XAllocID(GetBackEndDisplay(cl,screen));

       /* send the create context request to the back-end server */
       dpy = GetBackEndDisplay(cl,screen);
       if (glxc->pFBConfig) {
	     /*Since for a certain visual both RGB and COLOR INDEX
	      *can be on then the only parmeter to choose the renderType
	      * should be the class of the colormap since all 4 first 
	      * classes does not support RGB mode only COLOR INDEX ,
	      * and so TrueColor and DirectColor does not support COLOR INDEX*/
	     int renderType =  glxc->pFBConfig->renderType;  
	     if ( pVisual ) {
		 switch ( pVisual->class ){
		     case PseudoColor:
                     case StaticColor:
                     case GrayScale:
                     case StaticGray:
                         renderType = GLX_COLOR_INDEX_TYPE;
                         break;
                     case TrueColor:
                     case DirectColor:
                     default:
                         renderType = GLX_RGBA_TYPE;
                     break;
		 }
	     }
	  if ( __GLX_IS_VERSION_SUPPORTED(1,3) ) {
	     LockDisplay(dpy);
	     GetReq(GLXCreateNewContext,be_new_req);
	     be_new_req->reqType = dmxScreen->glxMajorOpcode;
	     be_new_req->glxCode = X_GLXCreateNewContext;
	     be_new_req->context = (unsigned int)glxc->real_ids[screen-from_screen];
	     be_new_req->fbconfig = (unsigned int)be_fbconfigId;
	     be_new_req->screen = DefaultScreen(dpy);
	     be_new_req->renderType = renderType;  

	     be_new_req->shareList = (shareglxc ? shareglxc->real_ids[screen-from_screen] : 0);
	     be_new_req->isDirect = 0;
	     UnlockDisplay(dpy);
	     glxc->real_vids[screen-from_screen] = be_fbconfigId;
	     sent = 1;
	  }
	  else if (glxIsExtensionSupported("GLX_SGIX_fbconfig")) {

	     xGLXCreateContextWithConfigSGIXReq *ext_req;
	     xGLXVendorPrivateReq *vpreq;
	     LockDisplay(dpy);
	     GetReqExtra(GLXVendorPrivate,
		         sz_xGLXCreateContextWithConfigSGIXReq - sz_xGLXVendorPrivateReq,
			vpreq);
	     ext_req = (xGLXCreateContextWithConfigSGIXReq *)vpreq;
	     ext_req->reqType = dmxScreen->glxMajorOpcode;
	     ext_req->glxCode = X_GLXVendorPrivate;
	     ext_req->vendorCode = X_GLXvop_CreateContextWithConfigSGIX;
	     ext_req->context = (unsigned int)glxc->real_ids[screen-from_screen];
	     ext_req->fbconfig = (unsigned int)be_fbconfigId;
	     ext_req->screen = DefaultScreen(dpy);
	     ext_req->renderType = renderType;  
	     ext_req->shareList = (shareglxc ? shareglxc->real_ids[screen-from_screen] : 0);
	     ext_req->isDirect = 0;
	     UnlockDisplay(dpy);
	     glxc->real_vids[screen-from_screen] = be_fbconfigId;
	     sent = 1;
	  }
       }

       if (!sent) {
	  LockDisplay(dpy);
	  GetReq(GLXCreateContext,be_req);
	  be_req->reqType = dmxScreen->glxMajorOpcode;
	  be_req->glxCode = X_GLXCreateContext;
	  be_req->context = (unsigned int)glxc->real_ids[screen-from_screen];
	  be_req->visual = (unsigned int)be_vid;
	  be_req->screen = DefaultScreen(dpy);
	  be_req->shareList = (shareglxc ? shareglxc->real_ids[screen-from_screen] : 0);
	  be_req->isDirect = 0;
	  UnlockDisplay(dpy);
	  glxc->real_vids[screen-from_screen] = be_vid;
       }
       SyncHandle();

    }

    /*
    ** Register this context as a resource.
    */
    if (!AddResource(gcId, __glXContextRes, (pointer)glxc)) {
       __glXFree( glxc->real_ids );
       __glXFree( glxc->real_vids );
       __glXFree( glxc );
	client->errorValue = gcId;
	return BadAlloc;
    }
    
    /*
    ** Finally, now that everything is working, setup the rest of the
    ** context.
    */
    glxc->id = gcId;
    glxc->share_id = shareList;
    glxc->idExists = GL_TRUE;
    glxc->isCurrent = GL_FALSE;

    return Success;
}

int __glXCreateContext(__GLXclientState *cl, GLbyte *pc)
{
    xGLXCreateContextReq *req = (xGLXCreateContextReq *) pc;

    return( CreateContext(cl, req->context,req->visual, None,
	                  req->screen, req->shareList, req->isDirect) );

}

int __glXCreateNewContext(__GLXclientState *cl, GLbyte *pc)
{
    xGLXCreateNewContextReq *req = (xGLXCreateNewContextReq *) pc;

    return( CreateContext(cl, req->context,None, req->fbconfig,
	                  req->screen, req->shareList, req->isDirect) );

}

int __glXCreateContextWithConfigSGIX(__GLXclientState *cl, GLbyte *pc)
{
    xGLXCreateContextWithConfigSGIXReq *req = (xGLXCreateContextWithConfigSGIXReq *) pc;

    return( CreateContext(cl, req->context, None, req->fbconfig,
	                  req->screen, req->shareList, req->isDirect) );

}

int __glXQueryMaxSwapBarriersSGIX(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXQueryMaxSwapBarriersSGIXReq *req =
	(xGLXQueryMaxSwapBarriersSGIXReq *)pc;
    xGLXQueryMaxSwapBarriersSGIXReply reply;

    reply.type = X_Reply;
    reply.sequenceNumber = client->sequence;
    reply.length = 0;
    reply.max = QueryMaxSwapBarriersSGIX(req->screen);

    if (client->swapped) {
	__glXSwapQueryMaxSwapBarriersSGIXReply(client, &reply);
    } else {
	WriteToClient(client, sz_xGLXQueryMaxSwapBarriersSGIXReply,
		      (char *)&reply);
    }

    return Success;
}

int __glXBindSwapBarrierSGIX(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXBindSwapBarrierSGIXReq *req = (xGLXBindSwapBarrierSGIXReq *)pc;
    DrawablePtr pDraw;
    __GLXpixmap *pGlxPixmap = NULL;
    __glXWindow *pGlxWindow = NULL;
    int rc;

    rc = dixLookupDrawable(&pDraw, req->drawable, client, 0, DixGetAttrAccess);
    if (rc != Success) {
	pGlxPixmap = (__GLXpixmap *) LookupIDByType(req->drawable,
						    __glXPixmapRes);
	if (pGlxPixmap) pDraw = pGlxPixmap->pDraw;
    }

    if (!pDraw && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
       pGlxWindow = (__glXWindow *) LookupIDByType(req->drawable,
						   __glXWindowRes);
       if (pGlxWindow) pDraw = pGlxWindow->pDraw;
    }

    if (!pDraw) {
       client->errorValue = req->drawable;
       return __glXBadDrawable;
    }

    return BindSwapBarrierSGIX(pDraw, req->barrier);
}

int __glXJoinSwapGroupSGIX(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXJoinSwapGroupSGIXReq *req = (xGLXJoinSwapGroupSGIXReq *)pc;
    DrawablePtr pDraw, pMember = NULL;
    __GLXpixmap *pGlxPixmap = NULL;
    __glXWindow *pGlxWindow = NULL;
    int rc;

    rc = dixLookupDrawable(&pDraw, req->drawable, client, 0, DixManageAccess);
    if (rc != Success) {
	pGlxPixmap = (__GLXpixmap *) LookupIDByType(req->drawable,
						    __glXPixmapRes);
	if (pGlxPixmap) pDraw = pGlxPixmap->pDraw;
    }

    if (!pDraw && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
       pGlxWindow = (__glXWindow *) LookupIDByType(req->drawable,
						   __glXWindowRes);
       if (pGlxWindow) pDraw = pGlxWindow->pDraw;
    }

    if (!pDraw) {
       client->errorValue = req->drawable;
       return __glXBadDrawable;
    }

    if (req->member != None) {
	rc = dixLookupDrawable(&pMember, req->member, client, 0,
			       DixGetAttrAccess);
	if (rc != Success) {
	    pGlxPixmap = (__GLXpixmap *) LookupIDByType(req->member,
							__glXPixmapRes);
	    if (pGlxPixmap) pMember = pGlxPixmap->pDraw;
	}

	if (!pMember && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
	    pGlxWindow = (__glXWindow *) LookupIDByType(req->member,
							__glXWindowRes);
	    if (pGlxWindow) pMember = pGlxWindow->pDraw;
	}

	if (!pMember) {
	    client->errorValue = req->member;
	    return __glXBadDrawable;
	}
    }

    return JoinSwapGroupSGIX(pDraw, pMember);
}


/*
** Destroy a GL context as an X resource.
*/
int __glXDestroyContext(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXDestroyContextReq *req = (xGLXDestroyContextReq *) pc;
    xGLXDestroyContextReq *be_req;
    GLXContextID gcId = req->context;
    __GLXcontext *glxc;
    int from_screen = 0;
    int to_screen = 0;
    int s;
    
    glxc = (__GLXcontext *) LookupIDByType(gcId, __glXContextRes);
    if (glxc) {
	/*
	** Just free the resource; don't actually destroy the context,
	** because it might be in use.  The
	** destroy method will be called by the resource destruction routine
	** if necessary.
	*/
	FreeResourceByType(gcId, __glXContextRes, FALSE);

	from_screen = to_screen = glxc->pScreen->myNum;

    } else {
	client->errorValue = gcId;
	return __glXBadContext;
    }

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    /*
     * send DestroyContext request to all back-end servers
     */
    for (s=from_screen; s<=to_screen; s++) {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);

       LockDisplay(dpy);
       GetReq(GLXDestroyContext,be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXDestroyContext;
       be_req->context = glxc->real_ids[s-from_screen];
       UnlockDisplay(dpy);
       SyncHandle();
    }

    return Success;
}

/*****************************************************************************/

/*
** For each client, the server keeps a table of all the contexts that are
** current for that client (each thread of a client may have its own current
** context).  These routines add, change, and lookup contexts in the table.
*/

/*
** Add a current context, and return the tag that will be used to refer to it.
*/
static int AddCurrentContext(__GLXclientState *cl, __GLXcontext *glxc, DrawablePtr pDraw)
{
    int i;
    int num = cl->numCurrentContexts;
    __GLXcontext **table = cl->currentContexts;

    if (!glxc) return -1;
    
    /*
    ** Try to find an empty slot and use it.
    */
    for (i=0; i < num; i++) {
	if (!table[i]) {
	    table[i] = glxc;
	    return i+1;
	}
    }
    /*
    ** Didn't find a free slot, so we'll have to grow the table.
    */
    if (!num) {
	table = (__GLXcontext **) __glXMalloc(sizeof(__GLXcontext *));
	cl->currentDrawables = (DrawablePtr *) __glXMalloc(sizeof(DrawablePtr));
	cl->be_currentCTag = (GLXContextTag *) __glXMalloc(screenInfo.numScreens *sizeof(GLXContextTag));
    } else {
	table = (__GLXcontext **) __glXRealloc(table,
					   (num+1)*sizeof(__GLXcontext *));
	cl->currentDrawables = (DrawablePtr *) __glXRealloc(
	                                          cl->currentDrawables ,
						  (num+1)*sizeof(DrawablePtr));
	cl->be_currentCTag = (GLXContextTag *) __glXRealloc(cl->be_currentCTag,
	            (num+1)*screenInfo.numScreens*sizeof(GLXContextTag));
    }
    table[num] = glxc;
    cl->currentDrawables[num] = pDraw;
    cl->currentContexts = table;
    cl->numCurrentContexts++;

    memset(cl->be_currentCTag + num*screenInfo.numScreens, 0, 
	         screenInfo.numScreens * sizeof(GLXContextTag));

    return num+1;
}

/*
** Given a tag, change the current context for the corresponding entry.
*/
static void ChangeCurrentContext(__GLXclientState *cl, __GLXcontext *glxc,
				GLXContextTag tag)
{
    __GLXcontext **table = cl->currentContexts;
    table[tag-1] = glxc;
}

/*
** Given a tag, and back-end screen number, retrives the current back-end
** tag.
*/
int GetCurrentBackEndTag(__GLXclientState *cl, GLXContextTag tag, int s)
{
   if (tag >0) {
      return( cl->be_currentCTag[ (tag-1)*screenInfo.numScreens + s ] );
   }
   else {
      return( 0 );
   }
}

/*
** Given a tag, and back-end screen number, sets the current back-end
** tag.
*/
static void SetCurrentBackEndTag(__GLXclientState *cl, GLXContextTag tag, int s, GLXContextTag be_tag)
{
   if (tag >0) {
      cl->be_currentCTag[ (tag-1)*screenInfo.numScreens + s ] = be_tag;
   }
}

/*
** For this implementation we have chosen to simply use the index of the
** context's entry in the table as the context tag.  A tag must be greater
** than 0.
*/
__GLXcontext *__glXLookupContextByTag(__GLXclientState *cl, GLXContextTag tag)
{
    int num = cl->numCurrentContexts;

    if (tag < 1 || tag > num) {
	return 0;
    } else {
	return cl->currentContexts[tag-1];
    }
}

DrawablePtr __glXLookupDrawableByTag(__GLXclientState *cl, GLXContextTag tag)
{
    int num = cl->numCurrentContexts;

    if (tag < 1 || tag > num) {
	return 0;
    } else {
	return cl->currentDrawables[tag-1];
    }
}

/*****************************************************************************/

static void StopUsingContext(__GLXcontext *glxc)
{
    if (glxc) {
	if (glxc == __glXLastContext) {
	    /* Tell server GL library */
	    __glXLastContext = 0;
	}
	glxc->isCurrent = GL_FALSE;
	if (!glxc->idExists) {
	    __glXFreeContext(glxc);
	}
    }
}

static void StartUsingContext(__GLXclientState *cl, __GLXcontext *glxc)
{
    glxc->isCurrent = GL_TRUE;
}

/*****************************************************************************/
/*
** Make an OpenGL context and drawable current.
*/
static int MakeCurrent(__GLXclientState *cl,
                       GLXDrawable drawable,
                       GLXDrawable readdrawable,
		       GLXContextID context,
		       GLXContextTag oldContextTag)
{
    ClientPtr client = cl->client;
    DrawablePtr pDraw = NULL;
    DrawablePtr pReadDraw = NULL;
    xGLXMakeCurrentReadSGIReply new_reply;
    xGLXMakeCurrentReq *be_req;
    xGLXMakeCurrentReply be_reply;
    xGLXMakeContextCurrentReq *be_new_req;
    xGLXMakeContextCurrentReply be_new_reply;
    GLXDrawable drawId = drawable;
    GLXDrawable readId = readdrawable;
    GLXContextID contextId = context;
    __GLXpixmap *pGlxPixmap = 0;
    __GLXpixmap *pReadGlxPixmap = 0;
    __GLXcontext *glxc, *prevglxc;
    GLXContextTag tag = oldContextTag;
    WindowPtr pWin = NULL;
    WindowPtr pReadWin = NULL;
    __glXWindow *pGlxWindow = NULL;
    __glXWindow *pGlxReadWindow = NULL;
    __glXPbuffer *pGlxPbuffer = NULL;
    __glXPbuffer *pGlxReadPbuffer = NULL;
#ifdef PANORAMIX
    PanoramiXRes *pXinDraw = NULL;
    PanoramiXRes *pXinReadDraw = NULL;
#endif
    int from_screen = 0;
    int to_screen = 0;
    int s, rc;

    /*
    ** If one is None and the other isn't, it's a bad match.
    */
    if ((drawId == None && contextId != None) ||
	(drawId != None && contextId == None)) {
	return BadMatch;
    }
    
    /*
    ** Lookup old context.  If we have one, it must be in a usable state.
    */
    if (tag != 0) {
	prevglxc = __glXLookupContextByTag(cl, tag);
	if (!prevglxc) {
	    /*
	    ** Tag for previous context is invalid.
	    */
	    return __glXBadContextTag;
	}
    } else {
	prevglxc = 0;
    }

    /*
    ** Lookup new context.  It must not be current for someone else.
    */
    if (contextId != None) {
	glxc = (__GLXcontext *) LookupIDByType(contextId, __glXContextRes);
	if (!glxc) {
	    client->errorValue = contextId;
	    return __glXBadContext;
	}
	if ((glxc != prevglxc) && glxc->isCurrent) {
	    /* Context is current to somebody else */
	    return BadAccess;
	}
    } else {
	/* Switching to no context.  Ignore new drawable. */
	glxc = 0;
    }

    if (drawId != None) {
	rc = dixLookupDrawable(&pDraw, drawId, client, 0, DixWriteAccess);
	if (rc == Success) {
	    if (pDraw->type == DRAWABLE_WINDOW) {
		/*
		** Drawable is an X Window.
		*/
	        VisualID vid;
		pWin = (WindowPtr)pDraw;
		vid = wVisual(pWin);

		new_reply.writeVid = (glxc->pFBConfig ? glxc->pFBConfig->id : vid);
		new_reply.writeType = GLX_WINDOW_TYPE;

		/*
		** Check if window and context are similar.
		*/
		if ((vid != glxc->pVisual->vid) ||
		    (pWin->drawable.pScreen != glxc->pScreen)) {
		    client->errorValue = drawId;
		    return BadMatch;
		}

		from_screen = to_screen = pWin->drawable.pScreen->myNum;

	    } else {
		/*
		** An X Pixmap is not allowed as a parameter (a GLX Pixmap
		** is, but it must first be created with glxCreateGLXPixmap).
		*/
		client->errorValue = drawId;
		return __glXBadDrawable;
	    }
	}

        if (!pDraw) {
	    pGlxPixmap = (__GLXpixmap *) LookupIDByType(drawId,
							__glXPixmapRes);
	    if (pGlxPixmap) {
		/*
		** Check if pixmap and context are similar.
		*/
		if (pGlxPixmap->pScreen != glxc->pScreen ||
		    pGlxPixmap->pGlxVisual != glxc->pGlxVisual) {
		    client->errorValue = drawId;
		    return BadMatch;
		}
		pDraw = pGlxPixmap->pDraw;

		new_reply.writeVid = (glxc->pFBConfig ? glxc->pFBConfig->id : 
		                      pGlxPixmap->pGlxVisual->vid);

		new_reply.writeType = GLX_PIXMAP_TYPE;

		from_screen = to_screen = pGlxPixmap->pScreen->myNum;

	    }
	}

	if (!pDraw && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
	   pGlxWindow = (__glXWindow *) LookupIDByType(drawId, __glXWindowRes);
            if (pGlxWindow) {
                /*
                ** Drawable is a GLXWindow.
                **
                ** Check if GLX window and context are similar.
                */
                if (pGlxWindow->pScreen != glxc->pScreen ||
                    pGlxWindow->pGlxFBConfig != glxc->pFBConfig) {
                    client->errorValue = drawId;
                    return BadMatch;
                }

                pDraw = pGlxWindow->pDraw;
                new_reply.writeVid = pGlxWindow->pGlxFBConfig->id;
                new_reply.writeType = GLX_GLXWINDOW_TYPE;
            }

	}

	if (!pDraw && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
	   pGlxPbuffer = (__glXPbuffer *)LookupIDByType(drawId, __glXPbufferRes);
	   if (pGlxPbuffer) {
                if (pGlxPbuffer->pScreen != glxc->pScreen ||
                    pGlxPbuffer->pFBConfig != glxc->pFBConfig) {
                    client->errorValue = drawId;
                    return BadMatch;
                }

		pDraw = (DrawablePtr)pGlxPbuffer;
                new_reply.writeVid = pGlxPbuffer->pFBConfig->id;
                new_reply.writeType = GLX_PBUFFER_TYPE;
	   }
	}

	if (!pDraw) {
	   /*
    	    ** Drawable is not a Window , GLXWindow or a GLXPixmap.
	    */
	   client->errorValue = drawId;
	   return __glXBadDrawable;
	}

    } else {
	pDraw = 0;
    }

    if (readId != None && readId != drawId ) {
	rc = dixLookupDrawable(&pReadDraw, readId, client, 0, DixReadAccess);
	if (rc == Success) {
	    if (pReadDraw->type == DRAWABLE_WINDOW) {
		/*
		** Drawable is an X Window.
		*/
	        VisualID vid;
		pReadWin = (WindowPtr)pDraw;
		vid = wVisual(pReadWin);

		new_reply.readVid = (glxc->pFBConfig ? glxc->pFBConfig->id : vid);
		new_reply.readType = GLX_WINDOW_TYPE;

		/*
		** Check if window and context are similar.
		*/
		if ((vid != glxc->pVisual->vid) ||
		    (pReadWin->drawable.pScreen != glxc->pScreen)) {
		    client->errorValue = readId;
		    return BadMatch;
		}

	    } else {

		/*
		** An X Pixmap is not allowed as a parameter (a GLX Pixmap
		** is, but it must first be created with glxCreateGLXPixmap).
		*/
		client->errorValue = readId;
		return __glXBadDrawable;
	    }
	}

	if (!pReadDraw) {
	    pReadGlxPixmap = (__GLXpixmap *) LookupIDByType(readId,
							__glXPixmapRes);
	    if (pReadGlxPixmap) {
		/*
		** Check if pixmap and context are similar.
		*/
		if (pReadGlxPixmap->pScreen != glxc->pScreen ||
		    pReadGlxPixmap->pGlxVisual != glxc->pGlxVisual) {
		    client->errorValue = readId;
		    return BadMatch;
		}
		pReadDraw = pReadGlxPixmap->pDraw;

		new_reply.readVid = (glxc->pFBConfig ? glxc->pFBConfig->id :
		                     pReadGlxPixmap->pGlxVisual->vid );
		new_reply.readType = GLX_PIXMAP_TYPE;

	    }
	}

	if (!pReadDraw && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
	   pGlxReadWindow = (__glXWindow *)
                                LookupIDByType(readId, __glXWindowRes);
            if (pGlxReadWindow) {
                /*
                ** Drawable is a GLXWindow.
                **
                ** Check if GLX window and context are similar.
                */
                if (pGlxReadWindow->pScreen != glxc->pScreen ||
                    pGlxReadWindow->pGlxFBConfig != glxc->pFBConfig) {
                    client->errorValue = readId;
                    return BadMatch;
                }

                pReadDraw = pGlxReadWindow->pDraw;
                new_reply.readVid = pGlxReadWindow->pGlxFBConfig->id;
                new_reply.readType = GLX_GLXWINDOW_TYPE;
            }
	}

	if (!pReadDraw && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
	   pGlxReadPbuffer = (__glXPbuffer *)LookupIDByType(readId, __glXPbufferRes);
	   if (pGlxReadPbuffer) {
                if (pGlxReadPbuffer->pScreen != glxc->pScreen ||
                    pGlxReadPbuffer->pFBConfig != glxc->pFBConfig) {
                    client->errorValue = drawId;
                    return BadMatch;
                }

		pReadDraw = (DrawablePtr)pGlxReadPbuffer;
                new_reply.readVid = pGlxReadPbuffer->pFBConfig->id;
                new_reply.readType = GLX_PBUFFER_TYPE;
	   }
	}

	if (!pReadDraw) {
	   /*
    	    ** Drawable is neither a Window nor a GLXPixmap.
	    */
	   client->errorValue = readId;
	   return __glXBadDrawable;
	}

    } else {
	pReadDraw = pDraw;
	pReadGlxPixmap = pGlxPixmap;
	pReadWin = pWin;
	new_reply.readVid = new_reply.writeVid;
	new_reply.readType = new_reply.writeType;
    }

    if (prevglxc) {

	if (prevglxc->pGlxPixmap) {
	    /*
	    ** The previous drawable was a glx pixmap, release it.
	    */
	    prevglxc->pGlxPixmap->refcnt--;
	    __glXFreeGLXPixmap( prevglxc->pGlxPixmap );
	    prevglxc->pGlxPixmap = 0;
	}

	if (prevglxc->pGlxReadPixmap) {
	    /*
	    ** The previous drawable was a glx pixmap, release it.
	    */
	    prevglxc->pGlxReadPixmap->refcnt--;
	    __glXFreeGLXPixmap( prevglxc->pGlxReadPixmap );
	    prevglxc->pGlxReadPixmap = 0;
	}

	if (prevglxc->pGlxWindow) {
	    /*
	    ** The previous drawable was a glx window, release it.
	    */
	    prevglxc->pGlxWindow->refcnt--;
	    __glXFreeGLXWindow( prevglxc->pGlxWindow );
	    prevglxc->pGlxWindow = 0;   
	}

	if (prevglxc->pGlxReadWindow) {
	    /*
	    ** The previous drawable was a glx window, release it.
	    */
	    prevglxc->pGlxReadWindow->refcnt--;
	    __glXFreeGLXWindow( prevglxc->pGlxReadWindow );
	    prevglxc->pGlxReadWindow = 0;   
	}

	if (prevglxc->pGlxPbuffer) {
	    /*
	    ** The previous drawable was a glx Pbuffer, release it.
	    */
	    prevglxc->pGlxPbuffer->refcnt--;
	    __glXFreeGLXPbuffer( prevglxc->pGlxPbuffer );
	    prevglxc->pGlxPbuffer = 0;   
	}

	if (prevglxc->pGlxReadPbuffer) {
	    /*
	    ** The previous drawable was a glx Pbuffer, release it.
	    */
	    prevglxc->pGlxReadPbuffer->refcnt--;
	    __glXFreeGLXPbuffer( prevglxc->pGlxReadPbuffer );
	    prevglxc->pGlxReadPbuffer = 0;   
	}

	ChangeCurrentContext(cl, glxc, tag);
	ChangeCurrentContext(cl, glxc, tag);
	StopUsingContext(prevglxc);
    } else {
	tag = AddCurrentContext(cl, glxc, pDraw);
    }
    if (glxc) {

       glxc->pGlxPixmap = pGlxPixmap;
       glxc->pGlxReadPixmap = pReadGlxPixmap;
       glxc->pGlxWindow = pGlxWindow;
       glxc->pGlxReadWindow = pGlxReadWindow;
       glxc->pGlxPbuffer = pGlxPbuffer;
       glxc->pGlxReadPbuffer = pGlxReadPbuffer;

	if (pGlxPixmap) {
	    pGlxPixmap->refcnt++;
	}

	if (pReadGlxPixmap) {
	    pReadGlxPixmap->refcnt++;
	}

	if (pGlxWindow) {
	   pGlxWindow->refcnt++;
	}

	if (pGlxReadWindow) {
	   pGlxReadWindow->refcnt++;
	}

	if (pGlxPbuffer) {
	   pGlxPbuffer->refcnt++;
	}

	if (pGlxReadPbuffer) {
	   pGlxReadPbuffer->refcnt++;
	}

	StartUsingContext(cl, glxc);
	new_reply.contextTag = tag;
    } else {
	new_reply.contextTag = 0;
    }
    new_reply.length = 0;
    new_reply.type = X_Reply;
    new_reply.sequenceNumber = client->sequence;

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;

       if (pDraw && new_reply.writeType != GLX_PBUFFER_TYPE) {
	  pXinDraw = (PanoramiXRes *)
	     SecurityLookupIDByClass(client, pDraw->id, XRC_DRAWABLE, DixReadAccess);
       }

       if (pReadDraw && pReadDraw != pDraw && 
	     new_reply.readType != GLX_PBUFFER_TYPE) {
	  pXinReadDraw = (PanoramiXRes *)
	     SecurityLookupIDByClass(client, pReadDraw->id, XRC_DRAWABLE, DixReadAccess);
       }
       else {
	  pXinReadDraw = pXinDraw;
       }
    }
#endif


    /* send the MakeCurrent request to all required
     * back-end servers.
     */
    for (s = from_screen; s<=to_screen; s++) {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);
       unsigned int be_draw = None;
       unsigned int be_read_draw = None;

       if (pGlxPixmap) {
	   be_draw = pGlxPixmap->be_xids[s];
       }
       else if (pGlxPbuffer) {
	  be_draw = pGlxPbuffer->be_xids[s];
       }
#ifdef PANORAMIX
       else if (pXinDraw) {
	  dixLookupWindow(&pWin, pXinDraw->info[s].id, client, DixReadAccess);
       }
#endif
       else if (pGlxWindow) {
	  pWin = (WindowPtr)pGlxWindow->pDraw;
       }

       if (pWin && be_draw == None) {
	   be_draw = (unsigned int)(DMX_GET_WINDOW_PRIV(pWin))->window;
	   if (!be_draw) {
	      /* it might be that the window did not created yet on the */
	      /* back-end server (lazy window creation option), force   */
	      /* creation of the window */
	      dmxCreateAndRealizeWindow( pWin, TRUE );
	      be_draw = (unsigned int)(DMX_GET_WINDOW_PRIV(pWin))->window;
	   }
       }

       /*
	* Before sending the MakeCurrent request - sync the
	* X11 connection to the back-end servers to make sure
	* that drawable is already created
	*/
       dmxSync( dmxScreen, 1 ); 

       if (drawId == readId) {
	  LockDisplay(dpy);
	  GetReq(GLXMakeCurrent, be_req);
	  be_req->reqType = dmxScreen->glxMajorOpcode;
	  be_req->glxCode = X_GLXMakeCurrent;
	  be_req->drawable = be_draw;
	  be_req->context = (unsigned int)(glxc ? glxc->real_ids[s-from_screen] : 0);
	  be_req->oldContextTag = GetCurrentBackEndTag(cl, tag, s);
	  if (!_XReply(dpy, (xReply *) &be_reply, 0, False)) {

	     /* The make current failed */
	     UnlockDisplay(dpy);
	     SyncHandle();
	     return( BE_TO_CLIENT_ERROR(dmxLastErrorEvent.error_code) );
	  }

	  UnlockDisplay(dpy);
       	  SyncHandle();

	  SetCurrentBackEndTag( cl, tag, s, be_reply.contextTag );
       }
       else {

	  if (pReadGlxPixmap) {
	     be_read_draw = pReadGlxPixmap->be_xids[s];
	  }
	  else if (pGlxReadPbuffer) {
	     be_read_draw = pGlxReadPbuffer->be_xids[s];
	  }
#ifdef PANORAMIX
	  else if (pXinReadDraw) {
	     dixLookupWindow(&pReadWin, pXinReadDraw->info[s].id, client,
			     DixReadAccess);
   	  }
#endif
	  else if (pGlxReadWindow) {
	     pReadWin = (WindowPtr)pGlxReadWindow->pDraw;
	  }

      	  if (pReadWin && be_read_draw == None) {
  	     be_read_draw = (unsigned int)(DMX_GET_WINDOW_PRIV(pReadWin))->window;
	   if (!be_read_draw) {
	      /* it might be that the window did not created yet on the */
	      /* back-end server (lazy window creation option), force   */
	      /* creation of the window */
	      dmxCreateAndRealizeWindow( pReadWin, TRUE );
	      be_read_draw = (unsigned int)(DMX_GET_WINDOW_PRIV(pReadWin))->window;
              dmxSync( dmxScreen, 1 ); 
	   }
     	  }

	  if ( __GLX_IS_VERSION_SUPPORTED(1,3) ) {
	     LockDisplay(dpy);
	     GetReq(GLXMakeContextCurrent, be_new_req);
	     be_new_req->reqType = dmxScreen->glxMajorOpcode;
	     be_new_req->glxCode = X_GLXMakeContextCurrent;
	     be_new_req->drawable = be_draw;
	     be_new_req->readdrawable = be_read_draw;
	     be_new_req->context = (unsigned int)(glxc ? glxc->real_ids[s-from_screen] : 0);
	     be_new_req->oldContextTag = GetCurrentBackEndTag(cl, tag, s);
	     if (!_XReply(dpy, (xReply *) &be_new_reply, 0, False)) {

		/* The make current failed */
		UnlockDisplay(dpy);
   		SyncHandle();
   		return( BE_TO_CLIENT_ERROR(dmxLastErrorEvent.error_code) );
   	     }

   	     UnlockDisplay(dpy);
      	     SyncHandle();

   	     SetCurrentBackEndTag( cl, tag, s, be_new_reply.contextTag );
	  }
	  else if (glxIsExtensionSupported("GLX_SGI_make_current_read")) {
	     xGLXMakeCurrentReadSGIReq *ext_req;
	     xGLXVendorPrivateWithReplyReq *vpreq;
             xGLXMakeCurrentReadSGIReply ext_reply;

	     LockDisplay(dpy);
	     GetReqExtra(GLXVendorPrivateWithReply, 
		         sz_xGLXMakeCurrentReadSGIReq - sz_xGLXVendorPrivateWithReplyReq,
		         vpreq);
	     ext_req = (xGLXMakeCurrentReadSGIReq *)vpreq;
	     ext_req->reqType = dmxScreen->glxMajorOpcode;
	     ext_req->glxCode = X_GLXVendorPrivateWithReply;
	     ext_req->vendorCode = X_GLXvop_MakeCurrentReadSGI;
	     ext_req->drawable = be_draw;
	     ext_req->readable = be_read_draw;
	     ext_req->context = (unsigned int)(glxc ? glxc->real_ids[s-from_screen] : 0);
	     ext_req->oldContextTag = GetCurrentBackEndTag(cl, tag, s);
	     if (!_XReply(dpy, (xReply *) &ext_reply, 0, False)) {

		/* The make current failed */
		UnlockDisplay(dpy);
   		SyncHandle();
   		return( BE_TO_CLIENT_ERROR(dmxLastErrorEvent.error_code) );
   	     }

   	     UnlockDisplay(dpy);
      	     SyncHandle();

   	     SetCurrentBackEndTag( cl, tag, s, ext_reply.contextTag );

	  }
	  else {
	     return BadMatch;
	  }
       }

       XFlush( dpy );
    }

    if (client->swapped) {
	__glXSwapMakeCurrentReply(client, &new_reply);
    } else {
	WriteToClient(client, sz_xGLXMakeContextCurrentReply, (char *)&new_reply);
    }

    return Success;
}

int __glXMakeCurrent(__GLXclientState *cl, GLbyte *pc)
{
    xGLXMakeCurrentReq *req = (xGLXMakeCurrentReq *) pc;

    return( MakeCurrent(cl, req->drawable, req->drawable,
	                req->context, req->oldContextTag ) );
}

int __glXMakeContextCurrent(__GLXclientState *cl, GLbyte *pc)
{
    xGLXMakeContextCurrentReq *req = (xGLXMakeContextCurrentReq *) pc;

    return( MakeCurrent(cl, req->drawable, req->readdrawable,
	                req->context, req->oldContextTag ) );
}

int __glXMakeCurrentReadSGI(__GLXclientState *cl, GLbyte *pc)
{
    xGLXMakeCurrentReadSGIReq *req = (xGLXMakeCurrentReadSGIReq *) pc;

    return( MakeCurrent(cl, req->drawable, req->readable,
	                req->context, req->oldContextTag ) );
}

int __glXIsDirect(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXIsDirectReq *req = (xGLXIsDirectReq *) pc;
    xGLXIsDirectReply reply;
    __GLXcontext *glxc;

    /*
    ** Find the GL context.
    */
    glxc = (__GLXcontext *) LookupIDByType(req->context, __glXContextRes);
    if (!glxc) {
	client->errorValue = req->context;
	return __glXBadContext;
    }

    reply.isDirect = 0;
    reply.length = 0;
    reply.type = X_Reply;
    reply.sequenceNumber = client->sequence;

    if (client->swapped) {
	__glXSwapIsDirectReply(client, &reply);
    } else {
	WriteToClient(client, sz_xGLXIsDirectReply, (char *)&reply);
    }

    return Success;
}

int __glXQueryVersion(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
/*    xGLXQueryVersionReq *req = (xGLXQueryVersionReq *) pc; */
    xGLXQueryVersionReply reply;

    /*
    ** Server should take into consideration the version numbers sent by the
    ** client if it wants to work with older clients; however, in this
    ** implementation the server just returns its version number.
    */
    reply.majorVersion = __glXVersionMajor;
    reply.minorVersion = __glXVersionMinor;
    reply.length = 0;
    reply.type = X_Reply;
    reply.sequenceNumber = client->sequence;

    if (client->swapped) {
	__glXSwapQueryVersionReply(client, &reply);
    } else {
	WriteToClient(client, sz_xGLXQueryVersionReply, (char *)&reply);
    }
    return Success;
}

int __glXWaitGL(__GLXclientState *cl, GLbyte *pc)
{
    xGLXWaitGLReq *req = (xGLXWaitGLReq *)pc;
    xGLXWaitGLReq *be_req = (xGLXWaitGLReq *)pc;
    int from_screen = 0;
    int to_screen = 0;
    int s;
    __GLXcontext *glxc = NULL;
    
    if (req->contextTag != 0) {
	glxc = __glXLookupContextByTag(cl, req->contextTag);
	if (glxc) {
	   from_screen = to_screen = glxc->pScreen->myNum;
	}
    }

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    for (s=from_screen; s<=to_screen; s++) {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);

       LockDisplay(dpy);
       GetReq(GLXWaitGL,be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXWaitGL;
       be_req->contextTag = (glxc ? GetCurrentBackEndTag(cl,req->contextTag,s) : 0);
       UnlockDisplay(dpy);
       SyncHandle();

       XSync(dpy, False);
    }
    
    return Success;
}

int __glXWaitX(__GLXclientState *cl, GLbyte *pc)
{
    xGLXWaitXReq *req = (xGLXWaitXReq *)pc;
    xGLXWaitXReq *be_req;
    int from_screen = 0;
    int to_screen = 0;
    int s;
    __GLXcontext *glxc = NULL;
    
    if (req->contextTag != 0) {
	glxc = __glXLookupContextByTag(cl, req->contextTag);
	if (glxc) {
	   from_screen = to_screen = glxc->pScreen->myNum;
	}
    }

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    for (s=from_screen; s<=to_screen; s++) {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);

       dmxSync( dmxScreen, 1 );

       LockDisplay(dpy);
       GetReq(GLXWaitX,be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXWaitX;
       be_req->contextTag = (glxc ? GetCurrentBackEndTag(cl,req->contextTag,s) : 0);
       UnlockDisplay(dpy);
       SyncHandle();

       XFlush( dpy );
    }

    return Success;
}

int __glXCopyContext(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXCopyContextReq *be_req;
    xGLXCopyContextReq *req = (xGLXCopyContextReq *) pc;
    GLXContextID source = req->source;
    GLXContextID dest = req->dest;
    GLXContextTag tag = req->contextTag;
    unsigned long mask = req->mask;
    __GLXcontext *src, *dst;
    int s;
    int from_screen = 0;
    int to_screen = 0;

    /*
    ** Check that each context exists.
    */
    src = (__GLXcontext *) LookupIDByType(source, __glXContextRes);
    if (!src) {
	client->errorValue = source;
	return __glXBadContext;
    }
    dst = (__GLXcontext *) LookupIDByType(dest, __glXContextRes);
    if (!dst) {
	client->errorValue = dest;
	return __glXBadContext;
    }

    /*
    ** They must be in the same address space, and same screen.
    */
    if (src->pGlxScreen != dst->pGlxScreen) {
	client->errorValue = source;
	return BadMatch;
    }

    /*
    ** The destination context must not be current for any client.
    */
    if (dst->isCurrent) {
	client->errorValue = dest;
	return BadAccess;
    }

    if (tag) {
	__GLXcontext *tagcx = __glXLookupContextByTag(cl, tag);
	
	if (!tagcx) {
	    return __glXBadContextTag;
	}
	if (tagcx != src) {
	    /*
	    ** This would be caused by a faulty implementation of the client
	    ** library.
	    */
	    return BadMatch;
	}
    }

    from_screen = to_screen = src->pScreen->myNum;

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    for (s=from_screen; s<=to_screen; s++) {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);

       LockDisplay(dpy);
       GetReq(GLXCopyContext,be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXCopyContext;
       be_req->source = (unsigned int)src->real_ids[s-from_screen];
       be_req->dest = (unsigned int)dst->real_ids[s-from_screen];
       be_req->mask = mask;
       be_req->contextTag = (tag ? GetCurrentBackEndTag(cl,req->contextTag,s) : 0);
       UnlockDisplay(dpy);
       SyncHandle();
    }

    return Success;
}

int __glXGetVisualConfigs(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXGetVisualConfigsReq *req = (xGLXGetVisualConfigsReq *) pc;
    xGLXGetVisualConfigsReply reply;
    __GLXscreenInfo *pGlxScreen;
    __GLXvisualConfig *pGlxVisual;
    CARD32 buf[__GLX_TOTAL_CONFIG];
    unsigned int screen;
    int i, p;

    screen = req->screen;
    if (screen > screenInfo.numScreens) {
	/* The client library must send a valid screen number. */
	client->errorValue = screen;
	return BadValue;
    }
    pGlxScreen = &__glXActiveScreens[screen];

    reply.numVisuals = pGlxScreen->numGLXVisuals;
    reply.numProps = __GLX_TOTAL_CONFIG;
    reply.length = (pGlxScreen->numGLXVisuals * __GLX_SIZE_CARD32 *
		    __GLX_TOTAL_CONFIG) >> 2;
    reply.type = X_Reply;
    reply.sequenceNumber = client->sequence;

    WriteToClient(client, sz_xGLXGetVisualConfigsReply, (char *)&reply);

    for (i=0; i < pGlxScreen->numVisuals; i++) {
	pGlxVisual = &pGlxScreen->pGlxVisual[i];
	if (!pGlxScreen->isGLXvis[i] || pGlxVisual->vid == 0) {
	    /* not a usable visual */
	    continue;
	}
	p = 0;
	buf[p++] = pGlxVisual->vid;
	buf[p++] = pGlxVisual->class;
	buf[p++] = pGlxVisual->rgba;

	buf[p++] = pGlxVisual->redSize;
	buf[p++] = pGlxVisual->greenSize;
	buf[p++] = pGlxVisual->blueSize;
	buf[p++] = pGlxVisual->alphaSize;
	buf[p++] = pGlxVisual->accumRedSize;
	buf[p++] = pGlxVisual->accumGreenSize;
	buf[p++] = pGlxVisual->accumBlueSize;
	buf[p++] = pGlxVisual->accumAlphaSize;

	buf[p++] = pGlxVisual->doubleBuffer;
	buf[p++] = pGlxVisual->stereo;

	buf[p++] = pGlxVisual->bufferSize;
	buf[p++] = pGlxVisual->depthSize;
	buf[p++] = pGlxVisual->stencilSize;
	buf[p++] = pGlxVisual->auxBuffers;
	buf[p++] = pGlxVisual->level;
	/* 
	** Add token/value pairs for extensions.
	*/
	buf[p++] = GLX_VISUAL_CAVEAT_EXT;
	buf[p++] = pGlxVisual->visualRating;
	buf[p++] = GLX_TRANSPARENT_TYPE_EXT;
	buf[p++] = pGlxVisual->transparentPixel;
	buf[p++] = GLX_TRANSPARENT_RED_VALUE_EXT;
	buf[p++] = pGlxVisual->transparentRed;
	buf[p++] = GLX_TRANSPARENT_GREEN_VALUE_EXT;
	buf[p++] = pGlxVisual->transparentGreen;
	buf[p++] = GLX_TRANSPARENT_BLUE_VALUE_EXT;
	buf[p++] = pGlxVisual->transparentBlue;
	buf[p++] = GLX_TRANSPARENT_ALPHA_VALUE_EXT;
	buf[p++] = pGlxVisual->transparentAlpha;
	buf[p++] = GLX_TRANSPARENT_INDEX_VALUE_EXT;
	buf[p++] = pGlxVisual->transparentIndex;
	buf[p++] = GLX_SAMPLES_SGIS;
	buf[p++] = pGlxVisual->multiSampleSize;
	buf[p++] = GLX_SAMPLE_BUFFERS_SGIS;
	buf[p++] = pGlxVisual->nMultiSampleBuffers;
	buf[p++] = GLX_VISUAL_SELECT_GROUP_SGIX;
	buf[p++] = pGlxVisual->visualSelectGroup;

	WriteToClient(client, __GLX_SIZE_CARD32 * __GLX_TOTAL_CONFIG, 
		(char *)buf);
    }
    return Success;
}

/*
** Create a GLX Pixmap from an X Pixmap.
*/
static int CreateGLXPixmap(__GLXclientState *cl,
                    VisualID visual, GLXFBConfigID fbconfigId,
		    int screenNum, XID pixmapId, XID glxpixmapId )
{
    ClientPtr client = cl->client;
    xGLXCreateGLXPixmapReq *be_req;
    xGLXCreatePixmapReq *be_new_req;
    DrawablePtr pDraw;
    ScreenPtr pScreen;
    VisualPtr pVisual;
    __GLXpixmap *pGlxPixmap;
    __GLXscreenInfo *pGlxScreen;
    __GLXvisualConfig *pGlxVisual;
    __GLXFBConfig *pFBConfig;
    int i, s, rc;
    int from_screen, to_screen;
#ifdef PANORAMIX
    PanoramiXRes *pXinDraw = NULL;
#endif

    rc = dixLookupDrawable(&pDraw, pixmapId, client, M_DRAWABLE_PIXMAP,
			   DixAddAccess);
    if (rc != Success)
	return rc;

    /*
    ** Check if screen of visual matches screen of pixmap.
    */
    pScreen = pDraw->pScreen;
    if (screenNum != pScreen->myNum) {
	return BadMatch;
    }

    if (fbconfigId == NULL && visual == NULL) {
	  return BadValue;
    }

    if (fbconfigId != None) {
       pFBConfig = glxLookupFBConfig( fbconfigId );
       if (!pFBConfig) {
	  client->errorValue = fbconfigId;
	  return BadValue;
       }
       visual = pFBConfig->associatedVisualId;
    }
    else {
       pFBConfig = NULL;
    }

    if (visual != None) {
       /*
	** Find the VisualRec for this visual.
	*/
       pVisual = pScreen->visuals;
       for (i=0; i < pScreen->numVisuals; i++, pVisual++) {
	  if (pVisual->vid == visual) {
	     break;
	  }
       }
       if (i == pScreen->numVisuals) {
	  client->errorValue = visual;
	  return BadValue;
       }
       /*
	** Check if depth of visual matches depth of pixmap.
	*/
       if (pVisual->nplanes != pDraw->depth) {
	  client->errorValue = visual;
	  return BadMatch;
       }

       /*
	** Get configuration of the visual.
	*/
       pGlxScreen = &__glXActiveScreens[screenNum];
       pGlxVisual = pGlxScreen->pGlxVisual;
       for (i = 0; i < pGlxScreen->numVisuals; i++, pGlxVisual++) {
	  if (pGlxVisual->vid == visual) {
	     break;
	  }
       }
       if (i == pGlxScreen->numVisuals) {
	  /*
	   ** Visual not support on this screen by this OpenGL implementation.
	   */
	  client->errorValue = visual;
	  return BadValue;
       }


       /* find the FBConfig for that visual (if any) */
       if ( pFBConfig == NULL ) {
	   pFBConfig = glxLookupFBConfigByVID( visual );

	   if ( pFBConfig == NULL ) {
	      /*
               * visual does not have an FBConfig ???
	      client->errorValue = visual;
	      return BadValue;
	       */
	   }
       }
    }
    else {
       pVisual = NULL;
       pGlxVisual = NULL;
    }

    pGlxPixmap = (__GLXpixmap *) __glXMalloc(sizeof(__GLXpixmap));
    if (!pGlxPixmap) {
	return BadAlloc;
    }
    pGlxPixmap->be_xids = (XID *) __glXMalloc(sizeof(XID) * screenInfo.numScreens);
    if (!pGlxPixmap->be_xids) {
        __glXFree( pGlxPixmap );
	return BadAlloc;
    }

    pGlxPixmap->pDraw = pDraw;
    pGlxPixmap->pGlxScreen = pGlxScreen;
    pGlxPixmap->pGlxVisual = pGlxVisual;
    pGlxPixmap->pFBConfig = pFBConfig;
    pGlxPixmap->pScreen = pScreen;
    pGlxPixmap->idExists = True;
    pGlxPixmap->refcnt = 0;

    /*
    ** Bump the ref count on the X pixmap so it won't disappear.
    */
    ((PixmapPtr) pDraw)->refcnt++;

    /*
     * send the request to the back-end server(s)
     */
    from_screen = to_screen = screenNum;
#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;

       pXinDraw = (PanoramiXRes *)
	  SecurityLookupIDByClass(client, pDraw->id, XRC_DRAWABLE, DixReadAccess);
    }
#endif

    for (s=from_screen; s<=to_screen; s++) {

       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);
       Pixmap  be_pixmap;
       DrawablePtr pRealDraw = pDraw;

#ifdef PANORAMIX
       if (pXinDraw) {
	   dixLookupDrawable(&pRealDraw, pXinDraw->info[s].id, client, 0,
			     DixAddAccess);
       }
#endif

       be_pixmap = (DMX_GET_PIXMAP_PRIV((PixmapPtr)pRealDraw))->pixmap;

       /* make sure pixmap already created on back-end */
       dmxSync( dmxScreen, 1 );

       if ( pFBConfig && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
       	  __GLXFBConfig *be_FBConfig = glxLookupBackEndFBConfig( pFBConfig->id, s );
	  
	  LockDisplay(dpy);
	  pGlxPixmap->be_xids[s] = XAllocID(dpy);
	  GetReq(GLXCreatePixmap,be_new_req);
	  be_new_req->reqType = dmxScreen->glxMajorOpcode;
	  be_new_req->glxCode = X_GLXCreatePixmap;
	  be_new_req->screen = DefaultScreen(dpy);
	  be_new_req->fbconfig = be_FBConfig->id;
	  be_new_req->pixmap = (unsigned int)be_pixmap;
	  be_new_req->glxpixmap = (unsigned int)pGlxPixmap->be_xids[s];
	  be_new_req->numAttribs = 0;
	  UnlockDisplay(dpy);
	  SyncHandle();
       }
       else if (pFBConfig && glxIsExtensionSupported("GLX_SGIX_fbconfig")) {
	  __GLXFBConfig *be_FBConfig = glxLookupBackEndFBConfig( pFBConfig->id, s );
	  xGLXCreateGLXPixmapWithConfigSGIXReq *ext_req;
	  xGLXVendorPrivateReq *vpreq;
	  
	  LockDisplay(dpy);
	  pGlxPixmap->be_xids[s] = XAllocID(dpy);
	  GetReqExtra(GLXVendorPrivate,
		      sz_xGLXCreateGLXPixmapWithConfigSGIXReq-sz_xGLXVendorPrivateReq,
		      vpreq);
	  ext_req = (xGLXCreateGLXPixmapWithConfigSGIXReq *)vpreq;
	  ext_req->reqType = dmxScreen->glxMajorOpcode;
	  ext_req->glxCode = X_GLXVendorPrivate;
	  ext_req->vendorCode = X_GLXvop_CreateGLXPixmapWithConfigSGIX;
	  ext_req->screen = DefaultScreen(dpy);
	  ext_req->fbconfig = be_FBConfig->id;
	  ext_req->pixmap = (unsigned int)be_pixmap;
	  ext_req->glxpixmap = (unsigned int)pGlxPixmap->be_xids[s];
	  UnlockDisplay(dpy);
	  SyncHandle();
       }
       else if (pGlxVisual) {
	  LockDisplay(dpy);
	  pGlxPixmap->be_xids[s] = XAllocID(dpy);
	  GetReq(GLXCreateGLXPixmap,be_req);
	  be_req->reqType = dmxScreen->glxMajorOpcode;
	  be_req->glxCode = X_GLXCreateGLXPixmap;
	  be_req->screen = DefaultScreen(dpy);
	  be_req->visual = (unsigned int)glxMatchGLXVisualInConfigList( 
	 	pGlxVisual,
		dmxScreen->glxVisuals,
		dmxScreen->numGlxVisuals );
   	  be_req->pixmap = (unsigned int)be_pixmap;
      	  be_req->glxpixmap = (unsigned int)pGlxPixmap->be_xids[s];
	  UnlockDisplay(dpy);
	  SyncHandle();
       }
       else {
	  client->errorValue = ( visual ? visual : fbconfigId );
          __glXFree( pGlxPixmap );
	  return BadValue;
       }

       XFlush( dpy );
    }

    if (!(AddResource(glxpixmapId, __glXPixmapRes, pGlxPixmap))) {
        __glXFree( pGlxPixmap );
	return BadAlloc;
    }

    return Success;
}

int __glXCreateGLXPixmap(__GLXclientState *cl, GLbyte *pc)
{
    xGLXCreateGLXPixmapReq *req = (xGLXCreateGLXPixmapReq *) pc;

    return( CreateGLXPixmap(cl, req->visual, None,
	                    req->screen, req->pixmap, req->glxpixmap) );
}

int __glXCreatePixmap(__GLXclientState *cl, GLbyte *pc)
{
    xGLXCreatePixmapReq *req = (xGLXCreatePixmapReq *) pc;

    return( CreateGLXPixmap(cl, None, req->fbconfig,
	                    req->screen, req->pixmap, req->glxpixmap) );
}

int __glXDestroyGLXPixmap(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXDestroyGLXPixmapReq *req = (xGLXDestroyGLXPixmapReq *) pc;
    XID glxpixmap = req->glxpixmap;
    __GLXpixmap *pGlxPixmap;
    int s;
    int from_screen, to_screen;

    /*
    ** Check if it's a valid GLX pixmap.
    */
    pGlxPixmap = (__GLXpixmap *)LookupIDByType(glxpixmap, __glXPixmapRes);
    if (!pGlxPixmap) {
	client->errorValue = glxpixmap;
	return __glXBadPixmap;
    }
    FreeResource(glxpixmap, FALSE);

    /*
     * destroy the pixmap on the back-end server(s).
     */
    from_screen = to_screen = pGlxPixmap->pDraw->pScreen->myNum;
#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    for (s=from_screen; s<=to_screen; s++) {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);

       /* make sure pixmap exist in back-end */
       dmxSync( dmxScreen, 1 );

       LockDisplay(dpy);
       GetReq(GLXDestroyGLXPixmap,req);
       req->reqType = dmxScreen->glxMajorOpcode;
       req->glxCode = X_GLXDestroyGLXPixmap;
       req->glxpixmap = (unsigned int)pGlxPixmap->be_xids[s];
       UnlockDisplay(dpy);
       SyncHandle();
    }

    
    return Success;
}

/*****************************************************************************/

/*
** NOTE: There is no portable implementation for swap buffers as of
** this time that is of value.  Consequently, this code must be
** implemented by somebody other than SGI.
*/
int __glXDoSwapBuffers(__GLXclientState *cl, XID drawId, GLXContextTag tag)
{
    ClientPtr client = cl->client;
    DrawablePtr pDraw;
    xGLXSwapBuffersReq *be_req;
    WindowPtr pWin = NULL;
    __GLXpixmap *pGlxPixmap = NULL;
    __GLXcontext *glxc = NULL;
#ifdef PANORAMIX
    PanoramiXRes *pXinDraw = NULL;
#endif
    __glXWindow *pGlxWindow = NULL;
    int from_screen = 0;
    int to_screen = 0;
    int s, rc;
    
    /*
    ** Check that the GLX drawable is valid.
    */
    rc = dixLookupDrawable(&pDraw, drawId, client, 0, DixWriteAccess);
    if (rc == Success) {
        from_screen = to_screen = pDraw->pScreen->myNum;

	if (pDraw->type == DRAWABLE_WINDOW) {
	    /*
	    ** Drawable is an X window.
	    */
	   pWin = (WindowPtr)pDraw;
	} else {
	    /*
	    ** Drawable is an X pixmap, which is not allowed.
	    */
	    client->errorValue = drawId;
	    return __glXBadDrawable;
	}
    } 

    if (!pDraw) {
	pGlxPixmap = (__GLXpixmap *) LookupIDByType(drawId,
						    __glXPixmapRes);
	if (pGlxPixmap) {
	    /*
	    ** Drawable is a GLX pixmap.
	    */
	   pDraw = pGlxPixmap->pDraw;
	   from_screen = to_screen = pGlxPixmap->pScreen->myNum;
	}
    }

    if (!pDraw && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
       pGlxWindow = (__glXWindow *) LookupIDByType(drawId, __glXWindowRes);
       if (pGlxWindow) {
	  /*
	   ** Drawable is a GLXWindow.
	   */
	  pDraw = pGlxWindow->pDraw;
	  from_screen = to_screen = pGlxWindow->pScreen->myNum;
       }
    }

    if (!pDraw) {
       /*
	** Drawable is neither a X window nor a GLX pixmap.
	*/
       client->errorValue = drawId;
       return __glXBadDrawable;
    }

    if (tag) {
	glxc = __glXLookupContextByTag(cl, tag);
	if (!glxc) {
	    return __glXBadContextTag;
	}
    }

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
       pXinDraw = (PanoramiXRes *)
        SecurityLookupIDByClass(client, pDraw->id, XRC_DRAWABLE, DixReadAccess);
    }
#endif

    /* If requested, send a glFinish to all back-end servers before swapping. */
    if (dmxGLXFinishSwap) {
	for (s=from_screen; s<=to_screen; s++) {
	    Display *dpy = GetBackEndDisplay(cl,s);
	    DMXScreenInfo *dmxScreen = &dmxScreens[s];
	    xGLXSingleReq *finishReq;
	    xGLXSingleReply reply;

#define X_GLXSingle 0    /* needed by GetReq below */

	    LockDisplay(dpy);
	    GetReq(GLXSingle,finishReq);
	    finishReq->reqType = dmxScreen->glxMajorOpcode;
	    finishReq->glxCode = X_GLsop_Finish;
	    finishReq->contextTag = (tag ? GetCurrentBackEndTag(cl,tag,s) : 0);
	    (void) _XReply(dpy, (xReply*) &reply, 0, False);
	    UnlockDisplay(dpy);
	    SyncHandle();
	}
    }

    /* If requested, send an XSync to all back-end servers before swapping. */
    if (dmxGLXSyncSwap) {
	for (s=from_screen; s<=to_screen; s++)
	    XSync(GetBackEndDisplay(cl,s), False);
    }


    /* send the SwapBuffers request to all back-end servers */

    for (s=from_screen; s<=to_screen; s++) {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);
       unsigned int be_draw = 0;

       if (pGlxPixmap) {
	  be_draw = (unsigned int)pGlxPixmap->be_xids[s];
       }
#ifdef PANORAMIX
       else if (pXinDraw) {
	  dixLookupWindow(&pWin, pXinDraw->info[s].id, client, DixReadAccess);
       }
#endif
       else if (pGlxWindow) {
	  pWin = (WindowPtr)pGlxWindow->pDraw;
       }

       if (pWin && !be_draw) {
	   be_draw = (unsigned int)(DMX_GET_WINDOW_PRIV(pWin))->window;
	   if (!be_draw) {
	      /* it might be that the window did not created yet on the */
	      /* back-end server (lazy window creation option), force   */
	      /* creation of the window */
	      dmxCreateAndRealizeWindow( pWin, TRUE );
	      be_draw = (unsigned int)(DMX_GET_WINDOW_PRIV(pWin))->window;
	   }
       }

       dmxSync( dmxScreen, 1 );

       LockDisplay(dpy);
       GetReq(GLXSwapBuffers,be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXSwapBuffers;
       be_req->drawable = be_draw;
       be_req->contextTag = ( tag ? GetCurrentBackEndTag(cl,tag,s) : 0 );
       UnlockDisplay(dpy);
       SyncHandle();
       XFlush(dpy);
    }

    return Success;
}

int __glXSwapBuffers(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    DrawablePtr pDraw;
    xGLXSwapBuffersReq *req = (xGLXSwapBuffersReq *) pc;
    GLXContextTag tag = req->contextTag;
    XID drawId = req->drawable;
    __GLXpixmap *pGlxPixmap = NULL;
    __GLXcontext *glxc = NULL;
    __glXWindow *pGlxWindow = NULL;
    int rc;
    
    /*
    ** Check that the GLX drawable is valid.
    */
    rc = dixLookupDrawable(&pDraw, drawId, client, 0, DixWriteAccess);
    if (rc == Success) {
	if (pDraw->type != DRAWABLE_WINDOW) {
	    /*
	    ** Drawable is an X pixmap, which is not allowed.
	    */
	    client->errorValue = drawId;
	    return __glXBadDrawable;
	}
    } 

    if (!pDraw) {
	pGlxPixmap = (__GLXpixmap *) LookupIDByType(drawId,
						    __glXPixmapRes);
	if (pGlxPixmap) {
	    /*
	    ** Drawable is a GLX pixmap.
	    */
	   pDraw = pGlxPixmap->pDraw;
	}
    }

    if (!pDraw && __GLX_IS_VERSION_SUPPORTED(1,3) ) {
       pGlxWindow = (__glXWindow *) LookupIDByType(drawId, __glXWindowRes);
       if (pGlxWindow) {
	  /*
	   ** Drawable is a GLXWindow.
	   */
	  pDraw = pGlxWindow->pDraw;
       }
    }

    if (!pDraw) {
       /*
	** Drawable is neither a X window nor a GLX pixmap.
	*/
       client->errorValue = drawId;
       return __glXBadDrawable;
    }

    if (tag) {
	glxc = __glXLookupContextByTag(cl, tag);
	if (!glxc) {
	    return __glXBadContextTag;
	}
    }

    if (pDraw &&
	pDraw->type == DRAWABLE_WINDOW &&
	DMX_GET_WINDOW_PRIV((WindowPtr)pDraw)->swapGroup) {
	return SGSwapBuffers(cl, drawId, tag, pDraw);
    }

    return __glXDoSwapBuffers(cl, drawId, tag);
}


/************************************************************************/

/*
** Render and Renderlarge are not in the GLX API.  They are used by the GLX
** client library to send batches of GL rendering commands.
*/

/*
** Execute all the drawing commands in a request.
*/
int __glXRender(__GLXclientState *cl, GLbyte *pc)
{
    xGLXRenderReq *req;
    xGLXRenderReq *be_req;
    int size;
    __GLXcontext *glxc;
    int from_screen = 0;
    int to_screen = 0;
    int s;

    /*
    ** NOTE: much of this code also appears in the byteswapping version of this
    ** routine, __glXSwapRender().  Any changes made here should also be
    ** duplicated there.
    */
    
    req = (xGLXRenderReq *) pc;

    glxc = __glXLookupContextByTag(cl, req->contextTag);
    if (!glxc) {
	return 0;
    }
    from_screen = to_screen = glxc->pScreen->myNum;

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    pc += sz_xGLXRenderReq;
    size = (req->length << 2) - sz_xGLXRenderReq;

    /*
     * just forward the request to back-end server(s)
     */
    for (s=from_screen; s<=to_screen; s++) {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);

       LockDisplay(dpy);
       GetReq(GLXRender,be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXRender;
       be_req->length = req->length;
       be_req->contextTag = GetCurrentBackEndTag(cl,req->contextTag,s);
       _XSend(dpy, (const char *)pc, size);
       UnlockDisplay(dpy);
       SyncHandle();
    }

    return Success;
}

/*
** Execute a large rendering request (one that spans multiple X requests).
*/
int __glXRenderLarge(__GLXclientState *cl, GLbyte *pc)
{
    xGLXRenderLargeReq *req;
    xGLXRenderLargeReq *be_req;
    __GLXcontext *glxc;
    int from_screen = 0;
    int to_screen = 0;
    int s;

    /*
    ** NOTE: much of this code also appears in the byteswapping version of this
    ** routine, __glXSwapRenderLarge().  Any changes made here should also be
    ** duplicated there.
    */
    
    req = (xGLXRenderLargeReq *) pc;
    glxc = __glXLookupContextByTag(cl, req->contextTag);
    if (!glxc) {
	return 0;
    }
    from_screen = to_screen = glxc->pScreen->myNum;

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    pc += sz_xGLXRenderLargeReq;

    /*
     * just forward the request to back-end server(s)
     */
    for (s=from_screen; s<=to_screen; s++) {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);

       GetReq(GLXRenderLarge,be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXRenderLarge;
       be_req->contextTag = GetCurrentBackEndTag(cl,req->contextTag,s);
       be_req->length = req->length;
       be_req->requestNumber = req->requestNumber;
       be_req->requestTotal = req->requestTotal;
       be_req->dataBytes = req->dataBytes;
       Data(dpy, (const char *)pc, req->dataBytes);
       UnlockDisplay(dpy);
       SyncHandle();

    }

    return Success;
}


/************************************************************************/

int __glXVendorPrivate(__GLXclientState *cl, GLbyte *pc)
{
    xGLXVendorPrivateReq *req;

    req = (xGLXVendorPrivateReq *) pc;

    switch( req->vendorCode ) {

       case X_GLvop_DeleteTexturesEXT:
	  return __glXVForwardSingleReq( cl, pc );
	  break;

       case X_GLXvop_SwapIntervalSGI:
	  if (glxIsExtensionSupported("SGI_swap_control")) {
	     return __glXVForwardSingleReq( cl, pc );
	  }
	  else {
	     return Success;
	  }
	  break;

#if 0 /* glx 1.3 */
       case X_GLXvop_CreateGLXVideoSourceSGIX:
	  break;
       case X_GLXvop_DestroyGLXVideoSourceSGIX:
	  break;
       case X_GLXvop_CreateGLXPixmapWithConfigSGIX:
	  break;
       case X_GLXvop_DestroyGLXPbufferSGIX:
	  break;
       case X_GLXvop_ChangeDrawableAttributesSGIX:
	  break;
#endif

       case X_GLXvop_BindSwapBarrierSGIX:
	  return __glXBindSwapBarrierSGIX( cl, pc );
	  break;

       case X_GLXvop_JoinSwapGroupSGIX:
	  return __glXJoinSwapGroupSGIX( cl, pc );
	  break;

       case X_GLXvop_CreateContextWithConfigSGIX:
	  return __glXCreateContextWithConfigSGIX( cl, pc );
	  break;

       default:
	  /*
	   ** unsupported private request
	   */
	  cl->client->errorValue = req->vendorCode;
	  return __glXUnsupportedPrivateRequest;
    }

    cl->client->errorValue = req->vendorCode;
    return __glXUnsupportedPrivateRequest;

}

int __glXVendorPrivateWithReply(__GLXclientState *cl, GLbyte *pc)
{
    xGLXVendorPrivateWithReplyReq *req;

    req = (xGLXVendorPrivateWithReplyReq *) pc;

    switch( req->vendorCode ) {

       case X_GLvop_GetConvolutionFilterEXT:
       case X_GLvop_GetConvolutionParameterfvEXT:
       case X_GLvop_GetConvolutionParameterivEXT:
       case X_GLvop_GetSeparableFilterEXT:
       case X_GLvop_GetHistogramEXT:
       case X_GLvop_GetHistogramParameterivEXT:
       case X_GLvop_GetMinmaxEXT:
       case X_GLvop_GetMinmaxParameterfvEXT:
       case X_GLvop_GetMinmaxParameterivEXT:
       case X_GLvop_AreTexturesResidentEXT:
       case X_GLvop_IsTextureEXT:
	  return( __glXVForwardPipe0WithReply(cl, pc) );
	  break;

       case X_GLvop_GenTexturesEXT:
	  return( __glXVForwardAllWithReply(cl, pc) );
	  break;


#if 0 /* glx1.3 */
       case X_GLvop_GetDetailTexFuncSGIS:
       case X_GLvop_GetSharpenTexFuncSGIS:
       case X_GLvop_GetColorTableSGI:
       case X_GLvop_GetColorTableParameterfvSGI:
       case X_GLvop_GetColorTableParameterivSGI:
       case X_GLvop_GetTexFilterFuncSGIS:
       case X_GLvop_GetInstrumentsSGIX:
       case X_GLvop_InstrumentsBufferSGIX:
       case X_GLvop_PollInstrumentsSGIX:
       case X_GLvop_FlushRasterSGIX:
       case X_GLXvop_CreateGLXPbufferSGIX:
       case X_GLXvop_GetDrawableAttributesSGIX:
       case X_GLXvop_QueryHyperpipeNetworkSGIX:
       case X_GLXvop_QueryHyperpipeConfigSGIX:
       case X_GLXvop_HyperpipeConfigSGIX:
       case X_GLXvop_DestroyHyperpipeConfigSGIX:
#endif
       case X_GLXvop_QueryMaxSwapBarriersSGIX:
	  return( __glXQueryMaxSwapBarriersSGIX(cl, pc) );
	  break;

       case X_GLXvop_GetFBConfigsSGIX:
	  return( __glXGetFBConfigsSGIX(cl, pc) );
	  break;

       case X_GLXvop_MakeCurrentReadSGI:
	  return( __glXMakeCurrentReadSGI(cl, pc) );
	  break;

       case X_GLXvop_QueryContextInfoEXT:
	  return( __glXQueryContextInfoEXT(cl,pc) );
	  break;

       default:
	  /*
	   ** unsupported private request
	   */
	  cl->client->errorValue = req->vendorCode;
	  return __glXUnsupportedPrivateRequest;
    }

}

int __glXQueryExtensionsString(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXQueryExtensionsStringReq *req = (xGLXQueryExtensionsStringReq *) pc;
    xGLXQueryExtensionsStringReply reply;
    GLint screen;
    size_t length;
    int len, numbytes;
    char *be_buf;

#ifdef FWD_QUERY_REQ
    xGLXQueryExtensionsStringReq *be_req;
    xGLXQueryExtensionsStringReply be_reply;
    DMXScreenInfo *dmxScreen;
    Display *dpy;
    int slop;
#endif

    screen = req->screen;

    /*
    ** Check if screen exists.
    */
    if ((screen < 0) || (screen >= screenInfo.numScreens)) {
	client->errorValue = screen;
	return BadValue;
    }

#ifdef FWD_QUERY_REQ
    dmxScreen = &dmxScreens[screen];

    /* Send the glXQueryServerString request */
    dpy = GetBackEndDisplay(cl,screen);
    LockDisplay(dpy);
    GetReq(GLXQueryExtensionsString,be_req);
    be_req->reqType = dmxScreen->glxMajorOpcode;
    be_req->glxCode = X_GLXQueryServerString;
    be_req->screen = DefaultScreen(dpy);
    _XReply(dpy, (xReply*) &be_reply, 0, False);
    len = (int)be_reply.length;
    numbytes = (int)be_reply.n;
    slop = numbytes * __GLX_SIZE_INT8 & 3;
    be_buf = (char *)Xalloc(numbytes);
    if (!be_buf) {
        /* Throw data on the floor */
        _XEatData(dpy, len);
    } else {
        _XRead(dpy, (char *)be_buf, numbytes);
        if (slop) _XEatData(dpy,4-slop);
    }
    UnlockDisplay(dpy);
    SyncHandle();

#else

    be_buf = __glXGetServerString(GLX_EXTENSIONS);
    numbytes = strlen(be_buf) + 1;
    len = __GLX_PAD(numbytes) >> 2;

#endif

    length = len;
    reply.type = X_Reply;
    reply.sequenceNumber = client->sequence;
    reply.length = len;
    reply.n = numbytes;

    if (client->swapped) {
        glxSwapQueryExtensionsStringReply(client, &reply, be_buf);
    } else {
        WriteToClient(client, sz_xGLXQueryExtensionsStringReply,(char *)&reply);
        WriteToClient(client, (int)(length << 2), (char *)be_buf);
    }

    return Success;
}

int __glXQueryServerString(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXQueryServerStringReq *req = (xGLXQueryServerStringReq *) pc;
    xGLXQueryServerStringReply reply;
    int name;
    GLint screen;
    size_t length;
    int len, numbytes;
    char *be_buf;
#ifdef FWD_QUERY_REQ
    xGLXQueryServerStringReq *be_req;
    xGLXQueryServerStringReply be_reply;
    DMXScreenInfo *dmxScreen;
    Display *dpy;
    int  slop;
#endif

    name = req->name;
    screen = req->screen;
    /*
    ** Check if screen exists.
    */
    if ((screen < 0) || (screen >= screenInfo.numScreens)) {
	client->errorValue = screen;
	return BadValue;
    }

#ifdef FWD_QUERY_REQ
    dmxScreen = &dmxScreens[screen];

    /* Send the glXQueryServerString request */
    dpy = GetBackEndDisplay(cl,screen);
    LockDisplay(dpy);
    GetReq(GLXQueryServerString,be_req);
    be_req->reqType = dmxScreen->glxMajorOpcode;
    be_req->glxCode = X_GLXQueryServerString;
    be_req->screen = DefaultScreen(dpy);
    be_req->name = name;
    _XReply(dpy, (xReply*) &be_reply, 0, False);
    len = (int)be_reply.length;
    numbytes = (int)be_reply.n;
    slop = numbytes * __GLX_SIZE_INT8 & 3;
    be_buf = (char *)Xalloc(numbytes);
    if (!be_buf) {
        /* Throw data on the floor */
        _XEatData(dpy, len);
    } else {
        _XRead(dpy, (char *)be_buf, numbytes);
        if (slop) _XEatData(dpy,4-slop);
    }
    UnlockDisplay(dpy);
    SyncHandle();

#else
    be_buf = __glXGetServerString(name);
    numbytes = strlen(be_buf) + 1;
    len = __GLX_PAD(numbytes) >> 2;
#endif

    length = len;
    reply.type = X_Reply;
    reply.sequenceNumber = client->sequence;
    reply.length = length;
    reply.n = numbytes;

    if (client->swapped) {
        glxSwapQueryServerStringReply(client, &reply, be_buf);
    } else {
        WriteToClient(client, sz_xGLXQueryServerStringReply, (char *)&reply);
        WriteToClient(client, (int)(length << 2), be_buf);
    }

    return Success;
}

int __glXClientInfo(__GLXclientState *cl, GLbyte *pc)
{
    xGLXClientInfoReq *req = (xGLXClientInfoReq *) pc;
    xGLXClientInfoReq *be_req;
    const char *buf;
    int from_screen = 0;
    int to_screen = 0;
    int s;
   
    cl->GLClientmajorVersion = req->major;
    cl->GLClientminorVersion = req->minor;
    if (cl->GLClientextensions) __glXFree(cl->GLClientextensions);
    buf = (const char *)(req+1);
    cl->GLClientextensions = strdup(buf);

    to_screen = screenInfo.numScreens - 1;

    for (s=from_screen; s<=to_screen; s++)
    {
       DMXScreenInfo *dmxScreen = &dmxScreens[s];
       Display *dpy = GetBackEndDisplay(cl,s);

       LockDisplay(dpy);
       GetReq(GLXClientInfo,be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXClientInfo;
       be_req->major = req->major;
       be_req->minor = req->minor;
       be_req->length = req->length;
       be_req->numbytes = req->numbytes;
       Data(dpy, buf, req->numbytes);

       UnlockDisplay(dpy);
       SyncHandle();
    }

    return Success;
}

int __glXUseXFont(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXUseXFontReq *req;
    xGLXUseXFontReq *be_req;
    FontPtr pFont;
    __GLXcontext *glxc = NULL;
    int from_screen = 0;
    int to_screen = 0;
    int s;
    dmxFontPrivPtr  pFontPriv;
    DMXScreenInfo *dmxScreen;
    Display *dpy;

    req = (xGLXUseXFontReq *) pc;

    if (req->contextTag != 0) {
	glxc = __glXLookupContextByTag(cl, req->contextTag);
	if (glxc) {
	   from_screen = to_screen = glxc->pScreen->myNum;
	}
    }

    /*
    ** Font can actually be either the ID of a font or the ID of a GC
    ** containing a font.
    */
    pFont = (FontPtr)LookupIDByType(req->font, RT_FONT);
    if (!pFont) {
        GC *pGC = (GC *)LookupIDByType(req->font, RT_GC);
        if (!pGC) {
	    client->errorValue = req->font;
            return BadFont;
	}
	pFont = pGC->font;
    }

    pFontPriv = FontGetPrivate(pFont, dmxFontPrivateIndex);

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif


    for (s=from_screen; s<=to_screen; s++) {
       dmxScreen = &dmxScreens[s];
       dpy = GetBackEndDisplay(cl,s);

       dmxSync( dmxScreen, 1 );

       LockDisplay(dpy);
       GetReq(GLXUseXFont,be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXUseXFont;
       be_req->contextTag = (glxc ? GetCurrentBackEndTag(cl,req->contextTag,s) : 0);
       be_req->font = pFontPriv->font[s]->fid;
       be_req->first = req->first;
       be_req->count = req->count;
       be_req->listBase = req->listBase;
       UnlockDisplay(dpy);
       SyncHandle();

       XSync( dpy, False );
    }

    return Success;
}

/*
 * start GLX 1.3 here
 */

int __glXGetFBConfigs(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXGetFBConfigsReq *req = (xGLXGetFBConfigsReq *) pc;
    xGLXGetFBConfigsReply reply;
    __GLXFBConfig *pFBConfig;
    CARD32 buf[2 * __GLX_TOTAL_FBCONFIG_PROPS];
    int numAttribs = __GLX_TOTAL_FBCONFIG_PROPS;
    unsigned int screen = req->screen;
    int numFBConfigs, i, p;
    __GLXscreenInfo *pGlxScreen;

    if (screen > screenInfo.numScreens) {
	/* The client library must send a valid screen number. */
	client->errorValue = screen;
	return BadValue;
    }

    pGlxScreen = &__glXActiveScreens[screen];
    numFBConfigs = __glXNumFBConfigs;

    reply.numFBConfigs = numFBConfigs;
    reply.numAttribs = numAttribs;
    reply.length = (numFBConfigs * 2 * numAttribs * __GLX_SIZE_CARD32) >> 2;
    reply.type = X_Reply;
    reply.sequenceNumber = client->sequence;

    if (client->swapped) {
	__GLX_DECLARE_SWAP_VARIABLES;
	__GLX_SWAP_SHORT(&reply.sequenceNumber);
	__GLX_SWAP_INT(&reply.length);
	__GLX_SWAP_INT(&reply.numFBConfigs);
	__GLX_SWAP_INT(&reply.numAttribs);
    }
    WriteToClient(client, sz_xGLXGetFBConfigsReply, (char *)&reply);

    for (i=0; i < numFBConfigs; i++) {
       int associatedVisualId = 0;
       int drawableTypeIndex;
       pFBConfig = __glXFBConfigs[ i * (screenInfo.numScreens+1) ];

	p = 0;
	/* core attributes */
	buf[p++] = GLX_FBCONFIG_ID;
	buf[p++] = pFBConfig->id;
	buf[p++] = GLX_BUFFER_SIZE;
	buf[p++] = pFBConfig->indexBits;
	buf[p++] = GLX_LEVEL;
	buf[p++] = pFBConfig->level;
	buf[p++] = GLX_DOUBLEBUFFER;
	buf[p++] = pFBConfig->doubleBufferMode;
	buf[p++] = GLX_STEREO;
	buf[p++] = pFBConfig->stereoMode;
	buf[p++] = GLX_AUX_BUFFERS;
	buf[p++] = pFBConfig->maxAuxBuffers;
	buf[p++] = GLX_RED_SIZE;
	buf[p++] = pFBConfig->redBits;
	buf[p++] = GLX_GREEN_SIZE;
	buf[p++] = pFBConfig->greenBits;
	buf[p++] = GLX_BLUE_SIZE;
	buf[p++] = pFBConfig->blueBits;
	buf[p++] = GLX_ALPHA_SIZE;
	buf[p++] = pFBConfig->alphaBits;
	buf[p++] = GLX_DEPTH_SIZE;
	buf[p++] = pFBConfig->depthBits;
	buf[p++] = GLX_STENCIL_SIZE;
	buf[p++] = pFBConfig->stencilBits;
	buf[p++] = GLX_ACCUM_RED_SIZE;
	buf[p++] = pFBConfig->accumRedBits;
	buf[p++] = GLX_ACCUM_GREEN_SIZE;
	buf[p++] = pFBConfig->accumGreenBits;
	buf[p++] = GLX_ACCUM_BLUE_SIZE;
	buf[p++] = pFBConfig->accumBlueBits;
	buf[p++] = GLX_ACCUM_ALPHA_SIZE;
	buf[p++] = pFBConfig->accumAlphaBits;
	buf[p++] = GLX_RENDER_TYPE;
	buf[p++] = pFBConfig->renderType;
	buf[p++] = GLX_DRAWABLE_TYPE;
	drawableTypeIndex = p;
	buf[p++] = pFBConfig->drawableType;
	buf[p++] = GLX_X_VISUAL_TYPE;
	buf[p++] = pFBConfig->visualType;
	buf[p++] = GLX_CONFIG_CAVEAT;
	buf[p++] = pFBConfig->visualCaveat;
	buf[p++] = GLX_TRANSPARENT_TYPE;
	buf[p++] = pFBConfig->transparentType;
	buf[p++] = GLX_TRANSPARENT_RED_VALUE;
	buf[p++] = pFBConfig->transparentRed;
	buf[p++] = GLX_TRANSPARENT_GREEN_VALUE;
	buf[p++] = pFBConfig->transparentGreen;
	buf[p++] = GLX_TRANSPARENT_BLUE_VALUE;
	buf[p++] = pFBConfig->transparentBlue;
	buf[p++] = GLX_TRANSPARENT_ALPHA_VALUE;
	buf[p++] = pFBConfig->transparentAlpha;
	buf[p++] = GLX_TRANSPARENT_INDEX_VALUE;
	buf[p++] = pFBConfig->transparentIndex;
	buf[p++] = GLX_MAX_PBUFFER_WIDTH;
	buf[p++] = pFBConfig->maxPbufferWidth;
	buf[p++] = GLX_MAX_PBUFFER_HEIGHT;
	buf[p++] = pFBConfig->maxPbufferHeight;
	buf[p++] = GLX_MAX_PBUFFER_PIXELS;
	buf[p++] = pFBConfig->maxPbufferPixels;

	/*
         * find the visual of the back-end server and match a visual
	 * on the proxy.
	 * do only once - if a visual is not yet associated.
	 */
	if (pFBConfig->associatedVisualId == (unsigned int)-1) {
	   DMXScreenInfo *dmxScreen = &dmxScreens[screen];
	   __GLXFBConfig *be_pFBConfig = __glXFBConfigs[ i * (screenInfo.numScreens+1)+screen+1 ];
	   __GLXvisualConfig *pGlxVisual = NULL;
	   int v;
	   int found = 0;
	   for (v=0; v<dmxScreen->numGlxVisuals; v++) {
	      if (dmxScreen->glxVisuals[v].vid == be_pFBConfig->associatedVisualId) {
		 pGlxVisual = &dmxScreen->glxVisuals[v];
		 break;
	      }
	   }

	   if (pGlxVisual) {
	      for (v=0; v<pGlxScreen->numVisuals; v++) {
		 if (glxVisualsMatch(&pGlxScreen->pGlxVisual[v], pGlxVisual)) {
		    associatedVisualId = pGlxScreen->pGlxVisual[v].vid;
		    found = 1;
		    break;
		 }
	      }
	   }

	   if (!found) {
	      associatedVisualId = 0;
	      pFBConfig->drawableType &= ~(GLX_WINDOW_BIT);
	      buf[drawableTypeIndex] = pFBConfig->drawableType;
	   }
#ifdef PANORAMIX
	   else if (!noPanoramiXExtension) {
	      /* convert the associated visualId to the panoramix one */
	      pFBConfig->associatedVisualId =
		  PanoramiXTranslateVisualID(screen, v);
	   }
#endif
	}
	else {
	   associatedVisualId = pFBConfig->associatedVisualId;
	}

	buf[p++] = GLX_VISUAL_ID;
	buf[p++] = associatedVisualId;

	/* SGIS_multisample attributes */
	buf[p++] = GLX_SAMPLES_SGIS;
	buf[p++] = pFBConfig->multiSampleSize;
	buf[p++] = GLX_SAMPLE_BUFFERS_SGIS;
	buf[p++] = pFBConfig->nMultiSampleBuffers;

	/* SGIX_pbuffer specific attributes */
	buf[p++] = GLX_OPTIMAL_PBUFFER_WIDTH_SGIX;
	buf[p++] = pFBConfig->optimalPbufferWidth;
	buf[p++] = GLX_OPTIMAL_PBUFFER_HEIGHT_SGIX;
	buf[p++] = pFBConfig->optimalPbufferHeight;

	buf[p++] = GLX_VISUAL_SELECT_GROUP_SGIX;
	buf[p++] = pFBConfig->visualSelectGroup;

	if (client->swapped) {
	    __GLX_DECLARE_SWAP_VARIABLES;
	    __GLX_SWAP_INT_ARRAY((int *)buf, 2*numAttribs);
	}
	WriteToClient(client, 2*numAttribs * __GLX_SIZE_CARD32, (char *)buf);
    }
    return Success;
}

int __glXGetFBConfigsSGIX(__GLXclientState *cl, GLbyte *pc)
{
   xGLXGetFBConfigsSGIXReq *req = (xGLXGetFBConfigsSGIXReq *)pc;
   xGLXGetFBConfigsReq new_req;

   new_req.reqType = req->reqType;
   new_req.glxCode = req->glxCode;
   new_req.length = req->length;
   new_req.screen = req->screen;

   return( __glXGetFBConfigs( cl, (GLbyte *)&new_req ) );
}


int __glXCreateWindow(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXCreateWindowReq *req = (xGLXCreateWindowReq *) pc;
    int screen = req->screen;
    GLXFBConfigID fbconfigId = req->fbconfig;
    XID windowId = req->window;
    XID glxwindowId = req->glxwindow;
    DrawablePtr pDraw;
    ScreenPtr pScreen;
    __glXWindow *pGlxWindow;
    __GLXFBConfig *pGlxFBConfig = NULL;
    VisualPtr pVisual;
    VisualID visId;
    int i, rc;

    /*
    ** Check if windowId is valid 
    */
    rc = dixLookupDrawable(&pDraw, windowId, client, M_DRAWABLE_WINDOW,
			   DixAddAccess);
    if (rc != Success)
	return rc;

    /*
    ** Check if screen of window matches screen of fbconfig.
    */
    pScreen = pDraw->pScreen;
    if (screen != pScreen->myNum) {
	return BadMatch;
    }

    /*
    ** Find the FBConfigRec for this fbconfigid.
    */
    if (!(pGlxFBConfig = glxLookupFBConfig(fbconfigId))) {
	client->errorValue = fbconfigId;
	return __glXBadFBConfig;
    }
    visId = pGlxFBConfig->associatedVisualId;

    /*
    ** Check if the fbconfig supports rendering to windows 
    */
    if( !(pGlxFBConfig->drawableType & GLX_WINDOW_BIT) ) {
	return BadMatch;	
    }

    if (visId != None) {
       /*
	** Check if the visual ID is valid for this screen.
	*/
       pVisual = pScreen->visuals;
       for (i = 0; i < pScreen->numVisuals; i++, pVisual++) {
	  if (pVisual->vid == visId) {
	     break;
	  }
       }
       if (i == pScreen->numVisuals) {
	  client->errorValue = visId;
	  return BadValue;
       }
	
        /*
        ** Check if color buffer depth of fbconfig matches depth 
	** of window.
        */
        if (pVisual->nplanes != pDraw->depth) {
	    return BadMatch;
	}
    } else
	/*
	** The window was created with no visual that corresponds
	** to fbconfig 
	*/
	return BadMatch;

    /*
    ** Check if there is already a fbconfig associated with this window
    */
    if ( LookupIDByType(glxwindowId, __glXWindowRes) ) {
	client->errorValue = glxwindowId;
	return BadAlloc;
    }

    pGlxWindow = (__glXWindow *) xalloc(sizeof(__glXWindow));
    if (!pGlxWindow) {
	return BadAlloc;
    }

    /*
    ** Register this GLX window as a resource
    */
    if (!(AddResource(glxwindowId, __glXWindowRes, pGlxWindow))) {
	return BadAlloc;
    }

    pGlxWindow->pDraw = pDraw;
    pGlxWindow->type = GLX_GLXWINDOW_TYPE;
    pGlxWindow->idExists = True;
    pGlxWindow->refcnt = 0;
    pGlxWindow->pGlxFBConfig = pGlxFBConfig;
    pGlxWindow->pScreen = pScreen;

    return Success;
}

int __glXDestroyWindow(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXDestroyWindowReq *req = (xGLXDestroyWindowReq *) pc;
    XID glxwindow = req->glxwindow;

    /*
    ** Check if it's a valid GLX window.
    */
    if (!LookupIDByType(glxwindow, __glXWindowRes)) {
	client->errorValue = glxwindow;
	return __glXBadDrawable;
    }
    /*
    ** The glx window destructor will check whether it's current before
    ** freeing anything.
    */
    FreeResource(glxwindow, RT_NONE);	

    return Success;
}

int __glXQueryContext(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    __GLXcontext *ctx;
    xGLXQueryContextReq *req;
    xGLXQueryContextReply reply;
    int nProps;
    int *sendBuf, *pSendBuf;
    int nReplyBytes;

    req = (xGLXQueryContextReq *)pc;
    ctx = (__GLXcontext *) LookupIDByType(req->context, __glXContextRes);
    if (!ctx) {
        client->errorValue = req->context;
        return __glXBadContext;
    }

    nProps = 3;

    reply.length = nProps << 1;
    reply.type = X_Reply;
    reply.sequenceNumber = client->sequence;
    reply.n = nProps;

    nReplyBytes = reply.length << 2;
    sendBuf = (int *)xalloc(nReplyBytes);
    pSendBuf = sendBuf;
    *pSendBuf++ = GLX_FBCONFIG_ID;
    *pSendBuf++ = (int)(ctx->pFBConfig->id);
    *pSendBuf++ = GLX_RENDER_TYPE;
    *pSendBuf++ = (int)(ctx->pFBConfig->renderType);
    *pSendBuf++ = GLX_SCREEN;
    *pSendBuf++ = (int)(ctx->pScreen->myNum);

    if (client->swapped) {
        __glXSwapQueryContextReply(client, &reply, sendBuf);
    } else {
        WriteToClient(client, sz_xGLXQueryContextReply, (char *)&reply);
        WriteToClient(client, nReplyBytes, (char *)sendBuf);
    }
    xfree((char *)sendBuf);

    return Success;
}

int __glXQueryContextInfoEXT(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    __GLXcontext *ctx;
    xGLXQueryContextInfoEXTReq *req;
    xGLXQueryContextInfoEXTReply reply;
    int nProps;
    int *sendBuf, *pSendBuf;
    int nReplyBytes;

    req = (xGLXQueryContextInfoEXTReq *)pc;
    ctx = (__GLXcontext *) SecurityLookupIDByType(client, req->context, __glXContextRes, DixReadAccess);
    if (!ctx) {
        client->errorValue = req->context;
        return __glXBadContext;
    }

    nProps = 4;

    reply.length = nProps << 1;
    reply.type = X_Reply;
    reply.sequenceNumber = client->sequence;
    reply.n = nProps;

    nReplyBytes = reply.length << 2;
    sendBuf = (int *)xalloc(nReplyBytes);
    pSendBuf = sendBuf;
    *pSendBuf++ = GLX_SHARE_CONTEXT_EXT;
    *pSendBuf++ = (int)(ctx->share_id);
    *pSendBuf++ = GLX_VISUAL_ID_EXT;
    *pSendBuf++ = (int)(ctx->pVisual ? ctx->pVisual->vid : 0);
    *pSendBuf++ = GLX_SCREEN_EXT;
    *pSendBuf++ = (int)(ctx->pScreen->myNum);
    *pSendBuf++ = GLX_FBCONFIG_ID;
    *pSendBuf++ = (int)(ctx->pFBConfig ? ctx->pFBConfig->id : 0);

    if (client->swapped) {
        __glXSwapQueryContextInfoEXTReply(client, &reply, sendBuf);
    } else {
        WriteToClient(client, sz_xGLXQueryContextInfoEXTReply, (char *)&reply);
        WriteToClient(client, nReplyBytes, (char *)sendBuf);
    }
    xfree((char *)sendBuf);

    return Success;
}

int __glXCreatePbuffer(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXCreatePbufferReq *req = (xGLXCreatePbufferReq *)pc;
    xGLXCreatePbufferReq *be_req;
    int screen = req->screen;
    GLXFBConfigID fbconfigId = req->fbconfig;
    GLXPbuffer pbuffer = req->pbuffer;
    __glXPbuffer *pGlxPbuffer;
    int numAttribs = req->numAttribs;
    int *attr;
    ScreenPtr pScreen;
    __GLXFBConfig *pGlxFBConfig;
    __GLXFBConfig *be_pGlxFBConfig;
    XID be_xid;
    Display *dpy;
    DMXScreenInfo *dmxScreen;
    int s;
    int from_screen, to_screen;

   /*
    ** Look up screen and FBConfig.
    */
    if (screen > screenInfo.numScreens) {
        /* The client library must send a valid screen number. */
        client->errorValue = screen;
        return BadValue;
    }
    pScreen = screenInfo.screens[screen];

    /*
    ** Find the FBConfigRec for this fbconfigid.
    */
    if (!(pGlxFBConfig = glxLookupFBConfig(fbconfigId))) {
	client->errorValue = fbconfigId;
	return __glXBadFBConfig;
    }

    /*
    ** Create the GLX part of the Pbuffer.
    */
    pGlxPbuffer = (__glXPbuffer *) xalloc(sizeof(__glXPbuffer));
    if (!pGlxPbuffer) {
        return BadAlloc;
    }

    pGlxPbuffer->be_xids = (XID *) xalloc( sizeof(XID) * screenInfo.numScreens );
    if (!pGlxPbuffer->be_xids) {
        xfree(pGlxPbuffer);
        return BadAlloc;
    }

    /*
     * Allocate an XID on the back-end server(s) and send him the request
     */
    from_screen = to_screen = screen;
#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    for (s=from_screen; s<=to_screen; s++) {
       dpy = GetBackEndDisplay(cl,s);
       be_xid = XAllocID(dpy);
       dmxScreen = &dmxScreens[s];
       be_pGlxFBConfig = glxLookupBackEndFBConfig( pGlxFBConfig->id, s );
       
       attr = (int *)( req+1 );

       LockDisplay(dpy);
       GetReqExtra(GLXCreatePbuffer, 2 * numAttribs * __GLX_SIZE_CARD32, be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXCreatePbuffer;
       be_req->screen = be_pGlxFBConfig->screen;
       be_req->fbconfig = be_pGlxFBConfig->id;
       be_req->pbuffer = be_xid;
       be_req->numAttribs = numAttribs;
    
       /* Send attributes */
       if ( attr != NULL ) {
	  CARD32 *pc = (CARD32 *)(be_req + 1);

	  while (numAttribs-- > 0) {
	     *pc++ = *attr++;     /* token */
	     *pc++ = *attr++;     /* value */
	  }
       }

       UnlockDisplay(dpy);
       SyncHandle();

       pGlxPbuffer->be_xids[s] = be_xid;
    }


    pGlxPbuffer->idExists = True;
    pGlxPbuffer->refcnt = 0;
    pGlxPbuffer->pFBConfig = pGlxFBConfig;
    pGlxPbuffer->pScreen = pScreen;

    /*
    ** Register the resource.
    */
    if (!(AddResource(pbuffer, __glXPbufferRes, pGlxPbuffer))) {
        return BadAlloc;
    }

    return Success;

}

int __glXDestroyPbuffer(__GLXclientState *cl, GLbyte *pc)
{
    ClientPtr client = cl->client;
    xGLXDestroyPbufferReq *req = (xGLXDestroyPbufferReq *) pc;
    xGLXDestroyPbufferReq *be_req;
    GLXPbuffer pbuffer = req->pbuffer;
    Display *dpy;
    int screen;
    DMXScreenInfo *dmxScreen;
    __glXPbuffer *pGlxPbuffer;
    int s;
    int from_screen, to_screen;

    /*
    ** Check if it's a valid Pbuffer
    */
    pGlxPbuffer = (__glXPbuffer *)LookupIDByType(pbuffer, __glXPbufferRes);
    if (!pGlxPbuffer) {
	client->errorValue = pbuffer;
	return __glXBadPbuffer;
    }

    screen = pGlxPbuffer->pScreen->myNum;

    from_screen = to_screen = screen;
#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    for (s=from_screen; s<=to_screen; s++) {
       dpy = GetBackEndDisplay(cl,s);
       dmxScreen = &dmxScreens[s];

       /* send the destroy request to the back-end server */
       LockDisplay(dpy);
       GetReq(GLXDestroyPbuffer, be_req);
       be_req->reqType = dmxScreen->glxMajorOpcode;
       be_req->glxCode = X_GLXDestroyPbuffer;
       be_req->pbuffer = pGlxPbuffer->be_xids[s];
       UnlockDisplay(dpy);
       SyncHandle();
    }

    FreeResource(pbuffer, RT_NONE);	

    return Success;
}

int __glXGetDrawableAttributes(__GLXclientState *cl, GLbyte *pc)
{
   xGLXGetDrawableAttributesReq *req = (xGLXGetDrawableAttributesReq *)pc;
   xGLXGetDrawableAttributesReq *be_req;
   xGLXGetDrawableAttributesReply reply;
   ClientPtr client = cl->client;
   GLXDrawable drawId = req->drawable;
   GLXDrawable be_drawable = 0;
   DrawablePtr pDraw = NULL;
   Display *dpy;
   int screen, rc;
   DMXScreenInfo *dmxScreen;
   CARD32 *attribs = NULL;
   int attribs_size;
#ifdef PANORAMIX
    PanoramiXRes *pXinDraw = NULL;
#endif

   if (drawId != None) {
      rc = dixLookupDrawable(&pDraw, drawId, client, 0, DixGetAttrAccess);
      if (rc == Success) {
	 if (pDraw->type == DRAWABLE_WINDOW) {
		WindowPtr pWin = (WindowPtr)pDraw;
		be_drawable = 0;
		screen = pWin->drawable.pScreen->myNum;

	 }
	 else {
	    /*
	     ** Drawable is not a Window , GLXWindow or a GLXPixmap.
	     */
	    client->errorValue = drawId;
	    return __glXBadDrawable;
	 }
      }

      if (!pDraw) {
	 __GLXpixmap *pGlxPixmap = (__GLXpixmap *) LookupIDByType(drawId,
							__glXPixmapRes);
	 if (pGlxPixmap) {
		pDraw = pGlxPixmap->pDraw;
		screen = pGlxPixmap->pScreen->myNum;
                be_drawable = pGlxPixmap->be_xids[screen];
	 }
      }

      if (!pDraw) {
	 __glXWindow *pGlxWindow = (__glXWindow *) LookupIDByType(drawId, __glXWindowRes);
	 if (pGlxWindow) {
	    pDraw = pGlxWindow->pDraw;
	    screen = pGlxWindow->pScreen->myNum;
	    be_drawable = 0;
     	 }
      }

      if (!pDraw) {
	 __glXPbuffer *pGlxPbuffer = (__glXPbuffer *)LookupIDByType(drawId, __glXPbufferRes);
	 if (pGlxPbuffer) {
    	    pDraw = (DrawablePtr)pGlxPbuffer;
	    screen = pGlxPbuffer->pScreen->myNum;
	    be_drawable = pGlxPbuffer->be_xids[screen];
	 }
      }


      if (!pDraw) {
	 /*
	  ** Drawable is not a Window , GLXWindow or a GLXPixmap.
	  */
	 client->errorValue = drawId;
	 return __glXBadDrawable;
      }
    }


   /* if the drawable is a window or GLXWindow - 
    * we need to find the base id on the back-end server
    */
   if (!be_drawable) {
       WindowPtr pWin = (WindowPtr)pDraw;

#ifdef PANORAMIX
       if (!noPanoramiXExtension) {
	  pXinDraw = (PanoramiXRes *)
	     SecurityLookupIDByClass(client, pDraw->id, XRC_DRAWABLE, DixReadAccess);
	  if (!pXinDraw) {
	     client->errorValue = drawId;
	     return __glXBadDrawable;
	  }

	  dixLookupWindow(&pWin, pXinDraw->info[screen].id, client,
			  DixReadAccess);
       }
#endif

       if (pWin) {
	   be_drawable = (unsigned int)(DMX_GET_WINDOW_PRIV(pWin))->window;
	   if (!be_drawable) {
	      /* it might be that the window did not created yet on the */
	      /* back-end server (lazy window creation option), force   */
	      /* creation of the window */
	      dmxCreateAndRealizeWindow( pWin, TRUE );
	      be_drawable = (unsigned int)(DMX_GET_WINDOW_PRIV(pWin))->window;
	   }
       }
       else {
	  client->errorValue = drawId;
	  return __glXBadDrawable;
       }
   }


   /* send the request to the back-end server */
   dpy = GetBackEndDisplay(cl,screen);
   dmxScreen = &dmxScreens[screen];

   /* make sure drawable exists on back-end */
   dmxSync( dmxScreen, 1 );

   LockDisplay(dpy);
   GetReq(GLXGetDrawableAttributes, be_req);
   be_req->reqType = dmxScreen->glxMajorOpcode;
   be_req->glxCode = X_GLXGetDrawableAttributes;
   be_req->drawable = be_drawable;
   be_req->length = req->length;
   if (!_XReply(dpy, (xReply *) &reply, 0, False)) {
      UnlockDisplay(dpy);
      SyncHandle();
      return( BE_TO_CLIENT_ERROR(dmxLastErrorEvent.error_code) );
   }

   if (reply.numAttribs) {
      attribs_size = 2 * reply.numAttribs * __GLX_SIZE_CARD32;
      attribs = (CARD32 *) Xalloc(attribs_size);
      if (attribs == NULL) {
	 UnlockDisplay(dpy);
	 SyncHandle();
	 return BadAlloc;
      }

      _XRead(dpy, (char *) attribs, attribs_size);
   }

   UnlockDisplay(dpy);
   SyncHandle();


   /* send the reply back to the client */
   reply.sequenceNumber = client->sequence;
   if (client->swapped) {
      __glXSwapGetDrawableAttributesReply(client, &reply, (int *)attribs);
   }
   else {
      WriteToClient(client, sz_xGLXGetDrawableAttributesReply, (char *)&reply);
      WriteToClient(client, attribs_size, (char *)attribs);
   }

   Xfree(attribs);

   return Success;
}

int __glXChangeDrawableAttributes(__GLXclientState *cl, GLbyte *pc)
{
   xGLXChangeDrawableAttributesReq *req = (xGLXChangeDrawableAttributesReq *)pc;
   xGLXChangeDrawableAttributesReq *be_req;
   ClientPtr client = cl->client;
   GLXDrawable drawId = req->drawable;
   GLXDrawable be_drawable = 0;
   DrawablePtr pDraw = NULL;
   Display *dpy;
   int screen, rc;
   DMXScreenInfo *dmxScreen;
   char *attrbuf;
#ifdef PANORAMIX
    PanoramiXRes *pXinDraw = NULL;
    PanoramiXRes *pXinReadDraw = NULL;
#endif

   if (drawId != None) {
      rc = dixLookupDrawable(&pDraw, drawId, client, 0, DixSetAttrAccess);
      if (rc == Success) {
	 if (pDraw->type == DRAWABLE_WINDOW) {
		WindowPtr pWin = (WindowPtr)pDraw;
		be_drawable = 0;
		screen = pWin->drawable.pScreen->myNum;

	 }
	 else {
	    /*
	     ** Drawable is not a Window , GLXWindow or a GLXPixmap.
	     */
	    client->errorValue = drawId;
	    return __glXBadDrawable;
	 }
      }

      if (!pDraw) {
	 __GLXpixmap *pGlxPixmap = (__GLXpixmap *) LookupIDByType(drawId,
							__glXPixmapRes);
	 if (pGlxPixmap) {
		pDraw = pGlxPixmap->pDraw;
		screen = pGlxPixmap->pScreen->myNum;
                be_drawable = pGlxPixmap->be_xids[screen];
	 }
      }

      if (!pDraw) {
	 __glXWindow *pGlxWindow = (__glXWindow *) LookupIDByType(drawId, __glXWindowRes);
	 if (pGlxWindow) {
	    pDraw = pGlxWindow->pDraw;
	    screen = pGlxWindow->pScreen->myNum;
	    be_drawable = 0;
     	 }
      }

      if (!pDraw) {
	 __glXPbuffer *pGlxPbuffer = (__glXPbuffer *)LookupIDByType(drawId, __glXPbufferRes);
	 if (pGlxPbuffer) {
    	    pDraw = (DrawablePtr)pGlxPbuffer;
	    screen = pGlxPbuffer->pScreen->myNum;
	    be_drawable = pGlxPbuffer->be_xids[screen];
	 }
      }


      if (!pDraw) {
	 /*
	  ** Drawable is not a Window , GLXWindow or a GLXPixmap.
	  */
	 client->errorValue = drawId;
	 return __glXBadDrawable;
      }
    }


   /* if the drawable is a window or GLXWindow - 
    * we need to find the base id on the back-end server
    */
   if (!be_drawable) {
       WindowPtr pWin = (WindowPtr)pDraw;

#ifdef PANORAMIX
       if (!noPanoramiXExtension) {
	  pXinDraw = (PanoramiXRes *)
	     SecurityLookupIDByClass(client, pDraw->id, XRC_DRAWABLE, DixReadAccess);
	  if (!pXinDraw) {
	     client->errorValue = drawId;
	     return __glXBadDrawable;
	  }

	  dixLookupWindow(&pWin, pXinDraw->info[screen].id, client,
			  DixReadAccess);
       }
#endif

       if (pWin) {
	   be_drawable = (unsigned int)(DMX_GET_WINDOW_PRIV(pWin))->window;
	   if (!be_drawable) {
	      /* it might be that the window did not created yet on the */
	      /* back-end server (lazy window creation option), force   */
	      /* creation of the window */
	      dmxCreateAndRealizeWindow( pWin, TRUE );
	      be_drawable = (unsigned int)(DMX_GET_WINDOW_PRIV(pWin))->window;
	   }
       }
       else {
	  client->errorValue = drawId;
	  return __glXBadDrawable;
       }
   }


   /* send the request to the back-end server */
   dpy = GetBackEndDisplay(cl,screen);
   dmxScreen = &dmxScreens[screen];

   /* make sure drawable exists on back-end */
   dmxSync( dmxScreen, 1 );

   LockDisplay(dpy);
   GetReqExtra(GLXChangeDrawableAttributes,
                        2 * req->numAttribs * __GLX_SIZE_CARD32, be_req);
   be_req->reqType = dmxScreen->glxMajorOpcode;
   be_req->glxCode = X_GLXChangeDrawableAttributes;
   be_req->drawable = be_drawable;
   be_req->numAttribs = req->numAttribs;
   be_req->length = req->length;

   UnlockDisplay(dpy);
   SyncHandle();

   return Success;
}

int __glXSendLargeCommand(__GLXclientState *cl, GLXContextTag contextTag)
{
   ClientPtr client = cl->client;
    xGLXRenderLargeReq *req;
    GLint maxSize, amount;
    GLint totalRequests, requestNumber;
    GLint dataLen;
    GLbyte *data;
    __GLXcontext *glxc;
    int s;
    int from_screen, to_screen;

    maxSize = cl->largeCmdMaxReqDataSize - (GLint)sizeof(xGLXRenderLargeReq);
    dataLen = cl->largeCmdBytesTotal;
    totalRequests = (dataLen / maxSize);
    if (dataLen % maxSize) totalRequests++;

    glxc = __glXLookupContextByTag(cl, contextTag);
    if (!glxc) {
       client->errorValue = contextTag;
       return __glXBadContext;
    }
    from_screen = to_screen = glxc->pScreen->myNum;

#ifdef PANORAMIX
    if (!noPanoramiXExtension) {
       from_screen = 0;
       to_screen = screenInfo.numScreens - 1;
    }
#endif

    /*
    ** Send enough requests until the whole array is sent.
    */
    requestNumber = 1;
    data = cl->largeCmdBuf;
    while (dataLen > 0) {
	amount = dataLen;
	if (amount > maxSize) {
	    amount = maxSize;
	}

	for (s=from_screen; s<=to_screen; s++) {

	   Display *dpy = GetBackEndDisplay(cl,s);
	   DMXScreenInfo *dmxScreen = &dmxScreens[s];

	   LockDisplay(dpy);
	   GetReq(GLXRenderLarge,req); 
	   req->reqType = dmxScreen->glxMajorOpcode;
	   req->glxCode = X_GLXRenderLarge; 
	   req->contextTag = GetCurrentBackEndTag(cl,contextTag,s);
	   req->length += (amount + 3) >> 2;
	   req->requestNumber = requestNumber++;
	   req->requestTotal = totalRequests;
	   req->dataBytes = amount;
	   Data(dpy, ((const char*)data), amount);
	   dataLen -= amount;
	   data = ((GLbyte *) data) + amount;
	   UnlockDisplay(dpy);
	   SyncHandle();
	}
    }

    return Success;
}