mirror of
https://github.com/X11Libre/xserver.git
synced 2026-03-24 10:14:52 +00:00
when the err_clear flag is set, the buffer memory will automatically be free()ed when allocation failed. This allows simplificatoin of caller's error pathes: the caller doesn't need to to call x_rpcbuf_clear() anymore and eg. can directly return out. Signed-off-by: Enrico Weigelt, metux IT consult <info@metux.net>
200 lines
4.3 KiB
C
200 lines
4.3 KiB
C
/* SPDX-License-Identifier: MIT OR X11
|
|
*
|
|
* Copyright © 2024 Enrico Weigelt, metux IT consult <info@metux.net>
|
|
*/
|
|
#include <dix-config.h>
|
|
|
|
#include <stddef.h>
|
|
|
|
#include "dix/rpcbuf_priv.h"
|
|
|
|
Bool x_rpcbuf_makeroom(struct x_rpcbuf *rpcbuf, size_t needed)
|
|
{
|
|
/* break out of alreay in error state */
|
|
if (rpcbuf->error)
|
|
return FALSE;
|
|
|
|
/* not allocated yet ? */
|
|
if (!rpcbuf->buffer) {
|
|
if (!(rpcbuf->buffer = calloc(1, XLIBRE_RPCBUF_CHUNK_SIZE)))
|
|
goto err;
|
|
rpcbuf->size = XLIBRE_RPCBUF_CHUNK_SIZE;
|
|
rpcbuf->wpos = 0;
|
|
}
|
|
|
|
/* still enough space */
|
|
if (rpcbuf->size > rpcbuf->wpos + needed)
|
|
return TRUE;
|
|
|
|
const size_t newsize = ((needed / XLIBRE_RPCBUF_CHUNK_SIZE) + 1)
|
|
* XLIBRE_RPCBUF_CHUNK_SIZE;
|
|
|
|
char *newbuf = realloc(rpcbuf->buffer, newsize);
|
|
if (!newbuf)
|
|
goto err;
|
|
|
|
memset(newbuf + rpcbuf->size, 0, newsize - rpcbuf->size);
|
|
rpcbuf->buffer = newbuf;
|
|
rpcbuf->size = newsize;
|
|
|
|
return TRUE;
|
|
|
|
err:
|
|
rpcbuf->error = TRUE;
|
|
if (rpcbuf->err_clear) {
|
|
free(rpcbuf->buffer);
|
|
rpcbuf->buffer = NULL;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
void x_rpcbuf_clear(struct x_rpcbuf *rpcbuf)
|
|
{
|
|
free(rpcbuf->buffer);
|
|
memset(rpcbuf, 0, sizeof(struct x_rpcbuf));
|
|
}
|
|
|
|
void x_rpcbuf_reset(struct x_rpcbuf *rpcbuf)
|
|
{
|
|
/* no need to reset if never been actually written to */
|
|
if ((!rpcbuf->buffer) || (!rpcbuf->size) || (!rpcbuf->wpos))
|
|
return;
|
|
|
|
/* clear memory, but don't free it */
|
|
memset(rpcbuf->buffer, 0, rpcbuf->size);
|
|
rpcbuf->wpos = 0;
|
|
}
|
|
|
|
void *x_rpcbuf_reserve(struct x_rpcbuf *rpcbuf, size_t needed)
|
|
{
|
|
if (!x_rpcbuf_makeroom(rpcbuf, needed))
|
|
return NULL;
|
|
|
|
void *pos = rpcbuf->buffer + rpcbuf->wpos;
|
|
rpcbuf->wpos += needed;
|
|
|
|
return pos;
|
|
}
|
|
|
|
Bool x_rpcbuf_write_string_pad(struct x_rpcbuf *rpcbuf, const char *str)
|
|
{
|
|
if (!str)
|
|
return TRUE;
|
|
|
|
size_t slen = strlen(str);
|
|
if (!slen)
|
|
return TRUE;
|
|
|
|
char *reserved = x_rpcbuf_reserve(rpcbuf, pad_to_int32(slen));
|
|
if (!reserved)
|
|
return FALSE;
|
|
|
|
memcpy(reserved, str, slen);
|
|
return TRUE;
|
|
}
|
|
|
|
Bool x_rpcbuf_write_CARD8(struct x_rpcbuf *rpcbuf, CARD8 value)
|
|
{
|
|
CARD8 *reserved = x_rpcbuf_reserve(rpcbuf, sizeof(value));
|
|
if (!reserved)
|
|
return FALSE;
|
|
|
|
*reserved = value;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
Bool x_rpcbuf_write_CARD16(struct x_rpcbuf *rpcbuf, CARD16 value)
|
|
{
|
|
CARD16 *reserved = x_rpcbuf_reserve(rpcbuf, sizeof(value));
|
|
if (!reserved)
|
|
return FALSE;
|
|
|
|
*reserved = value;
|
|
|
|
if (rpcbuf->swapped)
|
|
swaps(reserved);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
Bool x_rpcbuf_write_CARD32(struct x_rpcbuf *rpcbuf, CARD32 value)
|
|
{
|
|
CARD32 *reserved = x_rpcbuf_reserve(rpcbuf, sizeof(value));
|
|
if (!reserved)
|
|
return FALSE;
|
|
|
|
*reserved = value;
|
|
|
|
if (rpcbuf->swapped)
|
|
swapl(reserved);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
Bool x_rpcbuf_write_CARD8s(struct x_rpcbuf *rpcbuf, const CARD8 *values,
|
|
size_t count)
|
|
{
|
|
if ((!values) || (!count))
|
|
return TRUE;
|
|
|
|
INT16 *reserved = x_rpcbuf_reserve(rpcbuf, count);
|
|
if (!reserved)
|
|
return FALSE;
|
|
|
|
memcpy(reserved, values, count);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
Bool x_rpcbuf_write_CARD16s(struct x_rpcbuf *rpcbuf, const CARD16 *values,
|
|
size_t count)
|
|
{
|
|
if ((!values) || (!count))
|
|
return TRUE;
|
|
|
|
INT16 *reserved = x_rpcbuf_reserve(rpcbuf, sizeof(CARD16) * count);
|
|
if (!reserved)
|
|
return FALSE;
|
|
|
|
memcpy(reserved, values, sizeof(CARD16) * count);
|
|
|
|
if (rpcbuf->swapped)
|
|
SwapShorts(reserved, count);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
Bool x_rpcbuf_write_CARD32s(struct x_rpcbuf *rpcbuf, const CARD32 *values,
|
|
size_t count)
|
|
{
|
|
if ((!values) || (!count))
|
|
return TRUE;
|
|
|
|
CARD32 *reserved = x_rpcbuf_reserve(rpcbuf, sizeof(CARD32) * count);
|
|
if (!reserved)
|
|
return FALSE;
|
|
|
|
memcpy(reserved, values, sizeof(CARD32) * count);
|
|
|
|
if (rpcbuf->swapped)
|
|
SwapLongs(reserved, count);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
Bool x_rpcbuf_write_binary_pad(struct x_rpcbuf *rpcbuf, const void *values,
|
|
size_t size)
|
|
{
|
|
if ((!values) || (!size))
|
|
return TRUE;
|
|
|
|
void *reserved = x_rpcbuf_reserve(rpcbuf, pad_to_int32(size));
|
|
if (!reserved)
|
|
return FALSE;
|
|
|
|
memcpy(reserved, values, size);
|
|
|
|
return TRUE;
|
|
}
|