[#5505] Enforce Recycler limit when recycling from different threads
Motivation: Currently, the recycler max capacity it's only enforced on the thread-local stack which is used when the recycling happens on the same thread that requested the object. When the recycling happens in a different thread, then the objects will be queued into a linked list (where each node holds N objects, default=16). These objects are then transfered into the stack when new objects are requested and the stack is empty. The problem is that the queue doesn't have a max capacity and that can lead to bad scenarios. Eg: - Allocate 1M object from recycler - Recycle all of them from different thread - Recycler WeakOrderQueue will contain 1M objects - Reference graph will be very long to traverse and GC timeseems to be negatively impacted - Size of the queue will never shrink after this Modifications: Add some shared counter which is used to manage capacity limits when recycle from different thread then the allocation thread. We modify the counter whenever we allocate a new Link to reduce the overhead of increment / decrement it. Result: More predictable number of objects mantained in the recycler pool.
This commit is contained in:
parent
196540ca1d
commit
6b4dbe6207
@ -28,6 +28,9 @@ import java.util.Map;
|
|||||||
import java.util.WeakHashMap;
|
import java.util.WeakHashMap;
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
|
import static java.lang.Math.max;
|
||||||
|
import static java.lang.Math.min;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Light-weight object pool based on a thread-local stack.
|
* Light-weight object pool based on a thread-local stack.
|
||||||
*
|
*
|
||||||
@ -42,8 +45,10 @@ public abstract class Recycler<T> {
|
|||||||
private static final int OWN_THREAD_ID = ID_GENERATOR.getAndIncrement();
|
private static final int OWN_THREAD_ID = ID_GENERATOR.getAndIncrement();
|
||||||
// TODO: Some arbitrary large number - should adjust as we get more production experience.
|
// TODO: Some arbitrary large number - should adjust as we get more production experience.
|
||||||
private static final int DEFAULT_INITIAL_MAX_CAPACITY = 262144;
|
private static final int DEFAULT_INITIAL_MAX_CAPACITY = 262144;
|
||||||
|
|
||||||
private static final int DEFAULT_MAX_CAPACITY;
|
private static final int DEFAULT_MAX_CAPACITY;
|
||||||
private static final int INITIAL_CAPACITY;
|
private static final int INITIAL_CAPACITY;
|
||||||
|
private static final int MAX_SHARED_CAPACITY_FACTOR;
|
||||||
private static final int LINK_CAPACITY;
|
private static final int LINK_CAPACITY;
|
||||||
|
|
||||||
static {
|
static {
|
||||||
@ -55,30 +60,37 @@ public abstract class Recycler<T> {
|
|||||||
if (maxCapacity < 0) {
|
if (maxCapacity < 0) {
|
||||||
maxCapacity = DEFAULT_INITIAL_MAX_CAPACITY;
|
maxCapacity = DEFAULT_INITIAL_MAX_CAPACITY;
|
||||||
}
|
}
|
||||||
|
|
||||||
DEFAULT_MAX_CAPACITY = maxCapacity;
|
DEFAULT_MAX_CAPACITY = maxCapacity;
|
||||||
|
|
||||||
|
MAX_SHARED_CAPACITY_FACTOR = max(2,
|
||||||
|
SystemPropertyUtil.getInt("io.netty.recycler.maxSharedCapacityFactor",
|
||||||
|
2));
|
||||||
|
|
||||||
LINK_CAPACITY = MathUtil.findNextPositivePowerOfTwo(
|
LINK_CAPACITY = MathUtil.findNextPositivePowerOfTwo(
|
||||||
Math.max(SystemPropertyUtil.getInt("io.netty.recycler.linkCapacity", 16), 16));
|
max(SystemPropertyUtil.getInt("io.netty.recycler.linkCapacity", 16), 16));
|
||||||
|
|
||||||
if (logger.isDebugEnabled()) {
|
if (logger.isDebugEnabled()) {
|
||||||
if (DEFAULT_MAX_CAPACITY == 0) {
|
if (DEFAULT_MAX_CAPACITY == 0) {
|
||||||
logger.debug("-Dio.netty.recycler.maxCapacity.default: disabled");
|
logger.debug("-Dio.netty.recycler.maxCapacity.default: disabled");
|
||||||
|
logger.debug("-Dio.netty.recycler.maxSharedCapacityFactor: disabled");
|
||||||
logger.debug("-Dio.netty.recycler.linkCapacity: disabled");
|
logger.debug("-Dio.netty.recycler.linkCapacity: disabled");
|
||||||
} else {
|
} else {
|
||||||
logger.debug("-Dio.netty.recycler.maxCapacity.default: {}", DEFAULT_MAX_CAPACITY);
|
logger.debug("-Dio.netty.recycler.maxCapacity.default: {}", DEFAULT_MAX_CAPACITY);
|
||||||
|
logger.debug("-Dio.netty.recycler.maxSharedCapacityFactor: {}", MAX_SHARED_CAPACITY_FACTOR);
|
||||||
logger.debug("-Dio.netty.recycler.linkCapacity: {}", LINK_CAPACITY);
|
logger.debug("-Dio.netty.recycler.linkCapacity: {}", LINK_CAPACITY);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
INITIAL_CAPACITY = Math.min(DEFAULT_MAX_CAPACITY, 256);
|
INITIAL_CAPACITY = min(DEFAULT_MAX_CAPACITY, 256);
|
||||||
}
|
}
|
||||||
|
|
||||||
private final int maxCapacity;
|
private final int maxCapacity;
|
||||||
|
private final int maxSharedCapacityFactor;
|
||||||
|
|
||||||
private final FastThreadLocal<Stack<T>> threadLocal = new FastThreadLocal<Stack<T>>() {
|
private final FastThreadLocal<Stack<T>> threadLocal = new FastThreadLocal<Stack<T>>() {
|
||||||
@Override
|
@Override
|
||||||
protected Stack<T> initialValue() {
|
protected Stack<T> initialValue() {
|
||||||
return new Stack<T>(Recycler.this, Thread.currentThread(), maxCapacity);
|
return new Stack<T>(Recycler.this, Thread.currentThread(), maxCapacity, maxSharedCapacityFactor);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -87,7 +99,17 @@ public abstract class Recycler<T> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected Recycler(int maxCapacity) {
|
protected Recycler(int maxCapacity) {
|
||||||
this.maxCapacity = Math.max(0, maxCapacity);
|
this(maxCapacity, MAX_SHARED_CAPACITY_FACTOR);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected Recycler(int maxCapacity, int maxSharedCapacityFactor) {
|
||||||
|
if (maxCapacity <= 0) {
|
||||||
|
this.maxCapacity = 0;
|
||||||
|
this.maxSharedCapacityFactor = 1;
|
||||||
|
} else {
|
||||||
|
this.maxCapacity = maxCapacity;
|
||||||
|
this.maxSharedCapacityFactor = max(1, maxSharedCapacityFactor);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@SuppressWarnings("unchecked")
|
@SuppressWarnings("unchecked")
|
||||||
@ -188,6 +210,7 @@ public abstract class Recycler<T> {
|
|||||||
private WeakOrderQueue next;
|
private WeakOrderQueue next;
|
||||||
private final WeakReference<Thread> owner;
|
private final WeakReference<Thread> owner;
|
||||||
private final int id = ID_GENERATOR.getAndIncrement();
|
private final int id = ID_GENERATOR.getAndIncrement();
|
||||||
|
private final Stack<?> stack;
|
||||||
|
|
||||||
WeakOrderQueue(Stack<?> stack, Thread thread) {
|
WeakOrderQueue(Stack<?> stack, Thread thread) {
|
||||||
head = tail = new Link();
|
head = tail = new Link();
|
||||||
@ -196,6 +219,10 @@ public abstract class Recycler<T> {
|
|||||||
next = stack.head;
|
next = stack.head;
|
||||||
stack.head = this;
|
stack.head = this;
|
||||||
}
|
}
|
||||||
|
this.stack = stack;
|
||||||
|
// We allocated a Link so reserve the space
|
||||||
|
boolean reserved = stack.reserveSpace(LINK_CAPACITY);
|
||||||
|
assert reserved;
|
||||||
}
|
}
|
||||||
|
|
||||||
void add(DefaultHandle handle) {
|
void add(DefaultHandle handle) {
|
||||||
@ -204,7 +231,13 @@ public abstract class Recycler<T> {
|
|||||||
Link tail = this.tail;
|
Link tail = this.tail;
|
||||||
int writeIndex;
|
int writeIndex;
|
||||||
if ((writeIndex = tail.get()) == LINK_CAPACITY) {
|
if ((writeIndex = tail.get()) == LINK_CAPACITY) {
|
||||||
|
if (!stack.reserveSpace(LINK_CAPACITY)) {
|
||||||
|
// Drop it.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// We allocate a Link so reserve the space
|
||||||
this.tail = tail = tail.next = new Link();
|
this.tail = tail = tail.next = new Link();
|
||||||
|
|
||||||
writeIndex = tail.get();
|
writeIndex = tail.get();
|
||||||
}
|
}
|
||||||
tail.elements[writeIndex] = handle;
|
tail.elements[writeIndex] = handle;
|
||||||
@ -246,7 +279,7 @@ public abstract class Recycler<T> {
|
|||||||
|
|
||||||
if (expectedCapacity > dst.elements.length) {
|
if (expectedCapacity > dst.elements.length) {
|
||||||
final int actualCapacity = dst.increaseCapacity(expectedCapacity);
|
final int actualCapacity = dst.increaseCapacity(expectedCapacity);
|
||||||
srcEnd = Math.min(srcStart + actualCapacity - dstSize, srcEnd);
|
srcEnd = min(srcStart + actualCapacity - dstSize, srcEnd);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (srcStart != srcEnd) {
|
if (srcStart != srcEnd) {
|
||||||
@ -267,6 +300,9 @@ public abstract class Recycler<T> {
|
|||||||
dst.size = newDstSize;
|
dst.size = newDstSize;
|
||||||
|
|
||||||
if (srcEnd == LINK_CAPACITY && head.next != null) {
|
if (srcEnd == LINK_CAPACITY && head.next != null) {
|
||||||
|
// Add capacity back as the Link is GCed.
|
||||||
|
stack.reclaimSpace(LINK_CAPACITY);
|
||||||
|
|
||||||
this.head = head.next;
|
this.head = head.next;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -290,15 +326,35 @@ public abstract class Recycler<T> {
|
|||||||
private DefaultHandle[] elements;
|
private DefaultHandle[] elements;
|
||||||
private final int maxCapacity;
|
private final int maxCapacity;
|
||||||
private int size;
|
private int size;
|
||||||
|
private final AtomicInteger availableSharedCapacity;
|
||||||
|
|
||||||
private volatile WeakOrderQueue head;
|
private volatile WeakOrderQueue head;
|
||||||
private WeakOrderQueue cursor, prev;
|
private WeakOrderQueue cursor, prev;
|
||||||
|
|
||||||
Stack(Recycler<T> parent, Thread thread, int maxCapacity) {
|
Stack(Recycler<T> parent, Thread thread, int maxCapacity, int maxSharedCapacityFactor) {
|
||||||
this.parent = parent;
|
this.parent = parent;
|
||||||
this.thread = thread;
|
this.thread = thread;
|
||||||
this.maxCapacity = maxCapacity;
|
this.maxCapacity = maxCapacity;
|
||||||
elements = new DefaultHandle[Math.min(INITIAL_CAPACITY, maxCapacity)];
|
availableSharedCapacity = new AtomicInteger(max(maxCapacity / maxSharedCapacityFactor, LINK_CAPACITY));
|
||||||
|
elements = new DefaultHandle[min(INITIAL_CAPACITY, maxCapacity)];
|
||||||
|
}
|
||||||
|
|
||||||
|
boolean reserveSpace(int space) {
|
||||||
|
assert space >= 0;
|
||||||
|
for (;;) {
|
||||||
|
int available = availableSharedCapacity.get();
|
||||||
|
if (available < space) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (availableSharedCapacity.compareAndSet(available, available - space)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void reclaimSpace(int space) {
|
||||||
|
assert space >= 0;
|
||||||
|
availableSharedCapacity.addAndGet(space);
|
||||||
}
|
}
|
||||||
|
|
||||||
int increaseCapacity(int expectedCapacity) {
|
int increaseCapacity(int expectedCapacity) {
|
||||||
@ -308,7 +364,7 @@ public abstract class Recycler<T> {
|
|||||||
newCapacity <<= 1;
|
newCapacity <<= 1;
|
||||||
} while (newCapacity < expectedCapacity && newCapacity < maxCapacity);
|
} while (newCapacity < expectedCapacity && newCapacity < maxCapacity);
|
||||||
|
|
||||||
newCapacity = Math.min(newCapacity, maxCapacity);
|
newCapacity = min(newCapacity, maxCapacity);
|
||||||
if (newCapacity != elements.length) {
|
if (newCapacity != elements.length) {
|
||||||
elements = Arrays.copyOf(elements, newCapacity);
|
elements = Arrays.copyOf(elements, newCapacity);
|
||||||
}
|
}
|
||||||
@ -407,7 +463,7 @@ public abstract class Recycler<T> {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (size == elements.length) {
|
if (size == elements.length) {
|
||||||
elements = Arrays.copyOf(elements, Math.min(size << 1, maxCapacity));
|
elements = Arrays.copyOf(elements, min(size << 1, maxCapacity));
|
||||||
}
|
}
|
||||||
|
|
||||||
elements[size] = item;
|
elements[size] = item;
|
||||||
|
@ -18,6 +18,7 @@ package io.netty.util;
|
|||||||
import org.junit.Test;
|
import org.junit.Test;
|
||||||
|
|
||||||
import java.util.Random;
|
import java.util.Random;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
import static org.hamcrest.CoreMatchers.*;
|
import static org.hamcrest.CoreMatchers.*;
|
||||||
import static org.junit.Assert.*;
|
import static org.junit.Assert.*;
|
||||||
@ -199,6 +200,53 @@ public class RecyclerTest {
|
|||||||
assertThat(recycler.threadLocalSize(), is(0));
|
assertThat(recycler.threadLocalSize(), is(0));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testDiscardingExceedingElementsWithRecycleAtDifferentThread() throws Exception {
|
||||||
|
final int maxCapacity = 32;
|
||||||
|
final AtomicInteger instancesCount = new AtomicInteger(0);
|
||||||
|
|
||||||
|
final Recycler<HandledObject> recycler = new Recycler<HandledObject>(maxCapacity, 2) {
|
||||||
|
@Override
|
||||||
|
protected HandledObject newObject(Recycler.Handle handle) {
|
||||||
|
instancesCount.incrementAndGet();
|
||||||
|
return new HandledObject(handle);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Borrow 2 * maxCapacity objects.
|
||||||
|
final HandledObject[] array = new HandledObject[maxCapacity * 2];
|
||||||
|
for (int i = 0; i < array.length; i++) {
|
||||||
|
array[i] = recycler.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
assertEquals(array.length, instancesCount.get());
|
||||||
|
// Reset counter.
|
||||||
|
instancesCount.set(0);
|
||||||
|
|
||||||
|
// Recycle from other thread.
|
||||||
|
final Thread thread = new Thread() {
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
for (HandledObject object: array) {
|
||||||
|
recycler.recycle(object, object.handle);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
thread.start();
|
||||||
|
thread.join();
|
||||||
|
|
||||||
|
assertEquals(0, instancesCount.get());
|
||||||
|
|
||||||
|
// Borrow 2 * maxCapacity objects. Half of them should come from
|
||||||
|
// the recycler queue, the other half should be freshly allocated.
|
||||||
|
for (int i = 0; i < array.length; i++) {
|
||||||
|
recycler.get();
|
||||||
|
}
|
||||||
|
|
||||||
|
// The implementation uses maxCapacity / 2 as limit per WeakOrderQueue
|
||||||
|
assertEquals(array.length - maxCapacity / 2, instancesCount.get());
|
||||||
|
}
|
||||||
|
|
||||||
static final class HandledObject {
|
static final class HandledObject {
|
||||||
Recycler.Handle handle;
|
Recycler.Handle handle;
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user