527 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			527 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C
		
	
	
	
| /**************************************************************************
 | |
| 
 | |
|    Copyright 1998-1999 Precision Insight, Inc., Cedar Park, Texas.
 | |
|    Copyright 2000 VA Linux Systems, Inc.
 | |
|    Copyright (c) 2002, 2009-2012 Apple 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, sub license, 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 and this permission notice (including the
 | |
|    next paragraph) 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 NON-INFRINGEMENT.
 | |
|    IN NO EVENT SHALL PRECISION INSIGHT AND/OR ITS SUPPLIERS 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.
 | |
| 
 | |
| **************************************************************************/
 | |
| 
 | |
| /*
 | |
|  * Authors:
 | |
|  *   Kevin E. Martin <martin@valinux.com>
 | |
|  *   Jens Owen <jens@valinux.com>
 | |
|  *   Rickard E. (Rik) Faith <faith@valinux.com>
 | |
|  *   Jeremy Huddleston <jeremyhu@apple.com>
 | |
|  *
 | |
|  */
 | |
| 
 | |
| #ifdef HAVE_DIX_CONFIG_H
 | |
| #include <dix-config.h>
 | |
| #endif
 | |
| 
 | |
| #include <X11/X.h>
 | |
| #include <X11/Xproto.h>
 | |
| #include "misc.h"
 | |
| #include "dixstruct.h"
 | |
| #include "extnsionst.h"
 | |
| #include "colormapst.h"
 | |
| #include "cursorstr.h"
 | |
| #include "scrnintstr.h"
 | |
| #include "servermd.h"
 | |
| #define _APPLEDRI_SERVER_
 | |
| #include "appledristr.h"
 | |
| #include "swaprep.h"
 | |
| #include "dri.h"
 | |
| #include "dristruct.h"
 | |
| #include "xpr.h"
 | |
| #include "x-hash.h"
 | |
| #include "protocol-versions.h"
 | |
| 
 | |
| static int DRIErrorBase = 0;
 | |
| 
 | |
| static void
 | |
| AppleDRIResetProc(ExtensionEntry* extEntry);
 | |
| static int
 | |
| ProcAppleDRICreatePixmap(ClientPtr client);
 | |
| 
 | |
| static unsigned char DRIReqCode = 0;
 | |
| static int DRIEventBase = 0;
 | |
| 
 | |
| static void
 | |
| SNotifyEvent(xAppleDRINotifyEvent *from, xAppleDRINotifyEvent *to);
 | |
| 
 | |
| typedef struct _DRIEvent *DRIEventPtr;
 | |
| typedef struct _DRIEvent {
 | |
|     DRIEventPtr next;
 | |
|     ClientPtr client;
 | |
|     XID clientResource;
 | |
|     unsigned int mask;
 | |
| } DRIEventRec;
 | |
| 
 | |
| /*ARGSUSED*/
 | |
| static void
 | |
| AppleDRIResetProc(ExtensionEntry* extEntry)
 | |
| {
 | |
|     DRIReset();
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcAppleDRIQueryVersion(register ClientPtr client)
 | |
| {
 | |
|     xAppleDRIQueryVersionReply rep;
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xAppleDRIQueryVersionReq);
 | |
|     rep.type = X_Reply;
 | |
|     rep.length = 0;
 | |
|     rep.sequenceNumber = client->sequence;
 | |
|     rep.majorVersion = SERVER_APPLEDRI_MAJOR_VERSION;
 | |
|     rep.minorVersion = SERVER_APPLEDRI_MINOR_VERSION;
 | |
|     rep.patchVersion = SERVER_APPLEDRI_PATCH_VERSION;
 | |
|     if (client->swapped) {
 | |
|         swaps(&rep.sequenceNumber);
 | |
|         swapl(&rep.length);
 | |
|         swaps(&rep.majorVersion);
 | |
|         swaps(&rep.minorVersion);
 | |
|         swapl(&rep.patchVersion);
 | |
|     }
 | |
|     WriteToClient(client, sizeof(xAppleDRIQueryVersionReply), &rep);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| /* surfaces */
 | |
| 
 | |
| static int
 | |
| ProcAppleDRIQueryDirectRenderingCapable(register ClientPtr client)
 | |
| {
 | |
|     xAppleDRIQueryDirectRenderingCapableReply rep;
 | |
|     Bool isCapable;
 | |
| 
 | |
|     REQUEST(xAppleDRIQueryDirectRenderingCapableReq);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRIQueryDirectRenderingCapableReq);
 | |
|     rep.type = X_Reply;
 | |
|     rep.length = 0;
 | |
|     rep.sequenceNumber = client->sequence;
 | |
| 
 | |
|     if (stuff->screen >= screenInfo.numScreens) {
 | |
|         return BadValue;
 | |
|     }
 | |
| 
 | |
|     if (!DRIQueryDirectRenderingCapable(screenInfo.screens[stuff->screen],
 | |
|                                         &isCapable)) {
 | |
|         return BadValue;
 | |
|     }
 | |
|     rep.isCapable = isCapable;
 | |
| 
 | |
|     if (!client->local)
 | |
|         rep.isCapable = 0;
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         swaps(&rep.sequenceNumber);
 | |
|         swapl(&rep.length);
 | |
|     }
 | |
| 
 | |
|     WriteToClient(client,
 | |
|                   sizeof(xAppleDRIQueryDirectRenderingCapableReply),
 | |
|                   &rep);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcAppleDRIAuthConnection(register ClientPtr client)
 | |
| {
 | |
|     xAppleDRIAuthConnectionReply rep;
 | |
| 
 | |
|     REQUEST(xAppleDRIAuthConnectionReq);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRIAuthConnectionReq);
 | |
| 
 | |
|     rep.type = X_Reply;
 | |
|     rep.length = 0;
 | |
|     rep.sequenceNumber = client->sequence;
 | |
|     rep.authenticated = 1;
 | |
| 
 | |
|     if (!DRIAuthConnection(screenInfo.screens[stuff->screen],
 | |
|                            stuff->magic)) {
 | |
|         ErrorF("Failed to authenticate %u\n", (unsigned int)stuff->magic);
 | |
|         rep.authenticated = 0;
 | |
|     }
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         swaps(&rep.sequenceNumber);
 | |
|         swapl(&rep.length);
 | |
|         swapl(&rep.authenticated); /* Yes, this is a CARD32 ... sigh */
 | |
|     }
 | |
| 
 | |
|     WriteToClient(client, sizeof(xAppleDRIAuthConnectionReply), &rep);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static void
 | |
| surface_notify(void *_arg,
 | |
|                void *data)
 | |
| {
 | |
|     DRISurfaceNotifyArg *arg = _arg;
 | |
|     int client_index = (int)x_cvt_vptr_to_uint(data);
 | |
|     xAppleDRINotifyEvent se;
 | |
| 
 | |
|     if (client_index < 0 || client_index >= currentMaxClients)
 | |
|         return;
 | |
| 
 | |
|     se.type = DRIEventBase + AppleDRISurfaceNotify;
 | |
|     se.kind = arg->kind;
 | |
|     se.arg = arg->id;
 | |
|     se.time = currentTime.milliseconds;
 | |
|     WriteEventsToClient(clients[client_index], 1, (xEvent *)&se);
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcAppleDRICreateSurface(ClientPtr client)
 | |
| {
 | |
|     xAppleDRICreateSurfaceReply rep;
 | |
|     DrawablePtr pDrawable;
 | |
|     xp_surface_id sid;
 | |
|     unsigned int key[2];
 | |
|     int rc;
 | |
| 
 | |
|     REQUEST(xAppleDRICreateSurfaceReq);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRICreateSurfaceReq);
 | |
|     rep.type = X_Reply;
 | |
|     rep.length = 0;
 | |
|     rep.sequenceNumber = client->sequence;
 | |
| 
 | |
|     rc = dixLookupDrawable(&pDrawable, stuff->drawable, client, 0,
 | |
|                            DixReadAccess);
 | |
|     if (rc != Success)
 | |
|         return rc;
 | |
| 
 | |
|     rep.key_0 = rep.key_1 = rep.uid = 0;
 | |
| 
 | |
|     if (!DRICreateSurface(screenInfo.screens[stuff->screen],
 | |
|                           (Drawable)stuff->drawable, pDrawable,
 | |
|                           stuff->client_id, &sid, key,
 | |
|                           surface_notify,
 | |
|                           x_cvt_uint_to_vptr(client->index))) {
 | |
|         return BadValue;
 | |
|     }
 | |
| 
 | |
|     rep.key_0 = key[0];
 | |
|     rep.key_1 = key[1];
 | |
|     rep.uid = sid;
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         swaps(&rep.sequenceNumber);
 | |
|         swapl(&rep.length);
 | |
|         swapl(&rep.key_0);
 | |
|         swapl(&rep.key_1);
 | |
|         swapl(&rep.uid);
 | |
|     }
 | |
| 
 | |
|     WriteToClient(client, sizeof(xAppleDRICreateSurfaceReply), &rep);
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcAppleDRIDestroySurface(register ClientPtr client)
 | |
| {
 | |
|     int rc;
 | |
|     REQUEST(xAppleDRIDestroySurfaceReq);
 | |
|     DrawablePtr pDrawable;
 | |
|     REQUEST_SIZE_MATCH(xAppleDRIDestroySurfaceReq);
 | |
| 
 | |
|     rc = dixLookupDrawable(&pDrawable, stuff->drawable, client, 0,
 | |
|                            DixReadAccess);
 | |
|     if (rc != Success)
 | |
|         return rc;
 | |
| 
 | |
|     if (!DRIDestroySurface(screenInfo.screens[stuff->screen],
 | |
|                            (Drawable)stuff->drawable,
 | |
|                            pDrawable, NULL, NULL)) {
 | |
|         return BadValue;
 | |
|     }
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcAppleDRICreatePixmap(ClientPtr client)
 | |
| {
 | |
|     REQUEST(xAppleDRICreatePixmapReq);
 | |
|     DrawablePtr pDrawable;
 | |
|     int rc;
 | |
|     char path[PATH_MAX];
 | |
|     xAppleDRICreatePixmapReply rep;
 | |
|     int width, height, pitch, bpp;
 | |
|     void *ptr;
 | |
| 
 | |
|     REQUEST_SIZE_MATCH(xAppleDRICreatePixmapReq);
 | |
| 
 | |
|     rc = dixLookupDrawable(&pDrawable, stuff->drawable, client, 0,
 | |
|                            DixReadAccess);
 | |
| 
 | |
|     if (rc != Success)
 | |
|         return rc;
 | |
| 
 | |
|     if (!DRICreatePixmap(screenInfo.screens[stuff->screen],
 | |
|                          (Drawable)stuff->drawable,
 | |
|                          pDrawable,
 | |
|                          path, PATH_MAX)) {
 | |
|         return BadValue;
 | |
|     }
 | |
| 
 | |
|     if (!DRIGetPixmapData(pDrawable, &width, &height,
 | |
|                           &pitch, &bpp, &ptr)) {
 | |
|         return BadValue;
 | |
|     }
 | |
| 
 | |
|     rep.stringLength = strlen(path) + 1;
 | |
| 
 | |
|     rep.type = X_Reply;
 | |
|     rep.length = bytes_to_int32(rep.stringLength);
 | |
|     rep.sequenceNumber = client->sequence;
 | |
|     rep.width = width;
 | |
|     rep.height = height;
 | |
|     rep.pitch = pitch;
 | |
|     rep.bpp = bpp;
 | |
|     rep.size = pitch * height;
 | |
| 
 | |
|     if (sizeof(rep) != sz_xAppleDRICreatePixmapReply)
 | |
|         ErrorF("error sizeof(rep) is %zu\n", sizeof(rep));
 | |
| 
 | |
|     if (client->swapped) {
 | |
|         swaps(&rep.sequenceNumber);
 | |
|         swapl(&rep.length);
 | |
|         swapl(&rep.stringLength);
 | |
|         swapl(&rep.width);
 | |
|         swapl(&rep.height);
 | |
|         swapl(&rep.pitch);
 | |
|         swapl(&rep.bpp);
 | |
|         swapl(&rep.size);
 | |
|     }
 | |
| 
 | |
|     WriteToClient(client, sizeof(rep), &rep);
 | |
|     WriteToClient(client, rep.stringLength, path);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| static int
 | |
| ProcAppleDRIDestroyPixmap(ClientPtr client)
 | |
| {
 | |
|     DrawablePtr pDrawable;
 | |
|     int rc;
 | |
|     REQUEST(xAppleDRIDestroyPixmapReq);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRIDestroyPixmapReq);
 | |
| 
 | |
|     rc = dixLookupDrawable(&pDrawable, stuff->drawable, client, 0,
 | |
|                            DixReadAccess);
 | |
| 
 | |
|     if (rc != Success)
 | |
|         return rc;
 | |
| 
 | |
|     DRIDestroyPixmap(pDrawable);
 | |
| 
 | |
|     return Success;
 | |
| }
 | |
| 
 | |
| /* dispatch */
 | |
| 
 | |
| static int
 | |
| ProcAppleDRIDispatch(register ClientPtr client)
 | |
| {
 | |
|     REQUEST(xReq);
 | |
| 
 | |
|     switch (stuff->data) {
 | |
|     case X_AppleDRIQueryVersion:
 | |
|         return ProcAppleDRIQueryVersion(client);
 | |
| 
 | |
|     case X_AppleDRIQueryDirectRenderingCapable:
 | |
|         return ProcAppleDRIQueryDirectRenderingCapable(client);
 | |
|     }
 | |
| 
 | |
|     if (!client->local)
 | |
|         return DRIErrorBase + AppleDRIClientNotLocal;
 | |
| 
 | |
|     switch (stuff->data) {
 | |
|     case X_AppleDRIAuthConnection:
 | |
|         return ProcAppleDRIAuthConnection(client);
 | |
| 
 | |
|     case X_AppleDRICreateSurface:
 | |
|         return ProcAppleDRICreateSurface(client);
 | |
| 
 | |
|     case X_AppleDRIDestroySurface:
 | |
|         return ProcAppleDRIDestroySurface(client);
 | |
| 
 | |
|     case X_AppleDRICreatePixmap:
 | |
|         return ProcAppleDRICreatePixmap(client);
 | |
| 
 | |
|     case X_AppleDRIDestroyPixmap:
 | |
|         return ProcAppleDRIDestroyPixmap(client);
 | |
| 
 | |
|     default:
 | |
|         return BadRequest;
 | |
|     }
 | |
| }
 | |
| 
 | |
| static void
 | |
| SNotifyEvent(xAppleDRINotifyEvent *from,
 | |
|              xAppleDRINotifyEvent *to)
 | |
| {
 | |
|     to->type = from->type;
 | |
|     to->kind = from->kind;
 | |
|     cpswaps(from->sequenceNumber, to->sequenceNumber);
 | |
|     cpswapl(from->time, to->time);
 | |
|     cpswapl(from->arg, to->arg);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SProcAppleDRIQueryVersion(register ClientPtr client)
 | |
| {
 | |
|     REQUEST(xAppleDRIQueryVersionReq);
 | |
|     swaps(&stuff->length);
 | |
|     return ProcAppleDRIQueryVersion(client);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SProcAppleDRIQueryDirectRenderingCapable(register ClientPtr client)
 | |
| {
 | |
|     REQUEST(xAppleDRIQueryDirectRenderingCapableReq);
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRIQueryDirectRenderingCapableReq);
 | |
|     swapl(&stuff->screen);
 | |
|     return ProcAppleDRIQueryDirectRenderingCapable(client);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SProcAppleDRIAuthConnection(register ClientPtr client)
 | |
| {
 | |
|     REQUEST(xAppleDRIAuthConnectionReq);
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRIAuthConnectionReq);
 | |
|     swapl(&stuff->screen);
 | |
|     swapl(&stuff->magic);
 | |
|     return ProcAppleDRIAuthConnection(client);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SProcAppleDRICreateSurface(register ClientPtr client)
 | |
| {
 | |
|     REQUEST(xAppleDRICreateSurfaceReq);
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRICreateSurfaceReq);
 | |
|     swapl(&stuff->screen);
 | |
|     swapl(&stuff->drawable);
 | |
|     swapl(&stuff->client_id);
 | |
|     return ProcAppleDRICreateSurface(client);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SProcAppleDRIDestroySurface(register ClientPtr client)
 | |
| {
 | |
|     REQUEST(xAppleDRIDestroySurfaceReq);
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRIDestroySurfaceReq);
 | |
|     swapl(&stuff->screen);
 | |
|     swapl(&stuff->drawable);
 | |
|     return ProcAppleDRIDestroySurface(client);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SProcAppleDRICreatePixmap(register ClientPtr client)
 | |
| {
 | |
|     REQUEST(xAppleDRICreatePixmapReq);
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRICreatePixmapReq);
 | |
|     swapl(&stuff->screen);
 | |
|     swapl(&stuff->drawable);
 | |
|     return ProcAppleDRICreatePixmap(client);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SProcAppleDRIDestroyPixmap(register ClientPtr client)
 | |
| {
 | |
|     REQUEST(xAppleDRIDestroyPixmapReq);
 | |
|     swaps(&stuff->length);
 | |
|     REQUEST_SIZE_MATCH(xAppleDRIDestroyPixmapReq);
 | |
|     swapl(&stuff->drawable);
 | |
|     return ProcAppleDRIDestroyPixmap(client);
 | |
| }
 | |
| 
 | |
| static int
 | |
| SProcAppleDRIDispatch(register ClientPtr client)
 | |
| {
 | |
|     REQUEST(xReq);
 | |
| 
 | |
|     switch (stuff->data) {
 | |
|     case X_AppleDRIQueryVersion:
 | |
|         return SProcAppleDRIQueryVersion(client);
 | |
| 
 | |
|     case X_AppleDRIQueryDirectRenderingCapable:
 | |
|         return SProcAppleDRIQueryDirectRenderingCapable(client);
 | |
|     }
 | |
| 
 | |
|     if (!client->local)
 | |
|         return DRIErrorBase + AppleDRIClientNotLocal;
 | |
| 
 | |
|     switch (stuff->data) {
 | |
|     case X_AppleDRIAuthConnection:
 | |
|         return SProcAppleDRIAuthConnection(client);
 | |
| 
 | |
|     case X_AppleDRICreateSurface:
 | |
|         return SProcAppleDRICreateSurface(client);
 | |
| 
 | |
|     case X_AppleDRIDestroySurface:
 | |
|         return SProcAppleDRIDestroySurface(client);
 | |
| 
 | |
|     case X_AppleDRICreatePixmap:
 | |
|         return SProcAppleDRICreatePixmap(client);
 | |
| 
 | |
|     case X_AppleDRIDestroyPixmap:
 | |
|         return SProcAppleDRIDestroyPixmap(client);
 | |
| 
 | |
|     default:
 | |
|         return BadRequest;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void
 | |
| AppleDRIExtensionInit(void)
 | |
| {
 | |
|     ExtensionEntry* extEntry;
 | |
| 
 | |
|     if (DRIExtensionInit() &&
 | |
|         (extEntry = AddExtension(APPLEDRINAME,
 | |
|                                  AppleDRINumberEvents,
 | |
|                                  AppleDRINumberErrors,
 | |
|                                  ProcAppleDRIDispatch,
 | |
|                                  SProcAppleDRIDispatch,
 | |
|                                  AppleDRIResetProc,
 | |
|                                  StandardMinorOpcode))) {
 | |
|         size_t i;
 | |
|         DRIReqCode = (unsigned char)extEntry->base;
 | |
|         DRIErrorBase = extEntry->errorBase;
 | |
|         DRIEventBase = extEntry->eventBase;
 | |
|         for (i = 0; i < AppleDRINumberEvents; i++)
 | |
|             EventSwapVector[DRIEventBase + i] = (EventSwapPtr)SNotifyEvent;
 | |
|     }
 | |
| }
 |