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

410 lines
7.7 KiB
C
Raw Normal View History

/**
* WinPR: Windows Portable Runtime
* Object Pool
*
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
2022-02-16 12:08:00 +03:00
#include <winpr/config.h>
#include <winpr/crt.h>
2020-05-12 15:57:39 +03:00
#include <winpr/wlog.h>
#include <winpr/collections.h>
#include "../stream.h"
struct s_wStreamPool
2020-05-12 15:57:39 +03:00
{
size_t aSize;
size_t aCapacity;
wStream** aArray;
size_t uSize;
size_t uCapacity;
wStream** uArray;
CRITICAL_SECTION lock;
BOOL synchronized;
size_t defaultSize;
};
2020-10-30 12:17:27 +03:00
/**
* Lock the stream pool
*/
static INLINE void StreamPool_Lock(wStreamPool* pool)
{
WINPR_ASSERT(pool);
2020-10-30 12:17:27 +03:00
if (pool->synchronized)
EnterCriticalSection(&pool->lock);
}
/**
* Unlock the stream pool
*/
static INLINE void StreamPool_Unlock(wStreamPool* pool)
{
WINPR_ASSERT(pool);
2020-10-30 12:17:27 +03:00
if (pool->synchronized)
LeaveCriticalSection(&pool->lock);
}
2020-05-12 15:57:39 +03:00
static BOOL StreamPool_EnsureCapacity(wStreamPool* pool, size_t count, BOOL usedOrAvailable)
{
size_t new_cap = 0;
size_t* cap = NULL;
size_t* size = NULL;
wStream*** array = NULL;
2020-05-12 15:57:39 +03:00
WINPR_ASSERT(pool);
2020-05-19 20:27:58 +03:00
cap = (usedOrAvailable) ? &pool->uCapacity : &pool->aCapacity;
size = (usedOrAvailable) ? &pool->uSize : &pool->aSize;
array = (usedOrAvailable) ? &pool->uArray : &pool->aArray;
2020-05-12 15:57:39 +03:00
if (*cap == 0)
new_cap = *size + count;
else if (*size + count > *cap)
new_cap = *cap * 2;
else if ((*size + count) < *cap / 3)
new_cap = *cap / 2;
if (new_cap > 0)
{
wStream** new_arr = NULL;
2020-05-12 15:57:39 +03:00
if (*cap < *size + count)
*cap += count;
new_arr = (wStream**)realloc(*array, sizeof(wStream*) * new_cap);
if (!new_arr)
return FALSE;
*cap = new_cap;
*array = new_arr;
}
return TRUE;
}
/**
* Methods
*/
2020-05-12 15:57:39 +03:00
static void StreamPool_ShiftUsed(wStreamPool* pool, size_t index, INT64 count)
2013-04-12 20:20:20 +04:00
{
WINPR_ASSERT(pool);
2013-04-12 20:20:20 +04:00
if (count > 0)
{
2021-06-16 15:43:07 +03:00
const size_t pcount = (size_t)count;
StreamPool_EnsureCapacity(pool, pcount, TRUE);
2013-04-12 20:20:20 +04:00
2021-06-16 15:43:07 +03:00
MoveMemory(&pool->uArray[index + pcount], &pool->uArray[index],
2019-11-06 17:24:51 +03:00
(pool->uSize - index) * sizeof(wStream*));
2021-06-16 15:43:07 +03:00
pool->uSize += pcount;
2013-04-12 20:20:20 +04:00
}
else if (count < 0)
{
2021-06-16 15:43:07 +03:00
const size_t pcount = (size_t)-count;
2024-07-02 13:22:14 +03:00
const size_t off = index + pcount;
if (pool->uSize > off)
2013-11-08 02:37:58 +04:00
{
2021-06-16 15:43:07 +03:00
MoveMemory(&pool->uArray[index], &pool->uArray[index + pcount],
(pool->uSize - index - pcount) * sizeof(wStream*));
2013-11-08 02:37:58 +04:00
}
2021-06-16 15:43:07 +03:00
pool->uSize -= pcount;
2013-04-12 20:20:20 +04:00
}
}
/**
* Adds a used stream to the pool.
*/
2019-11-20 13:30:14 +03:00
static void StreamPool_AddUsed(wStreamPool* pool, wStream* s)
2013-04-12 20:20:20 +04:00
{
2020-05-12 15:57:39 +03:00
StreamPool_EnsureCapacity(pool, 1, TRUE);
2013-04-12 20:20:20 +04:00
pool->uArray[(pool->uSize)++] = s;
}
/**
* Removes a used stream from the pool.
*/
2019-11-20 13:30:14 +03:00
static void StreamPool_RemoveUsed(wStreamPool* pool, wStream* s)
2013-04-12 20:20:20 +04:00
{
WINPR_ASSERT(pool);
for (size_t index = 0; index < pool->uSize; index++)
2013-04-12 20:20:20 +04:00
{
if (pool->uArray[index] == s)
{
StreamPool_ShiftUsed(pool, index, -1);
2013-04-12 20:20:20 +04:00
break;
}
}
}
2020-05-12 15:57:39 +03:00
static void StreamPool_ShiftAvailable(wStreamPool* pool, size_t index, INT64 count)
{
WINPR_ASSERT(pool);
if (count > 0)
{
2021-06-16 15:43:07 +03:00
const size_t pcount = (size_t)count;
StreamPool_EnsureCapacity(pool, pcount, FALSE);
MoveMemory(&pool->aArray[index + pcount], &pool->aArray[index],
2019-11-06 17:24:51 +03:00
(pool->aSize - index) * sizeof(wStream*));
2021-06-16 15:43:07 +03:00
pool->aSize += pcount;
}
else if (count < 0)
{
2021-06-16 15:43:07 +03:00
const size_t pcount = (size_t)-count;
2024-07-02 13:22:14 +03:00
const size_t off = index + pcount;
if (pool->aSize > off)
2013-11-08 02:37:58 +04:00
{
2021-06-16 15:43:07 +03:00
MoveMemory(&pool->aArray[index], &pool->aArray[index + pcount],
(pool->aSize - index - pcount) * sizeof(wStream*));
2013-11-08 02:37:58 +04:00
}
2021-06-16 15:43:07 +03:00
pool->aSize -= pcount;
}
}
/**
* Gets a stream from the pool.
*/
wStream* StreamPool_Take(wStreamPool* pool, size_t size)
{
SSIZE_T foundIndex = -1;
wStream* s = NULL;
2020-10-30 12:17:27 +03:00
StreamPool_Lock(pool);
2013-04-12 21:44:23 +04:00
if (size == 0)
2013-04-12 20:20:20 +04:00
size = pool->defaultSize;
for (size_t index = 0; index < pool->aSize; index++)
{
s = pool->aArray[index];
if (Stream_Capacity(s) >= size)
{
2013-04-30 08:36:16 +04:00
foundIndex = index;
break;
}
}
2013-11-08 02:37:58 +04:00
if (foundIndex < 0)
{
s = Stream_New(NULL, size);
if (!s)
goto out_fail;
}
else if (s)
{
Stream_SetPosition(s, 0);
Stream_SetLength(s, Stream_Capacity(s));
StreamPool_ShiftAvailable(pool, foundIndex, -1);
}
if (s)
{
s->pool = pool;
s->count = 1;
StreamPool_AddUsed(pool, s);
}
2013-04-12 20:20:20 +04:00
out_fail:
2020-10-30 12:17:27 +03:00
StreamPool_Unlock(pool);
return s;
}
/**
* Returns an object to the pool.
*/
2022-06-24 17:14:25 +03:00
static void StreamPool_Remove(wStreamPool* pool, wStream* s)
{
StreamPool_EnsureCapacity(pool, 1, FALSE);
Stream_EnsureValidity(s);
for (size_t x = 0; x < pool->aSize; x++)
{
wStream* cs = pool->aArray[x];
WINPR_ASSERT(cs != s);
}
2022-06-24 17:14:25 +03:00
pool->aArray[(pool->aSize)++] = s;
StreamPool_RemoveUsed(pool, s);
}
static void StreamPool_ReleaseOrReturn(wStreamPool* pool, wStream* s)
{
StreamPool_Lock(pool);
if (s->count > 0)
s->count--;
if (s->count == 0)
StreamPool_Remove(pool, s);
StreamPool_Unlock(pool);
}
void StreamPool_Return(wStreamPool* pool, wStream* s)
{
WINPR_ASSERT(pool);
if (!s)
return;
2020-10-30 12:17:27 +03:00
StreamPool_Lock(pool);
2022-06-24 17:14:25 +03:00
StreamPool_Remove(pool, s);
2020-10-30 12:17:27 +03:00
StreamPool_Unlock(pool);
}
/**
* Increment stream reference count
*/
void Stream_AddRef(wStream* s)
{
WINPR_ASSERT(s);
if (s->pool)
{
StreamPool_Lock(s->pool);
s->count++;
StreamPool_Unlock(s->pool);
}
}
/**
* Decrement stream reference count
*/
void Stream_Release(wStream* s)
{
WINPR_ASSERT(s);
if (s->pool)
2022-06-24 17:14:25 +03:00
StreamPool_ReleaseOrReturn(s->pool, s);
}
2013-04-12 20:20:20 +04:00
/**
* Find stream in pool using pointer inside buffer
*/
wStream* StreamPool_Find(wStreamPool* pool, BYTE* ptr)
{
wStream* s = NULL;
BOOL found = FALSE;
StreamPool_Lock(pool);
2013-04-12 20:20:20 +04:00
for (size_t index = 0; index < pool->uSize; index++)
2013-04-12 20:20:20 +04:00
{
s = pool->uArray[index];
if ((ptr >= Stream_Buffer(s)) && (ptr < (Stream_Buffer(s) + Stream_Capacity(s))))
2013-04-12 20:20:20 +04:00
{
found = TRUE;
break;
}
}
2020-10-30 12:17:27 +03:00
StreamPool_Unlock(pool);
2013-04-12 20:20:20 +04:00
return (found) ? s : NULL;
}
/**
* Releases the streams currently cached in the pool.
*/
void StreamPool_Clear(wStreamPool* pool)
{
2020-10-30 12:17:27 +03:00
StreamPool_Lock(pool);
2013-04-12 20:20:20 +04:00
while (pool->aSize > 0)
{
wStream* s = pool->aArray[--pool->aSize];
Stream_Free(s, s->isAllocatedStream);
}
2020-05-12 15:57:39 +03:00
while (pool->uSize > 0)
{
wStream* s = pool->uArray[--pool->uSize];
Stream_Free(s, s->isAllocatedStream);
2020-05-12 15:57:39 +03:00
}
2020-10-30 12:17:27 +03:00
StreamPool_Unlock(pool);
}
/**
* Construction, Destruction
*/
2013-04-12 20:20:20 +04:00
wStreamPool* StreamPool_New(BOOL synchronized, size_t defaultSize)
{
wStreamPool* pool = NULL;
2019-11-06 17:24:51 +03:00
pool = (wStreamPool*)calloc(1, sizeof(wStreamPool));
if (pool)
{
pool->synchronized = synchronized;
2013-04-12 20:20:20 +04:00
pool->defaultSize = defaultSize;
2020-05-12 15:57:39 +03:00
if (!StreamPool_EnsureCapacity(pool, 32, FALSE))
goto fail;
if (!StreamPool_EnsureCapacity(pool, 32, TRUE))
goto fail;
InitializeCriticalSectionAndSpinCount(&pool->lock, 4000);
}
return pool;
2020-05-12 15:57:39 +03:00
fail:
WINPR_PRAGMA_DIAG_PUSH
WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
2020-05-12 15:57:39 +03:00
StreamPool_Free(pool);
WINPR_PRAGMA_DIAG_POP
2020-05-12 15:57:39 +03:00
return NULL;
}
void StreamPool_Free(wStreamPool* pool)
{
if (pool)
{
StreamPool_Clear(pool);
DeleteCriticalSection(&pool->lock);
2013-04-12 20:20:20 +04:00
free(pool->aArray);
free(pool->uArray);
free(pool);
}
}
2020-05-12 15:57:39 +03:00
char* StreamPool_GetStatistics(wStreamPool* pool, char* buffer, size_t size)
{
WINPR_ASSERT(pool);
if (!buffer || (size < 1))
2020-05-12 15:57:39 +03:00
return NULL;
(void)_snprintf(buffer, size - 1,
"aSize =%" PRIuz ", uSize =%" PRIuz "aCapacity=%" PRIuz
", uCapacity=%" PRIuz,
pool->aSize, pool->uSize, pool->aCapacity, pool->uCapacity);
2020-05-12 15:57:39 +03:00
buffer[size - 1] = '\0';
return buffer;
}