extension.c revision 05b261ec
105b261ecSmrg/***********************************************************
205b261ecSmrg
305b261ecSmrgCopyright 1987, 1998  The Open Group
405b261ecSmrg
505b261ecSmrgPermission to use, copy, modify, distribute, and sell this software and its
605b261ecSmrgdocumentation for any purpose is hereby granted without fee, provided that
705b261ecSmrgthe above copyright notice appear in all copies and that both that
805b261ecSmrgcopyright notice and this permission notice appear in supporting
905b261ecSmrgdocumentation.
1005b261ecSmrg
1105b261ecSmrgThe above copyright notice and this permission notice shall be included in
1205b261ecSmrgall copies or substantial portions of the Software.
1305b261ecSmrg
1405b261ecSmrgTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
1505b261ecSmrgIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
1605b261ecSmrgFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.  IN NO EVENT SHALL THE
1705b261ecSmrgOPEN GROUP BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
1805b261ecSmrgAN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
1905b261ecSmrgCONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
2005b261ecSmrg
2105b261ecSmrgExcept as contained in this notice, the name of The Open Group shall not be
2205b261ecSmrgused in advertising or otherwise to promote the sale, use or other dealings
2305b261ecSmrgin this Software without prior written authorization from The Open Group.
2405b261ecSmrg
2505b261ecSmrg
2605b261ecSmrgCopyright 1987 by Digital Equipment Corporation, Maynard, Massachusetts.
2705b261ecSmrg
2805b261ecSmrg                        All Rights Reserved
2905b261ecSmrg
3005b261ecSmrgPermission to use, copy, modify, and distribute this software and its
3105b261ecSmrgdocumentation for any purpose and without fee is hereby granted,
3205b261ecSmrgprovided that the above copyright notice appear in all copies and that
3305b261ecSmrgboth that copyright notice and this permission notice appear in
3405b261ecSmrgsupporting documentation, and that the name of Digital not be
3505b261ecSmrgused in advertising or publicity pertaining to distribution of the
3605b261ecSmrgsoftware without specific, written prior permission.
3705b261ecSmrg
3805b261ecSmrgDIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
3905b261ecSmrgALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
4005b261ecSmrgDIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
4105b261ecSmrgANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
4205b261ecSmrgWHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
4305b261ecSmrgARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
4405b261ecSmrgSOFTWARE.
4505b261ecSmrg
4605b261ecSmrg******************************************************************/
4705b261ecSmrg
4805b261ecSmrg#ifdef HAVE_DIX_CONFIG_H
4905b261ecSmrg#include <dix-config.h>
5005b261ecSmrg#endif
5105b261ecSmrg
5205b261ecSmrg#include <X11/X.h>
5305b261ecSmrg#define NEED_EVENTS
5405b261ecSmrg#define NEED_REPLIES
5505b261ecSmrg#include <X11/Xproto.h>
5605b261ecSmrg#include "misc.h"
5705b261ecSmrg#include "dixstruct.h"
5805b261ecSmrg#include "extnsionst.h"
5905b261ecSmrg#include "gcstruct.h"
6005b261ecSmrg#include "scrnintstr.h"
6105b261ecSmrg#include "dispatch.h"
6205b261ecSmrg#include "xace.h"
6305b261ecSmrg
6405b261ecSmrg#define EXTENSION_BASE  128
6505b261ecSmrg#define EXTENSION_EVENT_BASE  64
6605b261ecSmrg#define LAST_EVENT  128
6705b261ecSmrg#define LAST_ERROR 255
6805b261ecSmrg
6905b261ecSmrgstatic ExtensionEntry **extensions = (ExtensionEntry **)NULL;
7005b261ecSmrg
7105b261ecSmrgint lastEvent = EXTENSION_EVENT_BASE;
7205b261ecSmrgstatic int lastError = FirstExtensionError;
7305b261ecSmrgstatic unsigned int NumExtensions = 0;
7405b261ecSmrg
7505b261ecSmrgextern int extensionPrivateLen;
7605b261ecSmrgextern unsigned *extensionPrivateSizes;
7705b261ecSmrgextern unsigned totalExtensionSize;
7805b261ecSmrg
7905b261ecSmrgstatic void
8005b261ecSmrgInitExtensionPrivates(ExtensionEntry *ext)
8105b261ecSmrg{
8205b261ecSmrg    char *ptr;
8305b261ecSmrg    DevUnion *ppriv;
8405b261ecSmrg    unsigned *sizes;
8505b261ecSmrg    unsigned size;
8605b261ecSmrg    int i;
8705b261ecSmrg
8805b261ecSmrg    if (totalExtensionSize == sizeof(ExtensionEntry))
8905b261ecSmrg	ppriv = (DevUnion *)NULL;
9005b261ecSmrg    else
9105b261ecSmrg	ppriv = (DevUnion *)(ext + 1);
9205b261ecSmrg
9305b261ecSmrg    ext->devPrivates = ppriv;
9405b261ecSmrg    sizes = extensionPrivateSizes;
9505b261ecSmrg    ptr = (char *)(ppriv + extensionPrivateLen);
9605b261ecSmrg    for (i = extensionPrivateLen; --i >= 0; ppriv++, sizes++)
9705b261ecSmrg    {
9805b261ecSmrg	if ( (size = *sizes) )
9905b261ecSmrg	{
10005b261ecSmrg	    ppriv->ptr = (pointer)ptr;
10105b261ecSmrg	    ptr += size;
10205b261ecSmrg	}
10305b261ecSmrg	else
10405b261ecSmrg	    ppriv->ptr = (pointer)NULL;
10505b261ecSmrg    }
10605b261ecSmrg}
10705b261ecSmrg
10805b261ecSmrg_X_EXPORT ExtensionEntry *
10905b261ecSmrgAddExtension(char *name, int NumEvents, int NumErrors,
11005b261ecSmrg	     int (*MainProc)(ClientPtr c1),
11105b261ecSmrg	     int (*SwappedMainProc)(ClientPtr c2),
11205b261ecSmrg	     void (*CloseDownProc)(ExtensionEntry *e),
11305b261ecSmrg	     unsigned short (*MinorOpcodeProc)(ClientPtr c3))
11405b261ecSmrg{
11505b261ecSmrg    int i;
11605b261ecSmrg    ExtensionEntry *ext, **newexts;
11705b261ecSmrg
11805b261ecSmrg    if (!MainProc || !SwappedMainProc || !CloseDownProc || !MinorOpcodeProc)
11905b261ecSmrg        return((ExtensionEntry *) NULL);
12005b261ecSmrg    if ((lastEvent + NumEvents > LAST_EVENT) ||
12105b261ecSmrg	        (unsigned)(lastError + NumErrors > LAST_ERROR))
12205b261ecSmrg        return((ExtensionEntry *) NULL);
12305b261ecSmrg
12405b261ecSmrg    ext = (ExtensionEntry *) xalloc(totalExtensionSize);
12505b261ecSmrg    if (!ext)
12605b261ecSmrg	return((ExtensionEntry *) NULL);
12705b261ecSmrg    bzero(ext, totalExtensionSize);
12805b261ecSmrg    InitExtensionPrivates(ext);
12905b261ecSmrg    ext->name = (char *)xalloc(strlen(name) + 1);
13005b261ecSmrg    ext->num_aliases = 0;
13105b261ecSmrg    ext->aliases = (char **)NULL;
13205b261ecSmrg    if (!ext->name)
13305b261ecSmrg    {
13405b261ecSmrg	xfree(ext);
13505b261ecSmrg	return((ExtensionEntry *) NULL);
13605b261ecSmrg    }
13705b261ecSmrg    strcpy(ext->name,  name);
13805b261ecSmrg    i = NumExtensions;
13905b261ecSmrg    newexts = (ExtensionEntry **) xrealloc(extensions,
14005b261ecSmrg					   (i + 1) * sizeof(ExtensionEntry *));
14105b261ecSmrg    if (!newexts)
14205b261ecSmrg    {
14305b261ecSmrg	xfree(ext->name);
14405b261ecSmrg	xfree(ext);
14505b261ecSmrg	return((ExtensionEntry *) NULL);
14605b261ecSmrg    }
14705b261ecSmrg    NumExtensions++;
14805b261ecSmrg    extensions = newexts;
14905b261ecSmrg    extensions[i] = ext;
15005b261ecSmrg    ext->index = i;
15105b261ecSmrg    ext->base = i + EXTENSION_BASE;
15205b261ecSmrg    ext->CloseDown = CloseDownProc;
15305b261ecSmrg    ext->MinorOpcode = MinorOpcodeProc;
15405b261ecSmrg    ProcVector[i + EXTENSION_BASE] = MainProc;
15505b261ecSmrg    SwappedProcVector[i + EXTENSION_BASE] = SwappedMainProc;
15605b261ecSmrg    if (NumEvents)
15705b261ecSmrg    {
15805b261ecSmrg        ext->eventBase = lastEvent;
15905b261ecSmrg	ext->eventLast = lastEvent + NumEvents;
16005b261ecSmrg	lastEvent += NumEvents;
16105b261ecSmrg    }
16205b261ecSmrg    else
16305b261ecSmrg    {
16405b261ecSmrg        ext->eventBase = 0;
16505b261ecSmrg        ext->eventLast = 0;
16605b261ecSmrg    }
16705b261ecSmrg    if (NumErrors)
16805b261ecSmrg    {
16905b261ecSmrg        ext->errorBase = lastError;
17005b261ecSmrg	ext->errorLast = lastError + NumErrors;
17105b261ecSmrg	lastError += NumErrors;
17205b261ecSmrg    }
17305b261ecSmrg    else
17405b261ecSmrg    {
17505b261ecSmrg        ext->errorBase = 0;
17605b261ecSmrg        ext->errorLast = 0;
17705b261ecSmrg    }
17805b261ecSmrg
17905b261ecSmrg    return(ext);
18005b261ecSmrg}
18105b261ecSmrg
18205b261ecSmrg_X_EXPORT Bool AddExtensionAlias(char *alias, ExtensionEntry *ext)
18305b261ecSmrg{
18405b261ecSmrg    char *name;
18505b261ecSmrg    char **aliases;
18605b261ecSmrg
18705b261ecSmrg    aliases = (char **)xrealloc(ext->aliases,
18805b261ecSmrg				(ext->num_aliases + 1) * sizeof(char *));
18905b261ecSmrg    if (!aliases)
19005b261ecSmrg	return FALSE;
19105b261ecSmrg    ext->aliases = aliases;
19205b261ecSmrg    name = (char *)xalloc(strlen(alias) + 1);
19305b261ecSmrg    if (!name)
19405b261ecSmrg	return FALSE;
19505b261ecSmrg    strcpy(name,  alias);
19605b261ecSmrg    ext->aliases[ext->num_aliases] = name;
19705b261ecSmrg    ext->num_aliases++;
19805b261ecSmrg    return TRUE;
19905b261ecSmrg}
20005b261ecSmrg
20105b261ecSmrgstatic int
20205b261ecSmrgFindExtension(char *extname, int len)
20305b261ecSmrg{
20405b261ecSmrg    int i, j;
20505b261ecSmrg
20605b261ecSmrg    for (i=0; i<NumExtensions; i++)
20705b261ecSmrg    {
20805b261ecSmrg	if ((strlen(extensions[i]->name) == len) &&
20905b261ecSmrg	    !strncmp(extname, extensions[i]->name, len))
21005b261ecSmrg	    break;
21105b261ecSmrg	for (j = extensions[i]->num_aliases; --j >= 0;)
21205b261ecSmrg	{
21305b261ecSmrg	    if ((strlen(extensions[i]->aliases[j]) == len) &&
21405b261ecSmrg		!strncmp(extname, extensions[i]->aliases[j], len))
21505b261ecSmrg		break;
21605b261ecSmrg	}
21705b261ecSmrg	if (j >= 0) break;
21805b261ecSmrg    }
21905b261ecSmrg    return ((i == NumExtensions) ? -1 : i);
22005b261ecSmrg}
22105b261ecSmrg
22205b261ecSmrg/*
22305b261ecSmrg * CheckExtension returns the extensions[] entry for the requested
22405b261ecSmrg * extension name.  Maybe this could just return a Bool instead?
22505b261ecSmrg */
22605b261ecSmrg_X_EXPORT ExtensionEntry *
22705b261ecSmrgCheckExtension(const char *extname)
22805b261ecSmrg{
22905b261ecSmrg    int n;
23005b261ecSmrg
23105b261ecSmrg    n = FindExtension((char*)extname, strlen(extname));
23205b261ecSmrg    if (n != -1)
23305b261ecSmrg	return extensions[n];
23405b261ecSmrg    else
23505b261ecSmrg	return NULL;
23605b261ecSmrg}
23705b261ecSmrg
23805b261ecSmrg/*
23905b261ecSmrg * Added as part of Xace.
24005b261ecSmrg */
24105b261ecSmrgExtensionEntry *
24205b261ecSmrgGetExtensionEntry(int major)
24305b261ecSmrg{
24405b261ecSmrg    if (major < EXTENSION_BASE)
24505b261ecSmrg	return NULL;
24605b261ecSmrg    major -= EXTENSION_BASE;
24705b261ecSmrg    if (major >= NumExtensions)
24805b261ecSmrg	return NULL;
24905b261ecSmrg    return extensions[major];
25005b261ecSmrg}
25105b261ecSmrg
25205b261ecSmrg_X_EXPORT void
25305b261ecSmrgDeclareExtensionSecurity(char *extname, Bool secure)
25405b261ecSmrg{
25505b261ecSmrg    int i = FindExtension(extname, strlen(extname));
25605b261ecSmrg    if (i >= 0)
25705b261ecSmrg	XaceHook(XACE_DECLARE_EXT_SECURE, extensions[i], secure);
25805b261ecSmrg}
25905b261ecSmrg
26005b261ecSmrg_X_EXPORT unsigned short
26105b261ecSmrgStandardMinorOpcode(ClientPtr client)
26205b261ecSmrg{
26305b261ecSmrg    return ((xReq *)client->requestBuffer)->data;
26405b261ecSmrg}
26505b261ecSmrg
26605b261ecSmrg_X_EXPORT unsigned short
26705b261ecSmrgMinorOpcodeOfRequest(ClientPtr client)
26805b261ecSmrg{
26905b261ecSmrg    unsigned char major;
27005b261ecSmrg
27105b261ecSmrg    major = ((xReq *)client->requestBuffer)->reqType;
27205b261ecSmrg    if (major < EXTENSION_BASE)
27305b261ecSmrg	return 0;
27405b261ecSmrg    major -= EXTENSION_BASE;
27505b261ecSmrg    if (major >= NumExtensions)
27605b261ecSmrg	return 0;
27705b261ecSmrg    return (*extensions[major]->MinorOpcode)(client);
27805b261ecSmrg}
27905b261ecSmrg
28005b261ecSmrgvoid
28105b261ecSmrgCloseDownExtensions(void)
28205b261ecSmrg{
28305b261ecSmrg    int i,j;
28405b261ecSmrg
28505b261ecSmrg    for (i = NumExtensions - 1; i >= 0; i--)
28605b261ecSmrg    {
28705b261ecSmrg	(* extensions[i]->CloseDown)(extensions[i]);
28805b261ecSmrg	NumExtensions = i;
28905b261ecSmrg	xfree(extensions[i]->name);
29005b261ecSmrg	for (j = extensions[i]->num_aliases; --j >= 0;)
29105b261ecSmrg	    xfree(extensions[i]->aliases[j]);
29205b261ecSmrg	xfree(extensions[i]->aliases);
29305b261ecSmrg	xfree(extensions[i]);
29405b261ecSmrg    }
29505b261ecSmrg    xfree(extensions);
29605b261ecSmrg    extensions = (ExtensionEntry **)NULL;
29705b261ecSmrg    lastEvent = EXTENSION_EVENT_BASE;
29805b261ecSmrg    lastError = FirstExtensionError;
29905b261ecSmrg}
30005b261ecSmrg
30105b261ecSmrgint
30205b261ecSmrgProcQueryExtension(ClientPtr client)
30305b261ecSmrg{
30405b261ecSmrg    xQueryExtensionReply reply;
30505b261ecSmrg    int i;
30605b261ecSmrg    REQUEST(xQueryExtensionReq);
30705b261ecSmrg
30805b261ecSmrg    REQUEST_FIXED_SIZE(xQueryExtensionReq, stuff->nbytes);
30905b261ecSmrg
31005b261ecSmrg    reply.type = X_Reply;
31105b261ecSmrg    reply.length = 0;
31205b261ecSmrg    reply.major_opcode = 0;
31305b261ecSmrg    reply.sequenceNumber = client->sequence;
31405b261ecSmrg
31505b261ecSmrg    if ( ! NumExtensions )
31605b261ecSmrg        reply.present = xFalse;
31705b261ecSmrg    else
31805b261ecSmrg    {
31905b261ecSmrg	i = FindExtension((char *)&stuff[1], stuff->nbytes);
32005b261ecSmrg        if (i < 0 || !XaceHook(XACE_EXT_ACCESS, client, extensions[i]))
32105b261ecSmrg            reply.present = xFalse;
32205b261ecSmrg        else
32305b261ecSmrg        {
32405b261ecSmrg            reply.present = xTrue;
32505b261ecSmrg	    reply.major_opcode = extensions[i]->base;
32605b261ecSmrg	    reply.first_event = extensions[i]->eventBase;
32705b261ecSmrg	    reply.first_error = extensions[i]->errorBase;
32805b261ecSmrg	}
32905b261ecSmrg    }
33005b261ecSmrg    WriteReplyToClient(client, sizeof(xQueryExtensionReply), &reply);
33105b261ecSmrg    return(client->noClientException);
33205b261ecSmrg}
33305b261ecSmrg
33405b261ecSmrgint
33505b261ecSmrgProcListExtensions(ClientPtr client)
33605b261ecSmrg{
33705b261ecSmrg    xListExtensionsReply reply;
33805b261ecSmrg    char *bufptr, *buffer;
33905b261ecSmrg    int total_length = 0;
34005b261ecSmrg
34105b261ecSmrg    REQUEST_SIZE_MATCH(xReq);
34205b261ecSmrg
34305b261ecSmrg    reply.type = X_Reply;
34405b261ecSmrg    reply.nExtensions = 0;
34505b261ecSmrg    reply.length = 0;
34605b261ecSmrg    reply.sequenceNumber = client->sequence;
34705b261ecSmrg    buffer = NULL;
34805b261ecSmrg
34905b261ecSmrg    if ( NumExtensions )
35005b261ecSmrg    {
35105b261ecSmrg        int i, j;
35205b261ecSmrg
35305b261ecSmrg        for (i=0;  i<NumExtensions; i++)
35405b261ecSmrg	{
35505b261ecSmrg	    /* call callbacks to find out whether to show extension */
35605b261ecSmrg	    if (!XaceHook(XACE_EXT_ACCESS, client, extensions[i]))
35705b261ecSmrg		continue;
35805b261ecSmrg
35905b261ecSmrg	    total_length += strlen(extensions[i]->name) + 1;
36005b261ecSmrg	    reply.nExtensions += 1 + extensions[i]->num_aliases;
36105b261ecSmrg	    for (j = extensions[i]->num_aliases; --j >= 0;)
36205b261ecSmrg		total_length += strlen(extensions[i]->aliases[j]) + 1;
36305b261ecSmrg	}
36405b261ecSmrg        reply.length = (total_length + 3) >> 2;
36505b261ecSmrg	buffer = bufptr = (char *)ALLOCATE_LOCAL(total_length);
36605b261ecSmrg	if (!buffer)
36705b261ecSmrg	    return(BadAlloc);
36805b261ecSmrg        for (i=0;  i<NumExtensions; i++)
36905b261ecSmrg        {
37005b261ecSmrg	    int len;
37105b261ecSmrg	    if (!XaceHook(XACE_EXT_ACCESS, client, extensions[i]))
37205b261ecSmrg		continue;
37305b261ecSmrg
37405b261ecSmrg            *bufptr++ = len = strlen(extensions[i]->name);
37505b261ecSmrg	    memmove(bufptr, extensions[i]->name,  len);
37605b261ecSmrg	    bufptr += len;
37705b261ecSmrg	    for (j = extensions[i]->num_aliases; --j >= 0;)
37805b261ecSmrg	    {
37905b261ecSmrg		*bufptr++ = len = strlen(extensions[i]->aliases[j]);
38005b261ecSmrg		memmove(bufptr, extensions[i]->aliases[j],  len);
38105b261ecSmrg		bufptr += len;
38205b261ecSmrg	    }
38305b261ecSmrg	}
38405b261ecSmrg    }
38505b261ecSmrg    WriteReplyToClient(client, sizeof(xListExtensionsReply), &reply);
38605b261ecSmrg    if (reply.length)
38705b261ecSmrg    {
38805b261ecSmrg        WriteToClient(client, total_length, buffer);
38905b261ecSmrg    	DEALLOCATE_LOCAL(buffer);
39005b261ecSmrg    }
39105b261ecSmrg    return(client->noClientException);
39205b261ecSmrg}
39305b261ecSmrg
39405b261ecSmrg#ifdef XSERVER_DTRACE
39505b261ecSmrgvoid LoadExtensionNames(char **RequestNames) {
39605b261ecSmrg    int i;
39705b261ecSmrg
39805b261ecSmrg    for (i=0; i<NumExtensions; i++) {
39905b261ecSmrg	int r = extensions[i]->base;
40005b261ecSmrg
40105b261ecSmrg	if (RequestNames[r] == NULL) {
40205b261ecSmrg	    RequestNames[r] = strdup(extensions[i]->name);
40305b261ecSmrg	}
40405b261ecSmrg    }
40505b261ecSmrg}
40605b261ecSmrg#endif
407