/* * Copyright 2012 The Netty Project * * The Netty Project licenses this file to you under the Apache License, * version 2.0 (the "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the * License for the specific language governing permissions and limitations * under the License. */ package io.netty.buffer; import io.netty.util.internal.PlatformDependent; import io.netty.util.internal.StringUtil; import java.nio.ByteBuffer; abstract class PoolArena<T> { final PooledByteBufAllocator parent; private final int pageSize; private final int maxOrder; private final int pageShifts; private final int chunkSize; private final int subpageOverflowMask; private final PoolSubpage<T>[] tinySubpagePools; private final PoolSubpage<T>[] smallSubpagePools; private final PoolChunkList<T> q050; private final PoolChunkList<T> q025; private final PoolChunkList<T> q000; private final PoolChunkList<T> qInit; private final PoolChunkList<T> q075; private final PoolChunkList<T> q100; // TODO: Test if adding padding helps under contention //private long pad0, pad1, pad2, pad3, pad4, pad5, pad6, pad7; protected PoolArena(PooledByteBufAllocator parent, int pageSize, int maxOrder, int pageShifts, int chunkSize) { this.parent = parent; this.pageSize = pageSize; this.maxOrder = maxOrder; this.pageShifts = pageShifts; this.chunkSize = chunkSize; subpageOverflowMask = ~(pageSize - 1); tinySubpagePools = newSubpagePoolArray(512 >>> 4); for (int i = 0; i < tinySubpagePools.length; i ++) { tinySubpagePools[i] = newSubpagePoolHead(pageSize); } smallSubpagePools = newSubpagePoolArray(pageShifts - 9); for (int i = 0; i < smallSubpagePools.length; i ++) { smallSubpagePools[i] = newSubpagePoolHead(pageSize); } q100 = new PoolChunkList<T>(this, null, 100, Integer.MAX_VALUE); q075 = new PoolChunkList<T>(this, q100, 75, 100); q050 = new PoolChunkList<T>(this, q075, 50, 100); q025 = new PoolChunkList<T>(this, q050, 25, 75); q000 = new PoolChunkList<T>(this, q025, 1, 50); qInit = new PoolChunkList<T>(this, q000, Integer.MIN_VALUE, 25); q100.prevList = q075; q075.prevList = q050; q050.prevList = q025; q025.prevList = q000; q000.prevList = null; 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") private PoolSubpage<T>[] newSubpagePoolArray(int size) { return new PoolSubpage[size]; } PooledByteBuf<T> allocate(PoolThreadCache cache, int reqCapacity, int maxCapacity) { PooledByteBuf<T> buf = newByteBuf(maxCapacity); allocate(cache, buf, reqCapacity); return buf; } private void allocate(PoolThreadCache cache, PooledByteBuf<T> buf, final int reqCapacity) { final int normCapacity = normalizeCapacity(reqCapacity); if ((normCapacity & subpageOverflowMask) == 0) { // capacity < pageSize int tableIdx; PoolSubpage<T>[] table; if ((normCapacity & 0xFFFFFE00) == 0) { // < 512 tableIdx = normCapacity >>> 4; table = tinySubpagePools; } else { tableIdx = 0; int i = normCapacity >>> 10; while (i != 0) { i >>>= 1; tableIdx ++; } table = smallSubpagePools; } synchronized (this) { final PoolSubpage<T> head = table[tableIdx]; final PoolSubpage<T> s = head.next; if (s != head) { assert s.doNotDestroy && s.elemSize == normCapacity; long handle = s.allocate(); assert handle >= 0; s.chunk.initBufWithSubpage(buf, handle, reqCapacity); return; } } } else if (normCapacity > chunkSize) { allocateHuge(buf, reqCapacity); return; } allocateNormal(buf, reqCapacity, normCapacity); } private synchronized void allocateNormal(PooledByteBuf<T> buf, int reqCapacity, int normCapacity) { if (q050.allocate(buf, reqCapacity, normCapacity) || q025.allocate(buf, reqCapacity, normCapacity) || q000.allocate(buf, reqCapacity, normCapacity) || qInit.allocate(buf, reqCapacity, normCapacity) || q075.allocate(buf, reqCapacity, normCapacity) || q100.allocate(buf, reqCapacity, normCapacity)) { return; } // Add a new chunk. PoolChunk<T> c = newChunk(pageSize, maxOrder, pageShifts, chunkSize); long handle = c.allocate(normCapacity); assert handle > 0; c.initBuf(buf, handle, reqCapacity); qInit.add(c); } private void allocateHuge(PooledByteBuf<T> buf, int reqCapacity) { buf.initUnpooled(newUnpooledChunk(reqCapacity), reqCapacity); } void free(PoolChunk<T> chunk, long handle) { if (chunk.unpooled) { destroyChunk(chunk); } else { synchronized (this) { chunk.parent.free(chunk, handle); } } } PoolSubpage<T> findSubpagePoolHead(int elemSize) { int tableIdx; PoolSubpage<T>[] table; if ((elemSize & 0xFFFFFE00) == 0) { // < 512 tableIdx = elemSize >>> 4; table = tinySubpagePools; } else { tableIdx = 0; elemSize >>>= 10; while (elemSize != 0) { elemSize >>>= 1; tableIdx ++; } table = smallSubpagePools; } return table[tableIdx]; } private int normalizeCapacity(int reqCapacity) { if (reqCapacity < 0) { throw new IllegalArgumentException("capacity: " + reqCapacity + " (expected: 0+)"); } if (reqCapacity >= chunkSize) { return reqCapacity; } if ((reqCapacity & 0xFFFFFE00) != 0) { // >= 512 // Doubled int normalizedCapacity = reqCapacity; normalizedCapacity |= normalizedCapacity >>> 1; normalizedCapacity |= normalizedCapacity >>> 2; normalizedCapacity |= normalizedCapacity >>> 4; normalizedCapacity |= normalizedCapacity >>> 8; normalizedCapacity |= normalizedCapacity >>> 16; normalizedCapacity ++; if (normalizedCapacity < 0) { normalizedCapacity >>>= 1; } return normalizedCapacity; } // Quantum-spaced if ((reqCapacity & 15) == 0) { return reqCapacity; } return (reqCapacity & ~15) + 16; } void reallocate(PooledByteBuf<T> buf, int newCapacity, boolean freeOldMemory) { if (newCapacity < 0 || newCapacity > buf.maxCapacity()) { throw new IllegalArgumentException("newCapacity: " + newCapacity); } int oldCapacity = buf.length; if (oldCapacity == newCapacity) { return; } PoolChunk<T> oldChunk = buf.chunk; long oldHandle = buf.handle; T oldMemory = buf.memory; int oldOffset = buf.offset; int readerIndex = buf.readerIndex(); int writerIndex = buf.writerIndex(); allocate(parent.threadCache.get(), buf, newCapacity); if (newCapacity > oldCapacity) { memoryCopy( oldMemory, oldOffset, buf.memory, buf.offset, oldCapacity); } else if (newCapacity < oldCapacity) { if (readerIndex < newCapacity) { if (writerIndex > newCapacity) { writerIndex = newCapacity; } memoryCopy( oldMemory, oldOffset + readerIndex, buf.memory, buf.offset + readerIndex, writerIndex - readerIndex); } else { readerIndex = writerIndex = newCapacity; } } buf.setIndex(readerIndex, writerIndex); if (freeOldMemory) { free(oldChunk, oldHandle); } } protected abstract PoolChunk<T> newChunk(int pageSize, int maxOrder, int pageShifts, int chunkSize); protected abstract PoolChunk<T> newUnpooledChunk(int capacity); protected abstract PooledByteBuf<T> newByteBuf(int maxCapacity); protected abstract void memoryCopy(T src, int srcOffset, T dst, int dstOffset, int length); protected abstract void destroyChunk(PoolChunk<T> chunk); public synchronized String toString() { StringBuilder buf = new StringBuilder(); buf.append("Chunk(s) at 0~25%:"); buf.append(StringUtil.NEWLINE); buf.append(qInit); buf.append(StringUtil.NEWLINE); buf.append("Chunk(s) at 0~50%:"); buf.append(StringUtil.NEWLINE); buf.append(q000); buf.append(StringUtil.NEWLINE); buf.append("Chunk(s) at 25~75%:"); buf.append(StringUtil.NEWLINE); buf.append(q025); buf.append(StringUtil.NEWLINE); buf.append("Chunk(s) at 50~100%:"); buf.append(StringUtil.NEWLINE); buf.append(q050); buf.append(StringUtil.NEWLINE); buf.append("Chunk(s) at 75~100%:"); buf.append(StringUtil.NEWLINE); buf.append(q075); buf.append(StringUtil.NEWLINE); buf.append("Chunk(s) at 100%:"); buf.append(StringUtil.NEWLINE); buf.append(q100); buf.append(StringUtil.NEWLINE); buf.append("tiny subpages:"); for (int i = 1; i < tinySubpagePools.length; i ++) { PoolSubpage<T> head = tinySubpagePools[i]; if (head.next == head) { continue; } buf.append(StringUtil.NEWLINE); buf.append(i); buf.append(": "); PoolSubpage<T> s = head.next; for (;;) { buf.append(s); s = s.next; if (s == head) { break; } } } buf.append(StringUtil.NEWLINE); buf.append("small subpages:"); for (int i = 1; i < smallSubpagePools.length; i ++) { PoolSubpage<T> head = smallSubpagePools[i]; if (head.next == head) { continue; } buf.append(StringUtil.NEWLINE); buf.append(i); buf.append(": "); PoolSubpage<T> s = head.next; for (;;) { buf.append(s); s = s.next; if (s == head) { break; } } } buf.append(StringUtil.NEWLINE); return buf.toString(); } static final class HeapArena extends PoolArena<byte[]> { HeapArena(PooledByteBufAllocator parent, int pageSize, int maxOrder, int pageShifts, int chunkSize) { super(parent, pageSize, maxOrder, pageShifts, chunkSize); } @Override protected PoolChunk<byte[]> newChunk(int pageSize, int maxOrder, int pageShifts, int chunkSize) { return new PoolChunk<byte[]>(this, new byte[chunkSize], pageSize, maxOrder, pageShifts, chunkSize); } @Override protected PoolChunk<byte[]> newUnpooledChunk(int capacity) { return new PoolChunk<byte[]>(this, new byte[capacity], capacity); } @Override protected void destroyChunk(PoolChunk<byte[]> chunk) { // Rely on GC. } @Override protected PooledByteBuf<byte[]> newByteBuf(int maxCapacity) { return PooledHeapByteBuf.newInstance(maxCapacity); } @Override protected void memoryCopy(byte[] src, int srcOffset, byte[] dst, int dstOffset, int length) { if (length == 0) { return; } System.arraycopy(src, srcOffset, dst, dstOffset, length); } } static final class DirectArena extends PoolArena<ByteBuffer> { private static final boolean HAS_UNSAFE = PlatformDependent.hasUnsafe(); DirectArena(PooledByteBufAllocator parent, int pageSize, int maxOrder, int pageShifts, int chunkSize) { super(parent, pageSize, maxOrder, pageShifts, chunkSize); } @Override protected PoolChunk<ByteBuffer> newChunk(int pageSize, int maxOrder, int pageShifts, int chunkSize) { return new PoolChunk<ByteBuffer>( this, ByteBuffer.allocateDirect(chunkSize), pageSize, maxOrder, pageShifts, chunkSize); } @Override protected PoolChunk<ByteBuffer> newUnpooledChunk(int capacity) { return new PoolChunk<ByteBuffer>(this, ByteBuffer.allocateDirect(capacity), capacity); } @Override protected void destroyChunk(PoolChunk<ByteBuffer> chunk) { PlatformDependent.freeDirectBuffer(chunk.memory); } @Override protected PooledByteBuf<ByteBuffer> newByteBuf(int maxCapacity) { if (HAS_UNSAFE) { return PooledUnsafeDirectByteBuf.newInstance(maxCapacity); } else { return PooledDirectByteBuf.newInstance(maxCapacity); } } @Override protected void memoryCopy(ByteBuffer src, int srcOffset, ByteBuffer dst, int dstOffset, int length) { if (length == 0) { return; } if (HAS_UNSAFE) { PlatformDependent.copyMemory( PlatformDependent.directBufferAddress(src) + srcOffset, PlatformDependent.directBufferAddress(dst) + dstOffset, length); } else { // We must duplicate the NIO buffers because they may be accessed by other Netty buffers. src = src.duplicate(); dst = dst.duplicate(); src.position(srcOffset).limit(srcOffset + length); dst.position(dstOffset); dst.put(src); } } } }