extension.c revision 4642e01f
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"
624642e01fSmrg#include "privates.h"
634642e01fSmrg#include "registry.h"
6405b261ecSmrg#include "xace.h"
6505b261ecSmrg
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
7505b261ecSmrg_X_EXPORT ExtensionEntry *
7605b261ecSmrgAddExtension(char *name, int NumEvents, int NumErrors,
7705b261ecSmrg	     int (*MainProc)(ClientPtr c1),
7805b261ecSmrg	     int (*SwappedMainProc)(ClientPtr c2),
7905b261ecSmrg	     void (*CloseDownProc)(ExtensionEntry *e),
8005b261ecSmrg	     unsigned short (*MinorOpcodeProc)(ClientPtr c3))
8105b261ecSmrg{
8205b261ecSmrg    int i;
8305b261ecSmrg    ExtensionEntry *ext, **newexts;
8405b261ecSmrg
854642e01fSmrg    if (!MainProc || !SwappedMainProc || !MinorOpcodeProc)
8605b261ecSmrg        return((ExtensionEntry *) NULL);
8705b261ecSmrg    if ((lastEvent + NumEvents > LAST_EVENT) ||
8805b261ecSmrg	        (unsigned)(lastError + NumErrors > LAST_ERROR))
8905b261ecSmrg        return((ExtensionEntry *) NULL);
9005b261ecSmrg
914642e01fSmrg    ext = (ExtensionEntry *) xalloc(sizeof(ExtensionEntry));
9205b261ecSmrg    if (!ext)
9305b261ecSmrg	return((ExtensionEntry *) NULL);
9405b261ecSmrg    ext->name = (char *)xalloc(strlen(name) + 1);
9505b261ecSmrg    ext->num_aliases = 0;
9605b261ecSmrg    ext->aliases = (char **)NULL;
974642e01fSmrg    ext->devPrivates = NULL;
9805b261ecSmrg    if (!ext->name)
9905b261ecSmrg    {
10005b261ecSmrg	xfree(ext);
10105b261ecSmrg	return((ExtensionEntry *) NULL);
10205b261ecSmrg    }
10305b261ecSmrg    strcpy(ext->name,  name);
10405b261ecSmrg    i = NumExtensions;
10505b261ecSmrg    newexts = (ExtensionEntry **) xrealloc(extensions,
10605b261ecSmrg					   (i + 1) * sizeof(ExtensionEntry *));
10705b261ecSmrg    if (!newexts)
10805b261ecSmrg    {
10905b261ecSmrg	xfree(ext->name);
11005b261ecSmrg	xfree(ext);
11105b261ecSmrg	return((ExtensionEntry *) NULL);
11205b261ecSmrg    }
11305b261ecSmrg    NumExtensions++;
11405b261ecSmrg    extensions = newexts;
11505b261ecSmrg    extensions[i] = ext;
11605b261ecSmrg    ext->index = i;
11705b261ecSmrg    ext->base = i + EXTENSION_BASE;
11805b261ecSmrg    ext->CloseDown = CloseDownProc;
11905b261ecSmrg    ext->MinorOpcode = MinorOpcodeProc;
12005b261ecSmrg    ProcVector[i + EXTENSION_BASE] = MainProc;
12105b261ecSmrg    SwappedProcVector[i + EXTENSION_BASE] = SwappedMainProc;
12205b261ecSmrg    if (NumEvents)
12305b261ecSmrg    {
12405b261ecSmrg        ext->eventBase = lastEvent;
12505b261ecSmrg	ext->eventLast = lastEvent + NumEvents;
12605b261ecSmrg	lastEvent += NumEvents;
12705b261ecSmrg    }
12805b261ecSmrg    else
12905b261ecSmrg    {
13005b261ecSmrg        ext->eventBase = 0;
13105b261ecSmrg        ext->eventLast = 0;
13205b261ecSmrg    }
13305b261ecSmrg    if (NumErrors)
13405b261ecSmrg    {
13505b261ecSmrg        ext->errorBase = lastError;
13605b261ecSmrg	ext->errorLast = lastError + NumErrors;
13705b261ecSmrg	lastError += NumErrors;
13805b261ecSmrg    }
13905b261ecSmrg    else
14005b261ecSmrg    {
14105b261ecSmrg        ext->errorBase = 0;
14205b261ecSmrg        ext->errorLast = 0;
14305b261ecSmrg    }
14405b261ecSmrg
1454642e01fSmrg    RegisterExtensionNames(ext);
14605b261ecSmrg    return(ext);
14705b261ecSmrg}
14805b261ecSmrg
14905b261ecSmrg_X_EXPORT Bool AddExtensionAlias(char *alias, ExtensionEntry *ext)
15005b261ecSmrg{
15105b261ecSmrg    char *name;
15205b261ecSmrg    char **aliases;
15305b261ecSmrg
1544642e01fSmrg    if (!ext)
1554642e01fSmrg        return FALSE ;
15605b261ecSmrg    aliases = (char **)xrealloc(ext->aliases,
15705b261ecSmrg				(ext->num_aliases + 1) * sizeof(char *));
15805b261ecSmrg    if (!aliases)
15905b261ecSmrg	return FALSE;
16005b261ecSmrg    ext->aliases = aliases;
16105b261ecSmrg    name = (char *)xalloc(strlen(alias) + 1);
16205b261ecSmrg    if (!name)
16305b261ecSmrg	return FALSE;
16405b261ecSmrg    strcpy(name,  alias);
16505b261ecSmrg    ext->aliases[ext->num_aliases] = name;
16605b261ecSmrg    ext->num_aliases++;
16705b261ecSmrg    return TRUE;
16805b261ecSmrg}
16905b261ecSmrg
17005b261ecSmrgstatic int
17105b261ecSmrgFindExtension(char *extname, int len)
17205b261ecSmrg{
17305b261ecSmrg    int i, j;
17405b261ecSmrg
17505b261ecSmrg    for (i=0; i<NumExtensions; i++)
17605b261ecSmrg    {
17705b261ecSmrg	if ((strlen(extensions[i]->name) == len) &&
17805b261ecSmrg	    !strncmp(extname, extensions[i]->name, len))
17905b261ecSmrg	    break;
18005b261ecSmrg	for (j = extensions[i]->num_aliases; --j >= 0;)
18105b261ecSmrg	{
18205b261ecSmrg	    if ((strlen(extensions[i]->aliases[j]) == len) &&
18305b261ecSmrg		!strncmp(extname, extensions[i]->aliases[j], len))
18405b261ecSmrg		break;
18505b261ecSmrg	}
18605b261ecSmrg	if (j >= 0) break;
18705b261ecSmrg    }
18805b261ecSmrg    return ((i == NumExtensions) ? -1 : i);
18905b261ecSmrg}
19005b261ecSmrg
19105b261ecSmrg/*
19205b261ecSmrg * CheckExtension returns the extensions[] entry for the requested
19305b261ecSmrg * extension name.  Maybe this could just return a Bool instead?
19405b261ecSmrg */
19505b261ecSmrg_X_EXPORT ExtensionEntry *
19605b261ecSmrgCheckExtension(const char *extname)
19705b261ecSmrg{
19805b261ecSmrg    int n;
19905b261ecSmrg
20005b261ecSmrg    n = FindExtension((char*)extname, strlen(extname));
20105b261ecSmrg    if (n != -1)
20205b261ecSmrg	return extensions[n];
20305b261ecSmrg    else
20405b261ecSmrg	return NULL;
20505b261ecSmrg}
20605b261ecSmrg
20705b261ecSmrg/*
20805b261ecSmrg * Added as part of Xace.
20905b261ecSmrg */
21005b261ecSmrgExtensionEntry *
21105b261ecSmrgGetExtensionEntry(int major)
21205b261ecSmrg{
21305b261ecSmrg    if (major < EXTENSION_BASE)
21405b261ecSmrg	return NULL;
21505b261ecSmrg    major -= EXTENSION_BASE;
21605b261ecSmrg    if (major >= NumExtensions)
21705b261ecSmrg	return NULL;
21805b261ecSmrg    return extensions[major];
21905b261ecSmrg}
22005b261ecSmrg
22105b261ecSmrg_X_EXPORT unsigned short
22205b261ecSmrgStandardMinorOpcode(ClientPtr client)
22305b261ecSmrg{
22405b261ecSmrg    return ((xReq *)client->requestBuffer)->data;
22505b261ecSmrg}
22605b261ecSmrg
22705b261ecSmrg_X_EXPORT unsigned short
22805b261ecSmrgMinorOpcodeOfRequest(ClientPtr client)
22905b261ecSmrg{
23005b261ecSmrg    unsigned char major;
23105b261ecSmrg
23205b261ecSmrg    major = ((xReq *)client->requestBuffer)->reqType;
23305b261ecSmrg    if (major < EXTENSION_BASE)
23405b261ecSmrg	return 0;
23505b261ecSmrg    major -= EXTENSION_BASE;
23605b261ecSmrg    if (major >= NumExtensions)
23705b261ecSmrg	return 0;
23805b261ecSmrg    return (*extensions[major]->MinorOpcode)(client);
23905b261ecSmrg}
24005b261ecSmrg
24105b261ecSmrgvoid
24205b261ecSmrgCloseDownExtensions(void)
24305b261ecSmrg{
24405b261ecSmrg    int i,j;
24505b261ecSmrg
24605b261ecSmrg    for (i = NumExtensions - 1; i >= 0; i--)
24705b261ecSmrg    {
2484642e01fSmrg	if (extensions[i]->CloseDown)
2494642e01fSmrg	    extensions[i]->CloseDown(extensions[i]);
25005b261ecSmrg	NumExtensions = i;
25105b261ecSmrg	xfree(extensions[i]->name);
25205b261ecSmrg	for (j = extensions[i]->num_aliases; --j >= 0;)
25305b261ecSmrg	    xfree(extensions[i]->aliases[j]);
25405b261ecSmrg	xfree(extensions[i]->aliases);
2554642e01fSmrg	dixFreePrivates(extensions[i]->devPrivates);
25605b261ecSmrg	xfree(extensions[i]);
25705b261ecSmrg    }
25805b261ecSmrg    xfree(extensions);
25905b261ecSmrg    extensions = (ExtensionEntry **)NULL;
26005b261ecSmrg    lastEvent = EXTENSION_EVENT_BASE;
26105b261ecSmrg    lastError = FirstExtensionError;
26205b261ecSmrg}
26305b261ecSmrg
26405b261ecSmrgint
26505b261ecSmrgProcQueryExtension(ClientPtr client)
26605b261ecSmrg{
26705b261ecSmrg    xQueryExtensionReply reply;
26805b261ecSmrg    int i;
26905b261ecSmrg    REQUEST(xQueryExtensionReq);
27005b261ecSmrg
27105b261ecSmrg    REQUEST_FIXED_SIZE(xQueryExtensionReq, stuff->nbytes);
27205b261ecSmrg
27305b261ecSmrg    reply.type = X_Reply;
27405b261ecSmrg    reply.length = 0;
27505b261ecSmrg    reply.major_opcode = 0;
27605b261ecSmrg    reply.sequenceNumber = client->sequence;
27705b261ecSmrg
27805b261ecSmrg    if ( ! NumExtensions )
27905b261ecSmrg        reply.present = xFalse;
28005b261ecSmrg    else
28105b261ecSmrg    {
28205b261ecSmrg	i = FindExtension((char *)&stuff[1], stuff->nbytes);
2834642e01fSmrg        if (i < 0 || XaceHook(XACE_EXT_ACCESS, client, extensions[i]))
28405b261ecSmrg            reply.present = xFalse;
28505b261ecSmrg        else
28605b261ecSmrg        {
28705b261ecSmrg            reply.present = xTrue;
28805b261ecSmrg	    reply.major_opcode = extensions[i]->base;
28905b261ecSmrg	    reply.first_event = extensions[i]->eventBase;
29005b261ecSmrg	    reply.first_error = extensions[i]->errorBase;
29105b261ecSmrg	}
29205b261ecSmrg    }
29305b261ecSmrg    WriteReplyToClient(client, sizeof(xQueryExtensionReply), &reply);
29405b261ecSmrg    return(client->noClientException);
29505b261ecSmrg}
29605b261ecSmrg
29705b261ecSmrgint
29805b261ecSmrgProcListExtensions(ClientPtr client)
29905b261ecSmrg{
30005b261ecSmrg    xListExtensionsReply reply;
30105b261ecSmrg    char *bufptr, *buffer;
30205b261ecSmrg    int total_length = 0;
30305b261ecSmrg
30405b261ecSmrg    REQUEST_SIZE_MATCH(xReq);
30505b261ecSmrg
30605b261ecSmrg    reply.type = X_Reply;
30705b261ecSmrg    reply.nExtensions = 0;
30805b261ecSmrg    reply.length = 0;
30905b261ecSmrg    reply.sequenceNumber = client->sequence;
31005b261ecSmrg    buffer = NULL;
31105b261ecSmrg
31205b261ecSmrg    if ( NumExtensions )
31305b261ecSmrg    {
31405b261ecSmrg        int i, j;
31505b261ecSmrg
31605b261ecSmrg        for (i=0;  i<NumExtensions; i++)
31705b261ecSmrg	{
31805b261ecSmrg	    /* call callbacks to find out whether to show extension */
3194642e01fSmrg	    if (XaceHook(XACE_EXT_ACCESS, client, extensions[i]) != Success)
32005b261ecSmrg		continue;
32105b261ecSmrg
32205b261ecSmrg	    total_length += strlen(extensions[i]->name) + 1;
32305b261ecSmrg	    reply.nExtensions += 1 + extensions[i]->num_aliases;
32405b261ecSmrg	    for (j = extensions[i]->num_aliases; --j >= 0;)
32505b261ecSmrg		total_length += strlen(extensions[i]->aliases[j]) + 1;
32605b261ecSmrg	}
32705b261ecSmrg        reply.length = (total_length + 3) >> 2;
3284642e01fSmrg	buffer = bufptr = (char *)xalloc(total_length);
32905b261ecSmrg	if (!buffer)
33005b261ecSmrg	    return(BadAlloc);
33105b261ecSmrg        for (i=0;  i<NumExtensions; i++)
33205b261ecSmrg        {
33305b261ecSmrg	    int len;
3344642e01fSmrg	    if (XaceHook(XACE_EXT_ACCESS, client, extensions[i]) != Success)
33505b261ecSmrg		continue;
33605b261ecSmrg
33705b261ecSmrg            *bufptr++ = len = strlen(extensions[i]->name);
33805b261ecSmrg	    memmove(bufptr, extensions[i]->name,  len);
33905b261ecSmrg	    bufptr += len;
34005b261ecSmrg	    for (j = extensions[i]->num_aliases; --j >= 0;)
34105b261ecSmrg	    {
34205b261ecSmrg		*bufptr++ = len = strlen(extensions[i]->aliases[j]);
34305b261ecSmrg		memmove(bufptr, extensions[i]->aliases[j],  len);
34405b261ecSmrg		bufptr += len;
34505b261ecSmrg	    }
34605b261ecSmrg	}
34705b261ecSmrg    }
34805b261ecSmrg    WriteReplyToClient(client, sizeof(xListExtensionsReply), &reply);
34905b261ecSmrg    if (reply.length)
35005b261ecSmrg    {
35105b261ecSmrg        WriteToClient(client, total_length, buffer);
3524642e01fSmrg    	xfree(buffer);
35305b261ecSmrg    }
35405b261ecSmrg    return(client->noClientException);
35505b261ecSmrg}
356