2447 lines
		
	
	
		
			66 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			2447 lines
		
	
	
		
			66 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
|  * 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_DIX_CONFIG_H
 | |
| #include <dix-config.h>
 | |
| #endif
 | |
| 
 | |
| #include <string.h>
 | |
| #include <assert.h>
 | |
| 
 | |
| #include "glxserver.h"
 | |
| #include <GL/glxtokens.h>
 | |
| #include <unpack.h>
 | |
| #include "g_disptab.h"
 | |
| #include <pixmapstr.h>
 | |
| #include <windowstr.h>
 | |
| #include "glxutil.h"
 | |
| #include "glxext.h"
 | |
| #include "glapitable.h"
 | |
| #include "glapi.h"
 | |
| #include "glthread.h"
 | |
| #include "dispatch.h"
 | |
| #include "indirect_dispatch.h"
 | |
| #include "indirect_table.h"
 | |
| #include "indirect_util.h"
 | |
| 
 | |
| static int
 | |
| validGlxScreen(ClientPtr client, int screen, __GLXscreen **pGlxScreen, int *err)
 | |
| {
 | |
|     /*
 | |
|     ** Check if screen exists.
 | |
|     */
 | |
|     if (screen >= screenInfo.numScreens) {
 | |
| 	client->errorValue = screen;
 | |
| 	*err = BadValue;
 | |
| 	return FALSE;
 | |
|     }
 | |
|     *pGlxScreen = glxGetScreen(screenInfo.screens[screen]);
 | |
| 
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| static int
 | |
| validGlxFBConfig(ClientPtr client, __GLXscreen *pGlxScreen, XID id,
 | |
| 		 __GLXconfig **config, int *err)
 | |
| {
 | |
|     __GLXconfig *m;
 | |
| 
 | |
|     for (m = pGlxScreen->fbconfigs; m != NULL; m = m->next)
 | |
| 	if (m->fbconfigID == id) {
 | |
| 	    *config = m;
 | |
| 	    return TRUE;
 | |
| 	}
 | |
| 
 | |
|     client->errorValue = id;
 | |
|     *err = __glXError(GLXBadFBConfig);
 | |
| 
 | |
|     return FALSE;
 | |
| }
 | |
| 
 | |
| static int
 | |
| validGlxVisual(ClientPtr client, __GLXscreen *pGlxScreen, XID id,
 | |
| 	       __GLXconfig **config, int *err)
 | |
| {
 | |
|     int i;
 | |
| 
 | |
|     for (i = 0; i < pGlxScreen->numVisuals; i++)
 | |
|  	if (pGlxScreen->visuals[i]->visualID == id) {
 | |
| 	    *config = pGlxScreen->visuals[i];
 | |
| 	    return TRUE;
 | |
| 	}
 | |
| 
 | |
|     client->errorValue = id;
 | |
|     *err = BadValue;
 | |
| 
 | |
|     return FALSE;
 | |
| }
 | |
| 
 | |
| static int
 | |
| validGlxFBConfigForWindow(ClientPtr client, __GLXconfig *config,
 | |
| 			  DrawablePtr pDraw, int *err)
 | |
| {
 | |
|     ScreenPtr pScreen = pDraw->pScreen;
 | |
|     VisualPtr pVisual = NULL;
 | |
|     XID vid;
 | |
|     int i;
 | |
| 
 | |
|     vid = wVisual((WindowPtr)pDraw);
 | |
|     for (i = 0; i < pScreen->numVisuals; i++) {
 | |
| 	if (pScreen->visuals[i].vid == vid) {
 | |
| 	    pVisual = &pScreen->visuals[i];
 | |
| 	    break;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     /* FIXME: What exactly should we check here... */
 | |
|     if (pVisual->class != glxConvertToXVisualType(config->visualType) ||
 | |
| 	!(config->drawableType & GLX_WINDOW_BIT)) {
 | |
| 	client->errorValue = pDraw->id;
 | |
| 	*err = BadMatch;
 | |
| 	return FALSE;
 | |
|     }
 | |
| 
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| static int
 | |
| validGlxContext(ClientPtr client, XID id, int access_mode,
 | |
| 		__GLXcontext **context, int *err)
 | |
| {
 | |
|     *err = dixLookupResourceByType((pointer *) context, id,
 | |
| 				   __glXContextRes, client, access_mode);
 | |
|     if (*err != Success) {
 | |
| 	client->errorValue = id;
 | |
| 	if (*err == BadValue)
 | |
| 	    *err = __glXError(GLXBadContext);
 | |
| 	return FALSE;
 | |
|     }
 | |
| 
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| static int
 | |
| validGlxDrawable(ClientPtr client, XID id, int type, int access_mode,
 | |
| 		 __GLXdrawable **drawable, int *err)
 | |
| {
 | |
|     int rc;
 | |
| 
 | |
|     rc = dixLookupResourceByType((pointer *) drawable, id,
 | |
| 				 __glXDrawableRes, client, access_mode);
 | |
|     if (rc != Success && rc != BadValue) {
 | |
| 	*err = rc;
 | |
| 	client->errorValue = id;
 | |
| 	return FALSE;
 | |
|     }
 | |
| 
 | |
|     /* If the ID of the glx drawable we looked up doesn't match the id
 | |
|      * we looked for, it's because we looked it up under the X
 | |
|      * drawable ID (see DoCreateGLXDrawable). */
 | |
|     if (rc == BadValue ||
 | |
| 	(*drawable)->drawId != id ||
 | |
| 	(type != GLX_DRAWABLE_ANY && type != (*drawable)->type)) {
 | |
| 	client->errorValue = id;
 | |
| 	switch (type) {
 | |
| 	case GLX_DRAWABLE_WINDOW:
 | |
| 	    *err = __glXError(GLXBadWindow);
 | |
| 	    return FALSE;
 | |
| 	case GLX_DRAWABLE_PIXMAP:
 | |
| 	    *err = __glXError(GLXBadPixmap);
 | |
| 	    return FALSE;
 | |
| 	case GLX_DRAWABLE_PBUFFER:
 | |
| 	    *err = __glXError(GLXBadPbuffer);
 | |
| 	    return FALSE;
 | |
| 	case GLX_DRAWABLE_ANY:
 | |
| 	    *err = __glXError(GLXBadDrawable);
 | |
| 	    return FALSE;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| void
 | |
| __glXContextDestroy(__GLXcontext *context)
 | |
| {
 | |
|     __glXFlushContextCache();
 | |
| }
 | |
| 
 | |
| static void __glXdirectContextDestroy(__GLXcontext *context)
 | |
| {
 | |
|     __glXContextDestroy(context);
 | |
|     free(context);
 | |
| }
 | |
| 
 | |
| static __GLXcontext *__glXdirectContextCreate(__GLXscreen *screen,
 | |
| 					      __GLXconfig *modes,
 | |
| 					      __GLXcontext *shareContext)
 | |
| {
 | |
|     __GLXcontext *context;
 | |
| 
 | |
|     context = calloc(1, sizeof (__GLXcontext));
 | |
|     if (context == NULL)
 | |
| 	return NULL;
 | |
| 
 | |
|     context->destroy = __glXdirectContextDestroy;
 | |
| 
 | |
|     return context;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Create a GL context with the given properties.  This routine is used
 | |
|  * to implement \c glXCreateContext, \c glXCreateNewContext, and
 | |
|  * \c glXCreateContextWithConfigSGIX.  This works becuase of the hack way
 | |
|  * that GLXFBConfigs are implemented.  Basically, the FBConfigID is the
 | |
|  * same as the VisualID.
 | |
|  */
 | |
| 
 | |
| static int
 | |
| DoCreateContext(__GLXclientState *cl, GLXContextID gcId,
 | |
| 		GLXContextID shareList, __GLXconfig *config,
 | |
| 		__GLXscreen *pGlxScreen, GLboolean isDirect)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     __GLXcontext *glxc, *shareglxc;
 | |
|     int err;
 | |
|     
 | |
|     LEGAL_NEW_RESOURCE(gcId, client);
 | |
| 
 | |
|     /*
 | |
|     ** Find the display list space that we want to share.  
 | |
|     **
 | |
|     ** NOTE: In a multithreaded X server, we would need to keep a reference
 | |
|     ** count for each display list so that if one client detroyed a list that 
 | |
|     ** another client was using, the list would not really be freed until it 
 | |
|     ** was no longer in use.  Since this sample implementation has no support 
 | |
|     ** for multithreaded servers, we don't do this.  
 | |
|     */
 | |
|     if (shareList == None) {
 | |
| 	shareglxc = 0;
 | |
|     } else {
 | |
| 	if (!validGlxContext(client, shareList, DixReadAccess,
 | |
| 			     &shareglxc, &err))
 | |
| 	    return err;
 | |
| 
 | |
| 	if (shareglxc->isDirect) {
 | |
| 	    /*
 | |
| 	    ** NOTE: no support for sharing display lists between direct
 | |
| 	    ** contexts, even if they are in the same address space.
 | |
| 	    */
 | |
| #if 0
 | |
|             /* Disabling this code seems to allow shared display lists
 | |
|              * and texture objects to work.  We'll leave it disabled for now.
 | |
|              */
 | |
| 	    client->errorValue = shareList;
 | |
| 	    return BadMatch;
 | |
| #endif
 | |
| 	} else {
 | |
| 	    /*
 | |
| 	    ** Create an indirect context regardless of what the client asked
 | |
| 	    ** for; this way we can share display list space with shareList.
 | |
| 	    */
 | |
| 	    isDirect = GL_FALSE;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|     ** Allocate memory for the new context
 | |
|     */
 | |
|     if (!isDirect)
 | |
| 	glxc = pGlxScreen->createContext(pGlxScreen, config, shareglxc);
 | |
|     else
 | |
| 	glxc = __glXdirectContextCreate(pGlxScreen, config, shareglxc);
 | |
|     if (!glxc) {
 | |
| 	return BadAlloc;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|     ** Initially, setup the part of the context that could be used by
 | |
|     ** a GL core that needs windowing information (e.g., Mesa).
 | |
|     */
 | |
|     glxc->pGlxScreen = pGlxScreen;
 | |
|     glxc->config = config;
 | |
| 
 | |
|     /*
 | |
|     ** Register this context as a resource.
 | |
|     */
 | |
|     if (!AddResource(gcId, __glXContextRes, (pointer)glxc)) {
 | |
| 	(*glxc->destroy)(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;
 | |
|     glxc->isDirect = isDirect;
 | |
|     glxc->renderMode = GL_RENDER;
 | |
| 
 | |
|     __glXAddToContextList(glxc);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_CreateContext(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXCreateContextReq *req = (xGLXCreateContextReq *) pc;
 | |
|     __GLXconfig *config;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxScreen(cl->client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
|     if (!validGlxVisual(cl->client, pGlxScreen, req->visual, &config, &err))
 | |
| 	return err;
 | |
| 
 | |
|     return DoCreateContext(cl, req->context, req->shareList,
 | |
| 			   config, pGlxScreen, req->isDirect);
 | |
| }
 | |
| 
 | |
| int __glXDisp_CreateNewContext(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXCreateNewContextReq *req = (xGLXCreateNewContextReq *) pc;
 | |
|     __GLXconfig *config;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxScreen(cl->client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
|     if (!validGlxFBConfig(cl->client, pGlxScreen, req->fbconfig, &config, &err))
 | |
| 	return err;
 | |
| 
 | |
|     return DoCreateContext(cl, req->context, req->shareList,
 | |
| 			   config, pGlxScreen, req->isDirect);
 | |
| }
 | |
| 
 | |
| int __glXDisp_CreateContextWithConfigSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXCreateContextWithConfigSGIXReq *req = 
 | |
| 	(xGLXCreateContextWithConfigSGIXReq *) pc;
 | |
|     __GLXconfig *config;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxScreen(cl->client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
|     if (!validGlxFBConfig(cl->client, pGlxScreen, req->fbconfig, &config, &err))
 | |
| 	return err;
 | |
| 
 | |
|     return DoCreateContext(cl, req->context, req->shareList,
 | |
| 			   config, pGlxScreen, req->isDirect);
 | |
| }
 | |
| int __glXDisp_DestroyContext(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXDestroyContextReq *req = (xGLXDestroyContextReq *) pc;
 | |
|     __GLXcontext *glxc;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxContext(cl->client, req->context, DixDestroyAccess,
 | |
| 			 &glxc, &err))
 | |
| 	    return err;
 | |
| 
 | |
|     FreeResourceByType(req->context, __glXContextRes, FALSE);
 | |
|     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)
 | |
| {
 | |
|     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 **) malloc(sizeof(__GLXcontext *));
 | |
|     } else {
 | |
| 	table = (__GLXcontext **) realloc(table,
 | |
| 					   (num+1)*sizeof(__GLXcontext *));
 | |
|     }
 | |
|     table[num] = glxc;
 | |
|     cl->currentContexts = table;
 | |
|     cl->numCurrentContexts++;
 | |
|     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;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** 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];
 | |
|     }
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| 
 | |
| 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;
 | |
|     __glXLastContext = glxc;	
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * This is a helper function to handle the legacy (pre GLX 1.3) cases
 | |
|  * where passing an X window to glXMakeCurrent is valid.  Given a
 | |
|  * resource ID, look up the GLX drawable if available, otherwise, make
 | |
|  * sure it's an X window and create a GLX drawable one the fly.
 | |
|  */
 | |
| static __GLXdrawable *
 | |
| __glXGetDrawable(__GLXcontext *glxc, GLXDrawable drawId, ClientPtr client,
 | |
| 		 int *error)
 | |
| {
 | |
|     DrawablePtr pDraw;
 | |
|     __GLXdrawable *pGlxDraw;
 | |
|     int rc;
 | |
| 
 | |
|     if (validGlxDrawable(client, drawId, GLX_DRAWABLE_ANY,
 | |
| 			 DixWriteAccess, &pGlxDraw, &rc)) {
 | |
| 	if (glxc != NULL && pGlxDraw->config != glxc->config) {
 | |
| 	    client->errorValue = drawId;
 | |
| 	    *error = BadMatch;
 | |
| 	    return NULL;
 | |
| 	}
 | |
| 
 | |
| 	return pGlxDraw;
 | |
|     }
 | |
| 
 | |
|     /* No active context and an unknown drawable, bail. */
 | |
|     if (glxc == NULL) {
 | |
| 	    client->errorValue = drawId;
 | |
| 	    *error = BadMatch;
 | |
| 	    return NULL;
 | |
|     }
 | |
| 
 | |
|     /* The drawId wasn't a GLX drawable.  Make sure it's a window and
 | |
|      * create a GLXWindow for it.  Check that the drawable screen
 | |
|      * matches the context screen and that the context fbconfig is
 | |
|      * compatible with the window visual. */
 | |
| 
 | |
|     rc = dixLookupDrawable(&pDraw, drawId, client, 0, DixGetAttrAccess);
 | |
|     if (rc != Success || pDraw->type != DRAWABLE_WINDOW) {
 | |
| 	client->errorValue = drawId;
 | |
| 	*error = __glXError(GLXBadDrawable);
 | |
| 	return NULL;
 | |
|     }
 | |
| 
 | |
|     if (pDraw->pScreen != glxc->pGlxScreen->pScreen) {
 | |
| 	client->errorValue = pDraw->pScreen->myNum;
 | |
| 	*error = BadMatch;
 | |
| 	return NULL;
 | |
|     }
 | |
| 
 | |
|     if (!validGlxFBConfigForWindow(client, glxc->config, pDraw, error))
 | |
| 	return NULL;
 | |
| 
 | |
|     pGlxDraw = glxc->pGlxScreen->createDrawable(client, glxc->pGlxScreen,
 | |
| 						pDraw, drawId,
 | |
| 						GLX_DRAWABLE_WINDOW,
 | |
| 						drawId, glxc->config);
 | |
| 
 | |
|     /* since we are creating the drawablePrivate, drawId should be new */
 | |
|     if (!AddResource(drawId, __glXDrawableRes, pGlxDraw)) {
 | |
| 	pGlxDraw->destroy (pGlxDraw);
 | |
| 	*error = BadAlloc;
 | |
| 	return NULL;
 | |
|     }
 | |
| 
 | |
|     return pGlxDraw;
 | |
| }
 | |
| 
 | |
| /*****************************************************************************/
 | |
| /*
 | |
| ** Make an OpenGL context and drawable current.
 | |
| */
 | |
| 
 | |
| static int
 | |
| DoMakeCurrent(__GLXclientState *cl,
 | |
| 	      GLXDrawable drawId, GLXDrawable readId,
 | |
| 	      GLXContextID contextId, GLXContextTag tag)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXMakeCurrentReply reply;
 | |
|     __GLXcontext *glxc, *prevglxc;
 | |
|     __GLXdrawable *drawPriv = NULL;
 | |
|     __GLXdrawable *readPriv = NULL;
 | |
|     int error;
 | |
|     GLuint  mask;
 | |
| 
 | |
|     /*
 | |
|     ** If one is None and the other isn't, it's a bad match.
 | |
|     */
 | |
| 
 | |
|     mask  = (drawId == None)    ? (1 << 0) : 0;
 | |
|     mask |= (readId == None)    ? (1 << 1) : 0;
 | |
|     mask |= (contextId == None) ? (1 << 2) : 0;
 | |
| 
 | |
|     if ( (mask != 0x00) && (mask != 0x07) ) {
 | |
| 	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 __glXError(GLXBadContextTag);
 | |
| 	}
 | |
| 	if (prevglxc->renderMode != GL_RENDER) {
 | |
| 	    /* Oops.  Not in render mode render. */
 | |
| 	    client->errorValue = prevglxc->id;
 | |
| 	    return __glXError(GLXBadContextState);
 | |
| 	}
 | |
|     } else {
 | |
| 	prevglxc = 0;
 | |
|     }
 | |
| 
 | |
|     /*
 | |
|     ** Lookup new context.  It must not be current for someone else.
 | |
|     */
 | |
|     if (contextId != None) {
 | |
| 	int  status;
 | |
| 
 | |
| 	if (!validGlxContext(client, contextId, DixUseAccess, &glxc, &error))
 | |
| 	    return error;
 | |
| 	if ((glxc != prevglxc) && glxc->isCurrent) {
 | |
| 	    /* Context is current to somebody else */
 | |
| 	    return BadAccess;
 | |
| 	}
 | |
| 
 | |
| 	assert( drawId != None );
 | |
| 	assert( readId != None );
 | |
| 
 | |
| 	drawPriv = __glXGetDrawable(glxc, drawId, client, &status);
 | |
| 	if (drawPriv == NULL)
 | |
| 	    return status;
 | |
| 
 | |
| 	readPriv = __glXGetDrawable(glxc, readId, client, &status);
 | |
| 	if (readPriv == NULL)
 | |
| 	    return status;
 | |
| 
 | |
|     } else {
 | |
| 	/* Switching to no context.  Ignore new drawable. */
 | |
| 	glxc = 0;
 | |
| 	drawPriv = 0;
 | |
| 	readPriv = 0;
 | |
|     }
 | |
| 
 | |
| 
 | |
|     if (prevglxc) {
 | |
| 	/*
 | |
| 	** Flush the previous context if needed.
 | |
| 	*/
 | |
| 	if (__GLX_HAS_UNFLUSHED_CMDS(prevglxc)) {
 | |
| 	    if (__glXForceCurrent(cl, tag, (int *)&error)) {
 | |
| 		CALL_Flush( GET_DISPATCH(), () );
 | |
| 		__GLX_NOTE_FLUSHED_CMDS(prevglxc);
 | |
| 	    } else {
 | |
| 		return error;
 | |
| 	    }
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	** Make the previous context not current.
 | |
| 	*/
 | |
| 	if (!(*prevglxc->loseCurrent)(prevglxc)) {
 | |
| 	    return __glXError(GLXBadContext);
 | |
| 	}
 | |
| 	__glXFlushContextCache();
 | |
| 	if (!prevglxc->isDirect) {
 | |
| 	    prevglxc->drawPriv = NULL;
 | |
| 	    prevglxc->readPriv = NULL;
 | |
| 	}
 | |
|     }
 | |
| 	
 | |
| 
 | |
|     if ((glxc != 0) && !glxc->isDirect) {
 | |
| 
 | |
| 	glxc->drawPriv = drawPriv;
 | |
| 	glxc->readPriv = readPriv;
 | |
| 
 | |
| 	/* make the context current */
 | |
| 	if (!(*glxc->makeCurrent)(glxc)) {
 | |
| 	    glxc->drawPriv = NULL;
 | |
| 	    glxc->readPriv = NULL;
 | |
| 	    return __glXError(GLXBadContext);
 | |
| 	}
 | |
| 
 | |
| 	glxc->isCurrent = GL_TRUE;
 | |
|     }
 | |
| 
 | |
|     if (prevglxc) {
 | |
| 	ChangeCurrentContext(cl, glxc, tag);
 | |
| 	StopUsingContext(prevglxc);
 | |
|     } else {
 | |
| 	tag = AddCurrentContext(cl, glxc);
 | |
|     }
 | |
| 
 | |
|     if (glxc) {
 | |
| 	StartUsingContext(cl, glxc);
 | |
| 	reply.contextTag = tag;
 | |
|     } else {
 | |
| 	reply.contextTag = 0;
 | |
|     }
 | |
| 
 | |
|     reply.length = 0;
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
| 
 | |
|     if (client->swapped) {
 | |
| 	__glXSwapMakeCurrentReply(client, &reply);
 | |
|     } else {
 | |
| 	WriteToClient(client, sz_xGLXMakeCurrentReply, (char *)&reply);
 | |
|     }
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_MakeCurrent(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXMakeCurrentReq *req = (xGLXMakeCurrentReq *) pc;
 | |
| 
 | |
|     return DoMakeCurrent( cl, req->drawable, req->drawable,
 | |
| 			  req->context, req->oldContextTag );
 | |
| }
 | |
| 
 | |
| int __glXDisp_MakeContextCurrent(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXMakeContextCurrentReq *req = (xGLXMakeContextCurrentReq *) pc;
 | |
| 
 | |
|     return DoMakeCurrent( cl, req->drawable, req->readdrawable,
 | |
| 			  req->context, req->oldContextTag );
 | |
| }
 | |
| 
 | |
| int __glXDisp_MakeCurrentReadSGI(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXMakeCurrentReadSGIReq *req = (xGLXMakeCurrentReadSGIReq *) pc;
 | |
| 
 | |
|     return DoMakeCurrent( cl, req->drawable, req->readable,
 | |
| 			  req->context, req->oldContextTag );
 | |
| }
 | |
| 
 | |
| int __glXDisp_IsDirect(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXIsDirectReq *req = (xGLXIsDirectReq *) pc;
 | |
|     xGLXIsDirectReply reply;
 | |
|     __GLXcontext *glxc;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxContext(cl->client, req->context, DixReadAccess, &glxc, &err))
 | |
| 	return err;
 | |
| 
 | |
|     reply.isDirect = glxc->isDirect;
 | |
|     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 __glXDisp_QueryVersion(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXQueryVersionReq *req = (xGLXQueryVersionReq *) pc;
 | |
|     xGLXQueryVersionReply reply;
 | |
|     GLuint major, minor;
 | |
| 
 | |
|     major = req->majorVersion;
 | |
|     minor = req->minorVersion;
 | |
|     (void)major;
 | |
|     (void)minor;
 | |
| 
 | |
|     /*
 | |
|     ** 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 = glxMajorVersion;
 | |
|     reply.minorVersion = glxMinorVersion;
 | |
|     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 __glXDisp_WaitGL(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXWaitGLReq *req = (xGLXWaitGLReq *)pc;
 | |
|     GLXContextTag tag = req->contextTag;
 | |
|     __GLXcontext *glxc = NULL;
 | |
|     int error;
 | |
| 
 | |
|     if (tag) {
 | |
| 	glxc = __glXLookupContextByTag(cl, tag);
 | |
| 	if (!glxc)
 | |
| 	    return __glXError(GLXBadContextTag);
 | |
|     
 | |
| 	if (!__glXForceCurrent(cl, req->contextTag, &error))
 | |
| 	    return error;
 | |
| 
 | |
| 	CALL_Finish( GET_DISPATCH(), () );
 | |
|     }
 | |
| 
 | |
|     if (glxc && glxc->drawPriv->waitGL)
 | |
| 	(*glxc->drawPriv->waitGL)(glxc->drawPriv);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_WaitX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXWaitXReq *req = (xGLXWaitXReq *)pc;
 | |
|     GLXContextTag tag = req->contextTag;
 | |
|     __GLXcontext *glxc = NULL;
 | |
|     int error;
 | |
| 
 | |
|     if (tag) {
 | |
| 	glxc = __glXLookupContextByTag(cl, tag);
 | |
| 	if (!glxc)
 | |
| 	    return __glXError(GLXBadContextTag);
 | |
|     
 | |
| 	if (!__glXForceCurrent(cl, req->contextTag, &error))
 | |
| 	    return error;
 | |
|     }
 | |
| 
 | |
|     if (glxc && glxc->drawPriv->waitX)
 | |
| 	(*glxc->drawPriv->waitX)(glxc->drawPriv);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_CopyContext(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     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 error;
 | |
| 
 | |
|     if (!validGlxContext(cl->client, source, DixReadAccess, &src, &error))
 | |
| 	return error;
 | |
|     if (!validGlxContext(cl->client, dest, DixWriteAccess, &dst, &error))
 | |
| 	return error;
 | |
| 
 | |
|     /*
 | |
|     ** They must be in the same address space, and same screen.
 | |
|     ** NOTE: no support for direct rendering contexts here.
 | |
|     */
 | |
|     if (src->isDirect || dst->isDirect ||
 | |
| 	(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 __glXError(GLXBadContextTag);
 | |
| 	}
 | |
| 	if (tagcx != src) {
 | |
| 	    /*
 | |
| 	    ** This would be caused by a faulty implementation of the client
 | |
| 	    ** library.
 | |
| 	    */
 | |
| 	    return BadMatch;
 | |
| 	}
 | |
| 	/*
 | |
| 	** In this case, glXCopyContext is in both GL and X streams, in terms
 | |
| 	** of sequentiality.
 | |
| 	*/
 | |
| 	if (__glXForceCurrent(cl, tag, &error)) {
 | |
| 	    /*
 | |
| 	    ** Do whatever is needed to make sure that all preceding requests
 | |
| 	    ** in both streams are completed before the copy is executed.
 | |
| 	    */
 | |
| 	    CALL_Finish( GET_DISPATCH(), () );
 | |
| 	    __GLX_NOTE_FLUSHED_CMDS(tagcx);
 | |
| 	} else {
 | |
| 	    return error;
 | |
| 	}
 | |
|     }
 | |
|     /*
 | |
|     ** Issue copy.  The only reason for failure is a bad mask.
 | |
|     */
 | |
|     if (!(*dst->copy)(dst, src, mask)) {
 | |
| 	client->errorValue = mask;
 | |
| 	return BadValue;
 | |
|     }
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| enum {
 | |
|     GLX_VIS_CONFIG_UNPAIRED = 18,
 | |
|     GLX_VIS_CONFIG_PAIRED = 20
 | |
| };
 | |
| 
 | |
| enum {
 | |
|     GLX_VIS_CONFIG_TOTAL = GLX_VIS_CONFIG_UNPAIRED + GLX_VIS_CONFIG_PAIRED
 | |
| };
 | |
| 
 | |
| int __glXDisp_GetVisualConfigs(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXGetVisualConfigsReq *req = (xGLXGetVisualConfigsReq *) pc;
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXGetVisualConfigsReply reply;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     __GLXconfig *modes;
 | |
|     CARD32 buf[GLX_VIS_CONFIG_TOTAL];
 | |
|     int p, i, err;
 | |
|     __GLX_DECLARE_SWAP_VARIABLES;
 | |
|     __GLX_DECLARE_SWAP_ARRAY_VARIABLES;
 | |
| 
 | |
|     if (!validGlxScreen(cl->client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
| 
 | |
|     reply.numVisuals = pGlxScreen->numVisuals;
 | |
|     reply.numProps = GLX_VIS_CONFIG_TOTAL;
 | |
|     reply.length = (reply.numVisuals * __GLX_SIZE_CARD32 * GLX_VIS_CONFIG_TOTAL) >> 2;
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
| 
 | |
|     if (client->swapped) {
 | |
| 	__GLX_SWAP_SHORT(&reply.sequenceNumber);
 | |
| 	__GLX_SWAP_INT(&reply.length);
 | |
| 	__GLX_SWAP_INT(&reply.numVisuals);
 | |
| 	__GLX_SWAP_INT(&reply.numProps);
 | |
|     }
 | |
| 
 | |
|     WriteToClient(client, sz_xGLXGetVisualConfigsReply, (char *)&reply);
 | |
| 
 | |
|     for (i = 0; i < pGlxScreen->numVisuals; i++) {
 | |
| 	modes = pGlxScreen->visuals[i];
 | |
| 
 | |
| 	p = 0;
 | |
| 	buf[p++] = modes->visualID;
 | |
| 	buf[p++] = glxConvertToXVisualType( modes->visualType );
 | |
| 	buf[p++] = (modes->renderType & GLX_RGBA_BIT) ? GL_TRUE : GL_FALSE;
 | |
| 
 | |
| 	buf[p++] = modes->redBits;
 | |
| 	buf[p++] = modes->greenBits;
 | |
| 	buf[p++] = modes->blueBits;
 | |
| 	buf[p++] = modes->alphaBits;
 | |
| 	buf[p++] = modes->accumRedBits;
 | |
| 	buf[p++] = modes->accumGreenBits;
 | |
| 	buf[p++] = modes->accumBlueBits;
 | |
| 	buf[p++] = modes->accumAlphaBits;
 | |
| 
 | |
| 	buf[p++] = modes->doubleBufferMode;
 | |
| 	buf[p++] = modes->stereoMode;
 | |
| 
 | |
| 	buf[p++] = modes->rgbBits;
 | |
| 	buf[p++] = modes->depthBits;
 | |
| 	buf[p++] = modes->stencilBits;
 | |
| 	buf[p++] = modes->numAuxBuffers;
 | |
| 	buf[p++] = modes->level;
 | |
| 
 | |
| 	assert(p == GLX_VIS_CONFIG_UNPAIRED);
 | |
| 	/* 
 | |
| 	** Add token/value pairs for extensions.
 | |
| 	*/
 | |
| 	buf[p++] = GLX_VISUAL_CAVEAT_EXT;
 | |
| 	buf[p++] = modes->visualRating;
 | |
| 	buf[p++] = GLX_TRANSPARENT_TYPE;
 | |
| 	buf[p++] = modes->transparentPixel;
 | |
| 	buf[p++] = GLX_TRANSPARENT_RED_VALUE;
 | |
| 	buf[p++] = modes->transparentRed;
 | |
| 	buf[p++] = GLX_TRANSPARENT_GREEN_VALUE;
 | |
| 	buf[p++] = modes->transparentGreen;
 | |
| 	buf[p++] = GLX_TRANSPARENT_BLUE_VALUE;
 | |
| 	buf[p++] = modes->transparentBlue;
 | |
| 	buf[p++] = GLX_TRANSPARENT_ALPHA_VALUE;
 | |
| 	buf[p++] = modes->transparentAlpha;
 | |
| 	buf[p++] = GLX_TRANSPARENT_INDEX_VALUE;
 | |
| 	buf[p++] = modes->transparentIndex;
 | |
| 	buf[p++] = GLX_SAMPLES_SGIS;
 | |
| 	buf[p++] = modes->samples;
 | |
| 	buf[p++] = GLX_SAMPLE_BUFFERS_SGIS;
 | |
| 	buf[p++] = modes->sampleBuffers;
 | |
| 	buf[p++] = 0; /* copy over visualSelectGroup (GLX_VISUAL_SELECT_GROUP_SGIX)? */
 | |
| 	buf[p++] = 0;
 | |
| 
 | |
| 	assert(p == GLX_VIS_CONFIG_TOTAL);
 | |
| 	if (client->swapped) {
 | |
| 	    __GLX_SWAP_INT_ARRAY(buf, p);
 | |
| 	}
 | |
| 	WriteToClient(client, __GLX_SIZE_CARD32 * p, (char *)buf);
 | |
|     }
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| #define __GLX_TOTAL_FBCONFIG_ATTRIBS (36)
 | |
| #define __GLX_FBCONFIG_ATTRIBS_LENGTH (__GLX_TOTAL_FBCONFIG_ATTRIBS * 2)
 | |
| /**
 | |
|  * Send the set of GLXFBConfigs to the client.  There is not currently
 | |
|  * and interface into the driver on the server-side to get GLXFBConfigs,
 | |
|  * so we "invent" some based on the \c __GLXvisualConfig structures that
 | |
|  * the driver does supply.
 | |
|  * 
 | |
|  * The reply format for both \c glXGetFBConfigs and \c glXGetFBConfigsSGIX
 | |
|  * is the same, so this routine pulls double duty.
 | |
|  */
 | |
| 
 | |
| static int
 | |
| DoGetFBConfigs(__GLXclientState *cl, unsigned screen)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXGetFBConfigsReply reply;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     CARD32 buf[__GLX_FBCONFIG_ATTRIBS_LENGTH];
 | |
|     int p, err;
 | |
|     __GLXconfig *modes;
 | |
|     __GLX_DECLARE_SWAP_VARIABLES;
 | |
|     __GLX_DECLARE_SWAP_ARRAY_VARIABLES;
 | |
| 
 | |
|     if (!validGlxScreen(cl->client, screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
| 
 | |
|     reply.numFBConfigs = pGlxScreen->numFBConfigs;
 | |
|     reply.numAttribs = __GLX_TOTAL_FBCONFIG_ATTRIBS;
 | |
|     reply.length = (__GLX_FBCONFIG_ATTRIBS_LENGTH * reply.numFBConfigs);
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
| 
 | |
|     if (client->swapped) {
 | |
| 	__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 (modes = pGlxScreen->fbconfigs; modes != NULL; modes = modes->next) {
 | |
| 	p = 0;
 | |
| 
 | |
| #define WRITE_PAIR(tag,value) \
 | |
|     do { buf[p++] = tag ; buf[p++] = value ; } while( 0 )
 | |
| 
 | |
| 	WRITE_PAIR( GLX_VISUAL_ID,        modes->visualID );
 | |
| 	WRITE_PAIR( GLX_FBCONFIG_ID,      modes->fbconfigID );
 | |
| 	WRITE_PAIR( GLX_X_RENDERABLE,     GL_TRUE );
 | |
| 
 | |
| 	WRITE_PAIR( GLX_RGBA,
 | |
| 		    (modes->renderType & GLX_RGBA_BIT) ? GL_TRUE : GL_FALSE );
 | |
| 	WRITE_PAIR( GLX_RENDER_TYPE,      modes->renderType );
 | |
| 	WRITE_PAIR( GLX_DOUBLEBUFFER,     modes->doubleBufferMode );
 | |
| 	WRITE_PAIR( GLX_STEREO,           modes->stereoMode );
 | |
| 
 | |
| 	WRITE_PAIR( GLX_BUFFER_SIZE,      modes->rgbBits );
 | |
| 	WRITE_PAIR( GLX_LEVEL,            modes->level );
 | |
| 	WRITE_PAIR( GLX_AUX_BUFFERS,      modes->numAuxBuffers );
 | |
| 	WRITE_PAIR( GLX_RED_SIZE,         modes->redBits );
 | |
| 	WRITE_PAIR( GLX_GREEN_SIZE,       modes->greenBits );
 | |
| 	WRITE_PAIR( GLX_BLUE_SIZE,        modes->blueBits );
 | |
| 	WRITE_PAIR( GLX_ALPHA_SIZE,       modes->alphaBits );
 | |
| 	WRITE_PAIR( GLX_ACCUM_RED_SIZE,   modes->accumRedBits );
 | |
| 	WRITE_PAIR( GLX_ACCUM_GREEN_SIZE, modes->accumGreenBits );
 | |
| 	WRITE_PAIR( GLX_ACCUM_BLUE_SIZE,  modes->accumBlueBits );
 | |
| 	WRITE_PAIR( GLX_ACCUM_ALPHA_SIZE, modes->accumAlphaBits );
 | |
| 	WRITE_PAIR( GLX_DEPTH_SIZE,       modes->depthBits );
 | |
| 	WRITE_PAIR( GLX_STENCIL_SIZE,     modes->stencilBits );
 | |
| 	WRITE_PAIR( GLX_X_VISUAL_TYPE,    modes->visualType );
 | |
| 	WRITE_PAIR( GLX_CONFIG_CAVEAT, modes->visualRating );
 | |
| 	WRITE_PAIR( GLX_TRANSPARENT_TYPE, modes->transparentPixel );
 | |
| 	WRITE_PAIR( GLX_TRANSPARENT_RED_VALUE, modes->transparentRed );
 | |
| 	WRITE_PAIR( GLX_TRANSPARENT_GREEN_VALUE, modes->transparentGreen );
 | |
| 	WRITE_PAIR( GLX_TRANSPARENT_BLUE_VALUE, modes->transparentBlue );
 | |
| 	WRITE_PAIR( GLX_TRANSPARENT_ALPHA_VALUE, modes->transparentAlpha );
 | |
| 	WRITE_PAIR( GLX_TRANSPARENT_INDEX_VALUE, modes->transparentIndex );
 | |
| 	WRITE_PAIR( GLX_SWAP_METHOD_OML, modes->swapMethod );
 | |
| 	WRITE_PAIR( GLX_SAMPLES_SGIS, modes->samples );
 | |
| 	WRITE_PAIR( GLX_SAMPLE_BUFFERS_SGIS, modes->sampleBuffers );
 | |
| 	/* GLX_VISUAL_SELECT_GROUP_SGIX ? */
 | |
| 	WRITE_PAIR( GLX_DRAWABLE_TYPE, modes->drawableType );
 | |
| 	WRITE_PAIR( GLX_BIND_TO_TEXTURE_RGB_EXT, modes->bindToTextureRgb );
 | |
| 	WRITE_PAIR( GLX_BIND_TO_TEXTURE_RGBA_EXT, modes->bindToTextureRgba );
 | |
| 	WRITE_PAIR( GLX_BIND_TO_MIPMAP_TEXTURE_EXT, modes->bindToMipmapTexture );
 | |
| 	WRITE_PAIR( GLX_BIND_TO_TEXTURE_TARGETS_EXT, modes->bindToTextureTargets );
 | |
| 
 | |
| 	if (client->swapped) {
 | |
| 	    __GLX_SWAP_INT_ARRAY(buf, __GLX_FBCONFIG_ATTRIBS_LENGTH);
 | |
| 	}
 | |
| 	WriteToClient(client, __GLX_SIZE_CARD32 * __GLX_FBCONFIG_ATTRIBS_LENGTH,
 | |
| 		      (char *)buf);
 | |
|     }
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| 
 | |
| int __glXDisp_GetFBConfigs(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXGetFBConfigsReq *req = (xGLXGetFBConfigsReq *) pc;
 | |
|     return DoGetFBConfigs(cl, req->screen);
 | |
| }
 | |
| 
 | |
| int __glXDisp_GetFBConfigsSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXGetFBConfigsSGIXReq *req = (xGLXGetFBConfigsSGIXReq *) pc;
 | |
|     return DoGetFBConfigs(cl, req->screen);
 | |
| }
 | |
| 
 | |
| GLboolean
 | |
| __glXDrawableInit(__GLXdrawable *drawable,
 | |
| 		  __GLXscreen *screen, DrawablePtr pDraw, int type,
 | |
| 		  XID drawId, __GLXconfig *config)
 | |
| {
 | |
|     drawable->pDraw = pDraw;
 | |
|     drawable->type = type;
 | |
|     drawable->drawId = drawId;
 | |
|     drawable->config = config;
 | |
|     drawable->eventMask = 0;
 | |
| 
 | |
|     return GL_TRUE;
 | |
| }
 | |
| 
 | |
| void
 | |
| __glXDrawableRelease(__GLXdrawable *drawable)
 | |
| {
 | |
| }
 | |
| 
 | |
| static int 
 | |
| DoCreateGLXDrawable(ClientPtr client, __GLXscreen *pGlxScreen,
 | |
| 		    __GLXconfig *config, DrawablePtr pDraw, XID drawableId,
 | |
| 		    XID glxDrawableId, int type)
 | |
| {
 | |
|     __GLXdrawable *pGlxDraw;
 | |
| 
 | |
|     if (pGlxScreen->pScreen != pDraw->pScreen)
 | |
| 	return BadMatch;
 | |
| 
 | |
|     pGlxDraw = pGlxScreen->createDrawable(client, pGlxScreen, pDraw,
 | |
| 					  drawableId, type,
 | |
| 					  glxDrawableId, config);
 | |
|     if (pGlxDraw == NULL)
 | |
| 	return BadAlloc;
 | |
| 
 | |
|     if (!AddResource(glxDrawableId, __glXDrawableRes, pGlxDraw)) {
 | |
| 	pGlxDraw->destroy (pGlxDraw);
 | |
| 	return BadAlloc;
 | |
|     }
 | |
| 
 | |
|     /* Add the glx drawable under the XID of the underlying X drawable
 | |
|      * too.  That way we'll get a callback in DrawableGone and can
 | |
|      * clean up properly when the drawable is destroyed. */
 | |
|     if (drawableId != glxDrawableId &&
 | |
| 	!AddResource(pDraw->id, __glXDrawableRes, pGlxDraw)) {
 | |
| 	pGlxDraw->destroy (pGlxDraw);
 | |
| 	return BadAlloc;
 | |
|     }
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| DoCreateGLXPixmap(ClientPtr client, __GLXscreen *pGlxScreen, __GLXconfig *config,
 | |
| 		  XID drawableId, XID glxDrawableId)
 | |
| {
 | |
|     DrawablePtr pDraw;
 | |
|     int err;
 | |
| 
 | |
|     LEGAL_NEW_RESOURCE(glxDrawableId, client);
 | |
| 
 | |
|     err = dixLookupDrawable(&pDraw, drawableId, client, 0, DixAddAccess);
 | |
|     if (err != Success) {
 | |
| 	client->errorValue = drawableId;
 | |
| 	return err;
 | |
|     }
 | |
|     if (pDraw->type != DRAWABLE_PIXMAP) {
 | |
| 	client->errorValue = drawableId;
 | |
| 	return BadPixmap;
 | |
|     }
 | |
| 
 | |
|     err = DoCreateGLXDrawable(client, pGlxScreen, config, pDraw, drawableId,
 | |
| 			      glxDrawableId, GLX_DRAWABLE_PIXMAP);
 | |
| 
 | |
|     return err;
 | |
| }
 | |
| 
 | |
| static void
 | |
| determineTextureTarget(ClientPtr client, XID glxDrawableID,
 | |
| 		       CARD32 *attribs, CARD32 numAttribs)
 | |
| {
 | |
|     GLenum target = 0;
 | |
|     GLenum format = 0;
 | |
|     int i, err;
 | |
|     __GLXdrawable *pGlxDraw;
 | |
| 
 | |
|     if (!validGlxDrawable(client, glxDrawableID, GLX_DRAWABLE_PIXMAP,
 | |
| 			  DixWriteAccess, &pGlxDraw, &err))
 | |
| 	/* We just added it in CreatePixmap, so we should never get here. */
 | |
| 	return;
 | |
| 
 | |
|     for (i = 0; i < numAttribs; i++) {
 | |
| 	if (attribs[2 * i] == GLX_TEXTURE_TARGET_EXT) {
 | |
| 	    switch (attribs[2 * i + 1]) {
 | |
| 	    case GLX_TEXTURE_2D_EXT:
 | |
| 		target = GL_TEXTURE_2D;
 | |
| 		break;
 | |
| 	    case GLX_TEXTURE_RECTANGLE_EXT:
 | |
| 		target = GL_TEXTURE_RECTANGLE_ARB;
 | |
| 		break;
 | |
| 	    }
 | |
| 	}
 | |
| 
 | |
| 	if (attribs[2 * i] == GLX_TEXTURE_FORMAT_EXT)
 | |
| 		format = attribs[2 * i + 1];
 | |
|     }
 | |
|  
 | |
|     if (!target) {
 | |
| 	int w = pGlxDraw->pDraw->width, h = pGlxDraw->pDraw->height;
 | |
| 	
 | |
| 	if (h & (h - 1) || w & (w - 1))
 | |
| 	    target = GL_TEXTURE_RECTANGLE_ARB;
 | |
| 	else
 | |
| 	    target = GL_TEXTURE_2D;
 | |
|     }
 | |
| 
 | |
|     pGlxDraw->target = target;
 | |
|     pGlxDraw->format = format;
 | |
| }
 | |
| 
 | |
| int __glXDisp_CreateGLXPixmap(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXCreateGLXPixmapReq *req = (xGLXCreateGLXPixmapReq *) pc;
 | |
|     __GLXconfig *config;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxScreen(cl->client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
|     if (!validGlxVisual(cl->client, pGlxScreen, req->visual, &config, &err))
 | |
| 	return err;
 | |
| 
 | |
|     return DoCreateGLXPixmap(cl->client, pGlxScreen, config,
 | |
| 			     req->pixmap, req->glxpixmap);
 | |
| }
 | |
| 
 | |
| int __glXDisp_CreatePixmap(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXCreatePixmapReq *req = (xGLXCreatePixmapReq *) pc;
 | |
|     __GLXconfig *config;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxScreen(cl->client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
|     if (!validGlxFBConfig(cl->client, pGlxScreen, req->fbconfig, &config, &err))
 | |
| 	return err;
 | |
| 
 | |
|     err = DoCreateGLXPixmap(cl->client, pGlxScreen, config,
 | |
| 			    req->pixmap, req->glxpixmap);
 | |
|     if (err != Success)
 | |
| 	return err;
 | |
| 
 | |
|     determineTextureTarget(cl->client, req->glxpixmap,
 | |
| 			   (CARD32*) (req + 1), req->numAttribs);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_CreateGLXPixmapWithConfigSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXCreateGLXPixmapWithConfigSGIXReq *req = 
 | |
| 	(xGLXCreateGLXPixmapWithConfigSGIXReq *) pc;
 | |
|     __GLXconfig *config;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxScreen(cl->client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
|     if (!validGlxFBConfig(cl->client, pGlxScreen, req->fbconfig, &config, &err))
 | |
| 	return err;
 | |
| 
 | |
|     return DoCreateGLXPixmap(cl->client, pGlxScreen,
 | |
| 			     config, req->pixmap, req->glxpixmap);
 | |
| }
 | |
| 
 | |
| 
 | |
| static int DoDestroyDrawable(__GLXclientState *cl, XID glxdrawable, int type)
 | |
| {
 | |
|     __GLXdrawable *pGlxDraw;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxDrawable(cl->client, glxdrawable, type,
 | |
| 			  DixDestroyAccess, &pGlxDraw, &err))
 | |
| 	return err;
 | |
| 
 | |
|     FreeResource(glxdrawable, FALSE);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_DestroyGLXPixmap(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXDestroyGLXPixmapReq *req = (xGLXDestroyGLXPixmapReq *) pc;
 | |
| 
 | |
|     return DoDestroyDrawable(cl, req->glxpixmap, GLX_DRAWABLE_PIXMAP);
 | |
| }
 | |
| 
 | |
| int __glXDisp_DestroyPixmap(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXDestroyPixmapReq *req = (xGLXDestroyPixmapReq *) pc;
 | |
| 
 | |
|     return DoDestroyDrawable(cl, req->glxpixmap, GLX_DRAWABLE_PIXMAP);
 | |
| }
 | |
| 
 | |
| static int
 | |
| DoCreatePbuffer(ClientPtr client, int screenNum, XID fbconfigId,
 | |
| 		int width, int height, XID glxDrawableId)
 | |
| {
 | |
|     __GLXconfig	*config;
 | |
|     __GLXscreen		*pGlxScreen;
 | |
|     PixmapPtr		 pPixmap;
 | |
|     int			 err;
 | |
| 
 | |
|     LEGAL_NEW_RESOURCE(glxDrawableId, client);
 | |
| 
 | |
|     if (!validGlxScreen(client, screenNum, &pGlxScreen, &err))
 | |
| 	return err;
 | |
|     if (!validGlxFBConfig(client, pGlxScreen, fbconfigId, &config, &err))
 | |
| 	return err;
 | |
| 
 | |
|     __glXenterServer(GL_FALSE);
 | |
|     pPixmap = (*pGlxScreen->pScreen->CreatePixmap) (pGlxScreen->pScreen,
 | |
| 						    width, height, config->rgbBits, 0);
 | |
|     __glXleaveServer(GL_FALSE);
 | |
| 
 | |
|     /* Assign the pixmap the same id as the pbuffer and add it as a
 | |
|      * resource so it and the DRI2 drawable will be reclaimed when the
 | |
|      * pbuffer is destroyed. */
 | |
|     pPixmap->drawable.id = glxDrawableId;
 | |
|     if (!AddResource(pPixmap->drawable.id, RT_PIXMAP, pPixmap))
 | |
| 	return BadAlloc;
 | |
| 
 | |
|     return DoCreateGLXDrawable(client, pGlxScreen, config, &pPixmap->drawable,
 | |
| 			       glxDrawableId, glxDrawableId,
 | |
| 			       GLX_DRAWABLE_PBUFFER);
 | |
| }
 | |
| 
 | |
| int __glXDisp_CreatePbuffer(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXCreatePbufferReq	*req = (xGLXCreatePbufferReq *) pc;
 | |
|     CARD32			*attrs;
 | |
|     int				 width, height, i;
 | |
| 
 | |
|     attrs = (CARD32 *) (req + 1);
 | |
|     width = 0;
 | |
|     height = 0;
 | |
| 
 | |
|     for (i = 0; i < req->numAttribs; i++) {
 | |
| 	switch (attrs[i * 2]) {
 | |
| 	case GLX_PBUFFER_WIDTH:
 | |
| 	    width = attrs[i * 2 + 1];
 | |
| 	    break;
 | |
| 	case GLX_PBUFFER_HEIGHT:
 | |
| 	    height = attrs[i * 2 + 1];
 | |
| 	    break;
 | |
| 	case GLX_LARGEST_PBUFFER:
 | |
| 	case GLX_PRESERVED_CONTENTS:
 | |
| 	    /* FIXME: huh... */
 | |
| 	    break;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     return DoCreatePbuffer(cl->client, req->screen, req->fbconfig,
 | |
| 			   width, height, req->pbuffer);
 | |
| }
 | |
| 
 | |
| int __glXDisp_CreateGLXPbufferSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXCreateGLXPbufferSGIXReq *req = (xGLXCreateGLXPbufferSGIXReq *) pc;
 | |
| 
 | |
|     return DoCreatePbuffer(cl->client, req->screen, req->fbconfig,
 | |
| 			   req->width, req->height, req->pbuffer);
 | |
| }
 | |
| 
 | |
| int __glXDisp_DestroyPbuffer(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXDestroyPbufferReq *req = (xGLXDestroyPbufferReq *) pc;
 | |
| 
 | |
|     return DoDestroyDrawable(cl, req->pbuffer, GLX_DRAWABLE_PBUFFER);
 | |
| }
 | |
| 
 | |
| int __glXDisp_DestroyGLXPbufferSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXDestroyGLXPbufferSGIXReq *req = (xGLXDestroyGLXPbufferSGIXReq *) pc;
 | |
| 
 | |
|     return DoDestroyDrawable(cl, req->pbuffer, GLX_DRAWABLE_PBUFFER);
 | |
| }
 | |
| 
 | |
| static int
 | |
| DoChangeDrawableAttributes(ClientPtr client, XID glxdrawable,
 | |
| 			   int numAttribs, CARD32 *attribs)
 | |
| {
 | |
|     __GLXdrawable *pGlxDraw;
 | |
|     int i, err;
 | |
| 
 | |
|     if (!validGlxDrawable(client, glxdrawable, GLX_DRAWABLE_ANY,
 | |
| 			  DixSetAttrAccess, &pGlxDraw, &err))
 | |
| 	return err;
 | |
| 
 | |
|     for (i = 0; i < numAttribs; i++) {
 | |
| 	switch(attribs[i * 2]) {
 | |
| 	case GLX_EVENT_MASK:
 | |
| 	    /* All we do is to record the event mask so we can send it
 | |
| 	     * back when queried.  We never actually clobber the
 | |
| 	     * pbuffers, so we never need to send out the event. */
 | |
| 	    pGlxDraw->eventMask = attribs[i * 2 + 1];
 | |
| 	    break;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_ChangeDrawableAttributes(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXChangeDrawableAttributesReq *req =
 | |
| 	(xGLXChangeDrawableAttributesReq *) pc;
 | |
| 
 | |
|     return DoChangeDrawableAttributes(cl->client, req->drawable,
 | |
| 				      req->numAttribs, (CARD32 *) (req + 1));
 | |
| }
 | |
| 
 | |
| int __glXDisp_ChangeDrawableAttributesSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXChangeDrawableAttributesSGIXReq *req =
 | |
| 	(xGLXChangeDrawableAttributesSGIXReq *)pc;
 | |
| 
 | |
|     return DoChangeDrawableAttributes(cl->client, req->drawable,
 | |
| 				      req->numAttribs, (CARD32 *) (req + 1));
 | |
| }
 | |
| 
 | |
| int __glXDisp_CreateWindow(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXCreateWindowReq	*req = (xGLXCreateWindowReq *) pc;
 | |
|     __GLXconfig	*config;
 | |
|     __GLXscreen		*pGlxScreen;
 | |
|     ClientPtr		 client = cl->client;
 | |
|     DrawablePtr		 pDraw;
 | |
|     int			 err;
 | |
| 
 | |
|     LEGAL_NEW_RESOURCE(req->glxwindow, client);
 | |
| 
 | |
|     if (!validGlxScreen(client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
|     if (!validGlxFBConfig(client, pGlxScreen, req->fbconfig, &config, &err))
 | |
| 	return err;
 | |
| 
 | |
|     err = dixLookupDrawable(&pDraw, req->window, client, 0, DixAddAccess);
 | |
|     if (err != Success || pDraw->type != DRAWABLE_WINDOW) {
 | |
| 	client->errorValue = req->window;
 | |
| 	return BadWindow;
 | |
|     }
 | |
| 
 | |
|     if (!validGlxFBConfigForWindow(client, config, pDraw, &err))
 | |
| 	return err;
 | |
| 
 | |
|     return DoCreateGLXDrawable(client, pGlxScreen, config,
 | |
| 			       pDraw, req->window,
 | |
| 			       req->glxwindow, GLX_DRAWABLE_WINDOW);
 | |
| }
 | |
| 
 | |
| int __glXDisp_DestroyWindow(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXDestroyWindowReq *req = (xGLXDestroyWindowReq *) pc;
 | |
| 
 | |
|     return DoDestroyDrawable(cl, req->glxwindow, GLX_DRAWABLE_WINDOW);
 | |
| }
 | |
| 
 | |
| 
 | |
| /*****************************************************************************/
 | |
| 
 | |
| /*
 | |
| ** 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 __glXDisp_SwapBuffers(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXSwapBuffersReq *req = (xGLXSwapBuffersReq *) pc;
 | |
|     GLXContextTag tag = req->contextTag;
 | |
|     XID drawId = req->drawable;
 | |
|     __GLXcontext *glxc = NULL;
 | |
|     __GLXdrawable *pGlxDraw;
 | |
|     int error;
 | |
| 
 | |
|     if (tag) {
 | |
| 	glxc = __glXLookupContextByTag(cl, tag);
 | |
| 	if (!glxc) {
 | |
| 	    return __glXError(GLXBadContextTag);
 | |
| 	}
 | |
| 	/*
 | |
| 	** The calling thread is swapping its current drawable.  In this case,
 | |
| 	** glxSwapBuffers is in both GL and X streams, in terms of
 | |
| 	** sequentiality.
 | |
| 	*/
 | |
| 	if (__glXForceCurrent(cl, tag, &error)) {
 | |
| 	    /*
 | |
| 	    ** Do whatever is needed to make sure that all preceding requests
 | |
| 	    ** in both streams are completed before the swap is executed.
 | |
| 	    */
 | |
| 	    CALL_Finish( GET_DISPATCH(), () );
 | |
| 	    __GLX_NOTE_FLUSHED_CMDS(glxc);
 | |
| 	} else {
 | |
| 	    return error;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     pGlxDraw = __glXGetDrawable(glxc, drawId, client, &error);
 | |
|     if (pGlxDraw == NULL)
 | |
| 	return error;
 | |
| 
 | |
|     if (pGlxDraw->type == DRAWABLE_WINDOW &&
 | |
| 	(*pGlxDraw->swapBuffers)(cl->client, pGlxDraw) == GL_FALSE)
 | |
| 	return __glXError(GLXBadDrawable);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| 
 | |
| static int
 | |
| DoQueryContext(__GLXclientState *cl, GLXContextID gcId)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     __GLXcontext *ctx;
 | |
|     xGLXQueryContextInfoEXTReply reply;
 | |
|     int nProps;
 | |
|     int *sendBuf, *pSendBuf;
 | |
|     int nReplyBytes;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxContext(cl->client, gcId, DixReadAccess, &ctx, &err))
 | |
| 	return err;
 | |
| 
 | |
|     nProps = 3;
 | |
|     reply.length = nProps << 1;
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
|     reply.n = nProps;
 | |
| 
 | |
|     nReplyBytes = reply.length << 2;
 | |
|     sendBuf = (int *)malloc((size_t)nReplyBytes);
 | |
|     if (sendBuf == NULL) {
 | |
| 	return __glXError(GLXBadContext);	/* XXX: Is this correct? */
 | |
|     }
 | |
|     pSendBuf = sendBuf;
 | |
|     *pSendBuf++ = GLX_SHARE_CONTEXT_EXT;
 | |
|     *pSendBuf++ = (int)(ctx->share_id);
 | |
|     *pSendBuf++ = GLX_VISUAL_ID_EXT;
 | |
|     *pSendBuf++ = (int)(ctx->config->visualID);
 | |
|     *pSendBuf++ = GLX_SCREEN_EXT;
 | |
|     *pSendBuf++ = (int)(ctx->pGlxScreen->pScreen->myNum);
 | |
| 
 | |
|     if (client->swapped) {
 | |
| 	__glXSwapQueryContextInfoEXTReply(client, &reply, sendBuf);
 | |
|     } else {
 | |
| 	WriteToClient(client, sz_xGLXQueryContextInfoEXTReply, (char *)&reply);
 | |
| 	WriteToClient(client, nReplyBytes, (char *)sendBuf);
 | |
|     }
 | |
|     free((char *)sendBuf);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_QueryContextInfoEXT(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXQueryContextInfoEXTReq *req = (xGLXQueryContextInfoEXTReq *) pc;
 | |
| 
 | |
|     return DoQueryContext(cl, req->context);
 | |
| }
 | |
| 
 | |
| int __glXDisp_QueryContext(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXQueryContextReq *req = (xGLXQueryContextReq *) pc;
 | |
| 
 | |
|     return DoQueryContext(cl, req->context);
 | |
| }
 | |
| 
 | |
| int __glXDisp_BindTexImageEXT(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXVendorPrivateReq *req = (xGLXVendorPrivateReq *) pc;
 | |
|     ClientPtr		 client = cl->client;
 | |
|     __GLXcontext	*context;
 | |
|     __GLXdrawable	*pGlxDraw;
 | |
|     GLXDrawable		 drawId;
 | |
|     int			 buffer;
 | |
|     int			 error;
 | |
| 
 | |
|     pc += __GLX_VENDPRIV_HDR_SIZE;
 | |
| 
 | |
|     drawId = *((CARD32 *) (pc));
 | |
|     buffer = *((INT32 *)  (pc + 4));
 | |
| 
 | |
|     if (buffer != GLX_FRONT_LEFT_EXT)
 | |
| 	return __glXError(GLXBadPixmap);
 | |
| 
 | |
|     context = __glXForceCurrent (cl, req->contextTag, &error);
 | |
|     if (!context)
 | |
| 	return error;
 | |
| 
 | |
|     if (!validGlxDrawable(client, drawId, GLX_DRAWABLE_PIXMAP,
 | |
| 			  DixReadAccess, &pGlxDraw, &error))
 | |
| 	return error;
 | |
| 
 | |
|     if (!context->textureFromPixmap)
 | |
| 	return __glXError(GLXUnsupportedPrivateRequest);
 | |
| 
 | |
|     return context->textureFromPixmap->bindTexImage(context,
 | |
| 						    buffer,
 | |
| 						    pGlxDraw);
 | |
| }
 | |
| 
 | |
| int __glXDisp_ReleaseTexImageEXT(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXVendorPrivateReq *req = (xGLXVendorPrivateReq *) pc;
 | |
|     ClientPtr		 client = cl->client;
 | |
|     __GLXdrawable	*pGlxDraw;
 | |
|     __GLXcontext	*context;
 | |
|     GLXDrawable		 drawId;
 | |
|     int			 buffer;
 | |
|     int			 error;
 | |
| 
 | |
|     pc += __GLX_VENDPRIV_HDR_SIZE;
 | |
| 
 | |
|     drawId = *((CARD32 *) (pc));
 | |
|     buffer = *((INT32 *)  (pc + 4));
 | |
|     
 | |
|     context = __glXForceCurrent (cl, req->contextTag, &error);
 | |
|     if (!context)
 | |
| 	return error;
 | |
| 
 | |
|     if (!validGlxDrawable(client, drawId, GLX_DRAWABLE_PIXMAP,
 | |
| 			  DixReadAccess, &pGlxDraw, &error))
 | |
| 	return error;
 | |
| 
 | |
|     if (!context->textureFromPixmap)
 | |
| 	return __glXError(GLXUnsupportedPrivateRequest);
 | |
| 
 | |
|     return context->textureFromPixmap->releaseTexImage(context,
 | |
| 						       buffer,
 | |
| 						       pGlxDraw);
 | |
| }
 | |
| 
 | |
| int __glXDisp_CopySubBufferMESA(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXVendorPrivateReq *req = (xGLXVendorPrivateReq *) pc;
 | |
|     GLXContextTag         tag = req->contextTag;
 | |
|     __GLXcontext         *glxc = NULL;
 | |
|     __GLXdrawable        *pGlxDraw;
 | |
|     ClientPtr		  client = cl->client;
 | |
|     GLXDrawable		  drawId;
 | |
|     int                   error;
 | |
|     int                   x, y, width, height;
 | |
| 
 | |
|     (void) client;
 | |
|     (void) req;
 | |
| 
 | |
|     pc += __GLX_VENDPRIV_HDR_SIZE;
 | |
| 
 | |
|     drawId = *((CARD32 *) (pc));
 | |
|     x      = *((INT32 *)  (pc + 4));
 | |
|     y      = *((INT32 *)  (pc + 8));
 | |
|     width  = *((INT32 *)  (pc + 12));
 | |
|     height = *((INT32 *)  (pc + 16));
 | |
| 
 | |
|     if (tag) {
 | |
| 	glxc = __glXLookupContextByTag(cl, tag);
 | |
| 	if (!glxc) {
 | |
| 	    return __glXError(GLXBadContextTag);
 | |
| 	}
 | |
| 	/*
 | |
| 	** The calling thread is swapping its current drawable.  In this case,
 | |
| 	** glxSwapBuffers is in both GL and X streams, in terms of
 | |
| 	** sequentiality.
 | |
| 	*/
 | |
| 	if (__glXForceCurrent(cl, tag, &error)) {
 | |
| 	    /*
 | |
| 	    ** Do whatever is needed to make sure that all preceding requests
 | |
| 	    ** in both streams are completed before the swap is executed.
 | |
| 	    */
 | |
| 	    CALL_Finish( GET_DISPATCH(), () );
 | |
| 	    __GLX_NOTE_FLUSHED_CMDS(glxc);
 | |
| 	} else {
 | |
| 	    return error;
 | |
| 	}
 | |
|     }
 | |
| 
 | |
|     pGlxDraw = __glXGetDrawable(glxc, drawId, client, &error);
 | |
|     if (!pGlxDraw)
 | |
| 	return error;
 | |
| 
 | |
|     if (pGlxDraw == NULL ||
 | |
| 	pGlxDraw->type != GLX_DRAWABLE_WINDOW ||
 | |
| 	pGlxDraw->copySubBuffer == NULL)
 | |
| 	return __glXError(GLXBadDrawable);
 | |
| 
 | |
|     (*pGlxDraw->copySubBuffer)(pGlxDraw, x, y, width, height);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| /*
 | |
| ** Get drawable attributes
 | |
| */
 | |
| static int
 | |
| DoGetDrawableAttributes(__GLXclientState *cl, XID drawId)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXGetDrawableAttributesReply reply;
 | |
|     __GLXdrawable *pGlxDraw;
 | |
|     CARD32 attributes[6];
 | |
|     int numAttribs, error;
 | |
| 
 | |
|     if (!validGlxDrawable(client, drawId, GLX_DRAWABLE_ANY,
 | |
| 			  DixGetAttrAccess, &pGlxDraw, &error))
 | |
| 	return error;
 | |
| 
 | |
|     numAttribs = 3;
 | |
|     reply.length = numAttribs << 1;
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
|     reply.numAttribs = numAttribs;
 | |
| 
 | |
|     attributes[0] = GLX_TEXTURE_TARGET_EXT;
 | |
|     attributes[1] = pGlxDraw->target == GL_TEXTURE_2D ? GLX_TEXTURE_2D_EXT :
 | |
| 	GLX_TEXTURE_RECTANGLE_EXT;
 | |
|     attributes[2] = GLX_Y_INVERTED_EXT;
 | |
|     attributes[3] = GL_FALSE;
 | |
|     attributes[4] = GLX_EVENT_MASK;
 | |
|     attributes[5] = pGlxDraw->eventMask;
 | |
| 
 | |
|     if (client->swapped) {
 | |
| 	__glXSwapGetDrawableAttributesReply(client, &reply, attributes);
 | |
|     } else {
 | |
| 	WriteToClient(client, sz_xGLXGetDrawableAttributesReply,
 | |
| 		      (char *)&reply);
 | |
| 	WriteToClient(client, reply.length * sizeof (CARD32),
 | |
| 		      (char *)attributes);
 | |
|     }
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_GetDrawableAttributes(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXGetDrawableAttributesReq *req = (xGLXGetDrawableAttributesReq *)pc;
 | |
| 
 | |
|     return DoGetDrawableAttributes(cl, req->drawable);
 | |
| }
 | |
| 
 | |
| int __glXDisp_GetDrawableAttributesSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXGetDrawableAttributesSGIXReq *req =
 | |
| 	(xGLXGetDrawableAttributesSGIXReq *)pc;
 | |
|     
 | |
|     return DoGetDrawableAttributes(cl, req->drawable);
 | |
| }
 | |
| 
 | |
| /************************************************************************/
 | |
| 
 | |
| /*
 | |
| ** 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 __glXDisp_Render(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXRenderReq *req;
 | |
|     ClientPtr client= cl->client;
 | |
|     int left, cmdlen, error;
 | |
|     int commandsDone;
 | |
|     CARD16 opcode;
 | |
|     __GLXrenderHeader *hdr;
 | |
|     __GLXcontext *glxc;
 | |
|     __GLX_DECLARE_SWAP_VARIABLES;
 | |
| 
 | |
|     req = (xGLXRenderReq *) pc;
 | |
|     if (client->swapped) {
 | |
| 	__GLX_SWAP_SHORT(&req->length);
 | |
| 	__GLX_SWAP_INT(&req->contextTag);
 | |
|     }
 | |
| 
 | |
|     glxc = __glXForceCurrent(cl, req->contextTag, &error);
 | |
|     if (!glxc) {
 | |
| 	return error;
 | |
|     }
 | |
| 
 | |
|     commandsDone = 0;
 | |
|     pc += sz_xGLXRenderReq;
 | |
|     left = (req->length << 2) - sz_xGLXRenderReq;
 | |
|     while (left > 0) {
 | |
|         __GLXrenderSizeData entry;
 | |
|         int extra;
 | |
| 	__GLXdispatchRenderProcPtr proc;
 | |
| 	int err;
 | |
| 
 | |
| 	/*
 | |
| 	** Verify that the header length and the overall length agree.
 | |
| 	** Also, each command must be word aligned.
 | |
| 	*/
 | |
| 	hdr = (__GLXrenderHeader *) pc;
 | |
| 	if (client->swapped) {
 | |
| 	    __GLX_SWAP_SHORT(&hdr->length);
 | |
| 	    __GLX_SWAP_SHORT(&hdr->opcode);
 | |
| 	}
 | |
| 	cmdlen = hdr->length;
 | |
| 	opcode = hdr->opcode;
 | |
| 
 | |
| 	/*
 | |
| 	** Check for core opcodes and grab entry data.
 | |
| 	*/
 | |
| 	err = __glXGetProtocolSizeData(& Render_dispatch_info, opcode, & entry);
 | |
| 	proc = (__GLXdispatchRenderProcPtr)
 | |
| 	    __glXGetProtocolDecodeFunction(& Render_dispatch_info,
 | |
| 					   opcode, client->swapped);
 | |
| 
 | |
| 	if ((err < 0) || (proc == NULL)) {
 | |
| 	    client->errorValue = commandsDone;
 | |
| 	    return __glXError(GLXBadRenderRequest);
 | |
| 	}
 | |
| 
 | |
|         if (entry.varsize) {
 | |
|             /* variable size command */
 | |
|             extra = (*entry.varsize)(pc + __GLX_RENDER_HDR_SIZE,
 | |
| 				     client->swapped);
 | |
|             if (extra < 0) {
 | |
|                 extra = 0;
 | |
|             }
 | |
|             if (cmdlen != __GLX_PAD(entry.bytes + extra)) {
 | |
|                 return BadLength;
 | |
|             }
 | |
|         } else {
 | |
|             /* constant size command */
 | |
|             if (cmdlen != __GLX_PAD(entry.bytes)) {
 | |
|                 return BadLength;
 | |
|             }
 | |
|         }
 | |
| 	if (left < cmdlen) {
 | |
| 	    return BadLength;
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	** Skip over the header and execute the command.  We allow the
 | |
| 	** caller to trash the command memory.  This is useful especially
 | |
| 	** for things that require double alignment - they can just shift
 | |
| 	** the data towards lower memory (trashing the header) by 4 bytes
 | |
| 	** and achieve the required alignment.
 | |
| 	*/
 | |
| 	(*proc)(pc + __GLX_RENDER_HDR_SIZE);
 | |
| 	pc += cmdlen;
 | |
| 	left -= cmdlen;
 | |
| 	commandsDone++;
 | |
|     }
 | |
|     __GLX_NOTE_UNFLUSHED_CMDS(glxc);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| 
 | |
| /*
 | |
| ** Execute a large rendering request (one that spans multiple X requests).
 | |
| */
 | |
| int __glXDisp_RenderLarge(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXRenderLargeReq *req;
 | |
|     ClientPtr client= cl->client;
 | |
|     size_t dataBytes;
 | |
|     __GLXrenderLargeHeader *hdr;
 | |
|     __GLXcontext *glxc;
 | |
|     int error;
 | |
|     CARD16 opcode;
 | |
|     __GLX_DECLARE_SWAP_VARIABLES;
 | |
|     
 | |
|     req = (xGLXRenderLargeReq *) pc;
 | |
|     if (client->swapped) {
 | |
| 	__GLX_SWAP_SHORT(&req->length);
 | |
| 	__GLX_SWAP_INT(&req->contextTag);
 | |
| 	__GLX_SWAP_INT(&req->dataBytes);
 | |
| 	__GLX_SWAP_SHORT(&req->requestNumber);
 | |
| 	__GLX_SWAP_SHORT(&req->requestTotal);
 | |
|     }
 | |
| 
 | |
|     glxc = __glXForceCurrent(cl, req->contextTag, &error);
 | |
|     if (!glxc) {
 | |
| 	/* Reset in case this isn't 1st request. */
 | |
| 	__glXResetLargeCommandStatus(cl);
 | |
| 	return error;
 | |
|     }
 | |
|     dataBytes = req->dataBytes;
 | |
| 
 | |
|     /*
 | |
|     ** Check the request length.
 | |
|     */
 | |
|     if ((req->length << 2) != __GLX_PAD(dataBytes) + sz_xGLXRenderLargeReq) {
 | |
| 	client->errorValue = req->length;
 | |
| 	/* Reset in case this isn't 1st request. */
 | |
| 	__glXResetLargeCommandStatus(cl);
 | |
| 	return BadLength;
 | |
|     }
 | |
|     pc += sz_xGLXRenderLargeReq;
 | |
|     
 | |
|     if (cl->largeCmdRequestsSoFar == 0) {
 | |
| 	__GLXrenderSizeData entry;
 | |
| 	int extra;
 | |
| 	size_t cmdlen;
 | |
| 	int err;
 | |
| 
 | |
| 	/*
 | |
| 	** This is the first request of a multi request command.
 | |
| 	** Make enough space in the buffer, then copy the entire request.
 | |
| 	*/
 | |
| 	if (req->requestNumber != 1) {
 | |
| 	    client->errorValue = req->requestNumber;
 | |
| 	    return __glXError(GLXBadLargeRequest);
 | |
| 	}
 | |
| 
 | |
| 	hdr = (__GLXrenderLargeHeader *) pc;
 | |
| 	if (client->swapped) {
 | |
| 	    __GLX_SWAP_INT(&hdr->length);
 | |
| 	    __GLX_SWAP_INT(&hdr->opcode);
 | |
| 	}
 | |
| 	cmdlen = hdr->length;
 | |
| 	opcode = hdr->opcode;
 | |
| 
 | |
| 	/*
 | |
| 	** Check for core opcodes and grab entry data.
 | |
| 	*/
 | |
| 	err = __glXGetProtocolSizeData(& Render_dispatch_info, opcode, & entry);
 | |
| 	if (err < 0) {
 | |
| 	    client->errorValue = opcode;
 | |
| 	    return __glXError(GLXBadLargeRequest);
 | |
| 	}
 | |
| 
 | |
| 	if (entry.varsize) {
 | |
| 	    /*
 | |
| 	    ** If it's a variable-size command (a command whose length must
 | |
| 	    ** be computed from its parameters), all the parameters needed
 | |
| 	    ** will be in the 1st request, so it's okay to do this.
 | |
| 	    */
 | |
| 	    extra = (*entry.varsize)(pc + __GLX_RENDER_LARGE_HDR_SIZE,
 | |
| 				     client->swapped);
 | |
| 	    if (extra < 0) {
 | |
| 		extra = 0;
 | |
| 	    }
 | |
| 	    /* large command's header is 4 bytes longer, so add 4 */
 | |
| 	    if (cmdlen != __GLX_PAD(entry.bytes + 4 + extra)) {
 | |
| 		return BadLength;
 | |
| 	    }
 | |
| 	} else {
 | |
| 	    /* constant size command */
 | |
| 	    if (cmdlen != __GLX_PAD(entry.bytes + 4)) {
 | |
| 		return BadLength;
 | |
| 	    }
 | |
| 	}
 | |
| 	/*
 | |
| 	** Make enough space in the buffer, then copy the entire request.
 | |
| 	*/
 | |
| 	if (cl->largeCmdBufSize < cmdlen) {
 | |
| 	    if (!cl->largeCmdBuf) {
 | |
| 		cl->largeCmdBuf = (GLbyte *) malloc(cmdlen);
 | |
| 	    } else {
 | |
| 		cl->largeCmdBuf = (GLbyte *) realloc(cl->largeCmdBuf, cmdlen);
 | |
| 	    }
 | |
| 	    if (!cl->largeCmdBuf) {
 | |
| 		return BadAlloc;
 | |
| 	    }
 | |
| 	    cl->largeCmdBufSize = cmdlen;
 | |
| 	}
 | |
| 	memcpy(cl->largeCmdBuf, pc, dataBytes);
 | |
| 
 | |
| 	cl->largeCmdBytesSoFar = dataBytes;
 | |
| 	cl->largeCmdBytesTotal = cmdlen;
 | |
| 	cl->largeCmdRequestsSoFar = 1;
 | |
| 	cl->largeCmdRequestsTotal = req->requestTotal;
 | |
| 	return Success;
 | |
| 	
 | |
|     } else {
 | |
| 	/*
 | |
| 	** We are receiving subsequent (i.e. not the first) requests of a
 | |
| 	** multi request command.
 | |
| 	*/
 | |
| 
 | |
| 	/*
 | |
| 	** Check the request number and the total request count.
 | |
| 	*/
 | |
| 	if (req->requestNumber != cl->largeCmdRequestsSoFar + 1) {
 | |
| 	    client->errorValue = req->requestNumber;
 | |
| 	    __glXResetLargeCommandStatus(cl);
 | |
| 	    return __glXError(GLXBadLargeRequest);
 | |
| 	}
 | |
| 	if (req->requestTotal != cl->largeCmdRequestsTotal) {
 | |
| 	    client->errorValue = req->requestTotal;
 | |
| 	    __glXResetLargeCommandStatus(cl);
 | |
| 	    return __glXError(GLXBadLargeRequest);
 | |
| 	}
 | |
| 
 | |
| 	/*
 | |
| 	** Check that we didn't get too much data.
 | |
| 	*/
 | |
| 	if ((cl->largeCmdBytesSoFar + dataBytes) > cl->largeCmdBytesTotal) {
 | |
| 	    client->errorValue = dataBytes;
 | |
| 	    __glXResetLargeCommandStatus(cl);
 | |
| 	    return __glXError(GLXBadLargeRequest);
 | |
| 	}
 | |
| 	memcpy(cl->largeCmdBuf + cl->largeCmdBytesSoFar, pc, dataBytes);
 | |
| 	cl->largeCmdBytesSoFar += dataBytes;
 | |
| 	cl->largeCmdRequestsSoFar++;
 | |
| 
 | |
| 	if (req->requestNumber == cl->largeCmdRequestsTotal) {
 | |
| 	    __GLXdispatchRenderProcPtr proc;
 | |
| 
 | |
| 	    /*
 | |
| 	    ** This is the last request; it must have enough bytes to complete
 | |
| 	    ** the command.
 | |
| 	    */
 | |
| 	    /* NOTE: the two pad macros have been added below; they are needed
 | |
| 	    ** because the client library pads the total byte count, but not
 | |
| 	    ** the per-request byte counts.  The Protocol Encoding says the
 | |
| 	    ** total byte count should not be padded, so a proposal will be 
 | |
| 	    ** made to the ARB to relax the padding constraint on the total 
 | |
| 	    ** byte count, thus preserving backward compatibility.  Meanwhile, 
 | |
| 	    ** the padding done below fixes a bug that did not allow
 | |
| 	    ** large commands of odd sizes to be accepted by the server.
 | |
| 	    */
 | |
| 	    if (__GLX_PAD(cl->largeCmdBytesSoFar) !=
 | |
| 		__GLX_PAD(cl->largeCmdBytesTotal)) {
 | |
| 		client->errorValue = dataBytes;
 | |
| 		__glXResetLargeCommandStatus(cl);
 | |
| 		return __glXError(GLXBadLargeRequest);
 | |
| 	    }
 | |
| 	    hdr = (__GLXrenderLargeHeader *) cl->largeCmdBuf;
 | |
| 	    /*
 | |
| 	    ** The opcode and length field in the header had already been
 | |
| 	    ** swapped when the first request was received.
 | |
| 	    **
 | |
| 	    ** Use the opcode to index into the procedure table.
 | |
| 	    */
 | |
| 	    opcode = hdr->opcode;
 | |
| 
 | |
| 	    proc = (__GLXdispatchRenderProcPtr)
 | |
| 	      __glXGetProtocolDecodeFunction(& Render_dispatch_info, opcode,
 | |
| 					     client->swapped);
 | |
| 	    if (proc == NULL) {
 | |
| 		client->errorValue = opcode;
 | |
| 		return __glXError(GLXBadLargeRequest);
 | |
| 	    }
 | |
| 
 | |
| 	    /*
 | |
| 	    ** Skip over the header and execute the command.
 | |
| 	    */
 | |
| 	    (*proc)(cl->largeCmdBuf + __GLX_RENDER_LARGE_HDR_SIZE);
 | |
| 	    __GLX_NOTE_UNFLUSHED_CMDS(glxc);
 | |
| 
 | |
| 	    /*
 | |
| 	    ** Reset for the next RenderLarge series.
 | |
| 	    */
 | |
| 	    __glXResetLargeCommandStatus(cl);
 | |
| 	} else {
 | |
| 	    /*
 | |
| 	    ** This is neither the first nor the last request.
 | |
| 	    */
 | |
| 	}
 | |
| 	return Success;
 | |
|     }
 | |
| }
 | |
| 
 | |
| extern RESTYPE __glXSwapBarrierRes;
 | |
| 
 | |
| int __glXDisp_BindSwapBarrierSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXBindSwapBarrierSGIXReq *req = (xGLXBindSwapBarrierSGIXReq *) pc;
 | |
|     XID drawable = req->drawable;
 | |
|     int barrier = req->barrier;
 | |
|     DrawablePtr pDraw;
 | |
|     int screen, rc;
 | |
|     __GLXscreen *pGlxScreen;
 | |
| 
 | |
|     rc = dixLookupDrawable(&pDraw, drawable, client, 0, DixGetAttrAccess);
 | |
|     pGlxScreen = glxGetScreen(pDraw->pScreen);
 | |
|     if (rc == Success && (pDraw->type == DRAWABLE_WINDOW)) {
 | |
| 	screen = pDraw->pScreen->myNum;
 | |
|         if (pGlxScreen->swapBarrierFuncs) {
 | |
|             int ret = pGlxScreen->swapBarrierFuncs->bindSwapBarrierFunc(screen, drawable, barrier);
 | |
|             if (ret == Success) {
 | |
|                 if (barrier)
 | |
|                     /* add source for cleanup when drawable is gone */
 | |
|                     AddResource(drawable, __glXSwapBarrierRes, (pointer)(intptr_t)screen);
 | |
|                 else
 | |
|                     /* delete source */
 | |
|                     FreeResourceByType(drawable, __glXSwapBarrierRes, FALSE);
 | |
|             }
 | |
|             return ret;
 | |
|         }
 | |
|     }
 | |
|     client->errorValue = drawable;
 | |
|     return __glXError(GLXBadDrawable);
 | |
| }
 | |
| 
 | |
| 
 | |
| int __glXDisp_QueryMaxSwapBarriersSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXQueryMaxSwapBarriersSGIXReq *req =
 | |
|                                     (xGLXQueryMaxSwapBarriersSGIXReq *) pc;
 | |
|     xGLXQueryMaxSwapBarriersSGIXReply reply;
 | |
|     int screen = req->screen;
 | |
|     __GLXscreen *pGlxScreen;
 | |
| 
 | |
|     pGlxScreen = glxGetScreen(screenInfo.screens[screen]);
 | |
|     if (pGlxScreen->swapBarrierFuncs)
 | |
|         reply.max = pGlxScreen->swapBarrierFuncs->queryMaxSwapBarriersFunc(screen);
 | |
|     else
 | |
|         reply.max = 0;
 | |
| 
 | |
| 
 | |
|     reply.length = 0;
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         __GLX_DECLARE_SWAP_VARIABLES;
 | |
|         __GLX_SWAP_SHORT(&reply.sequenceNumber);
 | |
|     }
 | |
| 
 | |
|     WriteToClient(client, sz_xGLXQueryMaxSwapBarriersSGIXReply,
 | |
|                         (char *) &reply);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| #define GLX_BAD_HYPERPIPE_SGIX 92
 | |
| 
 | |
| int __glXDisp_QueryHyperpipeNetworkSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXQueryHyperpipeNetworkSGIXReq * req = (xGLXQueryHyperpipeNetworkSGIXReq *) pc;
 | |
|     xGLXQueryHyperpipeNetworkSGIXReply reply;
 | |
|     int screen = req->screen;
 | |
|     void *rdata = NULL;
 | |
| 
 | |
|     int length=0;
 | |
|     int npipes=0;
 | |
| 
 | |
|     int n= 0;
 | |
|     __GLXscreen *pGlxScreen;
 | |
| 
 | |
|     pGlxScreen = glxGetScreen(screenInfo.screens[screen]);
 | |
|     if (pGlxScreen->hyperpipeFuncs) {
 | |
|         rdata =
 | |
|             (pGlxScreen->hyperpipeFuncs->queryHyperpipeNetworkFunc(screen, &npipes, &n));
 | |
|     }
 | |
|     length = __GLX_PAD(n) >> 2;
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
|     reply.length = length;
 | |
|     reply.n = n;
 | |
|     reply.npipes = npipes;
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         __GLX_DECLARE_SWAP_VARIABLES;
 | |
|         __GLX_SWAP_SHORT(&reply.sequenceNumber);
 | |
|         __GLX_SWAP_INT(&reply.length);
 | |
|         __GLX_SWAP_INT(&reply.n);
 | |
|         __GLX_SWAP_INT(&reply.npipes);
 | |
|     }
 | |
|     WriteToClient(client, sz_xGLXQueryHyperpipeNetworkSGIXReply,
 | |
|                   (char *) &reply);
 | |
| 
 | |
|     WriteToClient(client, length << 2, (char *)rdata);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_DestroyHyperpipeConfigSGIX (__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXDestroyHyperpipeConfigSGIXReq * req =
 | |
|         (xGLXDestroyHyperpipeConfigSGIXReq *) pc;
 | |
|     xGLXDestroyHyperpipeConfigSGIXReply reply;
 | |
|     int screen = req->screen;
 | |
|     int  success = GLX_BAD_HYPERPIPE_SGIX;
 | |
|     int hpId ;
 | |
|     __GLXscreen *pGlxScreen;
 | |
| 
 | |
|     hpId = req->hpId;
 | |
| 
 | |
|     pGlxScreen = glxGetScreen(screenInfo.screens[screen]);
 | |
|     if (pGlxScreen->hyperpipeFuncs) {
 | |
|         success = pGlxScreen->hyperpipeFuncs->destroyHyperpipeConfigFunc(screen, hpId);
 | |
|     }
 | |
| 
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
|     reply.length = __GLX_PAD(0) >> 2;
 | |
|     reply.n = 0;
 | |
|     reply.success = success;
 | |
| 
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         __GLX_DECLARE_SWAP_VARIABLES;
 | |
|         __GLX_SWAP_SHORT(&reply.sequenceNumber);
 | |
|     }
 | |
|     WriteToClient(client,
 | |
|                   sz_xGLXDestroyHyperpipeConfigSGIXReply,
 | |
|                   (char *) &reply);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_QueryHyperpipeConfigSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXQueryHyperpipeConfigSGIXReq * req =
 | |
|         (xGLXQueryHyperpipeConfigSGIXReq *) pc;
 | |
|     xGLXQueryHyperpipeConfigSGIXReply reply;
 | |
|     int screen = req->screen;
 | |
|     void *rdata = NULL;
 | |
|     int length;
 | |
|     int npipes=0;
 | |
|     int n= 0;
 | |
|     int hpId;
 | |
|     __GLXscreen *pGlxScreen;
 | |
| 
 | |
|     hpId = req->hpId;
 | |
| 
 | |
|     pGlxScreen = glxGetScreen(screenInfo.screens[screen]);
 | |
|     if (pGlxScreen->hyperpipeFuncs) {
 | |
|         rdata = pGlxScreen->hyperpipeFuncs->queryHyperpipeConfigFunc(screen, hpId,&npipes, &n);
 | |
|     }
 | |
| 
 | |
|     length = __GLX_PAD(n) >> 2;
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
|     reply.length = length;
 | |
|     reply.n = n;
 | |
|     reply.npipes = npipes;
 | |
| 
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         __GLX_DECLARE_SWAP_VARIABLES;
 | |
|         __GLX_SWAP_SHORT(&reply.sequenceNumber);
 | |
|         __GLX_SWAP_INT(&reply.length);
 | |
|         __GLX_SWAP_INT(&reply.n);
 | |
|         __GLX_SWAP_INT(&reply.npipes);
 | |
|     }
 | |
| 
 | |
|     WriteToClient(client, sz_xGLXQueryHyperpipeConfigSGIXReply,
 | |
|                   (char *) &reply);
 | |
| 
 | |
|     WriteToClient(client, length << 2, (char *)rdata);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_HyperpipeConfigSGIX(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXHyperpipeConfigSGIXReq * req =
 | |
|         (xGLXHyperpipeConfigSGIXReq *) pc;
 | |
|     xGLXHyperpipeConfigSGIXReply reply;
 | |
|     int screen = req->screen;
 | |
|     void *rdata;
 | |
| 
 | |
|     int npipes=0, networkId;
 | |
|     int hpId=-1;
 | |
|     __GLXscreen *pGlxScreen;
 | |
| 
 | |
|     pGlxScreen = glxGetScreen(screenInfo.screens[screen]);
 | |
|     networkId = (int)req->networkId;
 | |
|     npipes = (int)req->npipes;
 | |
|     rdata = (void *)(req +1);
 | |
| 
 | |
|     if (pGlxScreen->hyperpipeFuncs) {
 | |
|         pGlxScreen->hyperpipeFuncs->hyperpipeConfigFunc(screen,networkId,
 | |
|                                                         &hpId, &npipes,
 | |
|                                                         (void *) rdata);
 | |
|     }
 | |
| 
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
|     reply.length = __GLX_PAD(0) >> 2;
 | |
|     reply.n = 0;
 | |
|     reply.npipes = npipes;
 | |
|     reply.hpId = hpId;
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         __GLX_DECLARE_SWAP_VARIABLES;
 | |
|         __GLX_SWAP_SHORT(&reply.sequenceNumber);
 | |
|         __GLX_SWAP_INT(&reply.npipes);
 | |
|         __GLX_SWAP_INT(&reply.hpId);
 | |
|     }
 | |
| 
 | |
|     WriteToClient(client, sz_xGLXHyperpipeConfigSGIXReply,
 | |
|                   (char *) &reply);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| 
 | |
| /************************************************************************/
 | |
| 
 | |
| /*
 | |
| ** No support is provided for the vendor-private requests other than
 | |
| ** allocating the entry points in the dispatch table.
 | |
| */
 | |
| 
 | |
| int __glXDisp_VendorPrivate(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXVendorPrivateReq *req = (xGLXVendorPrivateReq *) pc;
 | |
|     GLint vendorcode = req->vendorCode;
 | |
|     __GLXdispatchVendorPrivProcPtr proc;
 | |
| 
 | |
| 
 | |
|     proc = (__GLXdispatchVendorPrivProcPtr)
 | |
|       __glXGetProtocolDecodeFunction(& VendorPriv_dispatch_info,
 | |
| 				     vendorcode, 0);
 | |
|     if (proc != NULL) {
 | |
| 	(*proc)(cl, (GLbyte*)req);
 | |
| 	return Success;
 | |
|     }
 | |
| 
 | |
|     cl->client->errorValue = req->vendorCode;
 | |
|     return __glXError(GLXUnsupportedPrivateRequest);
 | |
| }
 | |
| 
 | |
| int __glXDisp_VendorPrivateWithReply(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXVendorPrivateReq *req = (xGLXVendorPrivateReq *) pc;
 | |
|     GLint vendorcode = req->vendorCode;
 | |
|     __GLXdispatchVendorPrivProcPtr proc;
 | |
| 
 | |
| 
 | |
|     proc = (__GLXdispatchVendorPrivProcPtr)
 | |
|       __glXGetProtocolDecodeFunction(& VendorPriv_dispatch_info,
 | |
| 				     vendorcode, 0);
 | |
|     if (proc != NULL) {
 | |
| 	return (*proc)(cl, (GLbyte*)req);
 | |
|     }
 | |
| 
 | |
|     cl->client->errorValue = vendorcode;
 | |
|     return __glXError(GLXUnsupportedPrivateRequest);
 | |
| }
 | |
| 
 | |
| int __glXDisp_QueryExtensionsString(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXQueryExtensionsStringReq *req = (xGLXQueryExtensionsStringReq *) pc;
 | |
|     xGLXQueryExtensionsStringReply reply;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     size_t n, length;
 | |
|     char *buf;
 | |
|     int err;
 | |
| 
 | |
|     if (!validGlxScreen(client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
| 
 | |
|     n = strlen(pGlxScreen->GLXextensions) + 1;
 | |
|     length = __GLX_PAD(n) >> 2;
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
|     reply.length = length;
 | |
|     reply.n = n;
 | |
| 
 | |
|     /* Allocate buffer to make sure it's a multiple of 4 bytes big.*/
 | |
|     buf = (char *) malloc(length << 2);
 | |
|     if (buf == NULL)
 | |
|         return BadAlloc;
 | |
|     memcpy(buf, pGlxScreen->GLXextensions, n);
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         glxSwapQueryExtensionsStringReply(client, &reply, buf);
 | |
|     } else {
 | |
|         WriteToClient(client, sz_xGLXQueryExtensionsStringReply,(char *)&reply);
 | |
|         WriteToClient(client, (int)(length << 2), (char *)buf);
 | |
|     }
 | |
| 
 | |
|     free(buf);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_QueryServerString(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     ClientPtr client = cl->client;
 | |
|     xGLXQueryServerStringReq *req = (xGLXQueryServerStringReq *) pc;
 | |
|     xGLXQueryServerStringReply reply;
 | |
|     size_t n, length;
 | |
|     const char *ptr;
 | |
|     char *buf;
 | |
|     __GLXscreen *pGlxScreen;
 | |
|     int err;
 | |
|     char ver_str[16];
 | |
| 
 | |
|     if (!validGlxScreen(client, req->screen, &pGlxScreen, &err))
 | |
| 	return err;
 | |
| 
 | |
|     switch(req->name) {
 | |
| 	case GLX_VENDOR:
 | |
| 	    ptr = pGlxScreen->GLXvendor;
 | |
| 	    break;
 | |
| 	case GLX_VERSION:
 | |
| 	    /* Return to the server version rather than the screen version
 | |
| 	     * to prevent confusion when they do not match.
 | |
| 	     */
 | |
| 	    snprintf(ver_str, 16, "%d.%d", glxMajorVersion, glxMinorVersion);
 | |
| 	    ptr = ver_str;
 | |
| 	    break;
 | |
| 	case GLX_EXTENSIONS:
 | |
| 	    ptr = pGlxScreen->GLXextensions;
 | |
| 	    break;
 | |
| 	default:
 | |
| 	    return BadValue; 
 | |
|     }
 | |
| 
 | |
|     n = strlen(ptr) + 1;
 | |
|     length = __GLX_PAD(n) >> 2;
 | |
|     reply.type = X_Reply;
 | |
|     reply.sequenceNumber = client->sequence;
 | |
|     reply.length = length;
 | |
|     reply.n = n;
 | |
| 
 | |
|     buf = (char *) malloc(length << 2);
 | |
|     if (buf == NULL) {
 | |
|         return BadAlloc;
 | |
|     }
 | |
|     memcpy(buf, ptr, n);
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         glxSwapQueryServerStringReply(client, &reply, buf);
 | |
|     } else {
 | |
|         WriteToClient(client, sz_xGLXQueryServerStringReply, (char *)&reply);
 | |
|         WriteToClient(client, (int)(length << 2), buf);
 | |
|     }
 | |
| 
 | |
|     free(buf);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| int __glXDisp_ClientInfo(__GLXclientState *cl, GLbyte *pc)
 | |
| {
 | |
|     xGLXClientInfoReq *req = (xGLXClientInfoReq *) pc;
 | |
|     const char *buf;
 | |
|    
 | |
|     cl->GLClientmajorVersion = req->major;
 | |
|     cl->GLClientminorVersion = req->minor;
 | |
|     free(cl->GLClientextensions);
 | |
|     buf = (const char *)(req+1);
 | |
|     cl->GLClientextensions = strdup(buf);
 | |
| 
 | |
|     return Success;
 | |
| }
 |