Blame winpr/libwinpr/utils/collections/StreamPool.c

Packit Service fa4841
/**
Packit Service fa4841
 * WinPR: Windows Portable Runtime
Packit Service fa4841
 * Object Pool
Packit Service fa4841
 *
Packit Service fa4841
 * Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit Service fa4841
 *
Packit Service fa4841
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service fa4841
 * you may not use this file except in compliance with the License.
Packit Service fa4841
 * You may obtain a copy of the License at
Packit Service fa4841
 *
Packit Service fa4841
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service fa4841
 *
Packit Service fa4841
 * Unless required by applicable law or agreed to in writing, software
Packit Service fa4841
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service fa4841
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service fa4841
 * See the License for the specific language governing permissions and
Packit Service fa4841
 * limitations under the License.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
#ifdef HAVE_CONFIG_H
Packit Service fa4841
#include "config.h"
Packit Service fa4841
#endif
Packit Service fa4841
Packit Service fa4841
#include <winpr/crt.h>
Packit Service fa4841
Packit Service fa4841
#include <winpr/collections.h>
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Methods
Packit Service fa4841
 */
Packit Service fa4841
Packit Service b1ea74
static void StreamPool_ShiftUsed(wStreamPool* pool, int index, int count)
Packit Service fa4841
{
Packit Service fa4841
	if (count > 0)
Packit Service fa4841
	{
Packit Service fa4841
		if (pool->uSize + count > pool->uCapacity)
Packit Service fa4841
		{
Packit Service fa4841
			int new_cap;
Packit Service b1ea74
			wStream** new_arr;
Packit Service fa4841
Packit Service fa4841
			new_cap = pool->uCapacity * 2;
Packit Service b1ea74
			new_arr = (wStream**)realloc(pool->uArray, sizeof(wStream*) * new_cap);
Packit Service fa4841
			if (!new_arr)
Packit Service fa4841
				return;
Packit Service fa4841
			pool->uCapacity = new_cap;
Packit Service fa4841
			pool->uArray = new_arr;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service b1ea74
		MoveMemory(&pool->uArray[index + count], &pool->uArray[index],
Packit Service b1ea74
		           (pool->uSize - index) * sizeof(wStream*));
Packit Service fa4841
		pool->uSize += count;
Packit Service fa4841
	}
Packit Service fa4841
	else if (count < 0)
Packit Service fa4841
	{
Packit Service fa4841
		if (pool->uSize - index + count > 0)
Packit Service fa4841
		{
Packit Service fa4841
			MoveMemory(&pool->uArray[index], &pool->uArray[index - count],
Packit Service b1ea74
			           (pool->uSize - index + count) * sizeof(wStream*));
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		pool->uSize += count;
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Adds a used stream to the pool.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service b1ea74
static void StreamPool_AddUsed(wStreamPool* pool, wStream* s)
Packit Service fa4841
{
Packit Service fa4841
	if ((pool->uSize + 1) >= pool->uCapacity)
Packit Service fa4841
	{
Packit Service fa4841
		int new_cap;
Packit Service b1ea74
		wStream** new_arr;
Packit Service fa4841
Packit Service fa4841
		new_cap = pool->uCapacity * 2;
Packit Service b1ea74
		new_arr = (wStream**)realloc(pool->uArray, sizeof(wStream*) * new_cap);
Packit Service fa4841
		if (!new_arr)
Packit Service fa4841
			return;
Packit Service fa4841
		pool->uCapacity = new_cap;
Packit Service fa4841
		pool->uArray = new_arr;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	pool->uArray[(pool->uSize)++] = s;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Removes a used stream from the pool.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service b1ea74
static void StreamPool_RemoveUsed(wStreamPool* pool, wStream* s)
Packit Service fa4841
{
Packit Service fa4841
	int index;
Packit Service fa4841
	BOOL found = FALSE;
Packit Service fa4841
Packit Service fa4841
	for (index = 0; index < pool->uSize; index++)
Packit Service fa4841
	{
Packit Service fa4841
		if (pool->uArray[index] == s)
Packit Service fa4841
		{
Packit Service fa4841
			found = TRUE;
Packit Service fa4841
			break;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (found)
Packit Service fa4841
		StreamPool_ShiftUsed(pool, index, -1);
Packit Service fa4841
}
Packit Service fa4841
Packit Service b1ea74
static void StreamPool_ShiftAvailable(wStreamPool* pool, int index, int count)
Packit Service fa4841
{
Packit Service fa4841
	if (count > 0)
Packit Service fa4841
	{
Packit Service fa4841
		if (pool->aSize + count > pool->aCapacity)
Packit Service fa4841
		{
Packit Service fa4841
			int new_cap;
Packit Service b1ea74
			wStream** new_arr;
Packit Service fa4841
Packit Service fa4841
			new_cap = pool->aCapacity * 2;
Packit Service b1ea74
			new_arr = (wStream**)realloc(pool->aArray, sizeof(wStream*) * new_cap);
Packit Service fa4841
			if (!new_arr)
Packit Service fa4841
				return;
Packit Service fa4841
			pool->aCapacity = new_cap;
Packit Service fa4841
			pool->aArray = new_arr;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service b1ea74
		MoveMemory(&pool->aArray[index + count], &pool->aArray[index],
Packit Service b1ea74
		           (pool->aSize - index) * sizeof(wStream*));
Packit Service fa4841
		pool->aSize += count;
Packit Service fa4841
	}
Packit Service fa4841
	else if (count < 0)
Packit Service fa4841
	{
Packit Service fa4841
		if (pool->aSize - index + count > 0)
Packit Service fa4841
		{
Packit Service fa4841
			MoveMemory(&pool->aArray[index], &pool->aArray[index - count],
Packit Service b1ea74
			           (pool->aSize - index + count) * sizeof(wStream*));
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		pool->aSize += count;
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Gets a stream from the pool.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
wStream* StreamPool_Take(wStreamPool* pool, size_t size)
Packit Service fa4841
{
Packit Service fa4841
	int index;
Packit Service fa4841
	int foundIndex;
Packit Service fa4841
	wStream* s = NULL;
Packit Service fa4841
Packit Service fa4841
	if (pool->synchronized)
Packit Service fa4841
		EnterCriticalSection(&pool->lock);
Packit Service fa4841
Packit Service fa4841
	if (size == 0)
Packit Service fa4841
		size = pool->defaultSize;
Packit Service fa4841
Packit Service fa4841
	foundIndex = -1;
Packit Service fa4841
Packit Service fa4841
	for (index = 0; index < pool->aSize; index++)
Packit Service fa4841
	{
Packit Service fa4841
		s = pool->aArray[index];
Packit Service fa4841
Packit Service fa4841
		if (Stream_Capacity(s) >= size)
Packit Service fa4841
		{
Packit Service fa4841
			foundIndex = index;
Packit Service fa4841
			break;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (foundIndex < 0)
Packit Service fa4841
	{
Packit Service fa4841
		s = Stream_New(NULL, size);
Packit Service fa4841
		if (!s)
Packit Service fa4841
			goto out_fail;
Packit Service fa4841
	}
Packit Service fa4841
	else
Packit Service fa4841
	{
Packit Service fa4841
		Stream_SetPosition(s, 0);
Packit Service fa4841
		Stream_SetLength(s, Stream_Capacity(s));
Packit Service fa4841
		StreamPool_ShiftAvailable(pool, foundIndex, -1);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (s)
Packit Service fa4841
	{
Packit Service fa4841
		s->pool = pool;
Packit Service fa4841
		s->count = 1;
Packit Service fa4841
		StreamPool_AddUsed(pool, s);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
out_fail:
Packit Service fa4841
	if (pool->synchronized)
Packit Service fa4841
		LeaveCriticalSection(&pool->lock);
Packit Service fa4841
Packit Service fa4841
	return s;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Returns an object to the pool.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
void StreamPool_Return(wStreamPool* pool, wStream* s)
Packit Service fa4841
{
Packit Service fa4841
	if (pool->synchronized)
Packit Service fa4841
		EnterCriticalSection(&pool->lock);
Packit Service fa4841
Packit Service fa4841
	if ((pool->aSize + 1) >= pool->aCapacity)
Packit Service fa4841
	{
Packit Service fa4841
		int new_cap;
Packit Service b1ea74
		wStream** new_arr;
Packit Service fa4841
Packit Service fa4841
		new_cap = pool->aCapacity * 2;
Packit Service b1ea74
		new_arr = (wStream**)realloc(pool->aArray, sizeof(wStream*) * new_cap);
Packit Service fa4841
		if (!new_arr)
Packit Service fa4841
			goto out_fail;
Packit Service fa4841
		pool->aCapacity = new_cap;
Packit Service fa4841
		pool->aArray = new_arr;
Packit Service fa4841
	}
Packit Service fa4841
	else if ((pool->aSize + 1) * 3 < pool->aCapacity)
Packit Service fa4841
	{
Packit Service fa4841
		int new_cap;
Packit Service b1ea74
		wStream** new_arr;
Packit Service fa4841
Packit Service fa4841
		new_cap = pool->aCapacity / 2;
Packit Service b1ea74
		new_arr = (wStream**)realloc(pool->aArray, sizeof(wStream*) * new_cap);
Packit Service fa4841
		if (!new_arr)
Packit Service fa4841
			goto out_fail;
Packit Service fa4841
		pool->aCapacity = new_cap;
Packit Service fa4841
		pool->aArray = new_arr;
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	pool->aArray[(pool->aSize)++] = s;
Packit Service fa4841
	StreamPool_RemoveUsed(pool, s);
Packit Service fa4841
Packit Service fa4841
out_fail:
Packit Service fa4841
	if (pool->synchronized)
Packit Service fa4841
		LeaveCriticalSection(&pool->lock);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Lock the stream pool
Packit Service fa4841
 */
Packit Service fa4841
Packit Service b1ea74
static void StreamPool_Lock(wStreamPool* pool)
Packit Service fa4841
{
Packit Service fa4841
	EnterCriticalSection(&pool->lock);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Unlock the stream pool
Packit Service fa4841
 */
Packit Service fa4841
Packit Service b1ea74
static void StreamPool_Unlock(wStreamPool* pool)
Packit Service fa4841
{
Packit Service fa4841
	LeaveCriticalSection(&pool->lock);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Increment stream reference count
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
void Stream_AddRef(wStream* s)
Packit Service fa4841
{
Packit Service fa4841
	if (s->pool)
Packit Service fa4841
	{
Packit Service fa4841
		StreamPool_Lock(s->pool);
Packit Service fa4841
		s->count++;
Packit Service fa4841
		StreamPool_Unlock(s->pool);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Decrement stream reference count
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
void Stream_Release(wStream* s)
Packit Service fa4841
{
Packit Service fa4841
	DWORD count;
Packit Service fa4841
Packit Service fa4841
	if (s->pool)
Packit Service fa4841
	{
Packit Service fa4841
		StreamPool_Lock(s->pool);
Packit Service fa4841
		count = --(s->count);
Packit Service fa4841
		StreamPool_Unlock(s->pool);
Packit Service fa4841
Packit Service fa4841
		if (count == 0)
Packit Service fa4841
			StreamPool_Return(s->pool, s);
Packit Service fa4841
	}
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Find stream in pool using pointer inside buffer
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
wStream* StreamPool_Find(wStreamPool* pool, BYTE* ptr)
Packit Service fa4841
{
Packit Service fa4841
	int index;
Packit Service fa4841
	wStream* s = NULL;
Packit Service fa4841
	BOOL found = FALSE;
Packit Service fa4841
Packit Service fa4841
	EnterCriticalSection(&pool->lock);
Packit Service fa4841
Packit Service fa4841
	for (index = 0; index < pool->uSize; index++)
Packit Service fa4841
	{
Packit Service fa4841
		s = pool->uArray[index];
Packit Service fa4841
Packit Service fa4841
		if ((ptr >= Stream_Buffer(s)) && (ptr < (Stream_Buffer(s) + Stream_Capacity(s))))
Packit Service fa4841
		{
Packit Service fa4841
			found = TRUE;
Packit Service fa4841
			break;
Packit Service fa4841
		}
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	LeaveCriticalSection(&pool->lock);
Packit Service fa4841
Packit Service fa4841
	return (found) ? s : NULL;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Find stream in pool and increment reference count
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
void StreamPool_AddRef(wStreamPool* pool, BYTE* ptr)
Packit Service fa4841
{
Packit Service fa4841
	wStream* s;
Packit Service fa4841
Packit Service fa4841
	s = StreamPool_Find(pool, ptr);
Packit Service fa4841
Packit Service fa4841
	if (s)
Packit Service fa4841
		Stream_AddRef(s);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Find stream in pool and decrement reference count
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
void StreamPool_Release(wStreamPool* pool, BYTE* ptr)
Packit Service fa4841
{
Packit Service fa4841
	wStream* s;
Packit Service fa4841
Packit Service fa4841
	s = StreamPool_Find(pool, ptr);
Packit Service fa4841
Packit Service fa4841
	if (s)
Packit Service fa4841
		Stream_Release(s);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Releases the streams currently cached in the pool.
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
void StreamPool_Clear(wStreamPool* pool)
Packit Service fa4841
{
Packit Service fa4841
	if (pool->synchronized)
Packit Service fa4841
		EnterCriticalSection(&pool->lock);
Packit Service fa4841
Packit Service fa4841
	while (pool->aSize > 0)
Packit Service fa4841
	{
Packit Service fa4841
		(pool->aSize)--;
Packit Service fa4841
		Stream_Free(pool->aArray[pool->aSize], TRUE);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	if (pool->synchronized)
Packit Service fa4841
		LeaveCriticalSection(&pool->lock);
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
/**
Packit Service fa4841
 * Construction, Destruction
Packit Service fa4841
 */
Packit Service fa4841
Packit Service fa4841
wStreamPool* StreamPool_New(BOOL synchronized, size_t defaultSize)
Packit Service fa4841
{
Packit Service fa4841
	wStreamPool* pool = NULL;
Packit Service fa4841
Packit Service b1ea74
	pool = (wStreamPool*)calloc(1, sizeof(wStreamPool));
Packit Service fa4841
Packit Service fa4841
	if (pool)
Packit Service fa4841
	{
Packit Service fa4841
		pool->synchronized = synchronized;
Packit Service fa4841
		pool->defaultSize = defaultSize;
Packit Service fa4841
Packit Service fa4841
		pool->aSize = 0;
Packit Service fa4841
		pool->aCapacity = 32;
Packit Service b1ea74
		pool->aArray = (wStream**)calloc(pool->aCapacity, sizeof(wStream*));
Packit Service fa4841
Packit Service fa4841
		if (!pool->aArray)
Packit Service fa4841
		{
Packit Service fa4841
			free(pool);
Packit Service fa4841
			return NULL;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		pool->uSize = 0;
Packit Service fa4841
		pool->uCapacity = 32;
Packit Service b1ea74
		pool->uArray = (wStream**)calloc(pool->uCapacity, sizeof(wStream*));
Packit Service fa4841
Packit Service fa4841
		if (!pool->uArray)
Packit Service fa4841
		{
Packit Service fa4841
			free(pool->aArray);
Packit Service fa4841
			free(pool);
Packit Service fa4841
			return NULL;
Packit Service fa4841
		}
Packit Service fa4841
Packit Service fa4841
		InitializeCriticalSectionAndSpinCount(&pool->lock, 4000);
Packit Service fa4841
	}
Packit Service fa4841
Packit Service fa4841
	return pool;
Packit Service fa4841
}
Packit Service fa4841
Packit Service fa4841
void StreamPool_Free(wStreamPool* pool)
Packit Service fa4841
{
Packit Service fa4841
	if (pool)
Packit Service fa4841
	{
Packit Service fa4841
		StreamPool_Clear(pool);
Packit Service fa4841
Packit Service fa4841
		DeleteCriticalSection(&pool->lock);
Packit Service fa4841
Packit Service fa4841
		free(pool->aArray);
Packit Service fa4841
		free(pool->uArray);
Packit Service fa4841
Packit Service fa4841
		free(pool);
Packit Service fa4841
	}
Packit Service fa4841
}