Files
xserver/dix/request_priv.h
Enrico Weigelt, metux IT consult 78b44589a1 include: misc.h: unexport various bit/byte counting inline functions
Those aren't used by any external drivers, so no need to keep
them in public header.

Signed-off-by: Enrico Weigelt, metux IT consult <info@metux.net>
2026-02-11 16:31:28 +01:00

162 lines
5.5 KiB
C

/* SPDX-License-Identifier: MIT OR X11
*
* Copyright © 2024 Enrico Weigelt, metux IT consult <info@metux.net>
*/
#ifndef _XSERVER_DIX_REQUEST_PRIV_H
#define _XSERVER_DIX_REQUEST_PRIV_H
#include <X11/Xproto.h>
#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 */
#include "os/osdep.h"
/*
* @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 */