1191 lines
35 KiB
C
1191 lines
35 KiB
C
/*
|
|
Copyright (c) 2002 XFree86 Inc
|
|
*/
|
|
|
|
#ifdef HAVE_DIX_CONFIG_H
|
|
#include <dix-config.h>
|
|
#endif
|
|
|
|
#include <stdio.h>
|
|
#include <string.h>
|
|
#include <X11/X.h>
|
|
#include <X11/Xproto.h>
|
|
#include <assert.h>
|
|
#include "misc.h"
|
|
#include "os.h"
|
|
#include "dixstruct.h"
|
|
#include "extnsionst.h"
|
|
#include "swaprep.h"
|
|
#include "registry.h"
|
|
#include <X11/extensions/XResproto.h>
|
|
#include "pixmapstr.h"
|
|
#include "windowstr.h"
|
|
#include "gcstruct.h"
|
|
#include "extinit.h"
|
|
#include "protocol-versions.h"
|
|
#include "client.h"
|
|
#include "list.h"
|
|
#include "misc.h"
|
|
#include <string.h>
|
|
#include "hashtable.h"
|
|
#include "picturestr.h"
|
|
|
|
#ifdef COMPOSITE
|
|
#include "compint.h"
|
|
#endif
|
|
|
|
/** @brief Holds fragments of responses for ConstructClientIds.
|
|
*
|
|
* note: there is no consideration for data alignment */
|
|
typedef struct {
|
|
struct xorg_list l;
|
|
int bytes;
|
|
/* data follows */
|
|
} FragmentList;
|
|
|
|
#define FRAGMENT_DATA(ptr) ((void*) ((char*) (ptr) + sizeof(FragmentList)))
|
|
|
|
/** @brief Holds structure for the generated response to
|
|
ProcXResQueryClientIds; used by ConstructClientId* -functions */
|
|
typedef struct {
|
|
int numIds;
|
|
int resultBytes;
|
|
struct xorg_list response;
|
|
int sentClientMasks[MAXCLIENTS];
|
|
} ConstructClientIdCtx;
|
|
|
|
/** @brief Holds the structure for information required to
|
|
generate the response to XResQueryResourceBytes. In addition
|
|
to response it contains information on the query as well,
|
|
as well as some volatile information required by a few
|
|
functions that cannot take that information directly
|
|
via a parameter, as they are called via already-existing
|
|
higher order functions. */
|
|
typedef struct {
|
|
ClientPtr sendClient;
|
|
int numSizes;
|
|
int resultBytes;
|
|
struct xorg_list response;
|
|
int status;
|
|
long numSpecs;
|
|
xXResResourceIdSpec *specs;
|
|
HashTable visitedResources;
|
|
|
|
/* Used by AddSubResourceSizeSpec when AddResourceSizeValue is
|
|
handling crossreferences */
|
|
HashTable visitedSubResources;
|
|
|
|
/* used when ConstructResourceBytesCtx is passed to
|
|
AddResourceSizeValue2 via FindClientResourcesByType */
|
|
RESTYPE resType;
|
|
|
|
/* used when ConstructResourceBytesCtx is passed to
|
|
AddResourceSizeValueByResource from ConstructResourceBytesByResource */
|
|
xXResResourceIdSpec *curSpec;
|
|
|
|
/** Used when iterating through a single resource's subresources
|
|
|
|
@see AddSubResourceSizeSpec */
|
|
xXResResourceSizeValue *sizeValue;
|
|
} ConstructResourceBytesCtx;
|
|
|
|
/** @brief Allocate and add a sequence of bytes at the end of a fragment list.
|
|
Call DestroyFragments to release the list.
|
|
|
|
@param frags A pointer to head of an initialized linked list
|
|
@param bytes Number of bytes to allocate
|
|
@return Returns a pointer to the allocated non-zeroed region
|
|
that is to be filled by the caller. On error (out of memory)
|
|
returns NULL and makes no changes to the list.
|
|
*/
|
|
static void *
|
|
AddFragment(struct xorg_list *frags, int bytes)
|
|
{
|
|
FragmentList *f = malloc(sizeof(FragmentList) + bytes);
|
|
if (!f) {
|
|
return NULL;
|
|
} else {
|
|
f->bytes = bytes;
|
|
xorg_list_add(&f->l, frags->prev);
|
|
return (char*) f + sizeof(*f);
|
|
}
|
|
}
|
|
|
|
/** @brief Sends all fragments in the list to the client. Does not
|
|
free anything.
|
|
|
|
@param client The client to send the fragments to
|
|
@param frags The head of the list of fragments
|
|
*/
|
|
static void
|
|
WriteFragmentsToClient(ClientPtr client, struct xorg_list *frags)
|
|
{
|
|
FragmentList *it;
|
|
xorg_list_for_each_entry(it, frags, l) {
|
|
WriteToClient(client, it->bytes, (char*) it + sizeof(*it));
|
|
}
|
|
}
|
|
|
|
/** @brief Frees a list of fragments. Does not free() root node.
|
|
|
|
@param frags The head of the list of fragments
|
|
*/
|
|
static void
|
|
DestroyFragments(struct xorg_list *frags)
|
|
{
|
|
FragmentList *it, *tmp;
|
|
xorg_list_for_each_entry_safe(it, tmp, frags, l) {
|
|
xorg_list_del(&it->l);
|
|
free(it);
|
|
}
|
|
}
|
|
|
|
/** @brief Constructs a context record for ConstructClientId* functions
|
|
to use */
|
|
static void
|
|
InitConstructClientIdCtx(ConstructClientIdCtx *ctx)
|
|
{
|
|
ctx->numIds = 0;
|
|
ctx->resultBytes = 0;
|
|
xorg_list_init(&ctx->response);
|
|
memset(ctx->sentClientMasks, 0, sizeof(ctx->sentClientMasks));
|
|
}
|
|
|
|
/** @brief Destroys a context record, releases all memory (except the storage
|
|
for *ctx itself) */
|
|
static void
|
|
DestroyConstructClientIdCtx(ConstructClientIdCtx *ctx)
|
|
{
|
|
DestroyFragments(&ctx->response);
|
|
}
|
|
|
|
static Bool
|
|
InitConstructResourceBytesCtx(ConstructResourceBytesCtx *ctx,
|
|
ClientPtr sendClient,
|
|
long numSpecs,
|
|
xXResResourceIdSpec *specs)
|
|
{
|
|
ctx->sendClient = sendClient;
|
|
ctx->numSizes = 0;
|
|
ctx->resultBytes = 0;
|
|
xorg_list_init(&ctx->response);
|
|
ctx->status = Success;
|
|
ctx->numSpecs = numSpecs;
|
|
ctx->specs = specs;
|
|
ctx->visitedResources = ht_create(sizeof(XID), 0,
|
|
ht_resourceid_hash, ht_resourceid_compare,
|
|
NULL);
|
|
|
|
if (!ctx->visitedResources) {
|
|
return FALSE;
|
|
} else {
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
static void
|
|
DestroyConstructResourceBytesCtx(ConstructResourceBytesCtx *ctx)
|
|
{
|
|
DestroyFragments(&ctx->response);
|
|
ht_destroy(ctx->visitedResources);
|
|
}
|
|
|
|
static int
|
|
ProcXResQueryVersion(ClientPtr client)
|
|
{
|
|
xXResQueryVersionReply rep = {
|
|
.type = X_Reply,
|
|
.sequenceNumber = client->sequence,
|
|
.length = 0,
|
|
.server_major = SERVER_XRES_MAJOR_VERSION,
|
|
.server_minor = SERVER_XRES_MINOR_VERSION
|
|
};
|
|
|
|
REQUEST_SIZE_MATCH(xXResQueryVersionReq);
|
|
|
|
if (client->swapped) {
|
|
swaps(&rep.sequenceNumber);
|
|
swapl(&rep.length);
|
|
swaps(&rep.server_major);
|
|
swaps(&rep.server_minor);
|
|
}
|
|
WriteToClient(client, sizeof(xXResQueryVersionReply), &rep);
|
|
return Success;
|
|
}
|
|
|
|
static int
|
|
ProcXResQueryClients(ClientPtr client)
|
|
{
|
|
/* REQUEST(xXResQueryClientsReq); */
|
|
xXResQueryClientsReply rep;
|
|
int *current_clients;
|
|
int i, num_clients;
|
|
|
|
REQUEST_SIZE_MATCH(xXResQueryClientsReq);
|
|
|
|
current_clients = malloc(currentMaxClients * sizeof(int));
|
|
|
|
num_clients = 0;
|
|
for (i = 0; i < currentMaxClients; i++) {
|
|
if (clients[i]) {
|
|
current_clients[num_clients] = i;
|
|
num_clients++;
|
|
}
|
|
}
|
|
|
|
rep = (xXResQueryClientsReply) {
|
|
.type = X_Reply,
|
|
.sequenceNumber = client->sequence,
|
|
.length = bytes_to_int32(num_clients * sz_xXResClient),
|
|
.num_clients = num_clients
|
|
};
|
|
if (client->swapped) {
|
|
swaps(&rep.sequenceNumber);
|
|
swapl(&rep.length);
|
|
swapl(&rep.num_clients);
|
|
}
|
|
WriteToClient(client, sizeof(xXResQueryClientsReply), &rep);
|
|
|
|
if (num_clients) {
|
|
xXResClient scratch;
|
|
|
|
for (i = 0; i < num_clients; i++) {
|
|
scratch.resource_base = clients[current_clients[i]]->clientAsMask;
|
|
scratch.resource_mask = RESOURCE_ID_MASK;
|
|
|
|
if (client->swapped) {
|
|
swapl(&scratch.resource_base);
|
|
swapl(&scratch.resource_mask);
|
|
}
|
|
WriteToClient(client, sz_xXResClient, &scratch);
|
|
}
|
|
}
|
|
|
|
free(current_clients);
|
|
|
|
return Success;
|
|
}
|
|
|
|
static void
|
|
ResFindAllRes(void *value, XID id, RESTYPE type, void *cdata)
|
|
{
|
|
int *counts = (int *) cdata;
|
|
|
|
counts[(type & TypeMask) - 1]++;
|
|
}
|
|
|
|
static int
|
|
ProcXResQueryClientResources(ClientPtr client)
|
|
{
|
|
REQUEST(xXResQueryClientResourcesReq);
|
|
xXResQueryClientResourcesReply rep;
|
|
int i, clientID, num_types;
|
|
int *counts;
|
|
|
|
REQUEST_SIZE_MATCH(xXResQueryClientResourcesReq);
|
|
|
|
clientID = CLIENT_ID(stuff->xid);
|
|
|
|
if ((clientID >= currentMaxClients) || !clients[clientID]) {
|
|
client->errorValue = stuff->xid;
|
|
return BadValue;
|
|
}
|
|
|
|
counts = calloc(lastResourceType + 1, sizeof(int));
|
|
|
|
FindAllClientResources(clients[clientID], ResFindAllRes, counts);
|
|
|
|
num_types = 0;
|
|
|
|
for (i = 0; i <= lastResourceType; i++) {
|
|
if (counts[i])
|
|
num_types++;
|
|
}
|
|
|
|
rep = (xXResQueryClientResourcesReply) {
|
|
.type = X_Reply,
|
|
.sequenceNumber = client->sequence,
|
|
.length = bytes_to_int32(num_types * sz_xXResType),
|
|
.num_types = num_types
|
|
};
|
|
if (client->swapped) {
|
|
swaps(&rep.sequenceNumber);
|
|
swapl(&rep.length);
|
|
swapl(&rep.num_types);
|
|
}
|
|
|
|
WriteToClient(client, sizeof(xXResQueryClientResourcesReply), &rep);
|
|
|
|
if (num_types) {
|
|
xXResType scratch;
|
|
const char *name;
|
|
|
|
for (i = 0; i < lastResourceType; i++) {
|
|
if (!counts[i])
|
|
continue;
|
|
|
|
name = LookupResourceName(i + 1);
|
|
if (strcmp(name, XREGISTRY_UNKNOWN))
|
|
scratch.resource_type = MakeAtom(name, strlen(name), TRUE);
|
|
else {
|
|
char buf[40];
|
|
|
|
snprintf(buf, sizeof(buf), "Unregistered resource %i", i + 1);
|
|
scratch.resource_type = MakeAtom(buf, strlen(buf), TRUE);
|
|
}
|
|
|
|
scratch.count = counts[i];
|
|
|
|
if (client->swapped) {
|
|
swapl(&scratch.resource_type);
|
|
swapl(&scratch.count);
|
|
}
|
|
WriteToClient(client, sz_xXResType, &scratch);
|
|
}
|
|
}
|
|
|
|
free(counts);
|
|
|
|
return Success;
|
|
}
|
|
|
|
static unsigned long
|
|
ResGetApproxPixmapBytes(PixmapPtr pix)
|
|
{
|
|
unsigned long nPixels;
|
|
float bytesPerPixel;
|
|
|
|
bytesPerPixel = (float)pix->drawable.bitsPerPixel / 8.0;
|
|
nPixels = pix->drawable.width * pix->drawable.height;
|
|
|
|
/* Divide by refcnt as pixmap could be shared between clients,
|
|
* so total pixmap mem is shared between these.
|
|
*/
|
|
return (nPixels * bytesPerPixel) / pix->refcnt;
|
|
}
|
|
|
|
static void
|
|
ResFindResourcePixmaps(void *value, XID id, RESTYPE type, void *cdata)
|
|
{
|
|
SizeType sizeFunc = GetResourceTypeSizeFunc(type);
|
|
ResourceSizeRec size = { 0, 0, 0 };
|
|
unsigned long *bytes = cdata;
|
|
|
|
sizeFunc(value, id, &size);
|
|
*bytes += size.pixmapRefSize;
|
|
}
|
|
|
|
static void
|
|
ResFindPixmaps(void *value, XID id, void *cdata)
|
|
{
|
|
unsigned long *bytes = (unsigned long *) cdata;
|
|
PixmapPtr pix = (PixmapPtr) value;
|
|
|
|
*bytes += ResGetApproxPixmapBytes(pix);
|
|
}
|
|
|
|
static void
|
|
ResFindWindowPixmaps(void *value, XID id, void *cdata)
|
|
{
|
|
unsigned long *bytes = (unsigned long *) cdata;
|
|
WindowPtr pWin = (WindowPtr) value;
|
|
|
|
if (pWin->backgroundState == BackgroundPixmap)
|
|
*bytes += ResGetApproxPixmapBytes(pWin->background.pixmap);
|
|
|
|
if (pWin->border.pixmap != NULL && !pWin->borderIsPixel)
|
|
*bytes += ResGetApproxPixmapBytes(pWin->border.pixmap);
|
|
}
|
|
|
|
static void
|
|
ResFindGCPixmaps(void *value, XID id, void *cdata)
|
|
{
|
|
unsigned long *bytes = (unsigned long *) cdata;
|
|
GCPtr pGC = (GCPtr) value;
|
|
|
|
if (pGC->stipple != NULL)
|
|
*bytes += ResGetApproxPixmapBytes(pGC->stipple);
|
|
|
|
if (pGC->tile.pixmap != NULL && !pGC->tileIsPixel)
|
|
*bytes += ResGetApproxPixmapBytes(pGC->tile.pixmap);
|
|
}
|
|
|
|
static void
|
|
ResFindPicturePixmaps(void *value, XID id, void *cdata)
|
|
{
|
|
#ifdef RENDER
|
|
ResFindResourcePixmaps(value, id, PictureType, cdata);
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
ResFindCompositeClientWindowPixmaps (void *value, XID id, void *cdata)
|
|
{
|
|
#ifdef COMPOSITE
|
|
ResFindResourcePixmaps(value, id, CompositeClientWindowType, cdata);
|
|
#endif
|
|
}
|
|
|
|
static int
|
|
ProcXResQueryClientPixmapBytes(ClientPtr client)
|
|
{
|
|
REQUEST(xXResQueryClientPixmapBytesReq);
|
|
xXResQueryClientPixmapBytesReply rep;
|
|
int clientID;
|
|
unsigned long bytes;
|
|
|
|
REQUEST_SIZE_MATCH(xXResQueryClientPixmapBytesReq);
|
|
|
|
clientID = CLIENT_ID(stuff->xid);
|
|
|
|
if ((clientID >= currentMaxClients) || !clients[clientID]) {
|
|
client->errorValue = stuff->xid;
|
|
return BadValue;
|
|
}
|
|
|
|
bytes = 0;
|
|
|
|
FindClientResourcesByType(clients[clientID], RT_PIXMAP, ResFindPixmaps,
|
|
(void *) (&bytes));
|
|
|
|
/*
|
|
* Make sure win background pixmaps also held to account.
|
|
*/
|
|
FindClientResourcesByType(clients[clientID], RT_WINDOW,
|
|
ResFindWindowPixmaps, (void *) (&bytes));
|
|
|
|
/*
|
|
* GC Tile & Stipple pixmaps too.
|
|
*/
|
|
FindClientResourcesByType(clients[clientID], RT_GC,
|
|
ResFindGCPixmaps, (void *) (&bytes));
|
|
|
|
#ifdef RENDER
|
|
/* Render extension picture pixmaps. */
|
|
FindClientResourcesByType(clients[clientID], PictureType,
|
|
ResFindPicturePixmaps,
|
|
(void *)(&bytes));
|
|
#endif
|
|
|
|
#ifdef COMPOSITE
|
|
/* Composite extension client window pixmaps. */
|
|
FindClientResourcesByType(clients[clientID], CompositeClientWindowType,
|
|
ResFindCompositeClientWindowPixmaps,
|
|
(void *)(&bytes));
|
|
#endif
|
|
|
|
rep = (xXResQueryClientPixmapBytesReply) {
|
|
.type = X_Reply,
|
|
.sequenceNumber = client->sequence,
|
|
.length = 0,
|
|
.bytes = bytes,
|
|
#ifdef _XSERVER64
|
|
.bytes_overflow = bytes >> 32
|
|
#else
|
|
.bytes_overflow = 0
|
|
#endif
|
|
};
|
|
if (client->swapped) {
|
|
swaps(&rep.sequenceNumber);
|
|
swapl(&rep.length);
|
|
swapl(&rep.bytes);
|
|
swapl(&rep.bytes_overflow);
|
|
}
|
|
WriteToClient(client, sizeof(xXResQueryClientPixmapBytesReply), &rep);
|
|
|
|
return Success;
|
|
}
|
|
|
|
/** @brief Finds out if a client's information need to be put into the
|
|
response; marks client having been handled, if that is the case.
|
|
|
|
@param client The client to send information about
|
|
@param mask The request mask (0 to send everything, otherwise a
|
|
bitmask of X_XRes*Mask)
|
|
@param ctx The context record that tells which clients and id types
|
|
have been already handled
|
|
@param sendMask Which id type are we now considering. One of X_XRes*Mask.
|
|
|
|
@return Returns TRUE if the client information needs to be on the
|
|
response, otherwise FALSE.
|
|
*/
|
|
static Bool
|
|
WillConstructMask(ClientPtr client, CARD32 mask,
|
|
ConstructClientIdCtx *ctx, int sendMask)
|
|
{
|
|
if ((!mask || (mask & sendMask))
|
|
&& !(ctx->sentClientMasks[client->index] & sendMask)) {
|
|
ctx->sentClientMasks[client->index] |= sendMask;
|
|
return TRUE;
|
|
} else {
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
/** @brief Constructs a response about a single client, based on a certain
|
|
client id spec
|
|
|
|
@param sendClient Which client wishes to receive this answer. Used for
|
|
byte endianess.
|
|
@param client Which client are we considering.
|
|
@param mask The client id spec mask indicating which information
|
|
we want about this client.
|
|
@param ctx The context record containing the constructed response
|
|
and information on which clients and masks have been
|
|
already handled.
|
|
|
|
@return Return TRUE if everything went OK, otherwise FALSE which indicates
|
|
a memory allocation problem.
|
|
*/
|
|
static Bool
|
|
ConstructClientIdValue(ClientPtr sendClient, ClientPtr client, CARD32 mask,
|
|
ConstructClientIdCtx *ctx)
|
|
{
|
|
xXResClientIdValue rep;
|
|
|
|
rep.spec.client = client->clientAsMask;
|
|
if (client->swapped) {
|
|
swapl (&rep.spec.client);
|
|
}
|
|
|
|
if (WillConstructMask(client, mask, ctx, X_XResClientXIDMask)) {
|
|
void *ptr = AddFragment(&ctx->response, sizeof(rep));
|
|
if (!ptr) {
|
|
return FALSE;
|
|
}
|
|
|
|
rep.spec.mask = X_XResClientXIDMask;
|
|
rep.length = 0;
|
|
if (sendClient->swapped) {
|
|
swapl (&rep.spec.mask);
|
|
/* swapl (&rep.length, n); - not required for rep.length = 0 */
|
|
}
|
|
|
|
memcpy(ptr, &rep, sizeof(rep));
|
|
|
|
ctx->resultBytes += sizeof(rep);
|
|
++ctx->numIds;
|
|
}
|
|
if (WillConstructMask(client, mask, ctx, X_XResLocalClientPIDMask)) {
|
|
pid_t pid = GetClientPid(client);
|
|
|
|
if (pid != -1) {
|
|
void *ptr = AddFragment(&ctx->response,
|
|
sizeof(rep) + sizeof(CARD32));
|
|
CARD32 *value = (void*) ((char*) ptr + sizeof(rep));
|
|
|
|
if (!ptr) {
|
|
return FALSE;
|
|
}
|
|
|
|
rep.spec.mask = X_XResLocalClientPIDMask;
|
|
rep.length = 4;
|
|
|
|
if (sendClient->swapped) {
|
|
swapl (&rep.spec.mask);
|
|
swapl (&rep.length);
|
|
}
|
|
|
|
if (sendClient->swapped) {
|
|
swapl (value);
|
|
}
|
|
memcpy(ptr, &rep, sizeof(rep));
|
|
*value = pid;
|
|
|
|
ctx->resultBytes += sizeof(rep) + sizeof(CARD32);
|
|
++ctx->numIds;
|
|
}
|
|
}
|
|
|
|
/* memory allocation errors earlier may return with FALSE */
|
|
return TRUE;
|
|
}
|
|
|
|
/** @brief Constructs a response about all clients, based on a client id specs
|
|
|
|
@param client Which client which we are constructing the response for.
|
|
@param numSpecs Number of client id specs in specs
|
|
@param specs Client id specs
|
|
|
|
@return Return Success if everything went OK, otherwise a Bad* (currently
|
|
BadAlloc or BadValue)
|
|
*/
|
|
static int
|
|
ConstructClientIds(ClientPtr client,
|
|
int numSpecs, xXResClientIdSpec* specs,
|
|
ConstructClientIdCtx *ctx)
|
|
{
|
|
int specIdx;
|
|
|
|
for (specIdx = 0; specIdx < numSpecs; ++specIdx) {
|
|
if (specs[specIdx].client == 0) {
|
|
int c;
|
|
for (c = 0; c < currentMaxClients; ++c) {
|
|
if (clients[c]) {
|
|
if (!ConstructClientIdValue(client, clients[c],
|
|
specs[specIdx].mask, ctx)) {
|
|
return BadAlloc;
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
int clientID = CLIENT_ID(specs[specIdx].client);
|
|
|
|
if ((clientID < currentMaxClients) && clients[clientID]) {
|
|
if (!ConstructClientIdValue(client, clients[clientID],
|
|
specs[specIdx].mask, ctx)) {
|
|
return BadAlloc;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* memory allocation errors earlier may return with BadAlloc */
|
|
return Success;
|
|
}
|
|
|
|
/** @brief Response to XResQueryClientIds request introduced in XResProto v1.2
|
|
|
|
@param client Which client which we are constructing the response for.
|
|
|
|
@return Returns the value returned from ConstructClientIds with the same
|
|
semantics
|
|
*/
|
|
static int
|
|
ProcXResQueryClientIds (ClientPtr client)
|
|
{
|
|
REQUEST(xXResQueryClientIdsReq);
|
|
|
|
xXResClientIdSpec *specs = (void*) ((char*) stuff + sizeof(*stuff));
|
|
int rc;
|
|
ConstructClientIdCtx ctx;
|
|
|
|
InitConstructClientIdCtx(&ctx);
|
|
|
|
REQUEST_AT_LEAST_SIZE(xXResQueryClientIdsReq);
|
|
REQUEST_FIXED_SIZE(xXResQueryClientIdsReq,
|
|
stuff->numSpecs * sizeof(specs[0]));
|
|
|
|
rc = ConstructClientIds(client, stuff->numSpecs, specs, &ctx);
|
|
|
|
if (rc == Success) {
|
|
xXResQueryClientIdsReply rep = {
|
|
.type = X_Reply,
|
|
.sequenceNumber = client->sequence,
|
|
.length = bytes_to_int32(ctx.resultBytes),
|
|
.numIds = ctx.numIds
|
|
};
|
|
|
|
assert((ctx.resultBytes & 3) == 0);
|
|
|
|
if (client->swapped) {
|
|
swaps (&rep.sequenceNumber);
|
|
swapl (&rep.length);
|
|
swapl (&rep.numIds);
|
|
}
|
|
|
|
WriteToClient(client, sizeof(rep), &rep);
|
|
WriteFragmentsToClient(client, &ctx.response);
|
|
}
|
|
|
|
DestroyConstructClientIdCtx(&ctx);
|
|
|
|
return rc;
|
|
}
|
|
|
|
/** @brief Swaps xXResResourceIdSpec endianess */
|
|
static void
|
|
SwapXResResourceIdSpec(xXResResourceIdSpec *spec)
|
|
{
|
|
swapl(&spec->resource);
|
|
swapl(&spec->type);
|
|
}
|
|
|
|
/** @brief Swaps xXResResourceSizeSpec endianess */
|
|
static void
|
|
SwapXResResourceSizeSpec(xXResResourceSizeSpec *size)
|
|
{
|
|
SwapXResResourceIdSpec(&size->spec);
|
|
swapl(&size->bytes);
|
|
swapl(&size->refCount);
|
|
swapl(&size->useCount);
|
|
}
|
|
|
|
/** @brief Swaps xXResResourceSizeValue endianess */
|
|
static void
|
|
SwapXResResourceSizeValue(xXResResourceSizeValue *rep)
|
|
{
|
|
SwapXResResourceSizeSpec(&rep->size);
|
|
swapl(&rep->numCrossReferences);
|
|
}
|
|
|
|
/** @brief Swaps the response bytes */
|
|
static void
|
|
SwapXResQueryResourceBytes(struct xorg_list *response)
|
|
{
|
|
struct xorg_list *it = response->next;
|
|
int c;
|
|
|
|
while (it != response) {
|
|
xXResResourceSizeValue *value = FRAGMENT_DATA(it);
|
|
it = it->next;
|
|
for (c = 0; c < value->numCrossReferences; ++c) {
|
|
xXResResourceSizeSpec *spec = FRAGMENT_DATA(it);
|
|
SwapXResResourceSizeSpec(spec);
|
|
it = it->next;
|
|
}
|
|
SwapXResResourceSizeValue(value);
|
|
}
|
|
}
|
|
|
|
/** @brief Adds xXResResourceSizeSpec describing a resource's size into
|
|
the buffer contained in the context. The resource is considered
|
|
to be a subresource.
|
|
|
|
@see AddResourceSizeValue
|
|
|
|
@param[in] value The X resource object on which to add information
|
|
about to the buffer
|
|
@param[in] id The ID of the X resource
|
|
@param[in] type The type of the X resource
|
|
@param[in/out] cdata The context object of type ConstructResourceBytesCtx.
|
|
Void pointer type is used here to satisfy the type
|
|
FindRes
|
|
*/
|
|
static void
|
|
AddSubResourceSizeSpec(void *value,
|
|
XID id,
|
|
RESTYPE type,
|
|
void *cdata)
|
|
{
|
|
ConstructResourceBytesCtx *ctx = cdata;
|
|
|
|
if (ctx->status == Success) {
|
|
xXResResourceSizeSpec **prevCrossRef =
|
|
ht_find(ctx->visitedSubResources, &value);
|
|
if (!prevCrossRef) {
|
|
Bool ok = TRUE;
|
|
xXResResourceSizeSpec *crossRef =
|
|
AddFragment(&ctx->response, sizeof(xXResResourceSizeSpec));
|
|
ok = ok && crossRef != NULL;
|
|
if (ok) {
|
|
xXResResourceSizeSpec **p;
|
|
p = ht_add(ctx->visitedSubResources, &value);
|
|
if (!p) {
|
|
ok = FALSE;
|
|
} else {
|
|
*p = crossRef;
|
|
}
|
|
}
|
|
if (!ok) {
|
|
ctx->status = BadAlloc;
|
|
} else {
|
|
SizeType sizeFunc = GetResourceTypeSizeFunc(type);
|
|
ResourceSizeRec size = { 0, 0, 0 };
|
|
sizeFunc(value, id, &size);
|
|
|
|
crossRef->spec.resource = id;
|
|
crossRef->spec.type = type;
|
|
crossRef->bytes = size.resourceSize;
|
|
crossRef->refCount = size.refCnt;
|
|
crossRef->useCount = 1;
|
|
|
|
++ctx->sizeValue->numCrossReferences;
|
|
|
|
ctx->resultBytes += sizeof(*crossRef);
|
|
}
|
|
} else {
|
|
/* if we have visited the subresource earlier (from current parent
|
|
resource), just increase its use count by one */
|
|
++(*prevCrossRef)->useCount;
|
|
}
|
|
}
|
|
}
|
|
|
|
/** @brief Adds xXResResourceSizeValue describing a resource's size into
|
|
the buffer contained in the context. In addition, the
|
|
subresources are iterated and added as xXResResourceSizeSpec's
|
|
by using AddSubResourceSizeSpec
|
|
|
|
@see AddSubResourceSizeSpec
|
|
|
|
@param[in] value The X resource object on which to add information
|
|
about to the buffer
|
|
@param[in] id The ID of the X resource
|
|
@param[in] type The type of the X resource
|
|
@param[in/out] cdata The context object of type ConstructResourceBytesCtx.
|
|
Void pointer type is used here to satisfy the type
|
|
FindRes
|
|
*/
|
|
static void
|
|
AddResourceSizeValue(void *ptr, XID id, RESTYPE type, void *cdata)
|
|
{
|
|
ConstructResourceBytesCtx *ctx = cdata;
|
|
if (ctx->status == Success &&
|
|
!ht_find(ctx->visitedResources, &id)) {
|
|
Bool ok = TRUE;
|
|
HashTable ht;
|
|
HtGenericHashSetupRec htSetup = {
|
|
.keySize = sizeof(void*)
|
|
};
|
|
|
|
/* it doesn't matter that we don't undo the work done here
|
|
* immediately. All but ht_init will be undone at the end
|
|
* of the request and there can happen no failure after
|
|
* ht_init, so we don't need to clean it up here in any
|
|
* special way */
|
|
|
|
xXResResourceSizeValue *value =
|
|
AddFragment(&ctx->response, sizeof(xXResResourceSizeValue));
|
|
if (!value) {
|
|
ok = FALSE;
|
|
}
|
|
ok = ok && ht_add(ctx->visitedResources, &id);
|
|
if (ok) {
|
|
ht = ht_create(htSetup.keySize,
|
|
sizeof(xXResResourceSizeSpec*),
|
|
ht_generic_hash, ht_generic_compare,
|
|
&htSetup);
|
|
ok = ok && ht;
|
|
}
|
|
|
|
if (!ok) {
|
|
ctx->status = BadAlloc;
|
|
} else {
|
|
SizeType sizeFunc = GetResourceTypeSizeFunc(type);
|
|
ResourceSizeRec size = { 0, 0, 0 };
|
|
|
|
sizeFunc(ptr, id, &size);
|
|
|
|
value->size.spec.resource = id;
|
|
value->size.spec.type = type;
|
|
value->size.bytes = size.resourceSize;
|
|
value->size.refCount = size.refCnt;
|
|
value->size.useCount = 1;
|
|
value->numCrossReferences = 0;
|
|
|
|
ctx->sizeValue = value;
|
|
ctx->visitedSubResources = ht;
|
|
FindSubResources(ptr, type, AddSubResourceSizeSpec, ctx);
|
|
ctx->visitedSubResources = NULL;
|
|
ctx->sizeValue = NULL;
|
|
|
|
ctx->resultBytes += sizeof(*value);
|
|
++ctx->numSizes;
|
|
|
|
ht_destroy(ht);
|
|
}
|
|
}
|
|
}
|
|
|
|
/** @brief A variant of AddResourceSizeValue that passes the resource type
|
|
through the context object to satisfy the type FindResType
|
|
|
|
@see AddResourceSizeValue
|
|
|
|
@param[in] ptr The resource
|
|
@param[in] id The resource ID
|
|
@param[in/out] cdata The context object that contains the resource type
|
|
*/
|
|
static void
|
|
AddResourceSizeValueWithResType(void *ptr, XID id, void *cdata)
|
|
{
|
|
ConstructResourceBytesCtx *ctx = cdata;
|
|
AddResourceSizeValue(ptr, id, ctx->resType, cdata);
|
|
}
|
|
|
|
/** @brief Adds the information of a resource into the buffer if it matches
|
|
the match condition.
|
|
|
|
@see AddResourceSizeValue
|
|
|
|
@param[in] ptr The resource
|
|
@param[in] id The resource ID
|
|
@param[in] type The resource type
|
|
@param[in/out] cdata The context object as a void pointer to satisfy the
|
|
type FindAllRes
|
|
*/
|
|
static void
|
|
AddResourceSizeValueByResource(void *ptr, XID id, RESTYPE type, void *cdata)
|
|
{
|
|
ConstructResourceBytesCtx *ctx = cdata;
|
|
xXResResourceIdSpec *spec = ctx->curSpec;
|
|
|
|
if ((!spec->type || spec->type == type) &&
|
|
(!spec->resource || spec->resource == id)) {
|
|
AddResourceSizeValue(ptr, id, type, ctx);
|
|
}
|
|
}
|
|
|
|
/** @brief Add all resources of the client into the result buffer
|
|
disregarding all those specifications that specify the
|
|
resource by its ID. Those are handled by
|
|
ConstructResourceBytesByResource
|
|
|
|
@see ConstructResourceBytesByResource
|
|
|
|
@param[in] aboutClient Which client is being considered
|
|
@param[in/out] ctx The context that contains the resource id
|
|
specifications as well as the result buffer
|
|
*/
|
|
static void
|
|
ConstructClientResourceBytes(ClientPtr aboutClient,
|
|
ConstructResourceBytesCtx *ctx)
|
|
{
|
|
int specIdx;
|
|
for (specIdx = 0; specIdx < ctx->numSpecs; ++specIdx) {
|
|
xXResResourceIdSpec* spec = ctx->specs + specIdx;
|
|
if (spec->resource) {
|
|
/* these specs are handled elsewhere */
|
|
} else if (spec->type) {
|
|
ctx->resType = spec->type;
|
|
FindClientResourcesByType(aboutClient, spec->type,
|
|
AddResourceSizeValueWithResType, ctx);
|
|
} else {
|
|
FindAllClientResources(aboutClient, AddResourceSizeValue, ctx);
|
|
}
|
|
}
|
|
}
|
|
|
|
/** @brief Add the sizes of all such resources that can are specified by
|
|
their ID in the resource id specification. The scan can
|
|
by limited to a client with the aboutClient parameter
|
|
|
|
@see ConstructResourceBytesByResource
|
|
|
|
@param[in] aboutClient Which client is being considered. This may be None
|
|
to mean all clients.
|
|
@param[in/out] ctx The context that contains the resource id
|
|
specifications as well as the result buffer. In
|
|
addition this function uses the curSpec field to
|
|
keep a pointer to the current resource id
|
|
specification in it, which can be used by
|
|
AddResourceSizeValueByResource .
|
|
*/
|
|
static void
|
|
ConstructResourceBytesByResource(XID aboutClient, ConstructResourceBytesCtx *ctx)
|
|
{
|
|
int specIdx;
|
|
for (specIdx = 0; specIdx < ctx->numSpecs; ++specIdx) {
|
|
xXResResourceIdSpec *spec = ctx->specs + specIdx;
|
|
if (spec->resource) {
|
|
int cid = CLIENT_ID(spec->resource);
|
|
if (cid < currentMaxClients &&
|
|
(aboutClient == None || cid == aboutClient)) {
|
|
ClientPtr client = clients[cid];
|
|
if (client) {
|
|
ctx->curSpec = spec;
|
|
FindAllClientResources(client,
|
|
AddResourceSizeValueByResource,
|
|
ctx);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/** @brief Build the resource size response for the given client
|
|
(or all if not specified) per the parameters set up
|
|
in the context object.
|
|
|
|
@param[in] aboutClient Which client to consider or None for all clients
|
|
@param[in/out] ctx The context object that contains the request as well
|
|
as the response buffer.
|
|
*/
|
|
static int
|
|
ConstructResourceBytes(XID aboutClient,
|
|
ConstructResourceBytesCtx *ctx)
|
|
{
|
|
if (aboutClient) {
|
|
int clientIdx = CLIENT_ID(aboutClient);
|
|
ClientPtr client = NullClient;
|
|
|
|
if ((clientIdx >= currentMaxClients) || !clients[clientIdx]) {
|
|
ctx->sendClient->errorValue = aboutClient;
|
|
return BadValue;
|
|
}
|
|
|
|
client = clients[clientIdx];
|
|
|
|
ConstructClientResourceBytes(client, ctx);
|
|
ConstructResourceBytesByResource(aboutClient, ctx);
|
|
} else {
|
|
int clientIdx;
|
|
|
|
ConstructClientResourceBytes(NULL, ctx);
|
|
|
|
for (clientIdx = 0; clientIdx < currentMaxClients; ++clientIdx) {
|
|
ClientPtr client = clients[clientIdx];
|
|
|
|
if (client) {
|
|
ConstructClientResourceBytes(client, ctx);
|
|
}
|
|
}
|
|
|
|
ConstructResourceBytesByResource(None, ctx);
|
|
}
|
|
|
|
|
|
return ctx->status;
|
|
}
|
|
|
|
/** @brief Implements the XResQueryResourceBytes of XResProto v1.2 */
|
|
static int
|
|
ProcXResQueryResourceBytes (ClientPtr client)
|
|
{
|
|
REQUEST(xXResQueryResourceBytesReq);
|
|
|
|
int rc;
|
|
ConstructResourceBytesCtx ctx;
|
|
|
|
REQUEST_AT_LEAST_SIZE(xXResQueryResourceBytesReq);
|
|
REQUEST_FIXED_SIZE(xXResQueryResourceBytesReq,
|
|
stuff->numSpecs * sizeof(ctx.specs[0]));
|
|
|
|
if (!InitConstructResourceBytesCtx(&ctx, client,
|
|
stuff->numSpecs,
|
|
(void*) ((char*) stuff +
|
|
sz_xXResQueryResourceBytesReq))) {
|
|
return BadAlloc;
|
|
}
|
|
|
|
rc = ConstructResourceBytes(stuff->client, &ctx);
|
|
|
|
if (rc == Success) {
|
|
xXResQueryResourceBytesReply rep = {
|
|
.type = X_Reply,
|
|
.sequenceNumber = client->sequence,
|
|
.length = bytes_to_int32(ctx.resultBytes),
|
|
.numSizes = ctx.numSizes
|
|
};
|
|
|
|
if (client->swapped) {
|
|
swaps (&rep.sequenceNumber);
|
|
swapl (&rep.length);
|
|
swapl (&rep.numSizes);
|
|
|
|
SwapXResQueryResourceBytes(&ctx.response);
|
|
}
|
|
|
|
WriteToClient(client, sizeof(rep), &rep);
|
|
WriteFragmentsToClient(client, &ctx.response);
|
|
}
|
|
|
|
DestroyConstructResourceBytesCtx(&ctx);
|
|
|
|
return rc;
|
|
}
|
|
|
|
static int
|
|
ProcResDispatch(ClientPtr client)
|
|
{
|
|
REQUEST(xReq);
|
|
switch (stuff->data) {
|
|
case X_XResQueryVersion:
|
|
return ProcXResQueryVersion(client);
|
|
case X_XResQueryClients:
|
|
return ProcXResQueryClients(client);
|
|
case X_XResQueryClientResources:
|
|
return ProcXResQueryClientResources(client);
|
|
case X_XResQueryClientPixmapBytes:
|
|
return ProcXResQueryClientPixmapBytes(client);
|
|
case X_XResQueryClientIds:
|
|
return ProcXResQueryClientIds(client);
|
|
case X_XResQueryResourceBytes:
|
|
return ProcXResQueryResourceBytes(client);
|
|
default: break;
|
|
}
|
|
|
|
return BadRequest;
|
|
}
|
|
|
|
static int
|
|
SProcXResQueryVersion(ClientPtr client)
|
|
{
|
|
REQUEST_SIZE_MATCH(xXResQueryVersionReq);
|
|
return ProcXResQueryVersion(client);
|
|
}
|
|
|
|
static int
|
|
SProcXResQueryClientResources(ClientPtr client)
|
|
{
|
|
REQUEST(xXResQueryClientResourcesReq);
|
|
REQUEST_SIZE_MATCH(xXResQueryClientResourcesReq);
|
|
swapl(&stuff->xid);
|
|
return ProcXResQueryClientResources(client);
|
|
}
|
|
|
|
static int
|
|
SProcXResQueryClientPixmapBytes(ClientPtr client)
|
|
{
|
|
REQUEST(xXResQueryClientPixmapBytesReq);
|
|
REQUEST_SIZE_MATCH(xXResQueryClientPixmapBytesReq);
|
|
swapl(&stuff->xid);
|
|
return ProcXResQueryClientPixmapBytes(client);
|
|
}
|
|
|
|
static int
|
|
SProcXResQueryClientIds (ClientPtr client)
|
|
{
|
|
REQUEST(xXResQueryClientIdsReq);
|
|
|
|
REQUEST_AT_LEAST_SIZE (xXResQueryClientIdsReq);
|
|
swapl(&stuff->numSpecs);
|
|
return ProcXResQueryClientIds(client);
|
|
}
|
|
|
|
/** @brief Implements the XResQueryResourceBytes of XResProto v1.2.
|
|
This variant byteswaps request contents before issuing the
|
|
rest of the work to ProcXResQueryResourceBytes */
|
|
static int
|
|
SProcXResQueryResourceBytes (ClientPtr client)
|
|
{
|
|
REQUEST(xXResQueryResourceBytesReq);
|
|
int c;
|
|
xXResResourceIdSpec *specs = (void*) ((char*) stuff + sizeof(*stuff));
|
|
|
|
swapl(&stuff->numSpecs);
|
|
REQUEST_AT_LEAST_SIZE(xXResQueryResourceBytesReq);
|
|
REQUEST_FIXED_SIZE(xXResQueryResourceBytesReq,
|
|
stuff->numSpecs * sizeof(specs[0]));
|
|
|
|
for (c = 0; c < stuff->numSpecs; ++c) {
|
|
SwapXResResourceIdSpec(specs + c);
|
|
}
|
|
|
|
return ProcXResQueryResourceBytes(client);
|
|
}
|
|
|
|
static int
|
|
SProcResDispatch (ClientPtr client)
|
|
{
|
|
REQUEST(xReq);
|
|
swaps(&stuff->length);
|
|
|
|
switch (stuff->data) {
|
|
case X_XResQueryVersion:
|
|
return SProcXResQueryVersion(client);
|
|
case X_XResQueryClients: /* nothing to swap */
|
|
return ProcXResQueryClients(client);
|
|
case X_XResQueryClientResources:
|
|
return SProcXResQueryClientResources(client);
|
|
case X_XResQueryClientPixmapBytes:
|
|
return SProcXResQueryClientPixmapBytes(client);
|
|
case X_XResQueryClientIds:
|
|
return SProcXResQueryClientIds(client);
|
|
case X_XResQueryResourceBytes:
|
|
return SProcXResQueryResourceBytes(client);
|
|
default: break;
|
|
}
|
|
|
|
return BadRequest;
|
|
}
|
|
|
|
void
|
|
ResExtensionInit(void)
|
|
{
|
|
(void) AddExtension(XRES_NAME, 0, 0,
|
|
ProcResDispatch, SProcResDispatch,
|
|
NULL, StandardMinorOpcode);
|
|
}
|