extension.c revision 6747b715
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#include <X11/Xproto.h>
5405b261ecSmrg#include "misc.h"
5505b261ecSmrg#include "dixstruct.h"
5605b261ecSmrg#include "extnsionst.h"
5705b261ecSmrg#include "gcstruct.h"
5805b261ecSmrg#include "scrnintstr.h"
5905b261ecSmrg#include "dispatch.h"
604642e01fSmrg#include "privates.h"
614642e01fSmrg#include "registry.h"
6205b261ecSmrg#include "xace.h"
6305b261ecSmrg
6405b261ecSmrg#define LAST_EVENT  128
6505b261ecSmrg#define LAST_ERROR 255
6605b261ecSmrg
6705b261ecSmrgstatic ExtensionEntry **extensions = (ExtensionEntry **)NULL;
6805b261ecSmrg
6905b261ecSmrgint lastEvent = EXTENSION_EVENT_BASE;
7005b261ecSmrgstatic int lastError = FirstExtensionError;
7105b261ecSmrgstatic unsigned int NumExtensions = 0;
7205b261ecSmrg
736747b715SmrgExtensionEntry *
7405b261ecSmrgAddExtension(char *name, int NumEvents, int NumErrors,
7505b261ecSmrg	     int (*MainProc)(ClientPtr c1),
7605b261ecSmrg	     int (*SwappedMainProc)(ClientPtr c2),
7705b261ecSmrg	     void (*CloseDownProc)(ExtensionEntry *e),
7805b261ecSmrg	     unsigned short (*MinorOpcodeProc)(ClientPtr c3))
7905b261ecSmrg{
8005b261ecSmrg    int i;
8105b261ecSmrg    ExtensionEntry *ext, **newexts;
8205b261ecSmrg
834642e01fSmrg    if (!MainProc || !SwappedMainProc || !MinorOpcodeProc)
8405b261ecSmrg        return((ExtensionEntry *) NULL);
8505b261ecSmrg    if ((lastEvent + NumEvents > LAST_EVENT) ||
866747b715Smrg	        (unsigned)(lastError + NumErrors > LAST_ERROR)) {
876747b715Smrg        LogMessage(X_ERROR, "Not enabling extension %s: maximum number of "
886747b715Smrg                   "events or errors exceeded.\n", name);
8905b261ecSmrg        return((ExtensionEntry *) NULL);
906747b715Smrg    }
9105b261ecSmrg
926747b715Smrg    ext = calloc(sizeof (ExtensionEntry), 1);
9305b261ecSmrg    if (!ext)
946747b715Smrg	return NULL;
956747b715Smrg    if (!dixAllocatePrivates(&ext->devPrivates, PRIVATE_EXTENSION)) {
966747b715Smrg	free(ext);
976747b715Smrg	return NULL;
986747b715Smrg    }
996747b715Smrg    ext->name = malloc(strlen(name) + 1);
10005b261ecSmrg    ext->num_aliases = 0;
10105b261ecSmrg    ext->aliases = (char **)NULL;
10205b261ecSmrg    if (!ext->name)
10305b261ecSmrg    {
1046747b715Smrg	dixFreePrivates(ext->devPrivates, PRIVATE_EXTENSION);
1056747b715Smrg	free(ext);
10605b261ecSmrg	return((ExtensionEntry *) NULL);
10705b261ecSmrg    }
10805b261ecSmrg    strcpy(ext->name,  name);
10905b261ecSmrg    i = NumExtensions;
1106747b715Smrg    newexts = (ExtensionEntry **) realloc(extensions,
11105b261ecSmrg					   (i + 1) * sizeof(ExtensionEntry *));
11205b261ecSmrg    if (!newexts)
11305b261ecSmrg    {
1146747b715Smrg	free(ext->name);
1156747b715Smrg	dixFreePrivates(ext->devPrivates, PRIVATE_EXTENSION);
1166747b715Smrg	free(ext);
11705b261ecSmrg	return((ExtensionEntry *) NULL);
11805b261ecSmrg    }
11905b261ecSmrg    NumExtensions++;
12005b261ecSmrg    extensions = newexts;
12105b261ecSmrg    extensions[i] = ext;
12205b261ecSmrg    ext->index = i;
12305b261ecSmrg    ext->base = i + EXTENSION_BASE;
12405b261ecSmrg    ext->CloseDown = CloseDownProc;
12505b261ecSmrg    ext->MinorOpcode = MinorOpcodeProc;
12605b261ecSmrg    ProcVector[i + EXTENSION_BASE] = MainProc;
12705b261ecSmrg    SwappedProcVector[i + EXTENSION_BASE] = SwappedMainProc;
12805b261ecSmrg    if (NumEvents)
12905b261ecSmrg    {
13005b261ecSmrg        ext->eventBase = lastEvent;
13105b261ecSmrg	ext->eventLast = lastEvent + NumEvents;
13205b261ecSmrg	lastEvent += NumEvents;
13305b261ecSmrg    }
13405b261ecSmrg    else
13505b261ecSmrg    {
13605b261ecSmrg        ext->eventBase = 0;
13705b261ecSmrg        ext->eventLast = 0;
13805b261ecSmrg    }
13905b261ecSmrg    if (NumErrors)
14005b261ecSmrg    {
14105b261ecSmrg        ext->errorBase = lastError;
14205b261ecSmrg	ext->errorLast = lastError + NumErrors;
14305b261ecSmrg	lastError += NumErrors;
14405b261ecSmrg    }
14505b261ecSmrg    else
14605b261ecSmrg    {
14705b261ecSmrg        ext->errorBase = 0;
14805b261ecSmrg        ext->errorLast = 0;
14905b261ecSmrg    }
15005b261ecSmrg
1514642e01fSmrg    RegisterExtensionNames(ext);
1526747b715Smrg    return ext;
15305b261ecSmrg}
15405b261ecSmrg
1556747b715SmrgBool AddExtensionAlias(char *alias, ExtensionEntry *ext)
15605b261ecSmrg{
15705b261ecSmrg    char *name;
15805b261ecSmrg    char **aliases;
15905b261ecSmrg
1604642e01fSmrg    if (!ext)
1614642e01fSmrg        return FALSE ;
1626747b715Smrg    aliases = (char **)realloc(ext->aliases,
16305b261ecSmrg				(ext->num_aliases + 1) * sizeof(char *));
16405b261ecSmrg    if (!aliases)
16505b261ecSmrg	return FALSE;
16605b261ecSmrg    ext->aliases = aliases;
1676747b715Smrg    name = malloc(strlen(alias) + 1);
16805b261ecSmrg    if (!name)
16905b261ecSmrg	return FALSE;
17005b261ecSmrg    strcpy(name,  alias);
17105b261ecSmrg    ext->aliases[ext->num_aliases] = name;
17205b261ecSmrg    ext->num_aliases++;
17305b261ecSmrg    return TRUE;
17405b261ecSmrg}
17505b261ecSmrg
17605b261ecSmrgstatic int
17705b261ecSmrgFindExtension(char *extname, int len)
17805b261ecSmrg{
17905b261ecSmrg    int i, j;
18005b261ecSmrg
18105b261ecSmrg    for (i=0; i<NumExtensions; i++)
18205b261ecSmrg    {
18305b261ecSmrg	if ((strlen(extensions[i]->name) == len) &&
18405b261ecSmrg	    !strncmp(extname, extensions[i]->name, len))
18505b261ecSmrg	    break;
18605b261ecSmrg	for (j = extensions[i]->num_aliases; --j >= 0;)
18705b261ecSmrg	{
18805b261ecSmrg	    if ((strlen(extensions[i]->aliases[j]) == len) &&
18905b261ecSmrg		!strncmp(extname, extensions[i]->aliases[j], len))
19005b261ecSmrg		break;
19105b261ecSmrg	}
19205b261ecSmrg	if (j >= 0) break;
19305b261ecSmrg    }
19405b261ecSmrg    return ((i == NumExtensions) ? -1 : i);
19505b261ecSmrg}
19605b261ecSmrg
19705b261ecSmrg/*
19805b261ecSmrg * CheckExtension returns the extensions[] entry for the requested
19905b261ecSmrg * extension name.  Maybe this could just return a Bool instead?
20005b261ecSmrg */
2016747b715SmrgExtensionEntry *
20205b261ecSmrgCheckExtension(const char *extname)
20305b261ecSmrg{
20405b261ecSmrg    int n;
20505b261ecSmrg
20605b261ecSmrg    n = FindExtension((char*)extname, strlen(extname));
20705b261ecSmrg    if (n != -1)
20805b261ecSmrg	return extensions[n];
20905b261ecSmrg    else
21005b261ecSmrg	return NULL;
21105b261ecSmrg}
21205b261ecSmrg
21305b261ecSmrg/*
21405b261ecSmrg * Added as part of Xace.
21505b261ecSmrg */
21605b261ecSmrgExtensionEntry *
21705b261ecSmrgGetExtensionEntry(int major)
21805b261ecSmrg{
21905b261ecSmrg    if (major < EXTENSION_BASE)
22005b261ecSmrg	return NULL;
22105b261ecSmrg    major -= EXTENSION_BASE;
22205b261ecSmrg    if (major >= NumExtensions)
22305b261ecSmrg	return NULL;
22405b261ecSmrg    return extensions[major];
22505b261ecSmrg}
22605b261ecSmrg
2276747b715Smrgunsigned short
22805b261ecSmrgStandardMinorOpcode(ClientPtr client)
22905b261ecSmrg{
23005b261ecSmrg    return ((xReq *)client->requestBuffer)->data;
23105b261ecSmrg}
23205b261ecSmrg
2336747b715Smrgunsigned short
23405b261ecSmrgMinorOpcodeOfRequest(ClientPtr client)
23505b261ecSmrg{
23605b261ecSmrg    unsigned char major;
23705b261ecSmrg
23805b261ecSmrg    major = ((xReq *)client->requestBuffer)->reqType;
23905b261ecSmrg    if (major < EXTENSION_BASE)
24005b261ecSmrg	return 0;
24105b261ecSmrg    major -= EXTENSION_BASE;
24205b261ecSmrg    if (major >= NumExtensions)
24305b261ecSmrg	return 0;
24405b261ecSmrg    return (*extensions[major]->MinorOpcode)(client);
24505b261ecSmrg}
24605b261ecSmrg
24705b261ecSmrgvoid
24805b261ecSmrgCloseDownExtensions(void)
24905b261ecSmrg{
25005b261ecSmrg    int i,j;
25105b261ecSmrg
25205b261ecSmrg    for (i = NumExtensions - 1; i >= 0; i--)
25305b261ecSmrg    {
2544642e01fSmrg	if (extensions[i]->CloseDown)
2554642e01fSmrg	    extensions[i]->CloseDown(extensions[i]);
25605b261ecSmrg	NumExtensions = i;
2576747b715Smrg	free(extensions[i]->name);
25805b261ecSmrg	for (j = extensions[i]->num_aliases; --j >= 0;)
2596747b715Smrg	    free(extensions[i]->aliases[j]);
2606747b715Smrg	free(extensions[i]->aliases);
2616747b715Smrg	dixFreePrivates(extensions[i]->devPrivates, PRIVATE_EXTENSION);
2626747b715Smrg	free(extensions[i]);
26305b261ecSmrg    }
2646747b715Smrg    free(extensions);
26505b261ecSmrg    extensions = (ExtensionEntry **)NULL;
26605b261ecSmrg    lastEvent = EXTENSION_EVENT_BASE;
26705b261ecSmrg    lastError = FirstExtensionError;
26805b261ecSmrg}
26905b261ecSmrg
27005b261ecSmrgint
27105b261ecSmrgProcQueryExtension(ClientPtr client)
27205b261ecSmrg{
27305b261ecSmrg    xQueryExtensionReply reply;
27405b261ecSmrg    int i;
27505b261ecSmrg    REQUEST(xQueryExtensionReq);
27605b261ecSmrg
27705b261ecSmrg    REQUEST_FIXED_SIZE(xQueryExtensionReq, stuff->nbytes);
2786747b715Smrg
2796747b715Smrg    memset(&reply, 0, sizeof(xQueryExtensionReply));
28005b261ecSmrg    reply.type = X_Reply;
28105b261ecSmrg    reply.length = 0;
28205b261ecSmrg    reply.major_opcode = 0;
28305b261ecSmrg    reply.sequenceNumber = client->sequence;
28405b261ecSmrg
28505b261ecSmrg    if ( ! NumExtensions )
28605b261ecSmrg        reply.present = xFalse;
28705b261ecSmrg    else
28805b261ecSmrg    {
28905b261ecSmrg	i = FindExtension((char *)&stuff[1], stuff->nbytes);
2904642e01fSmrg        if (i < 0 || XaceHook(XACE_EXT_ACCESS, client, extensions[i]))
29105b261ecSmrg            reply.present = xFalse;
29205b261ecSmrg        else
29305b261ecSmrg        {
29405b261ecSmrg            reply.present = xTrue;
29505b261ecSmrg	    reply.major_opcode = extensions[i]->base;
29605b261ecSmrg	    reply.first_event = extensions[i]->eventBase;
29705b261ecSmrg	    reply.first_error = extensions[i]->errorBase;
29805b261ecSmrg	}
29905b261ecSmrg    }
30005b261ecSmrg    WriteReplyToClient(client, sizeof(xQueryExtensionReply), &reply);
3016747b715Smrg    return Success;
30205b261ecSmrg}
30305b261ecSmrg
30405b261ecSmrgint
30505b261ecSmrgProcListExtensions(ClientPtr client)
30605b261ecSmrg{
30705b261ecSmrg    xListExtensionsReply reply;
30805b261ecSmrg    char *bufptr, *buffer;
30905b261ecSmrg    int total_length = 0;
31005b261ecSmrg
31105b261ecSmrg    REQUEST_SIZE_MATCH(xReq);
31205b261ecSmrg
3136747b715Smrg    memset(&reply, 0, sizeof(xListExtensionsReply));
31405b261ecSmrg    reply.type = X_Reply;
31505b261ecSmrg    reply.nExtensions = 0;
31605b261ecSmrg    reply.length = 0;
31705b261ecSmrg    reply.sequenceNumber = client->sequence;
31805b261ecSmrg    buffer = NULL;
31905b261ecSmrg
32005b261ecSmrg    if ( NumExtensions )
32105b261ecSmrg    {
32205b261ecSmrg        int i, j;
32305b261ecSmrg
32405b261ecSmrg        for (i=0;  i<NumExtensions; i++)
32505b261ecSmrg	{
32605b261ecSmrg	    /* call callbacks to find out whether to show extension */
3274642e01fSmrg	    if (XaceHook(XACE_EXT_ACCESS, client, extensions[i]) != Success)
32805b261ecSmrg		continue;
32905b261ecSmrg
33005b261ecSmrg	    total_length += strlen(extensions[i]->name) + 1;
33105b261ecSmrg	    reply.nExtensions += 1 + extensions[i]->num_aliases;
33205b261ecSmrg	    for (j = extensions[i]->num_aliases; --j >= 0;)
33305b261ecSmrg		total_length += strlen(extensions[i]->aliases[j]) + 1;
33405b261ecSmrg	}
3356747b715Smrg        reply.length = bytes_to_int32(total_length);
3366747b715Smrg	buffer = bufptr = malloc(total_length);
33705b261ecSmrg	if (!buffer)
3386747b715Smrg	    return BadAlloc;
33905b261ecSmrg        for (i=0;  i<NumExtensions; i++)
34005b261ecSmrg        {
34105b261ecSmrg	    int len;
3424642e01fSmrg	    if (XaceHook(XACE_EXT_ACCESS, client, extensions[i]) != Success)
34305b261ecSmrg		continue;
34405b261ecSmrg
34505b261ecSmrg            *bufptr++ = len = strlen(extensions[i]->name);
34605b261ecSmrg	    memmove(bufptr, extensions[i]->name,  len);
34705b261ecSmrg	    bufptr += len;
34805b261ecSmrg	    for (j = extensions[i]->num_aliases; --j >= 0;)
34905b261ecSmrg	    {
35005b261ecSmrg		*bufptr++ = len = strlen(extensions[i]->aliases[j]);
35105b261ecSmrg		memmove(bufptr, extensions[i]->aliases[j],  len);
35205b261ecSmrg		bufptr += len;
35305b261ecSmrg	    }
35405b261ecSmrg	}
35505b261ecSmrg    }
35605b261ecSmrg    WriteReplyToClient(client, sizeof(xListExtensionsReply), &reply);
35705b261ecSmrg    if (reply.length)
35805b261ecSmrg    {
35905b261ecSmrg        WriteToClient(client, total_length, buffer);
3606747b715Smrg        free(buffer);
36105b261ecSmrg    }
3626747b715Smrg    return Success;
36305b261ecSmrg}
364