/* 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=NULL; 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=NULL, *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) { REQUEST(xXResQueryVersionReq); xXResQueryVersionReply rep; REQUEST_SIZE_MATCH(xXResQueryVersionReq); rep.type = X_Reply; rep.sequenceNumber = client->sequence; rep.length = 0; rep.server_major = SERVER_XRES_MAJOR_VERSION; rep.server_minor = SERVER_XRES_MINOR_VERSION; 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.type = X_Reply; rep.sequenceNumber = client->sequence; rep.length = bytes_to_int32(num_clients * sz_xXResClient); rep.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(pointer value, XID id, RESTYPE type, pointer 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.type = X_Reply; rep.sequenceNumber = client->sequence; rep.length = bytes_to_int32(num_types * sz_xXResType); rep.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; int bytesPerPixel; bytesPerPixel = pix->drawable.bitsPerPixel >> 3; 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(pointer value, XID id, RESTYPE type, pointer cdata) { SizeType sizeFunc = GetResourceTypeSizeFunc(type); ResourceSizeRec size = { 0, 0, 0 }; unsigned long *bytes = cdata; sizeFunc(value, id, &size); *bytes += size.pixmapRefSize; } static void ResFindPixmaps(pointer value, XID id, pointer cdata) { unsigned long *bytes = (unsigned long *) cdata; PixmapPtr pix = (PixmapPtr) value; *bytes += ResGetApproxPixmapBytes(pix); } static void ResFindWindowPixmaps(pointer value, XID id, pointer 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(pointer value, XID id, pointer 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(pointer value, XID id, pointer cdata) { #ifdef RENDER ResFindResourcePixmaps(value, id, PictureType, cdata); #endif } static void ResFindCompositeClientWindowPixmaps (pointer value, XID id, pointer 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, (pointer) (&bytes)); /* * Make sure win background pixmaps also held to account. */ FindClientResourcesByType(clients[clientID], RT_WINDOW, ResFindWindowPixmaps, (pointer) (&bytes)); /* * GC Tile & Stipple pixmaps too. */ FindClientResourcesByType(clients[clientID], RT_GC, ResFindGCPixmaps, (pointer) (&bytes)); #ifdef RENDER /* Render extension picture pixmaps. */ FindClientResourcesByType(clients[clientID], PictureType, ResFindPicturePixmaps, (pointer)(&bytes)); #endif #ifdef COMPOSITE /* Composite extension client window pixmaps. */ FindClientResourcesByType(clients[clientID], CompositeClientWindowType, ResFindCompositeClientWindowPixmaps, (pointer)(&bytes)); #endif rep.type = X_Reply; rep.sequenceNumber = client->sequence; rep.length = 0; rep.bytes = bytes; #ifdef _XSERVER64 rep.bytes_overflow = bytes >> 32; #else rep.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; rep.type = X_Reply; rep.sequenceNumber = client->sequence; rep.length = bytes_to_int32(ctx.resultBytes); rep.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(pointer value, XID id, RESTYPE type, pointer 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(pointer ptr, XID id, RESTYPE type, pointer 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(pointer ptr, XID id, pointer 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(pointer ptr, XID id, RESTYPE type, pointer 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; rep.type = X_Reply; rep.sequenceNumber = client->sequence; rep.length = bytes_to_int32(ctx.resultBytes); rep.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(xXResQueryVersionReq); 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); }