20#include <winpr/config.h>
24#include <winpr/collections.h>
27#define MAX(a, b) ((a) > (b)) ? (a) : (b)
49 wBufferPoolItem* aArray;
53 wBufferPoolItem* uArray;
56static BOOL BufferPool_Lock(wBufferPool* pool)
61 if (pool->synchronized)
62 EnterCriticalSection(&pool->lock);
66static BOOL BufferPool_Unlock(wBufferPool* pool)
71 if (pool->synchronized)
72 LeaveCriticalSection(&pool->lock);
85static BOOL BufferPool_ShiftAvailable(wBufferPool* pool,
size_t index,
int count)
89 if (pool->aSize + count > pool->aCapacity)
91 wBufferPoolItem* newArray =
nullptr;
92 SSIZE_T newCapacity = pool->aSize + count;
93 newCapacity += (newCapacity + 2) / 2;
95 WINPR_ASSERT(newCapacity > 0);
96 if (pool->alignment > 0)
97 newArray = (wBufferPoolItem*)winpr_aligned_realloc(
99 sizeof(wBufferPoolItem) * WINPR_ASSERTING_INT_CAST(
size_t, newCapacity),
102 newArray = (wBufferPoolItem*)realloc(
104 sizeof(wBufferPoolItem) * WINPR_ASSERTING_INT_CAST(
size_t, newCapacity));
107 pool->aArray = newArray;
108 pool->aCapacity = newCapacity;
112 &pool->aArray[index + WINPR_ASSERTING_INT_CAST(
size_t, count)], &pool->aArray[index],
113 (WINPR_ASSERTING_INT_CAST(
size_t, pool->aSize) - index) *
sizeof(wBufferPoolItem));
114 pool->aSize += count;
119 &pool->aArray[index], &pool->aArray[index + WINPR_ASSERTING_INT_CAST(
size_t, -count)],
120 (WINPR_ASSERTING_INT_CAST(
size_t, pool->aSize) - index) *
sizeof(wBufferPoolItem));
121 pool->aSize += count;
126static BOOL BufferPool_ShiftUsed(wBufferPool* pool, SSIZE_T index, SSIZE_T count)
130 const SSIZE_T required = pool->uSize + count;
132 if ((required < count) || (required < pool->uSize))
135 if (required > pool->uCapacity)
137 SSIZE_T newUCapacity = pool->uCapacity;
140 if (newUCapacity > SSIZE_MAX - 128ll)
142 newUCapacity += 128ll;
143 }
while (newUCapacity <= required);
144 wBufferPoolItem* newUArray =
nullptr;
145 if (pool->alignment > 0)
146 newUArray = (wBufferPoolItem*)winpr_aligned_realloc(
148 sizeof(wBufferPoolItem) * WINPR_ASSERTING_INT_CAST(
size_t, newUCapacity),
151 newUArray = (wBufferPoolItem*)realloc(
153 sizeof(wBufferPoolItem) * WINPR_ASSERTING_INT_CAST(
size_t, newUCapacity));
156 pool->uCapacity = newUCapacity;
157 pool->uArray = newUArray;
160 MoveMemory(&pool->uArray[index + count], &pool->uArray[index],
161 WINPR_ASSERTING_INT_CAST(
size_t, pool->uSize - index) *
sizeof(wBufferPoolItem));
162 pool->uSize += count;
166 MoveMemory(&pool->uArray[index], &pool->uArray[index - count],
167 WINPR_ASSERTING_INT_CAST(
size_t, pool->uSize - index) *
sizeof(wBufferPoolItem));
168 pool->uSize += count;
177SSIZE_T BufferPool_GetPoolSize(wBufferPool* pool)
181 BufferPool_Lock(pool);
194 BufferPool_Unlock(pool);
203SSIZE_T BufferPool_GetBufferSize(wBufferPool* pool,
const void* buffer)
208 BufferPool_Lock(pool);
213 size = pool->fixedSize;
220 for (SSIZE_T index = 0; index < pool->uSize; index++)
222 if (pool->uArray[index].buffer == buffer)
224 size = pool->uArray[index].size;
231 BufferPool_Unlock(pool);
233 return (found) ? size : -1;
240void* BufferPool_Take(wBufferPool* pool, SSIZE_T size)
243 SSIZE_T maxIndex = 0;
244 SSIZE_T foundIndex = -1;
246 void* buffer =
nullptr;
248 BufferPool_Lock(pool);
255 buffer = pool->array[--(pool->size)];
260 buffer = winpr_aligned_malloc(WINPR_ASSERTING_INT_CAST(
size_t, pool->fixedSize),
263 buffer = malloc(WINPR_ASSERTING_INT_CAST(
size_t, pool->fixedSize));
277 size = pool->fixedSize;
279 for (SSIZE_T index = 0; index < pool->aSize; index++)
281 if (pool->aArray[index].size > maxSize)
284 maxSize = pool->aArray[index].size;
287 if (pool->aArray[index].size >= size)
295 if (!found && maxSize)
297 foundIndex = maxIndex;
308 buffer = winpr_aligned_malloc(WINPR_ASSERTING_INT_CAST(
size_t, size),
311 buffer = malloc(WINPR_ASSERTING_INT_CAST(
size_t, size));
319 buffer = pool->aArray[foundIndex].buffer;
323 void* newBuffer =
nullptr;
325 newBuffer = winpr_aligned_realloc(
326 buffer, WINPR_ASSERTING_INT_CAST(
size_t, size), pool->alignment);
328 newBuffer = realloc(buffer, WINPR_ASSERTING_INT_CAST(
size_t, size));
331 goto out_error_no_free;
336 if (!BufferPool_ShiftAvailable(pool, WINPR_ASSERTING_INT_CAST(
size_t, foundIndex), -1))
343 if (pool->uSize + 1 > pool->uCapacity)
345 size_t newUCapacity = WINPR_ASSERTING_INT_CAST(
size_t, pool->uCapacity);
346 newUCapacity += (newUCapacity + 2) / 2;
347 if (newUCapacity > SSIZE_MAX)
349 wBufferPoolItem* newUArray =
350 (wBufferPoolItem*)realloc(pool->uArray,
sizeof(wBufferPoolItem) * newUCapacity);
354 pool->uCapacity = (SSIZE_T)newUCapacity;
355 pool->uArray = newUArray;
358 pool->uArray[pool->uSize].buffer = buffer;
359 pool->uArray[pool->uSize].size = size;
363 BufferPool_Unlock(pool);
369 winpr_aligned_free(buffer);
373 BufferPool_Unlock(pool);
381BOOL BufferPool_Return(wBufferPool* pool,
void* buffer)
387 BufferPool_Lock(pool);
393 if ((pool->size + 1) >= pool->capacity)
395 SSIZE_T newCapacity = MAX(2, pool->size + (pool->size + 2) / 2 + 1);
396 void** newArray = (
void**)realloc(
397 (
void*)pool->array,
sizeof(
void*) * WINPR_ASSERTING_INT_CAST(
size_t, newCapacity));
401 pool->capacity = newCapacity;
402 pool->array = newArray;
405 pool->array[(pool->size)++] = buffer;
412 for (; index < pool->uSize; index++)
414 if (pool->uArray[index].buffer == buffer)
423 size = pool->uArray[index].size;
424 if (!BufferPool_ShiftUsed(pool, index, -1))
430 if ((pool->aSize + 1) >= pool->aCapacity)
432 SSIZE_T newCapacity = MAX(2, pool->aSize + (pool->aSize + 2) / 2 + 1);
433 wBufferPoolItem* newArray = (wBufferPoolItem*)realloc(
435 sizeof(wBufferPoolItem) * WINPR_ASSERTING_INT_CAST(
size_t, newCapacity));
439 pool->aCapacity = newCapacity;
440 pool->aArray = newArray;
443 pool->aArray[pool->aSize].buffer = buffer;
444 pool->aArray[pool->aSize].size = size;
451 BufferPool_Unlock(pool);
459void BufferPool_Clear(wBufferPool* pool)
461 BufferPool_Lock(pool);
467 while (pool->size > 0)
472 winpr_aligned_free(pool->array[pool->size]);
474 free(pool->array[pool->size]);
481 while (pool->aSize > 0)
486 winpr_aligned_free(pool->aArray[pool->aSize].buffer);
488 free(pool->aArray[pool->aSize].buffer);
491 while (pool->uSize > 0)
496 winpr_aligned_free(pool->uArray[pool->uSize].buffer);
498 free(pool->uArray[pool->uSize].buffer);
502 BufferPool_Unlock(pool);
509wBufferPool* BufferPool_New(BOOL
synchronized, SSIZE_T fixedSize, DWORD alignment)
511 wBufferPool* pool =
nullptr;
513 pool = (wBufferPool*)calloc(1,
sizeof(wBufferPool));
517 pool->fixedSize = fixedSize;
519 if (pool->fixedSize < 0)
522 pool->alignment = alignment;
523 pool->synchronized =
synchronized;
525 if (pool->synchronized)
527 if (!InitializeCriticalSectionAndSpinCount(&pool->lock, 4000))
538 (
void**)calloc(WINPR_ASSERTING_INT_CAST(
size_t, pool->capacity),
sizeof(
void*));
547 pool->aCapacity = 32;
548 pool->aArray = (wBufferPoolItem*)calloc(
549 WINPR_ASSERTING_INT_CAST(
size_t, pool->aCapacity),
sizeof(wBufferPoolItem));
554 pool->uCapacity = 32;
555 pool->uArray = (wBufferPoolItem*)calloc(
556 WINPR_ASSERTING_INT_CAST(
size_t, pool->uCapacity),
sizeof(wBufferPoolItem));
565 WINPR_PRAGMA_DIAG_PUSH
566 WINPR_PRAGMA_DIAG_IGNORED_MISMATCHED_DEALLOC
567 BufferPool_Free(pool);
568 WINPR_PRAGMA_DIAG_POP
572void BufferPool_Free(wBufferPool* pool)
576 BufferPool_Clear(pool);
578 if (pool->synchronized)
579 DeleteCriticalSection(&pool->lock);
585 free((
void*)pool->array);