Commit 07e428de authored by Rémi Denis-Courmont's avatar Rémi Denis-Courmont

picture_pool: split picture_pool_NonEmpty and picture_pool_Reset

parent 3ca607e8
...@@ -93,18 +93,25 @@ VLC_API void picture_pool_Delete( picture_pool_t * ); ...@@ -93,18 +93,25 @@ VLC_API void picture_pool_Delete( picture_pool_t * );
*/ */
VLC_API picture_t * picture_pool_Get( picture_pool_t * ) VLC_USED; VLC_API picture_t * picture_pool_Get( picture_pool_t * ) VLC_USED;
/**
* Forcefully return all pictures in the pool to free/unallocated state.
*
* @warning This can only be called when it is known that all pending
* references to the picture pool are stale, e.g. a decoder failed to
* release pictures properly when it terminated.
*/
void picture_pool_Reset( picture_pool_t * );
/** /**
* It forces the next picture_pool_Get to return a picture even if no * It forces the next picture_pool_Get to return a picture even if no
* pictures are free. * pictures are free.
* *
* If b_reset is true, all pictures will be marked as free.
*
* It does it by releasing itself the oldest used picture if none is * It does it by releasing itself the oldest used picture if none is
* available. * available.
* XXX it should be used with great care, the only reason you may need * XXX it should be used with great care, the only reason you may need
* it is to workaround a bug. * it is to workaround a bug.
*/ */
VLC_API void picture_pool_NonEmpty( picture_pool_t *, bool reset ); VLC_API void picture_pool_NonEmpty( picture_pool_t * );
/** /**
* It reserves picture_count pictures from the given pool and returns * It reserves picture_count pictures from the given pool and returns
......
...@@ -289,31 +289,43 @@ picture_t *picture_pool_Get(picture_pool_t *pool) ...@@ -289,31 +289,43 @@ picture_t *picture_pool_Get(picture_pool_t *pool)
return NULL; return NULL;
} }
void picture_pool_NonEmpty(picture_pool_t *pool, bool reset) void picture_pool_Reset(picture_pool_t *pool)
{ {
picture_t *old = NULL;
for (int i = 0; i < pool->picture_count; i++) { for (int i = 0; i < pool->picture_count; i++) {
if (pool->picture_reserved[i]) if (pool->picture_reserved[i])
continue; continue;
picture_t *picture = pool->picture[i]; picture_t *picture = pool->picture[i];
if (reset) {
if (atomic_load(&picture->gc.refcount) > 0) if (atomic_load(&picture->gc.refcount) > 0)
Unlock(picture); Unlock(picture);
atomic_store(&picture->gc.refcount, 0); atomic_store(&picture->gc.refcount, 0);
} else if (atomic_load(&picture->gc.refcount) == 0) {
return;
} else if (!old || picture->gc.p_sys->tick < old->gc.p_sys->tick) {
old = picture;
} }
}
void picture_pool_NonEmpty(picture_pool_t *pool)
{
picture_t *oldest = NULL;
for (int i = 0; i < pool->picture_count; i++) {
if (pool->picture_reserved[i])
continue;
picture_t *picture = pool->picture[i];
if (atomic_load(&picture->gc.refcount) == 0)
return; /* Nothing to do */
if (oldest == NULL || picture->gc.p_sys->tick < oldest->gc.p_sys->tick)
oldest = picture;
} }
if (!reset && old) {
if (atomic_load(&old->gc.refcount) > 0) if (oldest == NULL)
Unlock(old); return; /* Cannot fix! */
atomic_store(&old->gc.refcount, 0);
} if (atomic_load(&oldest->gc.refcount) > 0)
Unlock(oldest);
atomic_store(&oldest->gc.refcount, 0);
} }
int picture_pool_GetSize(picture_pool_t *pool) int picture_pool_GetSize(picture_pool_t *pool)
{ {
return pool->picture_count; return pool->picture_count;
......
...@@ -363,7 +363,7 @@ void vout_FixLeaks( vout_thread_t *vout ) ...@@ -363,7 +363,7 @@ void vout_FixLeaks( vout_thread_t *vout )
msg_Err(vout, "pictures leaked, trying to workaround"); msg_Err(vout, "pictures leaked, trying to workaround");
/* */ /* */
picture_pool_NonEmpty(vout->p->decoder_pool, false); picture_pool_NonEmpty(vout->p->decoder_pool);
vlc_mutex_unlock(&vout->p->picture_lock); vlc_mutex_unlock(&vout->p->picture_lock);
} }
...@@ -1232,7 +1232,7 @@ static void ThreadReset(vout_thread_t *vout) ...@@ -1232,7 +1232,7 @@ static void ThreadReset(vout_thread_t *vout)
{ {
ThreadFlush(vout, true, INT64_MAX); ThreadFlush(vout, true, INT64_MAX);
if (vout->p->decoder_pool) if (vout->p->decoder_pool)
picture_pool_NonEmpty(vout->p->decoder_pool, true); picture_pool_Reset(vout->p->decoder_pool);
vout->p->pause.is_on = false; vout->p->pause.is_on = false;
vout->p->pause.date = mdate(); vout->p->pause.date = mdate();
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment