You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1091 lines
35 KiB
C
1091 lines
35 KiB
C
/*
|
|
|
|
Copyright 1996, 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.
|
|
|
|
*/
|
|
|
|
#ifdef HAVE_DIX_CONFIG_H
|
|
#include <dix-config.h>
|
|
#endif
|
|
|
|
#include "scrnintstr.h"
|
|
#include "inputstr.h"
|
|
#include "windowstr.h"
|
|
#include "propertyst.h"
|
|
#include "colormapst.h"
|
|
#include "privates.h"
|
|
#include "registry.h"
|
|
#include "xacestr.h"
|
|
#include "securitysrv.h"
|
|
#include <X11/extensions/securproto.h>
|
|
#include "extinit.h"
|
|
#include "protocol-versions.h"
|
|
|
|
/* Extension stuff */
|
|
static int SecurityErrorBase; /* first Security error number */
|
|
static int SecurityEventBase; /* first Security event number */
|
|
|
|
RESTYPE SecurityAuthorizationResType; /* resource type for authorizations */
|
|
static RESTYPE RTEventClient;
|
|
|
|
static CallbackListPtr SecurityValidateGroupCallback = NULL;
|
|
|
|
/* Private state record */
|
|
static DevPrivateKeyRec stateKeyRec;
|
|
|
|
#define stateKey (&stateKeyRec)
|
|
|
|
/* This is what we store as client security state */
|
|
typedef struct {
|
|
unsigned int haveState :1;
|
|
unsigned int live :1;
|
|
unsigned int trustLevel :2;
|
|
XID authId;
|
|
} SecurityStateRec;
|
|
|
|
/* The only extensions that untrusted clients have access to */
|
|
static const char *SecurityTrustedExtensions[] = {
|
|
"XC-MISC",
|
|
"BIG-REQUESTS",
|
|
NULL
|
|
};
|
|
|
|
/*
|
|
* Access modes that untrusted clients are allowed on trusted objects.
|
|
*/
|
|
static const Mask SecurityResourceMask =
|
|
DixGetAttrAccess | DixReceiveAccess | DixListPropAccess |
|
|
DixGetPropAccess | DixListAccess;
|
|
static const Mask SecurityWindowExtraMask = DixRemoveAccess;
|
|
static const Mask SecurityRootWindowExtraMask =
|
|
DixReceiveAccess | DixSendAccess | DixAddAccess | DixRemoveAccess;
|
|
static const Mask SecurityDeviceMask =
|
|
DixGetAttrAccess | DixReceiveAccess | DixGetFocusAccess |
|
|
DixGrabAccess | DixSetAttrAccess | DixUseAccess;
|
|
static const Mask SecurityServerMask = DixGetAttrAccess | DixGrabAccess;
|
|
static const Mask SecurityClientMask = DixGetAttrAccess;
|
|
|
|
/* SecurityAudit
|
|
*
|
|
* Arguments:
|
|
* format is the formatting string to be used to interpret the
|
|
* remaining arguments.
|
|
*
|
|
* Returns: nothing.
|
|
*
|
|
* Side Effects:
|
|
* Writes the message to the log file if security logging is on.
|
|
*/
|
|
|
|
static void
|
|
_X_ATTRIBUTE_PRINTF(1, 2)
|
|
SecurityAudit(const char *format, ...)
|
|
{
|
|
va_list args;
|
|
|
|
if (auditTrailLevel < SECURITY_AUDIT_LEVEL)
|
|
return;
|
|
va_start(args, format);
|
|
VAuditF(format, args);
|
|
va_end(args);
|
|
} /* SecurityAudit */
|
|
|
|
/*
|
|
* Performs a Security permission check.
|
|
*/
|
|
static int
|
|
SecurityDoCheck(SecurityStateRec * subj, SecurityStateRec * obj,
|
|
Mask requested, Mask allowed)
|
|
{
|
|
if (!subj->haveState || !obj->haveState)
|
|
return Success;
|
|
if (subj->trustLevel == XSecurityClientTrusted)
|
|
return Success;
|
|
if (obj->trustLevel != XSecurityClientTrusted)
|
|
return Success;
|
|
if ((requested | allowed) == allowed)
|
|
return Success;
|
|
|
|
return BadAccess;
|
|
}
|
|
|
|
/*
|
|
* Labels initial server objects.
|
|
*/
|
|
static void
|
|
SecurityLabelInitial(void)
|
|
{
|
|
SecurityStateRec *state;
|
|
|
|
/* Do the serverClient */
|
|
state = dixLookupPrivate(&serverClient->devPrivates, stateKey);
|
|
state->trustLevel = XSecurityClientTrusted;
|
|
state->haveState = TRUE;
|
|
state->live = FALSE;
|
|
}
|
|
|
|
/*
|
|
* Looks up a request name
|
|
*/
|
|
static _X_INLINE const char *
|
|
SecurityLookupRequestName(ClientPtr client)
|
|
{
|
|
return LookupRequestName(client->majorOp, client->minorOp);
|
|
}
|
|
|
|
/* SecurityDeleteAuthorization
|
|
*
|
|
* Arguments:
|
|
* value is the authorization to delete.
|
|
* id is its resource ID.
|
|
*
|
|
* Returns: Success.
|
|
*
|
|
* Side Effects:
|
|
* Frees everything associated with the authorization.
|
|
*/
|
|
|
|
static int
|
|
SecurityDeleteAuthorization(void *value, XID id)
|
|
{
|
|
SecurityAuthorizationPtr pAuth = (SecurityAuthorizationPtr) value;
|
|
unsigned short name_len, data_len;
|
|
const char *name;
|
|
char *data;
|
|
int status;
|
|
int i;
|
|
OtherClientsPtr pEventClient;
|
|
|
|
/* Remove the auth using the os layer auth manager */
|
|
|
|
status = AuthorizationFromID(pAuth->id, &name_len, &name, &data_len, &data);
|
|
assert(status);
|
|
status = RemoveAuthorization(name_len, name, data_len, data);
|
|
assert(status);
|
|
(void) status;
|
|
|
|
/* free the auth timer if there is one */
|
|
|
|
if (pAuth->timer)
|
|
TimerFree(pAuth->timer);
|
|
|
|
/* send revoke events */
|
|
|
|
while ((pEventClient = pAuth->eventClients)) {
|
|
/* send revocation event event */
|
|
xSecurityAuthorizationRevokedEvent are = {
|
|
.type = SecurityEventBase + XSecurityAuthorizationRevoked,
|
|
.authId = pAuth->id
|
|
};
|
|
WriteEventsToClient(rClient(pEventClient), 1, (xEvent *) &are);
|
|
FreeResource(pEventClient->resource, RT_NONE);
|
|
}
|
|
|
|
/* kill all clients using this auth */
|
|
|
|
for (i = 1; i < currentMaxClients; i++)
|
|
if (clients[i]) {
|
|
SecurityStateRec *state;
|
|
|
|
state = dixLookupPrivate(&clients[i]->devPrivates, stateKey);
|
|
if (state->haveState && state->authId == pAuth->id)
|
|
CloseDownClient(clients[i]);
|
|
}
|
|
|
|
SecurityAudit("revoked authorization ID %lu\n", (unsigned long)pAuth->id);
|
|
free(pAuth);
|
|
return Success;
|
|
|
|
} /* SecurityDeleteAuthorization */
|
|
|
|
/* resource delete function for RTEventClient */
|
|
static int
|
|
SecurityDeleteAuthorizationEventClient(void *value, XID id)
|
|
{
|
|
OtherClientsPtr pEventClient, prev = NULL;
|
|
SecurityAuthorizationPtr pAuth = (SecurityAuthorizationPtr) value;
|
|
|
|
for (pEventClient = pAuth->eventClients;
|
|
pEventClient; pEventClient = pEventClient->next) {
|
|
if (pEventClient->resource == id) {
|
|
if (prev)
|
|
prev->next = pEventClient->next;
|
|
else
|
|
pAuth->eventClients = pEventClient->next;
|
|
free(pEventClient);
|
|
return Success;
|
|
}
|
|
prev = pEventClient;
|
|
}
|
|
/*NOTREACHED*/ return -1; /* make compiler happy */
|
|
} /* SecurityDeleteAuthorizationEventClient */
|
|
|
|
/* SecurityComputeAuthorizationTimeout
|
|
*
|
|
* Arguments:
|
|
* pAuth is the authorization for which we are computing the timeout
|
|
* seconds is the number of seconds we want to wait
|
|
*
|
|
* Returns:
|
|
* the number of milliseconds that the auth timer should be set to
|
|
*
|
|
* Side Effects:
|
|
* Sets pAuth->secondsRemaining to any "overflow" amount of time
|
|
* that didn't fit in 32 bits worth of milliseconds
|
|
*/
|
|
|
|
static CARD32
|
|
SecurityComputeAuthorizationTimeout(SecurityAuthorizationPtr pAuth,
|
|
unsigned int seconds)
|
|
{
|
|
/* maxSecs is the number of full seconds that can be expressed in
|
|
* 32 bits worth of milliseconds
|
|
*/
|
|
CARD32 maxSecs = (CARD32) (~0) / (CARD32) MILLI_PER_SECOND;
|
|
|
|
if (seconds > maxSecs) { /* only come here if we want to wait more than 49 days */
|
|
pAuth->secondsRemaining = seconds - maxSecs;
|
|
return maxSecs * MILLI_PER_SECOND;
|
|
}
|
|
else { /* by far the common case */
|
|
pAuth->secondsRemaining = 0;
|
|
return seconds * MILLI_PER_SECOND;
|
|
}
|
|
} /* SecurityStartAuthorizationTimer */
|
|
|
|
/* SecurityAuthorizationExpired
|
|
*
|
|
* This function is passed as an argument to TimerSet and gets called from
|
|
* the timer manager in the os layer when its time is up.
|
|
*
|
|
* Arguments:
|
|
* timer is the timer for this authorization.
|
|
* time is the current time.
|
|
* pval is the authorization whose time is up.
|
|
*
|
|
* Returns:
|
|
* A new time delay in milliseconds if the timer should wait some
|
|
* more, else zero.
|
|
*
|
|
* Side Effects:
|
|
* Frees the authorization resource if the timeout period is really
|
|
* over, otherwise recomputes pAuth->secondsRemaining.
|
|
*/
|
|
|
|
static CARD32
|
|
SecurityAuthorizationExpired(OsTimerPtr timer, CARD32 time, void *pval)
|
|
{
|
|
SecurityAuthorizationPtr pAuth = (SecurityAuthorizationPtr) pval;
|
|
|
|
assert(pAuth->timer == timer);
|
|
|
|
if (pAuth->secondsRemaining) {
|
|
return SecurityComputeAuthorizationTimeout(pAuth,
|
|
pAuth->secondsRemaining);
|
|
}
|
|
else {
|
|
FreeResource(pAuth->id, RT_NONE);
|
|
return 0;
|
|
}
|
|
} /* SecurityAuthorizationExpired */
|
|
|
|
/* SecurityStartAuthorizationTimer
|
|
*
|
|
* Arguments:
|
|
* pAuth is the authorization whose timer should be started.
|
|
*
|
|
* Returns: nothing.
|
|
*
|
|
* Side Effects:
|
|
* A timer is started, set to expire after the timeout period for
|
|
* this authorization. When it expires, the function
|
|
* SecurityAuthorizationExpired will be called.
|
|
*/
|
|
|
|
static void
|
|
SecurityStartAuthorizationTimer(SecurityAuthorizationPtr pAuth)
|
|
{
|
|
pAuth->timer = TimerSet(pAuth->timer, 0,
|
|
SecurityComputeAuthorizationTimeout(pAuth,
|
|
pAuth->timeout),
|
|
SecurityAuthorizationExpired, pAuth);
|
|
} /* SecurityStartAuthorizationTimer */
|
|
|
|
/* Proc functions all take a client argument, execute the request in
|
|
* client->requestBuffer, and return a protocol error status.
|
|
*/
|
|
|
|
static int
|
|
ProcSecurityQueryVersion(ClientPtr client)
|
|
{
|
|
/* REQUEST(xSecurityQueryVersionReq); */
|
|
xSecurityQueryVersionReply rep = {
|
|
.type = X_Reply,
|
|
.sequenceNumber = client->sequence,
|
|
.length = 0,
|
|
.majorVersion = SERVER_SECURITY_MAJOR_VERSION,
|
|
.minorVersion = SERVER_SECURITY_MINOR_VERSION
|
|
};
|
|
|
|
REQUEST_SIZE_MATCH(xSecurityQueryVersionReq);
|
|
|
|
if (client->swapped) {
|
|
swaps(&rep.sequenceNumber);
|
|
swaps(&rep.majorVersion);
|
|
swaps(&rep.minorVersion);
|
|
}
|
|
WriteToClient(client, SIZEOF(xSecurityQueryVersionReply), &rep);
|
|
return Success;
|
|
} /* ProcSecurityQueryVersion */
|
|
|
|
static int
|
|
SecurityEventSelectForAuthorization(SecurityAuthorizationPtr pAuth,
|
|
ClientPtr client, Mask mask)
|
|
{
|
|
OtherClients *pEventClient;
|
|
|
|
for (pEventClient = pAuth->eventClients;
|
|
pEventClient; pEventClient = pEventClient->next) {
|
|
if (SameClient(pEventClient, client)) {
|
|
if (mask == 0)
|
|
FreeResource(pEventClient->resource, RT_NONE);
|
|
else
|
|
pEventClient->mask = mask;
|
|
return Success;
|
|
}
|
|
}
|
|
|
|
pEventClient = malloc(sizeof(OtherClients));
|
|
if (!pEventClient)
|
|
return BadAlloc;
|
|
pEventClient->mask = mask;
|
|
pEventClient->resource = FakeClientID(client->index);
|
|
pEventClient->next = pAuth->eventClients;
|
|
if (!AddResource(pEventClient->resource, RTEventClient, (void *) pAuth)) {
|
|
free(pEventClient);
|
|
return BadAlloc;
|
|
}
|
|
pAuth->eventClients = pEventClient;
|
|
|
|
return Success;
|
|
} /* SecurityEventSelectForAuthorization */
|
|
|
|
static int
|
|
ProcSecurityGenerateAuthorization(ClientPtr client)
|
|
{
|
|
REQUEST(xSecurityGenerateAuthorizationReq);
|
|
int len; /* request length in CARD32s */
|
|
Bool removeAuth = FALSE; /* if bailout, call RemoveAuthorization? */
|
|
SecurityAuthorizationPtr pAuth = NULL; /* auth we are creating */
|
|
int err; /* error to return from this function */
|
|
XID authId; /* authorization ID assigned by os layer */
|
|
xSecurityGenerateAuthorizationReply rep; /* reply struct */
|
|
unsigned int trustLevel; /* trust level of new auth */
|
|
XID group; /* group of new auth */
|
|
CARD32 timeout; /* timeout of new auth */
|
|
CARD32 *values; /* list of supplied attributes */
|
|
char *protoname; /* auth proto name sent in request */
|
|
char *protodata; /* auth proto data sent in request */
|
|
unsigned int authdata_len; /* # bytes of generated auth data */
|
|
char *pAuthdata; /* generated auth data */
|
|
Mask eventMask; /* what events on this auth does client want */
|
|
|
|
/* check request length */
|
|
|
|
REQUEST_AT_LEAST_SIZE(xSecurityGenerateAuthorizationReq);
|
|
len = bytes_to_int32(SIZEOF(xSecurityGenerateAuthorizationReq));
|
|
len += bytes_to_int32(stuff->nbytesAuthProto);
|
|
len += bytes_to_int32(stuff->nbytesAuthData);
|
|
values = ((CARD32 *) stuff) + len;
|
|
len += Ones(stuff->valueMask);
|
|
if (client->req_len != len)
|
|
return BadLength;
|
|
|
|
/* check valuemask */
|
|
if (stuff->valueMask & ~XSecurityAllAuthorizationAttributes) {
|
|
client->errorValue = stuff->valueMask;
|
|
return BadValue;
|
|
}
|
|
|
|
/* check timeout */
|
|
timeout = 60;
|
|
if (stuff->valueMask & XSecurityTimeout) {
|
|
timeout = *values++;
|
|
}
|
|
|
|
/* check trustLevel */
|
|
trustLevel = XSecurityClientUntrusted;
|
|
if (stuff->valueMask & XSecurityTrustLevel) {
|
|
trustLevel = *values++;
|
|
if (trustLevel != XSecurityClientTrusted &&
|
|
trustLevel != XSecurityClientUntrusted) {
|
|
client->errorValue = trustLevel;
|
|
return BadValue;
|
|
}
|
|
}
|
|
|
|
/* check group */
|
|
group = None;
|
|
if (stuff->valueMask & XSecurityGroup) {
|
|
group = *values++;
|
|
if (SecurityValidateGroupCallback) {
|
|
SecurityValidateGroupInfoRec vgi;
|
|
|
|
vgi.group = group;
|
|
vgi.valid = FALSE;
|
|
CallCallbacks(&SecurityValidateGroupCallback, (void *) &vgi);
|
|
|
|
/* if nobody said they recognized it, it's an error */
|
|
|
|
if (!vgi.valid) {
|
|
client->errorValue = group;
|
|
return BadValue;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* check event mask */
|
|
eventMask = 0;
|
|
if (stuff->valueMask & XSecurityEventMask) {
|
|
eventMask = *values++;
|
|
if (eventMask & ~XSecurityAllEventMasks) {
|
|
client->errorValue = eventMask;
|
|
return BadValue;
|
|
}
|
|
}
|
|
|
|
protoname = (char *) &stuff[1];
|
|
protodata = protoname + bytes_to_int32(stuff->nbytesAuthProto);
|
|
|
|
/* call os layer to generate the authorization */
|
|
|
|
authId = GenerateAuthorization(stuff->nbytesAuthProto, protoname,
|
|
stuff->nbytesAuthData, protodata,
|
|
&authdata_len, &pAuthdata);
|
|
if ((XID) ~0L == authId) {
|
|
err = SecurityErrorBase + XSecurityBadAuthorizationProtocol;
|
|
goto bailout;
|
|
}
|
|
|
|
/* now that we've added the auth, remember to remove it if we have to
|
|
* abort the request for some reason (like allocation failure)
|
|
*/
|
|
removeAuth = TRUE;
|
|
|
|
/* associate additional information with this auth ID */
|
|
|
|
pAuth = malloc(sizeof(SecurityAuthorizationRec));
|
|
if (!pAuth) {
|
|
err = BadAlloc;
|
|
goto bailout;
|
|
}
|
|
|
|
/* fill in the auth fields */
|
|
|
|
pAuth->id = authId;
|
|
pAuth->timeout = timeout;
|
|
pAuth->group = group;
|
|
pAuth->trustLevel = trustLevel;
|
|
pAuth->refcnt = 0; /* the auth was just created; nobody's using it yet */
|
|
pAuth->secondsRemaining = 0;
|
|
pAuth->timer = NULL;
|
|
pAuth->eventClients = NULL;
|
|
|
|
/* handle event selection */
|
|
if (eventMask) {
|
|
err = SecurityEventSelectForAuthorization(pAuth, client, eventMask);
|
|
if (err != Success)
|
|
goto bailout;
|
|
}
|
|
|
|
if (!AddResource(authId, SecurityAuthorizationResType, pAuth)) {
|
|
err = BadAlloc;
|
|
goto bailout;
|
|
}
|
|
|
|
/* start the timer ticking */
|
|
|
|
if (pAuth->timeout != 0)
|
|
SecurityStartAuthorizationTimer(pAuth);
|
|
|
|
/* tell client the auth id and data */
|
|
|
|
rep = (xSecurityGenerateAuthorizationReply) {
|
|
.type = X_Reply,
|
|
.sequenceNumber = client->sequence,
|
|
.length = bytes_to_int32(authdata_len),
|
|
.authId = authId,
|
|
.dataLength = authdata_len
|
|
};
|
|
|
|
if (client->swapped) {
|
|
swapl(&rep.length);
|
|
swaps(&rep.sequenceNumber);
|
|
swapl(&rep.authId);
|
|
swaps(&rep.dataLength);
|
|
}
|
|
|
|
WriteToClient(client, SIZEOF(xSecurityGenerateAuthorizationReply), &rep);
|
|
WriteToClient(client, authdata_len, pAuthdata);
|
|
|
|
SecurityAudit
|
|
("client %d generated authorization %lu trust %d timeout %lu group %lu events %lu\n",
|
|
client->index, (unsigned long)pAuth->id, pAuth->trustLevel, (unsigned long)pAuth->timeout,
|
|
(unsigned long)pAuth->group, (unsigned long)eventMask);
|
|
|
|
/* the request succeeded; don't call RemoveAuthorization or free pAuth */
|
|
return Success;
|
|
|
|
bailout:
|
|
if (removeAuth)
|
|
RemoveAuthorization(stuff->nbytesAuthProto, protoname,
|
|
authdata_len, pAuthdata);
|
|
free(pAuth);
|
|
return err;
|
|
|
|
} /* ProcSecurityGenerateAuthorization */
|
|
|
|
static int
|
|
ProcSecurityRevokeAuthorization(ClientPtr client)
|
|
{
|
|
REQUEST(xSecurityRevokeAuthorizationReq);
|
|
SecurityAuthorizationPtr pAuth;
|
|
int rc;
|
|
|
|
REQUEST_SIZE_MATCH(xSecurityRevokeAuthorizationReq);
|
|
|
|
rc = dixLookupResourceByType((void **) &pAuth, stuff->authId,
|
|
SecurityAuthorizationResType, client,
|
|
DixDestroyAccess);
|
|
if (rc != Success)
|
|
return rc;
|
|
|
|
FreeResource(stuff->authId, RT_NONE);
|
|
return Success;
|
|
} /* ProcSecurityRevokeAuthorization */
|
|
|
|
static int
|
|
ProcSecurityDispatch(ClientPtr client)
|
|
{
|
|
REQUEST(xReq);
|
|
|
|
switch (stuff->data) {
|
|
case X_SecurityQueryVersion:
|
|
return ProcSecurityQueryVersion(client);
|
|
case X_SecurityGenerateAuthorization:
|
|
return ProcSecurityGenerateAuthorization(client);
|
|
case X_SecurityRevokeAuthorization:
|
|
return ProcSecurityRevokeAuthorization(client);
|
|
default:
|
|
return BadRequest;
|
|
}
|
|
} /* ProcSecurityDispatch */
|
|
|
|
static int _X_COLD
|
|
SProcSecurityQueryVersion(ClientPtr client)
|
|
{
|
|
REQUEST(xSecurityQueryVersionReq);
|
|
|
|
swaps(&stuff->length);
|
|
REQUEST_SIZE_MATCH(xSecurityQueryVersionReq);
|
|
swaps(&stuff->majorVersion);
|
|
swaps(&stuff->minorVersion);
|
|
return ProcSecurityQueryVersion(client);
|
|
} /* SProcSecurityQueryVersion */
|
|
|
|
static int _X_COLD
|
|
SProcSecurityGenerateAuthorization(ClientPtr client)
|
|
{
|
|
REQUEST(xSecurityGenerateAuthorizationReq);
|
|
CARD32 *values;
|
|
unsigned long nvalues;
|
|
int values_offset;
|
|
|
|
swaps(&stuff->length);
|
|
REQUEST_AT_LEAST_SIZE(xSecurityGenerateAuthorizationReq);
|
|
swaps(&stuff->nbytesAuthProto);
|
|
swaps(&stuff->nbytesAuthData);
|
|
swapl(&stuff->valueMask);
|
|
values_offset = bytes_to_int32(stuff->nbytesAuthProto) +
|
|
bytes_to_int32(stuff->nbytesAuthData);
|
|
if (values_offset >
|
|
stuff->length - bytes_to_int32(sz_xSecurityGenerateAuthorizationReq))
|
|
return BadLength;
|
|
values = (CARD32 *) (&stuff[1]) + values_offset;
|
|
nvalues = (((CARD32 *) stuff) + stuff->length) - values;
|
|
SwapLongs(values, nvalues);
|
|
return ProcSecurityGenerateAuthorization(client);
|
|
} /* SProcSecurityGenerateAuthorization */
|
|
|
|
static int _X_COLD
|
|
SProcSecurityRevokeAuthorization(ClientPtr client)
|
|
{
|
|
REQUEST(xSecurityRevokeAuthorizationReq);
|
|
|
|
swaps(&stuff->length);
|
|
REQUEST_SIZE_MATCH(xSecurityRevokeAuthorizationReq);
|
|
swapl(&stuff->authId);
|
|
return ProcSecurityRevokeAuthorization(client);
|
|
} /* SProcSecurityRevokeAuthorization */
|
|
|
|
static int _X_COLD
|
|
SProcSecurityDispatch(ClientPtr client)
|
|
{
|
|
REQUEST(xReq);
|
|
|
|
switch (stuff->data) {
|
|
case X_SecurityQueryVersion:
|
|
return SProcSecurityQueryVersion(client);
|
|
case X_SecurityGenerateAuthorization:
|
|
return SProcSecurityGenerateAuthorization(client);
|
|
case X_SecurityRevokeAuthorization:
|
|
return SProcSecurityRevokeAuthorization(client);
|
|
default:
|
|
return BadRequest;
|
|
}
|
|
} /* SProcSecurityDispatch */
|
|
|
|
static void _X_COLD
|
|
SwapSecurityAuthorizationRevokedEvent(xSecurityAuthorizationRevokedEvent * from,
|
|
xSecurityAuthorizationRevokedEvent * to)
|
|
{
|
|
to->type = from->type;
|
|
to->detail = from->detail;
|
|
cpswaps(from->sequenceNumber, to->sequenceNumber);
|
|
cpswapl(from->authId, to->authId);
|
|
}
|
|
|
|
/* SecurityCheckDeviceAccess
|
|
*
|
|
* Arguments:
|
|
* client is the client attempting to access a device.
|
|
* dev is the device being accessed.
|
|
* fromRequest is TRUE if the device access is a direct result of
|
|
* the client executing some request and FALSE if it is a
|
|
* result of the server trying to send an event (e.g. KeymapNotify)
|
|
* to the client.
|
|
* Returns:
|
|
* TRUE if the device access should be allowed, else FALSE.
|
|
*
|
|
* Side Effects:
|
|
* An audit message is generated if access is denied.
|
|
*/
|
|
|
|
static void
|
|
SecurityDevice(CallbackListPtr *pcbl, void *unused, void *calldata)
|
|
{
|
|
XaceDeviceAccessRec *rec = calldata;
|
|
SecurityStateRec *subj, *obj;
|
|
Mask requested = rec->access_mode;
|
|
Mask allowed = SecurityDeviceMask;
|
|
|
|
subj = dixLookupPrivate(&rec->client->devPrivates, stateKey);
|
|
obj = dixLookupPrivate(&serverClient->devPrivates, stateKey);
|
|
|
|
if (rec->dev != inputInfo.keyboard)
|
|
/* this extension only supports the core keyboard */
|
|
allowed = requested;
|
|
|
|
if (SecurityDoCheck(subj, obj, requested, allowed) != Success) {
|
|
SecurityAudit("Security denied client %d keyboard access on request "
|
|
"%s\n", rec->client->index,
|
|
SecurityLookupRequestName(rec->client));
|
|
rec->status = BadAccess;
|
|
}
|
|
}
|
|
|
|
/* SecurityResource
|
|
*
|
|
* This function gets plugged into client->CheckAccess and is called from
|
|
* SecurityLookupIDByType/Class to determine if the client can access the
|
|
* resource.
|
|
*
|
|
* Arguments:
|
|
* client is the client doing the resource access.
|
|
* id is the resource id.
|
|
* rtype is its type or class.
|
|
* access_mode represents the intended use of the resource; see
|
|
* resource.h.
|
|
* res is a pointer to the resource structure for this resource.
|
|
*
|
|
* Returns:
|
|
* If access is granted, the value of rval that was passed in, else FALSE.
|
|
*
|
|
* Side Effects:
|
|
* Disallowed resource accesses are audited.
|
|
*/
|
|
|
|
static void
|
|
SecurityResource(CallbackListPtr *pcbl, void *unused, void *calldata)
|
|
{
|
|
XaceResourceAccessRec *rec = calldata;
|
|
SecurityStateRec *subj, *obj;
|
|
int cid = CLIENT_ID(rec->id);
|
|
Mask requested = rec->access_mode;
|
|
Mask allowed = SecurityResourceMask;
|
|
|
|
subj = dixLookupPrivate(&rec->client->devPrivates, stateKey);
|
|
|
|
/* disable background None for untrusted windows */
|
|
if ((requested & DixCreateAccess) && (rec->rtype == RT_WINDOW))
|
|
if (subj->haveState && subj->trustLevel != XSecurityClientTrusted)
|
|
((WindowPtr) rec->res)->forcedBG = TRUE;
|
|
|
|
/* additional permissions for specific resource types */
|
|
if (rec->rtype == RT_WINDOW)
|
|
allowed |= SecurityWindowExtraMask;
|
|
|
|
/* special checks for server-owned resources */
|
|
if (cid == 0) {
|
|
if (rec->rtype & RC_DRAWABLE)
|
|
/* additional operations allowed on root windows */
|
|
allowed |= SecurityRootWindowExtraMask;
|
|
|
|
else if (rec->rtype == RT_COLORMAP)
|
|
/* allow access to default colormaps */
|
|
allowed = requested;
|
|
|
|
else
|
|
/* allow read access to other server-owned resources */
|
|
allowed |= DixReadAccess;
|
|
}
|
|
|
|
if (clients[cid] != NULL) {
|
|
obj = dixLookupPrivate(&clients[cid]->devPrivates, stateKey);
|
|
if (SecurityDoCheck(subj, obj, requested, allowed) == Success)
|
|
return;
|
|
}
|
|
|
|
SecurityAudit("Security: denied client %d access %lx to resource 0x%lx "
|
|
"of client %d on request %s\n", rec->client->index,
|
|
(unsigned long)requested, (unsigned long)rec->id, cid,
|
|
SecurityLookupRequestName(rec->client));
|
|
rec->status = BadAccess; /* deny access */
|
|
}
|
|
|
|
static void
|
|
SecurityExtension(CallbackListPtr *pcbl, void *unused, void *calldata)
|
|
{
|
|
XaceExtAccessRec *rec = calldata;
|
|
SecurityStateRec *subj;
|
|
int i = 0;
|
|
|
|
subj = dixLookupPrivate(&rec->client->devPrivates, stateKey);
|
|
|
|
if (subj->haveState && subj->trustLevel == XSecurityClientTrusted)
|
|
return;
|
|
|
|
while (SecurityTrustedExtensions[i])
|
|
if (!strcmp(SecurityTrustedExtensions[i++], rec->ext->name))
|
|
return;
|
|
|
|
SecurityAudit("Security: denied client %d access to extension "
|
|
"%s on request %s\n",
|
|
rec->client->index, rec->ext->name,
|
|
SecurityLookupRequestName(rec->client));
|
|
rec->status = BadAccess;
|
|
}
|
|
|
|
static void
|
|
SecurityServer(CallbackListPtr *pcbl, void *unused, void *calldata)
|
|
{
|
|
XaceServerAccessRec *rec = calldata;
|
|
SecurityStateRec *subj, *obj;
|
|
Mask requested = rec->access_mode;
|
|
Mask allowed = SecurityServerMask;
|
|
|
|
subj = dixLookupPrivate(&rec->client->devPrivates, stateKey);
|
|
obj = dixLookupPrivate(&serverClient->devPrivates, stateKey);
|
|
|
|
if (SecurityDoCheck(subj, obj, requested, allowed) != Success) {
|
|
SecurityAudit("Security: denied client %d access to server "
|
|
"configuration request %s\n", rec->client->index,
|
|
SecurityLookupRequestName(rec->client));
|
|
rec->status = BadAccess;
|
|
}
|
|
}
|
|
|
|
static void
|
|
SecurityClient(CallbackListPtr *pcbl, void *unused, void *calldata)
|
|
{
|
|
XaceClientAccessRec *rec = calldata;
|
|
SecurityStateRec *subj, *obj;
|
|
Mask requested = rec->access_mode;
|
|
Mask allowed = SecurityClientMask;
|
|
|
|
subj = dixLookupPrivate(&rec->client->devPrivates, stateKey);
|
|
obj = dixLookupPrivate(&rec->target->devPrivates, stateKey);
|
|
|
|
if (SecurityDoCheck(subj, obj, requested, allowed) != Success) {
|
|
SecurityAudit("Security: denied client %d access to client %d on "
|
|
"request %s\n", rec->client->index, rec->target->index,
|
|
SecurityLookupRequestName(rec->client));
|
|
rec->status = BadAccess;
|
|
}
|
|
}
|
|
|
|
static void
|
|
SecurityProperty(CallbackListPtr *pcbl, void *unused, void *calldata)
|
|
{
|
|
XacePropertyAccessRec *rec = calldata;
|
|
SecurityStateRec *subj, *obj;
|
|
ATOM name = (*rec->ppProp)->propertyName;
|
|
Mask requested = rec->access_mode;
|
|
Mask allowed = SecurityResourceMask | DixReadAccess;
|
|
|
|
subj = dixLookupPrivate(&rec->client->devPrivates, stateKey);
|
|
obj = dixLookupPrivate(&wClient(rec->pWin)->devPrivates, stateKey);
|
|
|
|
if (SecurityDoCheck(subj, obj, requested, allowed) != Success) {
|
|
SecurityAudit("Security: denied client %d access to property %s "
|
|
"(atom 0x%x) window 0x%lx of client %d on request %s\n",
|
|
rec->client->index, NameForAtom(name), name,
|
|
(unsigned long)rec->pWin->drawable.id, wClient(rec->pWin)->index,
|
|
SecurityLookupRequestName(rec->client));
|
|
rec->status = BadAccess;
|
|
}
|
|
}
|
|
|
|
static void
|
|
SecuritySend(CallbackListPtr *pcbl, void *unused, void *calldata)
|
|
{
|
|
XaceSendAccessRec *rec = calldata;
|
|
SecurityStateRec *subj, *obj;
|
|
|
|
if (rec->client) {
|
|
int i;
|
|
|
|
subj = dixLookupPrivate(&rec->client->devPrivates, stateKey);
|
|
obj = dixLookupPrivate(&wClient(rec->pWin)->devPrivates, stateKey);
|
|
|
|
if (SecurityDoCheck(subj, obj, DixSendAccess, 0) == Success)
|
|
return;
|
|
|
|
for (i = 0; i < rec->count; i++)
|
|
if (rec->events[i].u.u.type != UnmapNotify &&
|
|
rec->events[i].u.u.type != ConfigureRequest &&
|
|
rec->events[i].u.u.type != ClientMessage) {
|
|
|
|
SecurityAudit("Security: denied client %d from sending event "
|
|
"of type %s to window 0x%lx of client %d\n",
|
|
rec->client->index,
|
|
LookupEventName(rec->events[i].u.u.type),
|
|
(unsigned long)rec->pWin->drawable.id,
|
|
wClient(rec->pWin)->index);
|
|
rec->status = BadAccess;
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
SecurityReceive(CallbackListPtr *pcbl, void *unused, void *calldata)
|
|
{
|
|
XaceReceiveAccessRec *rec = calldata;
|
|
SecurityStateRec *subj, *obj;
|
|
|
|
subj = dixLookupPrivate(&rec->client->devPrivates, stateKey);
|
|
obj = dixLookupPrivate(&wClient(rec->pWin)->devPrivates, stateKey);
|
|
|
|
if (SecurityDoCheck(subj, obj, DixReceiveAccess, 0) == Success)
|
|
return;
|
|
|
|
SecurityAudit("Security: denied client %d from receiving an event "
|
|
"sent to window 0x%lx of client %d\n",
|
|
rec->client->index, (unsigned long)rec->pWin->drawable.id,
|
|
wClient(rec->pWin)->index);
|
|
rec->status = BadAccess;
|
|
}
|
|
|
|
/* SecurityClientStateCallback
|
|
*
|
|
* Arguments:
|
|
* pcbl is &ClientStateCallback.
|
|
* nullata is NULL.
|
|
* calldata is a pointer to a NewClientInfoRec (include/dixstruct.h)
|
|
* which contains information about client state changes.
|
|
*
|
|
* Returns: nothing.
|
|
*
|
|
* Side Effects:
|
|
*
|
|
* If a new client is connecting, its authorization ID is copied to
|
|
* client->authID. If this is a generated authorization, its reference
|
|
* count is bumped, its timer is cancelled if it was running, and its
|
|
* trustlevel is copied to TRUSTLEVEL(client).
|
|
*
|
|
* If a client is disconnecting and the client was using a generated
|
|
* authorization, the authorization's reference count is decremented, and
|
|
* if it is now zero, the timer for this authorization is started.
|
|
*/
|
|
|
|
static void
|
|
SecurityClientState(CallbackListPtr *pcbl, void *unused, void *calldata)
|
|
{
|
|
NewClientInfoRec *pci = calldata;
|
|
SecurityStateRec *state;
|
|
SecurityAuthorizationPtr pAuth;
|
|
int rc;
|
|
|
|
state = dixLookupPrivate(&pci->client->devPrivates, stateKey);
|
|
|
|
switch (pci->client->clientState) {
|
|
case ClientStateInitial:
|
|
state->trustLevel = XSecurityClientTrusted;
|
|
state->authId = None;
|
|
state->haveState = TRUE;
|
|
state->live = FALSE;
|
|
break;
|
|
|
|
case ClientStateRunning:
|
|
state->authId = AuthorizationIDOfClient(pci->client);
|
|
rc = dixLookupResourceByType((void **) &pAuth, state->authId,
|
|
SecurityAuthorizationResType, serverClient,
|
|
DixGetAttrAccess);
|
|
if (rc == Success) {
|
|
/* it is a generated authorization */
|
|
pAuth->refcnt++;
|
|
state->live = TRUE;
|
|
if (pAuth->refcnt == 1 && pAuth->timer)
|
|
TimerCancel(pAuth->timer);
|
|
|
|
state->trustLevel = pAuth->trustLevel;
|
|
}
|
|
break;
|
|
|
|
case ClientStateGone:
|
|
case ClientStateRetained:
|
|
rc = dixLookupResourceByType((void **) &pAuth, state->authId,
|
|
SecurityAuthorizationResType, serverClient,
|
|
DixGetAttrAccess);
|
|
if (rc == Success && state->live) {
|
|
/* it is a generated authorization */
|
|
pAuth->refcnt--;
|
|
state->live = FALSE;
|
|
if (pAuth->refcnt == 0)
|
|
SecurityStartAuthorizationTimer(pAuth);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
/* SecurityResetProc
|
|
*
|
|
* Arguments:
|
|
* extEntry is the extension information for the security extension.
|
|
*
|
|
* Returns: nothing.
|
|
*
|
|
* Side Effects:
|
|
* Performs any cleanup needed by Security at server shutdown time.
|
|
*/
|
|
|
|
static void
|
|
SecurityResetProc(ExtensionEntry * extEntry)
|
|
{
|
|
/* Unregister callbacks */
|
|
DeleteCallback(&ClientStateCallback, SecurityClientState, NULL);
|
|
|
|
XaceDeleteCallback(XACE_EXT_DISPATCH, SecurityExtension, NULL);
|
|
XaceDeleteCallback(XACE_RESOURCE_ACCESS, SecurityResource, NULL);
|
|
XaceDeleteCallback(XACE_DEVICE_ACCESS, SecurityDevice, NULL);
|
|
XaceDeleteCallback(XACE_PROPERTY_ACCESS, SecurityProperty, NULL);
|
|
XaceDeleteCallback(XACE_SEND_ACCESS, SecuritySend, NULL);
|
|
XaceDeleteCallback(XACE_RECEIVE_ACCESS, SecurityReceive, NULL);
|
|
XaceDeleteCallback(XACE_CLIENT_ACCESS, SecurityClient, NULL);
|
|
XaceDeleteCallback(XACE_EXT_ACCESS, SecurityExtension, NULL);
|
|
XaceDeleteCallback(XACE_SERVER_ACCESS, SecurityServer, NULL);
|
|
}
|
|
|
|
/* SecurityExtensionInit
|
|
*
|
|
* Arguments: none.
|
|
*
|
|
* Returns: nothing.
|
|
*
|
|
* Side Effects:
|
|
* Enables the Security extension if possible.
|
|
*/
|
|
|
|
void
|
|
SecurityExtensionInit(void)
|
|
{
|
|
ExtensionEntry *extEntry;
|
|
int ret = TRUE;
|
|
|
|
SecurityAuthorizationResType =
|
|
CreateNewResourceType(SecurityDeleteAuthorization,
|
|
"SecurityAuthorization");
|
|
|
|
RTEventClient =
|
|
CreateNewResourceType(SecurityDeleteAuthorizationEventClient,
|
|
"SecurityEventClient");
|
|
|
|
if (!SecurityAuthorizationResType || !RTEventClient)
|
|
return;
|
|
|
|
RTEventClient |= RC_NEVERRETAIN;
|
|
|
|
/* Allocate the private storage */
|
|
if (!dixRegisterPrivateKey
|
|
(stateKey, PRIVATE_CLIENT, sizeof(SecurityStateRec)))
|
|
FatalError("SecurityExtensionSetup: Can't allocate client private.\n");
|
|
|
|
/* Register callbacks */
|
|
ret &= AddCallback(&ClientStateCallback, SecurityClientState, NULL);
|
|
|
|
ret &= XaceRegisterCallback(XACE_EXT_DISPATCH, SecurityExtension, NULL);
|
|
ret &= XaceRegisterCallback(XACE_RESOURCE_ACCESS, SecurityResource, NULL);
|
|
ret &= XaceRegisterCallback(XACE_DEVICE_ACCESS, SecurityDevice, NULL);
|
|
ret &= XaceRegisterCallback(XACE_PROPERTY_ACCESS, SecurityProperty, NULL);
|
|
ret &= XaceRegisterCallback(XACE_SEND_ACCESS, SecuritySend, NULL);
|
|
ret &= XaceRegisterCallback(XACE_RECEIVE_ACCESS, SecurityReceive, NULL);
|
|
ret &= XaceRegisterCallback(XACE_CLIENT_ACCESS, SecurityClient, NULL);
|
|
ret &= XaceRegisterCallback(XACE_EXT_ACCESS, SecurityExtension, NULL);
|
|
ret &= XaceRegisterCallback(XACE_SERVER_ACCESS, SecurityServer, NULL);
|
|
|
|
if (!ret)
|
|
FatalError("SecurityExtensionSetup: Failed to register callbacks\n");
|
|
|
|
/* Add extension to server */
|
|
extEntry = AddExtension(SECURITY_EXTENSION_NAME,
|
|
XSecurityNumberEvents, XSecurityNumberErrors,
|
|
ProcSecurityDispatch, SProcSecurityDispatch,
|
|
SecurityResetProc, StandardMinorOpcode);
|
|
|
|
SecurityErrorBase = extEntry->errorBase;
|
|
SecurityEventBase = extEntry->eventBase;
|
|
|
|
EventSwapVector[SecurityEventBase + XSecurityAuthorizationRevoked] =
|
|
(EventSwapPtr) SwapSecurityAuthorizationRevokedEvent;
|
|
|
|
SetResourceTypeErrorValue(SecurityAuthorizationResType,
|
|
SecurityErrorBase + XSecurityBadAuthorization);
|
|
|
|
/* Label objects that were created before we could register ourself */
|
|
SecurityLabelInitial();
|
|
}
|