improved ZSTD_createCCtxPool() cancellation

use ZSTD_freeCCtxPool() to release the partially created pool.
avoids to duplicate logic.

Also : identified a new difficult corner case :
when freeing the Pool, all CCtx should be previously released back to the pool.
Otherwise, it means some CCtx are still in use.
There is currently no clear policy on what to do in such a case.
Note : it's supposed to never happen.
Since pool creation/usage is static, it has no external user,
which limits risks.
This commit is contained in:
Yann Collet 2017-01-11 15:44:26 +01:00
parent 47557ba2b2
commit 8ce1cc2bec

View File

@ -141,6 +141,15 @@ typedef struct {
/* assumption : CCtxPool invocation only from main thread */ /* assumption : CCtxPool invocation only from main thread */
/* note : all CCtx borrowed from the pool should be released back to the pool _before_ freeing the pool */
static void ZSTDMT_freeCCtxPool(ZSTDMT_CCtxPool* pool)
{
unsigned u;
for (u=0; u<pool->availCCtx; u++) /* note : availCCtx is supposed == totalCCtx; otherwise, some CCtx are still in use */
ZSTD_freeCCtx(pool->cctx[u]);
free(pool);
}
static ZSTDMT_CCtxPool* ZSTDMT_createCCtxPool(unsigned nbThreads) static ZSTDMT_CCtxPool* ZSTDMT_createCCtxPool(unsigned nbThreads)
{ {
ZSTDMT_CCtxPool* const cctxPool = (ZSTDMT_CCtxPool*) calloc(1, sizeof(ZSTDMT_CCtxPool) + nbThreads*sizeof(ZSTD_CCtx*)); ZSTDMT_CCtxPool* const cctxPool = (ZSTDMT_CCtxPool*) calloc(1, sizeof(ZSTDMT_CCtxPool) + nbThreads*sizeof(ZSTD_CCtx*));
@ -149,10 +158,8 @@ static ZSTDMT_CCtxPool* ZSTDMT_createCCtxPool(unsigned nbThreads)
for (threadNb=0; threadNb<nbThreads; threadNb++) { for (threadNb=0; threadNb<nbThreads; threadNb++) {
cctxPool->cctx[threadNb] = ZSTD_createCCtx(); cctxPool->cctx[threadNb] = ZSTD_createCCtx();
if (cctxPool->cctx[threadNb]==NULL) { /* failed cctx allocation : abort cctxPool creation */ if (cctxPool->cctx[threadNb]==NULL) { /* failed cctx allocation : abort cctxPool creation */
unsigned u; cctxPool->totalCCtx = cctxPool->availCCtx = threadNb;
for (u=0; u<threadNb; u++) ZSTDMT_freeCCtxPool(cctxPool);
free(cctxPool->cctx[u]);
free(cctxPool);
return NULL; return NULL;
} } } } } }
cctxPool->totalCCtx = cctxPool->availCCtx = nbThreads; cctxPool->totalCCtx = cctxPool->availCCtx = nbThreads;
@ -165,7 +172,7 @@ static ZSTD_CCtx* ZSTDMT_getCCtx(ZSTDMT_CCtxPool* pool)
pool->availCCtx--; pool->availCCtx--;
return pool->cctx[pool->availCCtx]; return pool->cctx[pool->availCCtx];
} }
/* should not be possible, since totalCCtx==nbThreads */ /* note : should not be possible, since totalCCtx==nbThreads */
return ZSTD_createCCtx(); return ZSTD_createCCtx();
} }
@ -174,18 +181,10 @@ static void ZSTDMT_releaseCCtx(ZSTDMT_CCtxPool* pool, ZSTD_CCtx* cctx)
if (pool->availCCtx < pool->totalCCtx) if (pool->availCCtx < pool->totalCCtx)
pool->cctx[pool->availCCtx++] = cctx; pool->cctx[pool->availCCtx++] = cctx;
else else
/* should not be possible, since totalCCtx==nbThreads */ /* note : should not be possible, since totalCCtx==nbThreads */
ZSTD_freeCCtx(cctx); ZSTD_freeCCtx(cctx);
} }
static void ZSTDMT_freeCCtxPool(ZSTDMT_CCtxPool* pool)
{
unsigned u;
for (u=0; u<pool->totalCCtx; u++)
ZSTD_freeCCtx(pool->cctx[u]);
free(pool);
}
struct ZSTDMT_CCtx_s { struct ZSTDMT_CCtx_s {
POOL_ctx* factory; POOL_ctx* factory;