Blame libfreerdp/utils/ringbuffer.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2014 Thincast Technologies GmbH
Packit 1fb8d4
 * Copyright 2014 Hardening <contact@hardening-consulting.com>
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 * http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/utils/ringbuffer.h>
Packit 1fb8d4
Packit 1fb8d4
#include <stdlib.h>
Packit 1fb8d4
#include <string.h>
Packit 1fb8d4
#include <assert.h>
Packit 1fb8d4
Packit 1fb8d4
#include <winpr/crt.h>
Packit 1fb8d4
#include <freerdp/log.h>
Packit 1fb8d4
Packit 1fb8d4
#define TAG FREERDP_TAG("utils.ringbuffer")
Packit 1fb8d4
Packit 1fb8d4
#ifdef WITH_DEBUG_RINGBUFFER
Packit 1fb8d4
#define DEBUG_RINGBUFFER(...) WLog_DBG(TAG, __VA_ARGS__)
Packit 1fb8d4
#else
Packit 1fb8d4
#define DEBUG_RINGBUFFER(...) do { } while (0)
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
BOOL ringbuffer_init(RingBuffer* rb, size_t initialSize)
Packit 1fb8d4
{
Packit 1fb8d4
	rb->buffer = malloc(initialSize);
Packit 1fb8d4
Packit 1fb8d4
	if (!rb->buffer)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	rb->readPtr = rb->writePtr = 0;
Packit 1fb8d4
	rb->initialSize = rb->size = rb->freeSize = initialSize;
Packit 1fb8d4
	DEBUG_RINGBUFFER("ringbuffer_init(%p)", (void*) rb);
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
size_t ringbuffer_used(const RingBuffer* rb)
Packit 1fb8d4
{
Packit 1fb8d4
	return rb->size - rb->freeSize;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
size_t ringbuffer_capacity(const RingBuffer* rb)
Packit 1fb8d4
{
Packit 1fb8d4
	return rb->size;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void ringbuffer_destroy(RingBuffer* rb)
Packit 1fb8d4
{
Packit 1fb8d4
	DEBUG_RINGBUFFER("ringbuffer_destroy(%p)", (void*) rb);
Packit 1fb8d4
	free(rb->buffer);
Packit 1fb8d4
	rb->buffer = NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL ringbuffer_realloc(RingBuffer* rb, size_t targetSize)
Packit 1fb8d4
{
Packit 1fb8d4
	BYTE* newData;
Packit 1fb8d4
	DEBUG_RINGBUFFER("ringbuffer_realloc(%p): targetSize: %"PRIdz"", (void*) rb, targetSize);
Packit 1fb8d4
Packit 1fb8d4
	if (rb->writePtr == rb->readPtr)
Packit 1fb8d4
	{
Packit 1fb8d4
		/* when no size is used we can realloc() and set the heads at the
Packit 1fb8d4
		 * beginning of the buffer
Packit 1fb8d4
		 */
Packit 1fb8d4
		newData = (BYTE*) realloc(rb->buffer, targetSize);
Packit 1fb8d4
Packit 1fb8d4
		if (!newData)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		rb->readPtr = rb->writePtr = 0;
Packit 1fb8d4
		rb->buffer = newData;
Packit 1fb8d4
	}
Packit 1fb8d4
	else if ((rb->writePtr >= rb->readPtr) && (rb->writePtr < targetSize))
Packit 1fb8d4
	{
Packit 1fb8d4
		/* we reallocate only if we're in that case, realloc don't touch read
Packit 1fb8d4
		 * and write heads
Packit 1fb8d4
		 *
Packit 1fb8d4
		 *        readPtr              writePtr
Packit 1fb8d4
		 *              |              |
Packit 1fb8d4
		 *              v              v
Packit 1fb8d4
		 * [............|XXXXXXXXXXXXXX|..........]
Packit 1fb8d4
		 */
Packit 1fb8d4
		newData = (BYTE*) realloc(rb->buffer, targetSize);
Packit 1fb8d4
Packit 1fb8d4
		if (!newData)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		rb->buffer = newData;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		/* in case of malloc the read head is moved at the beginning of the new buffer
Packit 1fb8d4
		 * and the write head is set accordingly
Packit 1fb8d4
		 */
Packit 1fb8d4
		newData = (BYTE*) malloc(targetSize);
Packit 1fb8d4
Packit 1fb8d4
		if (!newData)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
Packit 1fb8d4
		if (rb->readPtr < rb->writePtr)
Packit 1fb8d4
		{
Packit 1fb8d4
			/*        readPtr              writePtr
Packit 1fb8d4
			 *              |              |
Packit 1fb8d4
			 *              v              v
Packit 1fb8d4
			 * [............|XXXXXXXXXXXXXX|..........]
Packit 1fb8d4
			 */
Packit 1fb8d4
			memcpy(newData, rb->buffer + rb->readPtr, ringbuffer_used(rb));
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			/*        writePtr             readPtr
Packit 1fb8d4
			 *              |              |
Packit 1fb8d4
			 *              v              v
Packit 1fb8d4
			 * [XXXXXXXXXXXX|..............|XXXXXXXXXX]
Packit 1fb8d4
			 */
Packit 1fb8d4
			BYTE* dst = newData;
Packit 1fb8d4
			memcpy(dst, rb->buffer + rb->readPtr, rb->size - rb->readPtr);
Packit 1fb8d4
			dst += (rb->size - rb->readPtr);
Packit 1fb8d4
Packit 1fb8d4
			if (rb->writePtr)
Packit 1fb8d4
				memcpy(dst, rb->buffer, rb->writePtr);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		rb->writePtr = rb->size - rb->freeSize;
Packit 1fb8d4
		rb->readPtr = 0;
Packit 1fb8d4
		free(rb->buffer);
Packit 1fb8d4
		rb->buffer = newData;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	rb->freeSize += (targetSize - rb->size);
Packit 1fb8d4
	rb->size = targetSize;
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 *
Packit 1fb8d4
 * @param rb
Packit 1fb8d4
 * @param ptr
Packit 1fb8d4
 * @param sz
Packit 1fb8d4
 * @return
Packit 1fb8d4
 */
Packit 1fb8d4
BOOL ringbuffer_write(RingBuffer* rb, const BYTE* ptr, size_t sz)
Packit 1fb8d4
{
Packit 1fb8d4
	size_t toWrite;
Packit 1fb8d4
	size_t remaining;
Packit 1fb8d4
	DEBUG_RINGBUFFER("ringbuffer_write(%p): sz: %"PRIdz"", (void*) rb, sz);
Packit 1fb8d4
Packit 1fb8d4
	if ((rb->freeSize <= sz) && !ringbuffer_realloc(rb, rb->size + sz))
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	/*  the write could be split in two
Packit 1fb8d4
	 *    readHead        writeHead
Packit 1fb8d4
	 *      |               |
Packit 1fb8d4
	 *      v               v
Packit 1fb8d4
	 * [    ################        ]
Packit 1fb8d4
	 */
Packit 1fb8d4
	toWrite = sz;
Packit 1fb8d4
	remaining = sz;
Packit 1fb8d4
Packit 1fb8d4
	if (rb->size - rb->writePtr < sz)
Packit 1fb8d4
		toWrite = rb->size - rb->writePtr;
Packit 1fb8d4
Packit 1fb8d4
	if (toWrite)
Packit 1fb8d4
	{
Packit 1fb8d4
		memcpy(rb->buffer + rb->writePtr, ptr, toWrite);
Packit 1fb8d4
		remaining -= toWrite;
Packit 1fb8d4
		ptr += toWrite;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (remaining)
Packit 1fb8d4
		memcpy(rb->buffer, ptr, remaining);
Packit 1fb8d4
Packit 1fb8d4
	rb->writePtr = (rb->writePtr + sz) % rb->size;
Packit 1fb8d4
	rb->freeSize -= sz;
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BYTE* ringbuffer_ensure_linear_write(RingBuffer* rb, size_t sz)
Packit 1fb8d4
{
Packit 1fb8d4
	DEBUG_RINGBUFFER("ringbuffer_ensure_linear_write(%p): sz: %"PRIdz"", (void*) rb, sz);
Packit 1fb8d4
Packit 1fb8d4
	if (rb->freeSize < sz)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!ringbuffer_realloc(rb, rb->size + sz - rb->freeSize + 32))
Packit 1fb8d4
			return NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (rb->writePtr == rb->readPtr)
Packit 1fb8d4
	{
Packit 1fb8d4
		rb->writePtr = rb->readPtr = 0;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (rb->writePtr + sz < rb->size)
Packit 1fb8d4
		return rb->buffer + rb->writePtr;
Packit 1fb8d4
Packit 1fb8d4
	/*
Packit 1fb8d4
	 * to add:             .......
Packit 1fb8d4
	 * [          XXXXXXXXX  ]
Packit 1fb8d4
	 *
Packit 1fb8d4
	 * result:
Packit 1fb8d4
	 * [XXXXXXXXX.......     ]
Packit 1fb8d4
	 */
Packit 1fb8d4
	memmove(rb->buffer, rb->buffer + rb->readPtr, rb->writePtr - rb->readPtr);
Packit 1fb8d4
	rb->readPtr = 0;
Packit 1fb8d4
	rb->writePtr = rb->size - rb->freeSize;
Packit 1fb8d4
	return rb->buffer + rb->writePtr;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL ringbuffer_commit_written_bytes(RingBuffer* rb, size_t sz)
Packit 1fb8d4
{
Packit 1fb8d4
	DEBUG_RINGBUFFER("ringbuffer_commit_written_bytes(%p): sz: %"PRIdz"", (void*) rb, sz);
Packit 1fb8d4
Packit 1fb8d4
	if (sz < 1)
Packit 1fb8d4
		return TRUE;
Packit 1fb8d4
Packit 1fb8d4
	if (rb->writePtr + sz > rb->size)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	rb->writePtr = (rb->writePtr + sz) % rb->size;
Packit 1fb8d4
	rb->freeSize -= sz;
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
int ringbuffer_peek(const RingBuffer* rb, DataChunk chunks[2], size_t sz)
Packit 1fb8d4
{
Packit 1fb8d4
	size_t remaining = sz;
Packit 1fb8d4
	size_t toRead;
Packit 1fb8d4
	int chunkIndex = 0;
Packit 1fb8d4
	int status = 0;
Packit 1fb8d4
	DEBUG_RINGBUFFER("ringbuffer_peek(%p): sz: %"PRIdz"", (void*) rb, sz);
Packit 1fb8d4
Packit 1fb8d4
	if (sz < 1)
Packit 1fb8d4
		return 0;
Packit 1fb8d4
Packit 1fb8d4
	if ((rb->size - rb->freeSize) < sz)
Packit 1fb8d4
		remaining = rb->size - rb->freeSize;
Packit 1fb8d4
Packit 1fb8d4
	toRead = remaining;
Packit 1fb8d4
Packit 1fb8d4
	if ((rb->readPtr + remaining) > rb->size)
Packit 1fb8d4
		toRead = rb->size - rb->readPtr;
Packit 1fb8d4
Packit 1fb8d4
	if (toRead)
Packit 1fb8d4
	{
Packit 1fb8d4
		chunks[0].data = rb->buffer + rb->readPtr;
Packit 1fb8d4
		chunks[0].size = toRead;
Packit 1fb8d4
		remaining -= toRead;
Packit 1fb8d4
		chunkIndex++;
Packit 1fb8d4
		status++;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (remaining)
Packit 1fb8d4
	{
Packit 1fb8d4
		chunks[chunkIndex].data = rb->buffer;
Packit 1fb8d4
		chunks[chunkIndex].size = remaining;
Packit 1fb8d4
		status++;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void ringbuffer_commit_read_bytes(RingBuffer* rb, size_t sz)
Packit 1fb8d4
{
Packit 1fb8d4
	DEBUG_RINGBUFFER("ringbuffer_commit_read_bytes(%p): sz: %"PRIdz"", (void*) rb, sz);
Packit 1fb8d4
Packit 1fb8d4
	if (sz < 1)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	assert(rb->size - rb->freeSize >= sz);
Packit 1fb8d4
	rb->readPtr = (rb->readPtr + sz) % rb->size;
Packit 1fb8d4
	rb->freeSize += sz;
Packit 1fb8d4
Packit 1fb8d4
	/* when we reach a reasonable free size, we can go back to the original size */
Packit 1fb8d4
	if ((rb->size != rb->initialSize)
Packit 1fb8d4
	    && (ringbuffer_used(rb) < rb->initialSize / 2))
Packit 1fb8d4
		ringbuffer_realloc(rb, rb->initialSize);
Packit 1fb8d4
}