Issue #1861 - Limit total bytes pooled by ByteBufferPools.

Code cleanups + javadocs.

Signed-off-by: Simone Bordet <simone.bordet@gmail.com>
This commit is contained in:
Simone Bordet 2019-02-25 16:14:41 +01:00
parent 259e31856a
commit f2d15f12d5
5 changed files with 200 additions and 153 deletions

View File

@ -20,93 +20,115 @@ package org.eclipse.jetty.io;
import java.nio.ByteBuffer; import java.nio.ByteBuffer;
import org.eclipse.jetty.util.annotation.ManagedObject;
import org.eclipse.jetty.util.annotation.ManagedOperation;
/**
* <p>A ByteBuffer pool where ByteBuffers are held in queues that are held in array elements.</p>
* <p>Given a capacity {@code factor} of 1024, the first array element holds a queue of ByteBuffers
* each of capacity 1024, the second array element holds a queue of ByteBuffers each of capacity
* 2048, and so on.</p>
*/
@ManagedObject
public class ArrayByteBufferPool implements ByteBufferPool public class ArrayByteBufferPool implements ByteBufferPool
{ {
private final int _min; private final int _minCapacity;
private final int _maxQueue;
private final ByteBufferPool.Bucket[] _direct; private final ByteBufferPool.Bucket[] _direct;
private final ByteBufferPool.Bucket[] _indirect; private final ByteBufferPool.Bucket[] _indirect;
private final int _inc; private final int _factor;
/**
* Creates a new ArrayByteBufferPool with a default configuration.
*/
public ArrayByteBufferPool() public ArrayByteBufferPool()
{ {
this(-1,-1,-1,-1); this(-1, -1, -1, -1);
} }
public ArrayByteBufferPool(int minSize, int increment, int maxSize) /**
* Creates a new ArrayByteBufferPool with the given configuration.
*
* @param minCapacity the minimum ByteBuffer capacity
* @param factor the capacity factor
* @param maxCapacity the maximum ByteBuffer capacity
*/
public ArrayByteBufferPool(int minCapacity, int factor, int maxCapacity)
{ {
this(minSize,increment,maxSize,-1); this(minCapacity, factor, maxCapacity, -1);
} }
public ArrayByteBufferPool(int minSize, int increment, int maxSize, int maxQueue) /**
* Creates a new ArrayByteBufferPool with the given configuration.
*
* @param minCapacity the minimum ByteBuffer capacity
* @param factor the capacity factor
* @param maxCapacity the maximum ByteBuffer capacity
* @param maxQueueLength the maximum ByteBuffer queue length
*/
public ArrayByteBufferPool(int minCapacity, int factor, int maxCapacity, int maxQueueLength)
{ {
if (minSize<=0) if (minCapacity <= 0)
minSize=0; minCapacity = 0;
if (increment<=0) if (factor <= 0)
increment=1024; factor = 1024;
if (maxSize<=0) if (maxCapacity <= 0)
maxSize=64*1024; maxCapacity = 64 * 1024;
if (minSize>=increment) if ((maxCapacity % factor) != 0 || factor >= maxCapacity)
throw new IllegalArgumentException("minSize >= increment"); throw new IllegalArgumentException("The capacity factor must be a divisor of maxCapacity");
if ((maxSize%increment)!=0 || increment>=maxSize) _minCapacity = minCapacity;
throw new IllegalArgumentException("increment must be a divisor of maxSize"); _factor = factor;
_min=minSize;
_inc=increment;
_direct=new ByteBufferPool.Bucket[maxSize/increment]; int length = maxCapacity / factor;
_indirect=new ByteBufferPool.Bucket[maxSize/increment]; _direct = new ByteBufferPool.Bucket[length];
_maxQueue=maxQueue; _indirect = new ByteBufferPool.Bucket[length];
int size=0; int capacity = 0;
for (int i=0;i<_direct.length;i++) for (int i = 0; i < _direct.length; ++i)
{ {
size+=_inc; capacity += _factor;
_direct[i]=new ByteBufferPool.Bucket(this,size,_maxQueue); _direct[i] = new ByteBufferPool.Bucket(this, capacity, maxQueueLength);
_indirect[i]=new ByteBufferPool.Bucket(this,size,_maxQueue); _indirect[i] = new ByteBufferPool.Bucket(this, capacity, maxQueueLength);
} }
} }
@Override @Override
public ByteBuffer acquire(int size, boolean direct) public ByteBuffer acquire(int size, boolean direct)
{ {
ByteBufferPool.Bucket bucket = bucketFor(size,direct); ByteBufferPool.Bucket bucket = bucketFor(size, direct);
if (bucket==null) if (bucket == null)
return newByteBuffer(size,direct); return newByteBuffer(size, direct);
return bucket.acquire(direct); return bucket.acquire(direct);
} }
@Override @Override
public void release(ByteBuffer buffer) public void release(ByteBuffer buffer)
{ {
if (buffer!=null) if (buffer != null)
{ {
ByteBufferPool.Bucket bucket = bucketFor(buffer.capacity(),buffer.isDirect()); ByteBufferPool.Bucket bucket = bucketFor(buffer.capacity(), buffer.isDirect());
if (bucket!=null) if (bucket != null)
bucket.release(buffer); bucket.release(buffer);
} }
} }
@ManagedOperation(value = "Clears this ByteBufferPool", impact = "ACTION")
public void clear() public void clear()
{ {
for (int i=0;i<_direct.length;i++) for (int i = 0; i < _direct.length; ++i)
{ {
_direct[i].clear(); _direct[i].clear();
_indirect[i].clear(); _indirect[i].clear();
} }
} }
private ByteBufferPool.Bucket bucketFor(int size,boolean direct) private ByteBufferPool.Bucket bucketFor(int capacity, boolean direct)
{ {
if (size<=_min) if (capacity <= _minCapacity)
return null; return null;
int b=(size-1)/_inc; int b = (capacity - 1) / _factor;
if (b>=_direct.length) if (b >= _direct.length)
return null; return null;
ByteBufferPool.Bucket bucket = direct?_direct[b]:_indirect[b]; return bucketsFor(direct)[b];
return bucket;
} }
// Package local for testing // Package local for testing

View File

@ -56,6 +56,13 @@ public interface ByteBufferPool
*/ */
public void release(ByteBuffer buffer); public void release(ByteBuffer buffer);
/**
* <p>Creates a new ByteBuffer of the given capacity and the given directness.</p>
*
* @param capacity the ByteBuffer capacity
* @param direct the ByteBuffer directness
* @return a newly allocated ByteBuffer
*/
default ByteBuffer newByteBuffer(int capacity, boolean direct) default ByteBuffer newByteBuffer(int capacity, boolean direct)
{ {
return direct ? BufferUtil.allocateDirect(capacity) : BufferUtil.allocate(capacity); return direct ? BufferUtil.allocateDirect(capacity) : BufferUtil.allocate(capacity);
@ -131,10 +138,10 @@ public interface ByteBufferPool
private final int _capacity; private final int _capacity;
private final AtomicInteger _space; private final AtomicInteger _space;
public Bucket(ByteBufferPool pool, int bufferSize, int maxSize) public Bucket(ByteBufferPool pool, int capacity, int maxSize)
{ {
_pool = pool; _pool = pool;
_capacity = bufferSize; _capacity = capacity;
_space = maxSize > 0 ? new AtomicInteger(maxSize) : null; _space = maxSize > 0 ? new AtomicInteger(maxSize) : null;
} }
@ -204,7 +211,7 @@ public interface ByteBufferPool
@Override @Override
public String toString() public String toString()
{ {
return String.format("Bucket@%x{%d/%d}", hashCode(), size(), _capacity); return String.format("%s@%x{%d/%d}", getClass().getSimpleName(), hashCode(), size(), _capacity);
} }
} }
} }

View File

@ -22,36 +22,58 @@ import java.nio.ByteBuffer;
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap; import java.util.concurrent.ConcurrentMap;
import java.util.concurrent.atomic.AtomicInteger; import java.util.concurrent.atomic.AtomicInteger;
import java.util.function.Function;
import org.eclipse.jetty.util.BufferUtil; import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.annotation.ManagedObject;
import org.eclipse.jetty.util.annotation.ManagedOperation;
/**
* <p>A ByteBuffer pool where ByteBuffers are held in queues that are held in a Map.</p>
* <p>Given a capacity {@code factor} of 1024, the Map entry with key {@code 1} holds a
* queue of ByteBuffers each of capacity 1024, the Map entry with key {@code 2} holds a
* queue of ByteBuffers each of capacity 2048, and so on.</p>
*/
@ManagedObject
public class MappedByteBufferPool implements ByteBufferPool public class MappedByteBufferPool implements ByteBufferPool
{ {
private final ConcurrentMap<Integer, Bucket> directBuffers = new ConcurrentHashMap<>(); private final ConcurrentMap<Integer, Bucket> directBuffers = new ConcurrentHashMap<>();
private final ConcurrentMap<Integer, Bucket> heapBuffers = new ConcurrentHashMap<>(); private final ConcurrentMap<Integer, Bucket> heapBuffers = new ConcurrentHashMap<>();
private final int _factor; private final int _factor;
private final Function<Integer, Bucket> _newBucket; private final int _maxQueueLength;
/**
* Creates a new MappedByteBufferPool with a default configuration.
*/
public MappedByteBufferPool() public MappedByteBufferPool()
{ {
this(-1); this(-1);
} }
/**
* Creates a new MappedByteBufferPool with the given capacity factor.
*
* @param factor the capacity factor
*/
public MappedByteBufferPool(int factor) public MappedByteBufferPool(int factor)
{ {
this(factor,-1,null); this(factor, -1);
} }
public MappedByteBufferPool(int factor,int maxQueue) /**
* Creates a new MappedByteBufferPool with the given capacity factor and max queue length.
*
* @param factor the capacity factor
* @param maxQueueLength the maximum ByteBuffer queue length
*/
public MappedByteBufferPool(int factor, int maxQueueLength)
{ {
this(factor,maxQueue,null); _factor = factor <= 0 ? 1024 : factor;
_maxQueueLength = maxQueueLength;
} }
public MappedByteBufferPool(int factor,int maxQueue,Function<Integer, Bucket> newBucket) private Bucket newBucket(int key)
{ {
_factor = factor<=0?1024:factor; return new Bucket(this, key * _factor, _maxQueueLength);
_newBucket = newBucket!=null?newBucket:i->new Bucket(this,i*_factor,maxQueue);
} }
@Override @Override
@ -59,10 +81,9 @@ public class MappedByteBufferPool implements ByteBufferPool
{ {
int b = bucketFor(size); int b = bucketFor(size);
ConcurrentMap<Integer, Bucket> buffers = bucketsFor(direct); ConcurrentMap<Integer, Bucket> buffers = bucketsFor(direct);
Bucket bucket = buffers.get(b); Bucket bucket = buffers.get(b);
if (bucket==null) if (bucket == null)
return newByteBuffer(b*_factor, direct); return newByteBuffer(b * _factor, direct);
return bucket.acquire(direct); return bucket.acquire(direct);
} }
@ -73,15 +94,16 @@ public class MappedByteBufferPool implements ByteBufferPool
return; // nothing to do return; // nothing to do
// validate that this buffer is from this pool // validate that this buffer is from this pool
assert((buffer.capacity() % _factor) == 0); assert ((buffer.capacity() % _factor) == 0);
int b = bucketFor(buffer.capacity()); int b = bucketFor(buffer.capacity());
ConcurrentMap<Integer, Bucket> buckets = bucketsFor(buffer.isDirect()); ConcurrentMap<Integer, Bucket> buckets = bucketsFor(buffer.isDirect());
Bucket bucket = buckets.computeIfAbsent(b,_newBucket); Bucket bucket = buckets.computeIfAbsent(b, this::newBucket);
bucket.release(buffer); bucket.release(buffer);
} }
@ManagedOperation(value = "Clears this ByteBufferPool", impact = "ACTION")
public void clear() public void clear()
{ {
directBuffers.values().forEach(Bucket::clear); directBuffers.values().forEach(Bucket::clear);

View File

@ -18,32 +18,33 @@
package org.eclipse.jetty.io; package org.eclipse.jetty.io;
import static org.hamcrest.Matchers.greaterThanOrEqualTo;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.junit.jupiter.api.Assertions.assertTrue;
import java.nio.ByteBuffer; import java.nio.ByteBuffer;
import java.util.Arrays; import java.util.Arrays;
import org.eclipse.jetty.io.ByteBufferPool.Bucket; import org.eclipse.jetty.io.ByteBufferPool.Bucket;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
@SuppressWarnings("ReferenceEquality") import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.greaterThanOrEqualTo;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertNotSame;
import static org.junit.jupiter.api.Assertions.assertSame;
import static org.junit.jupiter.api.Assertions.assertTrue;
public class ArrayByteBufferPoolTest public class ArrayByteBufferPoolTest
{ {
@Test @Test
public void testMinimumRelease() throws Exception public void testMinimumRelease()
{ {
ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(10,100,1000); ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(10, 100, 1000);
ByteBufferPool.Bucket[] buckets = bufferPool.bucketsFor(true); ByteBufferPool.Bucket[] buckets = bufferPool.bucketsFor(true);
for (int size=1;size<=9;size++) for (int size = 1; size <= 9; size++)
{ {
ByteBuffer buffer = bufferPool.acquire(size, true); ByteBuffer buffer = bufferPool.acquire(size, true);
assertTrue(buffer.isDirect()); assertTrue(buffer.isDirect());
assertEquals(size,buffer.capacity()); assertEquals(size, buffer.capacity());
for (ByteBufferPool.Bucket bucket : buckets) for (ByteBufferPool.Bucket bucket : buckets)
assertTrue(bucket.isEmpty()); assertTrue(bucket.isEmpty());
@ -55,39 +56,12 @@ public class ArrayByteBufferPoolTest
} }
@Test @Test
public void testMaxRelease() throws Exception public void testMaxRelease()
{ {
ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(10,100,1000); ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(10, 100, 1000);
ByteBufferPool.Bucket[] buckets = bufferPool.bucketsFor(true); ByteBufferPool.Bucket[] buckets = bufferPool.bucketsFor(true);
for (int size=999;size<=1001;size++) for (int size = 999; size <= 1001; size++)
{
bufferPool.clear();
ByteBuffer buffer = bufferPool.acquire(size, true);
assertTrue(buffer.isDirect());
assertThat(buffer.capacity(),greaterThanOrEqualTo(size));
for (ByteBufferPool.Bucket bucket : buckets)
assertTrue(bucket.isEmpty());
bufferPool.release(buffer);
int pooled=0;
for (ByteBufferPool.Bucket bucket : buckets)
{
pooled+=bucket.size();
}
assertEquals(size<=1000,1==pooled);
}
}
@Test
public void testAcquireRelease() throws Exception
{
ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(10,100,1000);
ByteBufferPool.Bucket[] buckets = bufferPool.bucketsFor(true);
for (int size=390;size<=510;size++)
{ {
bufferPool.clear(); bufferPool.clear();
ByteBuffer buffer = bufferPool.acquire(size, true); ByteBuffer buffer = bufferPool.acquire(size, true);
@ -99,28 +73,54 @@ public class ArrayByteBufferPoolTest
bufferPool.release(buffer); bufferPool.release(buffer);
int pooled=0; int pooled = 0;
for (ByteBufferPool.Bucket bucket : buckets) for (ByteBufferPool.Bucket bucket : buckets)
{ {
if (!bucket.isEmpty()) pooled += bucket.size();
{
pooled+=bucket.size();
// TODO assertThat(bucket._bufferSize,greaterThanOrEqualTo(size));
// TODO assertThat(bucket._bufferSize,Matchers.lessThan(size+100));
}
} }
assertEquals(1,pooled); assertEquals(size <= 1000, 1 == pooled);
} }
} }
@Test @Test
@SuppressWarnings("ReferenceEquality") public void testAcquireRelease()
public void testAcquireReleaseAcquire() throws Exception
{ {
ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(10,100,1000); ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(10, 100, 1000);
ByteBufferPool.Bucket[] buckets = bufferPool.bucketsFor(true); ByteBufferPool.Bucket[] buckets = bufferPool.bucketsFor(true);
for (int size=390;size<=510;size++) for (int size = 390; size <= 510; size++)
{
bufferPool.clear();
ByteBuffer buffer = bufferPool.acquire(size, true);
assertTrue(buffer.isDirect());
assertThat(buffer.capacity(), greaterThanOrEqualTo(size));
for (ByteBufferPool.Bucket bucket : buckets)
assertTrue(bucket.isEmpty());
bufferPool.release(buffer);
int pooled = 0;
for (ByteBufferPool.Bucket bucket : buckets)
{
if (!bucket.isEmpty())
{
pooled += bucket.size();
// TODO assertThat(bucket._bufferSize,greaterThanOrEqualTo(size));
// TODO assertThat(bucket._bufferSize,Matchers.lessThan(size+100));
}
}
assertEquals(1, pooled);
}
}
@Test
public void testAcquireReleaseAcquire()
{
ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(10, 100, 1000);
ByteBufferPool.Bucket[] buckets = bufferPool.bucketsFor(true);
for (int size = 390; size <= 510; size++)
{ {
bufferPool.clear(); bufferPool.clear();
ByteBuffer buffer1 = bufferPool.acquire(size, true); ByteBuffer buffer1 = bufferPool.acquire(size, true);
@ -130,38 +130,37 @@ public class ArrayByteBufferPoolTest
ByteBuffer buffer3 = bufferPool.acquire(size, false); ByteBuffer buffer3 = bufferPool.acquire(size, false);
bufferPool.release(buffer3); bufferPool.release(buffer3);
int pooled=0; int pooled = 0;
for (ByteBufferPool.Bucket bucket : buckets) for (ByteBufferPool.Bucket bucket : buckets)
{ {
if (!bucket.isEmpty()) if (!bucket.isEmpty())
{ {
pooled+=bucket.size(); pooled += bucket.size();
// TODO assertThat(bucket._bufferSize,greaterThanOrEqualTo(size)); // TODO assertThat(bucket._bufferSize,greaterThanOrEqualTo(size));
// TODO assertThat(bucket._bufferSize,Matchers.lessThan(size+100)); // TODO assertThat(bucket._bufferSize,Matchers.lessThan(size+100));
} }
} }
assertEquals(1,pooled); assertEquals(1, pooled);
assertTrue(buffer1==buffer2); assertSame(buffer1, buffer2);
assertTrue(buffer1!=buffer3); assertNotSame(buffer1, buffer3);
} }
} }
@Test @Test
public void testMaxQueue() throws Exception public void testMaxQueue()
{ {
ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(-1,-1,-1,2); ArrayByteBufferPool bufferPool = new ArrayByteBufferPool(-1, -1, -1, 2);
ByteBuffer buffer1 = bufferPool.acquire(512, false); ByteBuffer buffer1 = bufferPool.acquire(512, false);
ByteBuffer buffer2 = bufferPool.acquire(512, false); ByteBuffer buffer2 = bufferPool.acquire(512, false);
ByteBuffer buffer3 = bufferPool.acquire(512, false); ByteBuffer buffer3 = bufferPool.acquire(512, false);
Bucket[] buckets = bufferPool.bucketsFor(false); Bucket[] buckets = bufferPool.bucketsFor(false);
Arrays.asList(buckets).forEach(b->assertEquals(0,b.size())); Arrays.asList(buckets).forEach(b -> assertEquals(0, b.size()));
bufferPool.release(buffer1); bufferPool.release(buffer1);
Bucket bucket=Arrays.asList(buckets).stream().filter(b->b.size()>0).findFirst().get(); Bucket bucket = Arrays.stream(buckets).filter(b -> b.size() > 0).findFirst().get();
assertEquals(1, bucket.size()); assertEquals(1, bucket.size());
bufferPool.release(buffer2); bufferPool.release(buffer2);
@ -170,5 +169,4 @@ public class ArrayByteBufferPoolTest
bufferPool.release(buffer3); bufferPool.release(buffer3);
assertEquals(2, bucket.size()); assertEquals(2, bucket.size());
} }
} }

View File

@ -26,21 +26,21 @@ import org.eclipse.jetty.util.BufferUtil;
import org.eclipse.jetty.util.StringUtil; import org.eclipse.jetty.util.StringUtil;
import org.junit.jupiter.api.Test; import org.junit.jupiter.api.Test;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.hamcrest.Matchers.containsString; import static org.hamcrest.Matchers.containsString;
import static org.hamcrest.Matchers.greaterThanOrEqualTo; import static org.hamcrest.Matchers.greaterThanOrEqualTo;
import static org.junit.jupiter.api.Assertions.assertEquals; import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertSame; import static org.junit.jupiter.api.Assertions.assertSame;
import static org.hamcrest.MatcherAssert.assertThat;
import static org.junit.jupiter.api.Assertions.assertTrue; import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail; import static org.junit.jupiter.api.Assertions.fail;
public class MappedByteBufferPoolTest public class MappedByteBufferPoolTest
{ {
@Test @Test
public void testAcquireRelease() throws Exception public void testAcquireRelease()
{ {
MappedByteBufferPool bufferPool = new MappedByteBufferPool(); MappedByteBufferPool bufferPool = new MappedByteBufferPool();
ConcurrentMap<Integer,Bucket> buckets = bufferPool.bucketsFor(true); ConcurrentMap<Integer, Bucket> buckets = bufferPool.bucketsFor(true);
int size = 512; int size = 512;
ByteBuffer buffer = bufferPool.acquire(size, true); ByteBuffer buffer = bufferPool.acquire(size, true);
@ -56,10 +56,10 @@ public class MappedByteBufferPoolTest
} }
@Test @Test
public void testAcquireReleaseAcquire() throws Exception public void testAcquireReleaseAcquire()
{ {
MappedByteBufferPool bufferPool = new MappedByteBufferPool(); MappedByteBufferPool bufferPool = new MappedByteBufferPool();
ConcurrentMap<Integer,Bucket> buckets = bufferPool.bucketsFor(false); ConcurrentMap<Integer, Bucket> buckets = bufferPool.bucketsFor(false);
ByteBuffer buffer1 = bufferPool.acquire(512, false); ByteBuffer buffer1 = bufferPool.acquire(512, false);
bufferPool.release(buffer1); bufferPool.release(buffer1);
@ -76,10 +76,10 @@ public class MappedByteBufferPoolTest
} }
@Test @Test
public void testAcquireReleaseClear() throws Exception public void testAcquireReleaseClear()
{ {
MappedByteBufferPool bufferPool = new MappedByteBufferPool(); MappedByteBufferPool bufferPool = new MappedByteBufferPool();
ConcurrentMap<Integer,Bucket> buckets = bufferPool.bucketsFor(true); ConcurrentMap<Integer, Bucket> buckets = bufferPool.bucketsFor(true);
ByteBuffer buffer = bufferPool.acquire(512, true); ByteBuffer buffer = bufferPool.acquire(512, true);
bufferPool.release(buffer); bufferPool.release(buffer);
@ -96,11 +96,9 @@ public class MappedByteBufferPoolTest
* In a scenario where MappedByteBufferPool is being used improperly, * In a scenario where MappedByteBufferPool is being used improperly,
* such as releasing a buffer that wasn't created/acquired by the * such as releasing a buffer that wasn't created/acquired by the
* MappedByteBufferPool, an assertion is tested for. * MappedByteBufferPool, an assertion is tested for.
*
* @throws Exception test failure
*/ */
@Test @Test
public void testReleaseAssertion() throws Exception public void testReleaseAssertion()
{ {
int factor = 1024; int factor = 1024;
MappedByteBufferPool bufferPool = new MappedByteBufferPool(factor); MappedByteBufferPool bufferPool = new MappedByteBufferPool(factor);
@ -129,18 +127,18 @@ public class MappedByteBufferPoolTest
{ {
MappedByteBufferPool pool = new MappedByteBufferPool.Tagged(); MappedByteBufferPool pool = new MappedByteBufferPool.Tagged();
ByteBuffer buffer = pool.acquire(1024,false); ByteBuffer buffer = pool.acquire(1024, false);
assertThat(BufferUtil.toDetailString(buffer),containsString("@T00000001")); assertThat(BufferUtil.toDetailString(buffer), containsString("@T00000001"));
buffer = pool.acquire(1024,false); buffer = pool.acquire(1024, false);
assertThat(BufferUtil.toDetailString(buffer),containsString("@T00000002")); assertThat(BufferUtil.toDetailString(buffer), containsString("@T00000002"));
} }
@Test @Test
public void testMaxQueue() throws Exception public void testMaxQueue()
{ {
MappedByteBufferPool bufferPool = new MappedByteBufferPool(-1,2); MappedByteBufferPool bufferPool = new MappedByteBufferPool(-1, 2);
ConcurrentMap<Integer,Bucket> buckets = bufferPool.bucketsFor(false); ConcurrentMap<Integer, Bucket> buckets = bufferPool.bucketsFor(false);
ByteBuffer buffer1 = bufferPool.acquire(512, false); ByteBuffer buffer1 = bufferPool.acquire(512, false);
ByteBuffer buffer2 = bufferPool.acquire(512, false); ByteBuffer buffer2 = bufferPool.acquire(512, false);
@ -149,7 +147,7 @@ public class MappedByteBufferPoolTest
bufferPool.release(buffer1); bufferPool.release(buffer1);
assertEquals(1, buckets.size()); assertEquals(1, buckets.size());
Bucket bucket=buckets.values().iterator().next(); Bucket bucket = buckets.values().iterator().next();
assertEquals(1, bucket.size()); assertEquals(1, bucket.size());
bufferPool.release(buffer2); bufferPool.release(buffer2);