| package com.airbnb.lottie.snapshots.utils |
| |
| import android.util.Log |
| import com.airbnb.lottie.L |
| import kotlinx.coroutines.ExperimentalCoroutinesApi |
| import java.util.* |
| import kotlin.collections.HashSet |
| |
| class ObjectPool<T>(private val factory: () -> T) { |
| |
| private val semaphore = SuspendingSemaphore(MAX_RELEASED_OBJECTS) |
| private val objects = Collections.synchronizedList(ArrayList<T>()) |
| private val releasedObjects = HashSet<T>() |
| |
| @Synchronized |
| fun acquire(): T { |
| val blockedStartTime = System.currentTimeMillis() |
| semaphore.acquire() |
| val waitingTimeMs = System.currentTimeMillis() - blockedStartTime |
| if (waitingTimeMs > 100) { |
| Log.d(L.TAG, "Waited ${waitingTimeMs}ms for an object.") |
| } |
| |
| val obj = synchronized(objects) { |
| objects.firstOrNull()?.also { objects.remove(it) } |
| } ?: factory() |
| |
| releasedObjects += obj |
| return obj |
| } |
| |
| @Synchronized |
| fun release(obj: T) { |
| val removed = releasedObjects.remove(obj) |
| if (!removed) throw IllegalArgumentException("Unable to find original obj.") |
| |
| objects.add(obj) |
| semaphore.release() |
| } |
| |
| companion object { |
| // The maximum number of objects that are allowed out at a time. |
| // If this limit is reached a thread must wait for another bitmap to be returned. |
| // Bitmaps are expensive, and if we aren't careful we can easily allocate too many objects |
| // since coroutines run parallelized. |
| private const val MAX_RELEASED_OBJECTS = 10 |
| } |
| } |