package io.undertow.util;
import java.util.concurrent.LinkedBlockingDeque;
import java.util.concurrent.atomic.AtomicIntegerFieldUpdater;
import java.util.function.Consumer;
import java.util.function.Supplier;
import io.undertow.UndertowMessages;
public class SimpleObjectPool<T> implements ObjectPool {
private final Supplier<T> supplier;
private final Consumer<T> recycler;
private final Consumer<T> consumer;
private final LinkedBlockingDeque<T> pool;
public SimpleObjectPool(int poolSize, Supplier<T> supplier, Consumer<T> recycler, Consumer<T> consumer) {
this.supplier = supplier;
this.recycler = recycler;
this.consumer = consumer;
pool = new LinkedBlockingDeque<T>(poolSize);
}
public SimpleObjectPool(int poolSize, Supplier<T> supplier, Consumer<T> consumer) {
this(poolSize, supplier, object -> {}, consumer);
}
@Override
public PooledObject<T> allocate() {
T obj = pool.poll();
if(obj == null) {
obj = supplier.get();
}
return new SimplePooledObject<>(obj, this);
}
private static final class SimplePooledObject<T> implements PooledObject<T> {
private static final AtomicIntegerFieldUpdater<SimplePooledObject> closedUpdater =
AtomicIntegerFieldUpdater.newUpdater(SimplePooledObject.class, "closed");
private volatile int closed;
private final T object;
private final SimpleObjectPool<T> objectPool;
SimplePooledObject(T object, SimpleObjectPool<T> objectPool) {
this.object = object;
this.objectPool = objectPool;
}
@Override
public T getObject() {
if (closedUpdater.get(this) != 0) {
throw UndertowMessages.MESSAGES.objectIsClosed();
}
return object;
}
@Override
public void close() {
if (closedUpdater.compareAndSet(this, 0, 1)) {
objectPool.recycler.accept(object);
if (!objectPool.pool.offer(object)) {
objectPool.consumer.accept(object);
}
}
}
}
}