Fix a bug in PoolArena and PoolSubpage where subpage pools are not updated correctly

- Make PoolSubpage a linked list node in the pool
- Now that a subpage is added to and removed from the pool correctly, allocating a subpage from the pool became vastly simpler.
This commit is contained in:
Trustin Lee 2013-03-05 23:55:41 +09:00
parent 6e5bb87219
commit 5f2c2cdc9b
3 changed files with 90 additions and 53 deletions

View File

@ -20,8 +20,6 @@ import io.netty.util.internal.PlatformDependent;
import io.netty.util.internal.StringUtil; import io.netty.util.internal.StringUtil;
import java.nio.ByteBuffer; import java.nio.ByteBuffer;
import java.util.ArrayDeque;
import java.util.Deque;
abstract class PoolArena<T> { abstract class PoolArena<T> {
@ -33,8 +31,8 @@ abstract class PoolArena<T> {
private final int chunkSize; private final int chunkSize;
private final int subpageOverflowMask; private final int subpageOverflowMask;
private final Deque<PoolSubpage<T>>[] tinySubpagePools; private final PoolSubpage<T>[] tinySubpagePools;
private final Deque<PoolSubpage<T>>[] smallSubpagePools; private final PoolSubpage<T>[] smallSubpagePools;
private final PoolChunkList<T> q050; private final PoolChunkList<T> q050;
private final PoolChunkList<T> q025; private final PoolChunkList<T> q025;
@ -56,12 +54,12 @@ abstract class PoolArena<T> {
tinySubpagePools = newSubpagePoolArray(512 >>> 4); tinySubpagePools = newSubpagePoolArray(512 >>> 4);
for (int i = 0; i < tinySubpagePools.length; i ++) { for (int i = 0; i < tinySubpagePools.length; i ++) {
tinySubpagePools[i] = new ArrayDeque<PoolSubpage<T>>(); tinySubpagePools[i] = newSubpagePoolHead(pageSize);
} }
smallSubpagePools = newSubpagePoolArray(pageShifts - 9); smallSubpagePools = newSubpagePoolArray(pageShifts - 9);
for (int i = 0; i < smallSubpagePools.length; i ++) { for (int i = 0; i < smallSubpagePools.length; i ++) {
smallSubpagePools[i] = new ArrayDeque<PoolSubpage<T>>(); smallSubpagePools[i] = newSubpagePoolHead(pageSize);
} }
q100 = new PoolChunkList<T>(this, null, 100, Integer.MAX_VALUE); q100 = new PoolChunkList<T>(this, null, 100, Integer.MAX_VALUE);
@ -79,9 +77,16 @@ abstract class PoolArena<T> {
qInit.prevList = qInit; qInit.prevList = qInit;
} }
private PoolSubpage<T> newSubpagePoolHead(int pageSize) {
PoolSubpage<T> head = new PoolSubpage<T>(pageSize);
head.prev = head;
head.next = head;
return head;
}
@SuppressWarnings("unchecked") @SuppressWarnings("unchecked")
private Deque<PoolSubpage<T>>[] newSubpagePoolArray(int size) { private PoolSubpage<T>[] newSubpagePoolArray(int size) {
return new Deque[size]; return new PoolSubpage[size];
} }
PooledByteBuf<T> allocate(PoolThreadCache cache, int reqCapacity, int maxCapacity) { PooledByteBuf<T> allocate(PoolThreadCache cache, int reqCapacity, int maxCapacity) {
@ -94,7 +99,7 @@ abstract class PoolArena<T> {
final int normCapacity = normalizeCapacity(reqCapacity); final int normCapacity = normalizeCapacity(reqCapacity);
if ((normCapacity & subpageOverflowMask) == 0) { // capacity < pageSize if ((normCapacity & subpageOverflowMask) == 0) { // capacity < pageSize
int tableIdx; int tableIdx;
Deque<PoolSubpage<T>>[] table; PoolSubpage<T>[] table;
if ((normCapacity & 0xFFFFFE00) == 0) { // < 512 if ((normCapacity & 0xFFFFFE00) == 0) { // < 512
tableIdx = normCapacity >>> 4; tableIdx = normCapacity >>> 4;
table = tinySubpagePools; table = tinySubpagePools;
@ -109,28 +114,16 @@ abstract class PoolArena<T> {
} }
synchronized (this) { synchronized (this) {
Deque<PoolSubpage<T>> subpages = table[tableIdx]; final PoolSubpage<T> head = table[tableIdx];
for (;;) { final PoolSubpage<T> s = head.next;
PoolSubpage<T> s = subpages.peekFirst(); if (s != head) {
if (s == null) { assert s.doNotDestroy && s.elemSize == normCapacity;
break;
}
if (!s.doNotDestroy || s.elemSize != normCapacity) {
// The subpage has been destroyed or being used for different element size.
subpages.removeFirst();
continue;
}
long handle = s.allocate(); long handle = s.allocate();
if (handle < 0) { assert handle >= 0;
subpages.removeFirst();
} else {
s.chunk.initBufWithSubpage(buf, handle, reqCapacity); s.chunk.initBufWithSubpage(buf, handle, reqCapacity);
return; return;
} }
} }
}
} else if (normCapacity > chunkSize) { } else if (normCapacity > chunkSize) {
allocateHuge(buf, reqCapacity); allocateHuge(buf, reqCapacity);
return; return;
@ -142,7 +135,7 @@ abstract class PoolArena<T> {
private synchronized void allocateNormal(PooledByteBuf<T> buf, int reqCapacity, int normCapacity) { private synchronized void allocateNormal(PooledByteBuf<T> buf, int reqCapacity, int normCapacity) {
if (q050.allocate(buf, reqCapacity, normCapacity) || q025.allocate(buf, reqCapacity, normCapacity) || if (q050.allocate(buf, reqCapacity, normCapacity) || q025.allocate(buf, reqCapacity, normCapacity) ||
q000.allocate(buf, reqCapacity, normCapacity) || qInit.allocate(buf, reqCapacity, normCapacity) || q000.allocate(buf, reqCapacity, normCapacity) || qInit.allocate(buf, reqCapacity, normCapacity) ||
q075.allocate(buf, reqCapacity, normCapacity)) { q075.allocate(buf, reqCapacity, normCapacity) || q100.allocate(buf, reqCapacity, normCapacity)) {
return; return;
} }
@ -166,10 +159,9 @@ abstract class PoolArena<T> {
} }
} }
void addSubpage(PoolSubpage<T> subpage) { PoolSubpage<T> findSubpagePoolHead(int elemSize) {
int tableIdx; int tableIdx;
int elemSize = subpage.elemSize; PoolSubpage<T>[] table;
Deque<PoolSubpage<T>>[] table;
if ((elemSize & 0xFFFFFE00) == 0) { // < 512 if ((elemSize & 0xFFFFFE00) == 0) { // < 512
tableIdx = elemSize >>> 4; tableIdx = elemSize >>> 4;
table = tinySubpagePools; table = tinySubpagePools;
@ -183,7 +175,7 @@ abstract class PoolArena<T> {
table = smallSubpagePools; table = smallSubpagePools;
} }
table[tableIdx].addFirst(subpage); return table[tableIdx];
} }
private int normalizeCapacity(int reqCapacity) { private int normalizeCapacity(int reqCapacity) {
@ -288,28 +280,42 @@ abstract class PoolArena<T> {
buf.append(StringUtil.NEWLINE); buf.append(StringUtil.NEWLINE);
buf.append("tiny subpages:"); buf.append("tiny subpages:");
for (int i = 1; i < tinySubpagePools.length; i ++) { for (int i = 1; i < tinySubpagePools.length; i ++) {
Deque<PoolSubpage<T>> subpages = tinySubpagePools[i]; PoolSubpage<T> head = tinySubpagePools[i];
if (subpages.isEmpty()) { if (head.next == head) {
continue; continue;
} }
buf.append(StringUtil.NEWLINE); buf.append(StringUtil.NEWLINE);
buf.append(i); buf.append(i);
buf.append(": "); buf.append(": ");
buf.append(subpages); PoolSubpage<T> s = head.next;
for (;;) {
buf.append(s);
s = s.next;
if (s == head) {
break;
}
}
} }
buf.append(StringUtil.NEWLINE); buf.append(StringUtil.NEWLINE);
buf.append("small subpages:"); buf.append("small subpages:");
for (int i = 1; i < smallSubpagePools.length; i ++) { for (int i = 1; i < smallSubpagePools.length; i ++) {
Deque<PoolSubpage<T>> subpages = smallSubpagePools[i]; PoolSubpage<T> head = smallSubpagePools[i];
if (subpages.isEmpty()) { if (head.next == head) {
continue; continue;
} }
buf.append(StringUtil.NEWLINE); buf.append(StringUtil.NEWLINE);
buf.append(i); buf.append(i);
buf.append(": "); buf.append(": ");
buf.append(subpages); PoolSubpage<T> s = head.next;
for (;;) {
buf.append(s);
s = s.next;
if (s == head) {
break;
}
}
} }
buf.append(StringUtil.NEWLINE); buf.append(StringUtil.NEWLINE);

View File

@ -214,7 +214,6 @@ final class PoolChunk<T> {
} else { } else {
subpage.init(normCapacity); subpage.init(normCapacity);
} }
arena.addSubpage(subpage);
return subpage.allocate(); return subpage.allocate();
} }

View File

@ -24,6 +24,9 @@ final class PoolSubpage<T> {
final int pageSize; final int pageSize;
final long[] bitmap; final long[] bitmap;
PoolSubpage<T> prev;
PoolSubpage<T> next;
boolean doNotDestroy; boolean doNotDestroy;
int elemSize; int elemSize;
int maxNumElems; int maxNumElems;
@ -34,6 +37,16 @@ final class PoolSubpage<T> {
// TODO: Test if adding padding helps under contention // TODO: Test if adding padding helps under contention
//private long pad0, pad1, pad2, pad3, pad4, pad5, pad6, pad7; //private long pad0, pad1, pad2, pad3, pad4, pad5, pad6, pad7;
/** Special constructor that creates a linked list head */
PoolSubpage(int pageSize) {
chunk = null;
memoryMapIdx = -1;
runOffset = -1;
elemSize = -1;
this.pageSize = pageSize;
bitmap = null;
}
PoolSubpage(PoolChunk<T> chunk, int memoryMapIdx, int runOffset, int pageSize, int elemSize) { PoolSubpage(PoolChunk<T> chunk, int memoryMapIdx, int runOffset, int pageSize, int elemSize) {
this.chunk = chunk; this.chunk = chunk;
this.memoryMapIdx = memoryMapIdx; this.memoryMapIdx = memoryMapIdx;
@ -46,10 +59,7 @@ final class PoolSubpage<T> {
void init(int elemSize) { void init(int elemSize) {
doNotDestroy = true; doNotDestroy = true;
this.elemSize = elemSize; this.elemSize = elemSize;
if (elemSize == 0) { if (elemSize != 0) {
return;
}
maxNumElems = numAvail = pageSize / elemSize; maxNumElems = numAvail = pageSize / elemSize;
nextAvail = 0; nextAvail = 0;
bitmapLength = maxNumElems >>> 6; bitmapLength = maxNumElems >>> 6;
@ -62,6 +72,9 @@ final class PoolSubpage<T> {
} }
} }
addToPool();
}
/** /**
* Returns the bitmap index of the subpage allocation. * Returns the bitmap index of the subpage allocation.
*/ */
@ -81,6 +94,7 @@ final class PoolSubpage<T> {
bitmap[q] |= 1L << r; bitmap[q] |= 1L << r;
if (-- numAvail == 0) { if (-- numAvail == 0) {
removeFromPool();
nextAvail = -1; nextAvail = -1;
} else { } else {
nextAvail = findNextAvailable(); nextAvail = findNextAvailable();
@ -105,7 +119,7 @@ final class PoolSubpage<T> {
if (numAvail ++ == 0) { if (numAvail ++ == 0) {
nextAvail = bitmapIdx; nextAvail = bitmapIdx;
chunk.arena.addSubpage(this); addToPool();
return true; return true;
} }
@ -113,10 +127,28 @@ final class PoolSubpage<T> {
return true; return true;
} else { } else {
doNotDestroy = false; doNotDestroy = false;
removeFromPool();
return false; return false;
} }
} }
private void addToPool() {
PoolSubpage<T> head = chunk.arena.findSubpagePoolHead(elemSize);
assert prev == null && next == null;
prev = head;
next = head.next;
next.prev = this;
head.next = this;
}
private void removeFromPool() {
assert prev != null && next != null;
prev.next = next;
next.prev = prev;
next = null;
prev = null;
}
private int findNextAvailable() { private int findNextAvailable() {
int newNextAvail = -1; int newNextAvail = -1;
loop: loop: