372 lines
12 KiB
C
372 lines
12 KiB
C
/***********************************************************
|
|
|
|
Copyright 1987, 1998 The Open Group
|
|
|
|
Permission to use, copy, modify, distribute, and sell this software and its
|
|
documentation for any purpose is hereby granted without fee, provided that
|
|
the above copyright notice appear in all copies and that both that
|
|
copyright notice and this permission notice appear in supporting
|
|
documentation.
|
|
|
|
The above copyright notice and this permission notice shall be included in
|
|
all copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
OPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
|
AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
Except as contained in this notice, the name of The Open Group shall not be
|
|
used in advertising or otherwise to promote the sale, use or other dealings
|
|
in this Software without prior written authorization from The Open Group.
|
|
|
|
Copyright 1987 by Digital Equipment Corporation, Maynard, Massachusetts.
|
|
|
|
All Rights Reserved
|
|
|
|
Permission to use, copy, modify, and distribute this software and its
|
|
documentation for any purpose and without fee is hereby granted,
|
|
provided that the above copyright notice appear in all copies and that
|
|
both that copyright notice and this permission notice appear in
|
|
supporting documentation, and that the name of Digital not be
|
|
used in advertising or publicity pertaining to distribution of the
|
|
software without specific, written prior permission.
|
|
|
|
DIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
|
|
ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
|
|
DIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
|
|
ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
|
|
WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
|
|
ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
|
|
SOFTWARE.
|
|
|
|
******************************************************************/
|
|
|
|
#include <dix-config.h>
|
|
|
|
#include <X11/X.h>
|
|
#include <X11/Xproto.h>
|
|
|
|
#include "dix/extension_priv.h"
|
|
#include "dix/registry_priv.h"
|
|
|
|
#include "misc.h"
|
|
#include "dixstruct.h"
|
|
#include "extnsionst.h"
|
|
#include "gcstruct.h"
|
|
#include "scrnintstr.h"
|
|
#include "dispatch.h"
|
|
#include "privates.h"
|
|
#include "xace.h"
|
|
|
|
#define LAST_ERROR 255
|
|
|
|
static ExtensionEntry **extensions = (ExtensionEntry **) NULL;
|
|
|
|
int lastEvent = EXTENSION_EVENT_BASE;
|
|
static int lastError = FirstExtensionError;
|
|
static unsigned int NumExtensions = RESERVED_EXTENSIONS;
|
|
|
|
static struct { const char *name; int id; } reservedExt[] = {
|
|
{ "BIG-REQUESTS", EXTENSION_MAJOR_BIG_REQUESTS },
|
|
{ "Apple-WM", EXTENSION_MAJOR_APPLE_WM },
|
|
{ "Apple-DRI", EXTENSION_MAJOR_APPLE_DRI },
|
|
{ "Composite", EXTENSION_MAJOR_COMPOSITE },
|
|
{ "DAMAGE", EXTENSION_MAJOR_DAMAGE },
|
|
{ "DOUBLE-BUFFER", EXTENSION_MAJOR_DOUBLE_BUFFER },
|
|
{ "DPMS", EXTENSION_MAJOR_DPMS },
|
|
{ "DRI2", EXTENSION_MAJOR_DRI2 },
|
|
{ "DRI3", EXTENSION_MAJOR_DRI3 },
|
|
{ "Generic Event Extension", EXTENSION_MAJOR_GENERIC_EVENT },
|
|
{ "GLX", EXTENSION_MAJOR_GLX },
|
|
{ "MIT-SCREEN-SAVER", EXTENSION_MAJOR_MIT_SCREEN_SAVER },
|
|
{ "NAMESPACE", EXTENSION_MAJOR_NAMESPACE },
|
|
{ "Present", EXTENSION_MAJOR_PRESENT },
|
|
{ "RANDR", EXTENSION_MAJOR_RANDR },
|
|
{ "RECORD", EXTENSION_MAJOR_RECORD },
|
|
{ "RENDER", EXTENSION_MAJOR_RENDER },
|
|
{ "SECURITY", EXTENSION_MAJOR_SECURITY },
|
|
{ "SELinux", EXTENSION_MAJOR_SELINUX },
|
|
{ "SHAPE", EXTENSION_MAJOR_SHAPE },
|
|
{ "MIT-SHM", EXTENSION_MAJOR_SHM },
|
|
{ "SYNC", EXTENSION_MAJOR_SYNC },
|
|
{ "Windows-DRI", EXTENSION_MAJOR_WINDOWS_DRI },
|
|
{ "XFIXES", EXTENSION_MAJOR_XFIXES },
|
|
{ "XFree86-Bigfont", EXTENSION_MAJOR_XF86_BIGFONT },
|
|
{ "XFree86-DGA", EXTENSION_MAJOR_XF86_DGA },
|
|
{ "XFree86-DRI", EXTENSION_MAJOR_XF86_DRI },
|
|
{ "XFree86-VidModeExtension", EXTENSION_MAJOR_XF86_VIDMODE },
|
|
{ "XC-MISC", EXTENSION_MAJOR_XC_MISC },
|
|
{ "XInputExtension", EXTENSION_MAJOR_XINPUT },
|
|
{ "XINERAMA", EXTENSION_MAJOR_XINERAMA },
|
|
{ "XKEYBOARD", EXTENSION_MAJOR_XKEYBOARD },
|
|
{ "X-Resource", EXTENSION_MAJOR_XRESOURCE },
|
|
{ "XTEST", EXTENSION_MAJOR_XTEST },
|
|
{ "XVideo", EXTENSION_MAJOR_XVIDEO },
|
|
{ "XVideo-MotionCompensation", EXTENSION_MAJOR_XVMC },
|
|
};
|
|
|
|
static int checkReserved(const char* name)
|
|
{
|
|
for (int i=0; i<ARRAY_SIZE(reservedExt); i++) {
|
|
if (strcmp(name, reservedExt[i].name) == 0) {
|
|
if (reservedExt[i].id < (RESERVED_EXTENSIONS + EXTENSION_BASE))
|
|
return reservedExt[i].id;
|
|
FatalError("BUG: RESERVED_EXTENSIONS too small for %d\n", reservedExt[i].id);
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
ExtensionEntry *
|
|
AddExtension(const char *name, int NumEvents, int NumErrors,
|
|
int (*MainProc) (ClientPtr c1),
|
|
int (*SwappedMainProc) (ClientPtr c2),
|
|
void (*CloseDownProc) (ExtensionEntry * e),
|
|
unsigned short (*MinorOpcodeProc) (ClientPtr c3))
|
|
{
|
|
if (!extensions)
|
|
extensions = calloc(NumExtensions, sizeof(ExtensionEntry*));
|
|
if (!extensions)
|
|
return NULL;
|
|
|
|
if (!MainProc || !SwappedMainProc || !MinorOpcodeProc)
|
|
return ((ExtensionEntry *) NULL);
|
|
if ((lastEvent + NumEvents > MAXEVENTS) ||
|
|
(unsigned) (lastError + NumErrors > LAST_ERROR)) {
|
|
LogMessage(X_ERROR, "Not enabling extension %s: maximum number of "
|
|
"events or errors exceeded.\n", name);
|
|
return ((ExtensionEntry *) NULL);
|
|
}
|
|
|
|
ExtensionEntry *ext = calloc(1, sizeof(ExtensionEntry));
|
|
if (!ext)
|
|
return NULL;
|
|
if (!dixAllocatePrivates(&ext->devPrivates, PRIVATE_EXTENSION))
|
|
goto badalloc;
|
|
ext->name = strdup(name);
|
|
if (!ext->name)
|
|
goto badalloc;
|
|
|
|
int i = checkReserved(ext->name);
|
|
if (i == -1) {
|
|
i = NumExtensions;
|
|
ExtensionEntry **newexts = reallocarray(extensions, i + 1, sizeof(ExtensionEntry *));
|
|
if (!newexts)
|
|
goto badalloc;
|
|
|
|
NumExtensions++;
|
|
extensions = newexts;
|
|
} else {
|
|
i = i - EXTENSION_BASE;
|
|
}
|
|
|
|
extensions[i] = ext;
|
|
ext->index = i;
|
|
ext->base = i + EXTENSION_BASE;
|
|
ext->CloseDown = CloseDownProc;
|
|
ext->MinorOpcode = MinorOpcodeProc;
|
|
ProcVector[i + EXTENSION_BASE] = MainProc;
|
|
SwappedProcVector[i + EXTENSION_BASE] = SwappedMainProc;
|
|
if (NumEvents) {
|
|
ext->eventBase = lastEvent;
|
|
ext->eventLast = lastEvent + NumEvents;
|
|
lastEvent += NumEvents;
|
|
}
|
|
else {
|
|
ext->eventBase = 0;
|
|
ext->eventLast = 0;
|
|
}
|
|
if (NumErrors) {
|
|
ext->errorBase = lastError;
|
|
ext->errorLast = lastError + NumErrors;
|
|
lastError += NumErrors;
|
|
}
|
|
else {
|
|
ext->errorBase = 0;
|
|
ext->errorLast = 0;
|
|
}
|
|
|
|
#ifdef X_REGISTRY_REQUEST
|
|
RegisterExtensionNames(ext);
|
|
#endif
|
|
return ext;
|
|
|
|
badalloc:
|
|
if (ext) {
|
|
free((char*)ext->name);
|
|
dixFreePrivates(ext->devPrivates, PRIVATE_EXTENSION);
|
|
free(ext);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* CheckExtension returns the extensions[] entry for the requested
|
|
* extension name. Maybe this could just return a Bool instead?
|
|
*/
|
|
ExtensionEntry *
|
|
CheckExtension(const char *extname)
|
|
{
|
|
if (!extensions)
|
|
return NULL;
|
|
|
|
for (int i = 0; i < NumExtensions; i++) {
|
|
if (extensions[i] &&
|
|
extensions[i]->name &&
|
|
strcmp(extensions[i]->name, extname) == 0) {
|
|
return extensions[i];
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
/*
|
|
* Added as part of Xace.
|
|
*/
|
|
ExtensionEntry *
|
|
GetExtensionEntry(int major)
|
|
{
|
|
if ((major < EXTENSION_BASE) || !extensions)
|
|
return NULL;
|
|
major -= EXTENSION_BASE;
|
|
if (major >= NumExtensions)
|
|
return NULL;
|
|
return extensions[major];
|
|
}
|
|
|
|
unsigned short
|
|
StandardMinorOpcode(ClientPtr client)
|
|
{
|
|
return ((xReq *) client->requestBuffer)->data;
|
|
}
|
|
|
|
void
|
|
CloseDownExtensions(void)
|
|
{
|
|
if (!extensions)
|
|
return;
|
|
|
|
for (int i = NumExtensions - 1; i >= 0; i--) {
|
|
if (!extensions[i])
|
|
continue;
|
|
if (extensions[i]->CloseDown)
|
|
extensions[i]->CloseDown(extensions[i]);
|
|
NumExtensions = i;
|
|
free((void *) extensions[i]->name);
|
|
dixFreePrivates(extensions[i]->devPrivates, PRIVATE_EXTENSION);
|
|
free(extensions[i]);
|
|
extensions[i] = NULL;
|
|
}
|
|
free(extensions);
|
|
extensions = (ExtensionEntry **) NULL;
|
|
NumExtensions = RESERVED_EXTENSIONS;
|
|
lastEvent = EXTENSION_EVENT_BASE;
|
|
lastError = FirstExtensionError;
|
|
}
|
|
|
|
static Bool
|
|
ExtensionAvailable(ClientPtr client, ExtensionEntry *ext)
|
|
{
|
|
if (!ext)
|
|
return FALSE;
|
|
if (XaceHookExtAccess(client, ext) != Success)
|
|
return FALSE;
|
|
if (!ext->base)
|
|
return FALSE;
|
|
return TRUE;
|
|
}
|
|
|
|
int
|
|
ProcQueryExtension(ClientPtr client)
|
|
{
|
|
REQUEST(xQueryExtensionReq);
|
|
REQUEST_FIXED_SIZE(xQueryExtensionReq, stuff->nbytes);
|
|
|
|
xQueryExtensionReply rep = {
|
|
.type = X_Reply,
|
|
.sequenceNumber = client->sequence,
|
|
.length = 0,
|
|
.major_opcode = 0
|
|
};
|
|
|
|
if (!NumExtensions || !extensions)
|
|
rep.present = xFalse;
|
|
else {
|
|
char extname[PATH_MAX] = { 0 };
|
|
strncpy(extname, (char *) &stuff[1], min(stuff->nbytes, sizeof(extname)-1));
|
|
ExtensionEntry *extEntry = CheckExtension(extname);
|
|
|
|
if (!extEntry || !ExtensionAvailable(client, extEntry))
|
|
rep.present = xFalse;
|
|
else {
|
|
rep.present = xTrue;
|
|
rep.major_opcode = extEntry->base;
|
|
rep.first_event = extEntry->eventBase;
|
|
rep.first_error = extEntry->errorBase;
|
|
}
|
|
}
|
|
|
|
if (client->swapped) {
|
|
swaps(&rep.sequenceNumber);
|
|
}
|
|
WriteToClient(client, sizeof(rep), &rep);
|
|
return Success;
|
|
}
|
|
|
|
int
|
|
ProcListExtensions(ClientPtr client)
|
|
{
|
|
char *bufptr, *buffer;
|
|
int total_length = 0;
|
|
|
|
REQUEST_SIZE_MATCH(xReq);
|
|
|
|
xListExtensionsReply rep = {
|
|
.type = X_Reply,
|
|
.nExtensions = 0,
|
|
.sequenceNumber = client->sequence,
|
|
.length = 0
|
|
};
|
|
buffer = NULL;
|
|
|
|
if (NumExtensions && extensions) {
|
|
int i;
|
|
|
|
for (i = 0; i < NumExtensions; i++) {
|
|
/* call callbacks to find out whether to show extension */
|
|
if (!ExtensionAvailable(client, extensions[i]))
|
|
continue;
|
|
|
|
total_length += strlen(extensions[i]->name) + 1;
|
|
rep.nExtensions += 1;
|
|
}
|
|
rep.length = bytes_to_int32(total_length);
|
|
buffer = bufptr = calloc(1, total_length);
|
|
if (!buffer)
|
|
return BadAlloc;
|
|
for (i = 0; i < NumExtensions; i++) {
|
|
int len;
|
|
|
|
if (!ExtensionAvailable(client, extensions[i]))
|
|
continue;
|
|
|
|
*bufptr++ = len = strlen(extensions[i]->name);
|
|
memcpy(bufptr, extensions[i]->name, len);
|
|
bufptr += len;
|
|
}
|
|
}
|
|
|
|
if (client->swapped) {
|
|
swaps(&rep.sequenceNumber);
|
|
swapl(&rep.length);
|
|
}
|
|
WriteToClient(client, sizeof(rep), &rep);
|
|
WriteToClient(client, total_length, buffer);
|
|
|
|
free(buffer);
|
|
return Success;
|
|
}
|