/* SPDX-License-Identifier: MIT OR X11 * * Copyright © 2024 Enrico Weigelt, metux IT consult */ #ifndef _XSERVER_DIX_REQUEST_PRIV_H #define _XSERVER_DIX_REQUEST_PRIV_H #include #include "dix/rpcbuf_priv.h" /* x_rpcbuf_t */ #include "include/dix.h" #include "include/dixstruct.h" #include "include/misc.h" /* bytes_to_int32 */ #include "include/os.h" /* WriteToClient */ /* * @brief write rpc buffer to client and then clear it * * @param pClient the client to write buffer to * @param rpcbuf the buffer whose contents will be written * @return the result of WriteToClient() call */ static inline ssize_t WriteRpcbufToClient(ClientPtr pClient, x_rpcbuf_t *rpcbuf) { /* explicitly casting between (s)size_t and int - should be safe, since payloads are always small enough to easily fit into int. */ ssize_t ret = WriteToClient(pClient, (int)rpcbuf->wpos, rpcbuf->buffer); x_rpcbuf_clear(rpcbuf); return ret; } /* compute the amount of extra units a reply header needs. * * all reply header structs are at least the size of xGenericReply * we have to count how many units the header is bigger than xGenericReply * */ #define X_REPLY_HEADER_UNITS(hdrtype) \ (bytes_to_int32((sizeof(hdrtype) - sizeof(xGenericReply)))) static inline int __write_reply_hdr_and_rpcbuf( ClientPtr pClient, void *hdrData, size_t hdrLen, x_rpcbuf_t *rpcbuf) { if (rpcbuf->error) return BadAlloc; xGenericReply *reply = hdrData; reply->type = X_Reply; reply->length = (bytes_to_int32(hdrLen - sizeof(xGenericReply))) + x_rpcbuf_wsize_units(rpcbuf); reply->sequenceNumber = (CARD16)pClient->sequence; /* shouldn't go above 64k */ if (pClient->swapped) { swaps(&reply->sequenceNumber); swapl(&reply->length); } WriteToClient(pClient, (int)hdrLen, hdrData); WriteRpcbufToClient(pClient, rpcbuf); return Success; } static inline int __write_reply_hdr_simple( ClientPtr pClient, void *hdrData, size_t hdrLen) { xGenericReply *reply = hdrData; reply->type = X_Reply; reply->length = (bytes_to_int32(hdrLen - sizeof(xGenericReply))); reply->sequenceNumber = (CARD16)pClient->sequence; /* shouldn't go above 64k */ if (pClient->swapped) { swaps(&reply->sequenceNumber); swapl(&reply->length); } WriteToClient(pClient, (int)hdrLen, hdrData); return Success; } /* * send reply with header struct (not pointer!) along with rpcbuf payload * * @param client pointer to the client (ClientPtr) * @param hdrstruct the header struct (not pointer, the struct itself!) * @param rpcbuf the rpcbuf to send (not pointer, the struct itself!) * return X11 result code */ #define X_SEND_REPLY_WITH_RPCBUF(client, hdrstruct, rpcbuf) \ __write_reply_hdr_and_rpcbuf(client, &(hdrstruct), sizeof(hdrstruct), &(rpcbuf)); /* * send reply with header struct (not pointer!) without any payload * * @param client pointer to the client (ClientPtr) * @param hdrstruct the header struct (not pointer, the struct itself!) * @return X11 result code (=Success) */ #define X_SEND_REPLY_SIMPLE(client, hdrstruct) \ __write_reply_hdr_simple(client, &(hdrstruct), sizeof(hdrstruct)); /* * macros for request handlers * * these are handling request packet checking and swapping of multi-byte * values, if necessary. (length field is already swapped earlier) */ /* declare request struct and check size */ #define X_REQUEST_HEAD_STRUCT(type) \ REQUEST(type); \ if (stuff == NULL) return (BadLength); \ REQUEST_SIZE_MATCH(type); /* declare request struct and check size (at least as big) */ #define X_REQUEST_HEAD_AT_LEAST(type) \ REQUEST(type); \ if (stuff == NULL) return (BadLength); \ REQUEST_AT_LEAST_SIZE(type); \ /* declare request struct, do NOT check size !*/ #define X_REQUEST_HEAD_NO_CHECK(type) \ REQUEST(type); \ if (stuff == NULL) return (BadLength); \ /* swap a CARD16 request struct field if necessary */ #define X_REQUEST_FIELD_CARD16(field) \ do { if (client->swapped) swaps(&stuff->field); } while (0) /* swap a CARD32 request struct field if necessary */ #define X_REQUEST_FIELD_CARD32(field) \ do { if (client->swapped) swapl(&stuff->field); } while (0) /* swap a CARD64 request struct field if necessary */ #define X_REQUEST_FIELD_CARD64(field) \ do { if (client->swapped) swapll(&stuff->field); } while (0) /* swap CARD16 rest of request (after the struct) */ #define X_REQUEST_REST_CARD16() \ do { if (client->swapped) SwapRestS(stuff); } while (0) /* swap CARD32 rest of request (after the struct) */ #define X_REQUEST_REST_CARD32() \ do { if (client->swapped) SwapRestL(stuff); } while (0) /* swap CARD16 rest of request (after the struct) - check fixed count */ #define X_REQUEST_REST_COUNT_CARD16(count) \ REQUEST_FIXED_SIZE(*stuff, count * sizeof(CARD16)); \ CARD16 *request_rest = (CARD16 *) (&stuff[1]); \ do { if (client->swapped) SwapShorts((signed short*)request_rest, count); } while (0) /* swap CARD32 rest of request (after the struct) - check fixed count */ #define X_REQUEST_REST_COUNT_CARD32(count) \ REQUEST_FIXED_SIZE(*stuff, count * sizeof(CARD32)); \ CARD32 *request_rest = (CARD32 *) (&stuff[1]); \ do { if (client->swapped) SwapLongs(request_rest, count); } while (0) \ #endif /* _XSERVER_DIX_REQUEST_PRIV_H */