extension.c revision 9ace9065
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 } 999ace9065Smrg ext->name = strdup(name); 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 i = NumExtensions; 1096747b715Smrg newexts = (ExtensionEntry **) realloc(extensions, 11005b261ecSmrg (i + 1) * sizeof(ExtensionEntry *)); 11105b261ecSmrg if (!newexts) 11205b261ecSmrg { 1136747b715Smrg free(ext->name); 1146747b715Smrg dixFreePrivates(ext->devPrivates, PRIVATE_EXTENSION); 1156747b715Smrg free(ext); 11605b261ecSmrg return((ExtensionEntry *) NULL); 11705b261ecSmrg } 11805b261ecSmrg NumExtensions++; 11905b261ecSmrg extensions = newexts; 12005b261ecSmrg extensions[i] = ext; 12105b261ecSmrg ext->index = i; 12205b261ecSmrg ext->base = i + EXTENSION_BASE; 12305b261ecSmrg ext->CloseDown = CloseDownProc; 12405b261ecSmrg ext->MinorOpcode = MinorOpcodeProc; 12505b261ecSmrg ProcVector[i + EXTENSION_BASE] = MainProc; 12605b261ecSmrg SwappedProcVector[i + EXTENSION_BASE] = SwappedMainProc; 12705b261ecSmrg if (NumEvents) 12805b261ecSmrg { 12905b261ecSmrg ext->eventBase = lastEvent; 13005b261ecSmrg ext->eventLast = lastEvent + NumEvents; 13105b261ecSmrg lastEvent += NumEvents; 13205b261ecSmrg } 13305b261ecSmrg else 13405b261ecSmrg { 13505b261ecSmrg ext->eventBase = 0; 13605b261ecSmrg ext->eventLast = 0; 13705b261ecSmrg } 13805b261ecSmrg if (NumErrors) 13905b261ecSmrg { 14005b261ecSmrg ext->errorBase = lastError; 14105b261ecSmrg ext->errorLast = lastError + NumErrors; 14205b261ecSmrg lastError += NumErrors; 14305b261ecSmrg } 14405b261ecSmrg else 14505b261ecSmrg { 14605b261ecSmrg ext->errorBase = 0; 14705b261ecSmrg ext->errorLast = 0; 14805b261ecSmrg } 14905b261ecSmrg 1504642e01fSmrg RegisterExtensionNames(ext); 1516747b715Smrg return ext; 15205b261ecSmrg} 15305b261ecSmrg 1546747b715SmrgBool AddExtensionAlias(char *alias, ExtensionEntry *ext) 15505b261ecSmrg{ 15605b261ecSmrg char *name; 15705b261ecSmrg char **aliases; 15805b261ecSmrg 1594642e01fSmrg if (!ext) 1604642e01fSmrg return FALSE ; 1616747b715Smrg aliases = (char **)realloc(ext->aliases, 16205b261ecSmrg (ext->num_aliases + 1) * sizeof(char *)); 16305b261ecSmrg if (!aliases) 16405b261ecSmrg return FALSE; 16505b261ecSmrg ext->aliases = aliases; 1669ace9065Smrg name = strdup(alias); 16705b261ecSmrg if (!name) 16805b261ecSmrg return FALSE; 16905b261ecSmrg ext->aliases[ext->num_aliases] = name; 17005b261ecSmrg ext->num_aliases++; 17105b261ecSmrg return TRUE; 17205b261ecSmrg} 17305b261ecSmrg 17405b261ecSmrgstatic int 17505b261ecSmrgFindExtension(char *extname, int len) 17605b261ecSmrg{ 17705b261ecSmrg int i, j; 17805b261ecSmrg 17905b261ecSmrg for (i=0; i<NumExtensions; i++) 18005b261ecSmrg { 18105b261ecSmrg if ((strlen(extensions[i]->name) == len) && 18205b261ecSmrg !strncmp(extname, extensions[i]->name, len)) 18305b261ecSmrg break; 18405b261ecSmrg for (j = extensions[i]->num_aliases; --j >= 0;) 18505b261ecSmrg { 18605b261ecSmrg if ((strlen(extensions[i]->aliases[j]) == len) && 18705b261ecSmrg !strncmp(extname, extensions[i]->aliases[j], len)) 18805b261ecSmrg break; 18905b261ecSmrg } 19005b261ecSmrg if (j >= 0) break; 19105b261ecSmrg } 19205b261ecSmrg return ((i == NumExtensions) ? -1 : i); 19305b261ecSmrg} 19405b261ecSmrg 19505b261ecSmrg/* 19605b261ecSmrg * CheckExtension returns the extensions[] entry for the requested 19705b261ecSmrg * extension name. Maybe this could just return a Bool instead? 19805b261ecSmrg */ 1996747b715SmrgExtensionEntry * 20005b261ecSmrgCheckExtension(const char *extname) 20105b261ecSmrg{ 20205b261ecSmrg int n; 20305b261ecSmrg 20405b261ecSmrg n = FindExtension((char*)extname, strlen(extname)); 20505b261ecSmrg if (n != -1) 20605b261ecSmrg return extensions[n]; 20705b261ecSmrg else 20805b261ecSmrg return NULL; 20905b261ecSmrg} 21005b261ecSmrg 21105b261ecSmrg/* 21205b261ecSmrg * Added as part of Xace. 21305b261ecSmrg */ 21405b261ecSmrgExtensionEntry * 21505b261ecSmrgGetExtensionEntry(int major) 21605b261ecSmrg{ 21705b261ecSmrg if (major < EXTENSION_BASE) 21805b261ecSmrg return NULL; 21905b261ecSmrg major -= EXTENSION_BASE; 22005b261ecSmrg if (major >= NumExtensions) 22105b261ecSmrg return NULL; 22205b261ecSmrg return extensions[major]; 22305b261ecSmrg} 22405b261ecSmrg 2256747b715Smrgunsigned short 22605b261ecSmrgStandardMinorOpcode(ClientPtr client) 22705b261ecSmrg{ 22805b261ecSmrg return ((xReq *)client->requestBuffer)->data; 22905b261ecSmrg} 23005b261ecSmrg 2316747b715Smrgunsigned short 23205b261ecSmrgMinorOpcodeOfRequest(ClientPtr client) 23305b261ecSmrg{ 23405b261ecSmrg unsigned char major; 23505b261ecSmrg 23605b261ecSmrg major = ((xReq *)client->requestBuffer)->reqType; 23705b261ecSmrg if (major < EXTENSION_BASE) 23805b261ecSmrg return 0; 23905b261ecSmrg major -= EXTENSION_BASE; 24005b261ecSmrg if (major >= NumExtensions) 24105b261ecSmrg return 0; 24205b261ecSmrg return (*extensions[major]->MinorOpcode)(client); 24305b261ecSmrg} 24405b261ecSmrg 24505b261ecSmrgvoid 24605b261ecSmrgCloseDownExtensions(void) 24705b261ecSmrg{ 24805b261ecSmrg int i,j; 24905b261ecSmrg 25005b261ecSmrg for (i = NumExtensions - 1; i >= 0; i--) 25105b261ecSmrg { 2524642e01fSmrg if (extensions[i]->CloseDown) 2534642e01fSmrg extensions[i]->CloseDown(extensions[i]); 25405b261ecSmrg NumExtensions = i; 2556747b715Smrg free(extensions[i]->name); 25605b261ecSmrg for (j = extensions[i]->num_aliases; --j >= 0;) 2576747b715Smrg free(extensions[i]->aliases[j]); 2586747b715Smrg free(extensions[i]->aliases); 2596747b715Smrg dixFreePrivates(extensions[i]->devPrivates, PRIVATE_EXTENSION); 2606747b715Smrg free(extensions[i]); 26105b261ecSmrg } 2626747b715Smrg free(extensions); 26305b261ecSmrg extensions = (ExtensionEntry **)NULL; 26405b261ecSmrg lastEvent = EXTENSION_EVENT_BASE; 26505b261ecSmrg lastError = FirstExtensionError; 26605b261ecSmrg} 26705b261ecSmrg 26805b261ecSmrgint 26905b261ecSmrgProcQueryExtension(ClientPtr client) 27005b261ecSmrg{ 27105b261ecSmrg xQueryExtensionReply reply; 27205b261ecSmrg int i; 27305b261ecSmrg REQUEST(xQueryExtensionReq); 27405b261ecSmrg 27505b261ecSmrg REQUEST_FIXED_SIZE(xQueryExtensionReq, stuff->nbytes); 2766747b715Smrg 2776747b715Smrg memset(&reply, 0, sizeof(xQueryExtensionReply)); 27805b261ecSmrg reply.type = X_Reply; 27905b261ecSmrg reply.length = 0; 28005b261ecSmrg reply.major_opcode = 0; 28105b261ecSmrg reply.sequenceNumber = client->sequence; 28205b261ecSmrg 28305b261ecSmrg if ( ! NumExtensions ) 28405b261ecSmrg reply.present = xFalse; 28505b261ecSmrg else 28605b261ecSmrg { 28705b261ecSmrg i = FindExtension((char *)&stuff[1], stuff->nbytes); 2884642e01fSmrg if (i < 0 || XaceHook(XACE_EXT_ACCESS, client, extensions[i])) 28905b261ecSmrg reply.present = xFalse; 29005b261ecSmrg else 29105b261ecSmrg { 29205b261ecSmrg reply.present = xTrue; 29305b261ecSmrg reply.major_opcode = extensions[i]->base; 29405b261ecSmrg reply.first_event = extensions[i]->eventBase; 29505b261ecSmrg reply.first_error = extensions[i]->errorBase; 29605b261ecSmrg } 29705b261ecSmrg } 29805b261ecSmrg WriteReplyToClient(client, sizeof(xQueryExtensionReply), &reply); 2996747b715Smrg return Success; 30005b261ecSmrg} 30105b261ecSmrg 30205b261ecSmrgint 30305b261ecSmrgProcListExtensions(ClientPtr client) 30405b261ecSmrg{ 30505b261ecSmrg xListExtensionsReply reply; 30605b261ecSmrg char *bufptr, *buffer; 30705b261ecSmrg int total_length = 0; 30805b261ecSmrg 30905b261ecSmrg REQUEST_SIZE_MATCH(xReq); 31005b261ecSmrg 3116747b715Smrg memset(&reply, 0, sizeof(xListExtensionsReply)); 31205b261ecSmrg reply.type = X_Reply; 31305b261ecSmrg reply.nExtensions = 0; 31405b261ecSmrg reply.length = 0; 31505b261ecSmrg reply.sequenceNumber = client->sequence; 31605b261ecSmrg buffer = NULL; 31705b261ecSmrg 31805b261ecSmrg if ( NumExtensions ) 31905b261ecSmrg { 32005b261ecSmrg int i, j; 32105b261ecSmrg 32205b261ecSmrg for (i=0; i<NumExtensions; i++) 32305b261ecSmrg { 32405b261ecSmrg /* call callbacks to find out whether to show extension */ 3254642e01fSmrg if (XaceHook(XACE_EXT_ACCESS, client, extensions[i]) != Success) 32605b261ecSmrg continue; 32705b261ecSmrg 32805b261ecSmrg total_length += strlen(extensions[i]->name) + 1; 32905b261ecSmrg reply.nExtensions += 1 + extensions[i]->num_aliases; 33005b261ecSmrg for (j = extensions[i]->num_aliases; --j >= 0;) 33105b261ecSmrg total_length += strlen(extensions[i]->aliases[j]) + 1; 33205b261ecSmrg } 3336747b715Smrg reply.length = bytes_to_int32(total_length); 3346747b715Smrg buffer = bufptr = malloc(total_length); 33505b261ecSmrg if (!buffer) 3366747b715Smrg return BadAlloc; 33705b261ecSmrg for (i=0; i<NumExtensions; i++) 33805b261ecSmrg { 33905b261ecSmrg int len; 3404642e01fSmrg if (XaceHook(XACE_EXT_ACCESS, client, extensions[i]) != Success) 34105b261ecSmrg continue; 34205b261ecSmrg 34305b261ecSmrg *bufptr++ = len = strlen(extensions[i]->name); 34405b261ecSmrg memmove(bufptr, extensions[i]->name, len); 34505b261ecSmrg bufptr += len; 34605b261ecSmrg for (j = extensions[i]->num_aliases; --j >= 0;) 34705b261ecSmrg { 34805b261ecSmrg *bufptr++ = len = strlen(extensions[i]->aliases[j]); 34905b261ecSmrg memmove(bufptr, extensions[i]->aliases[j], len); 35005b261ecSmrg bufptr += len; 35105b261ecSmrg } 35205b261ecSmrg } 35305b261ecSmrg } 35405b261ecSmrg WriteReplyToClient(client, sizeof(xListExtensionsReply), &reply); 35505b261ecSmrg if (reply.length) 35605b261ecSmrg WriteToClient(client, total_length, buffer); 3579ace9065Smrg 3589ace9065Smrg free(buffer); 3596747b715Smrg return Success; 36005b261ecSmrg} 361