Add PooledSlicedByteBuf and PooledDuplicatedByteBuf

Motivation:

At the moment when calling slice(...) or duplicate(...) on a Pooled*ByteBuf a new SlicedByteBuf or DuplicatedByteBuf. This can create a lot of GC.

Modifications:

Add PooledSlicedByteBuf and PooledDuplicatedByteBuf which will be used when a PooledByteBuf is used.

Result:

Less GC.
This commit is contained in:
Norman Maurer 2015-05-11 08:35:12 +02:00
parent 981292ffc0
commit 8a1a0b2c1e
6 changed files with 162 additions and 8 deletions

View File

@ -47,12 +47,20 @@ public abstract class AbstractDerivedByteBuf extends AbstractByteBuf {
@Override @Override
public final boolean release() { public final boolean release() {
return unwrap().release(); if (unwrap().release()) {
deallocate();
return true;
}
return false;
} }
@Override @Override
public final boolean release(int decrement) { public final boolean release(int decrement) {
return unwrap().release(decrement); if (unwrap().release(decrement)) {
deallocate();
return true;
}
return false;
} }
@Override @Override
@ -64,4 +72,9 @@ public abstract class AbstractDerivedByteBuf extends AbstractByteBuf {
public ByteBuffer nioBuffer(int index, int length) { public ByteBuffer nioBuffer(int index, int length) {
return unwrap().nioBuffer(index, length); return unwrap().nioBuffer(index, length);
} }
/**
* Called when the wrapped {@link ByteBuf} was released due calling of {@link #release()} or {@link #release(int)}.
*/
protected void deallocate() { }
} }

View File

@ -31,11 +31,19 @@ import java.nio.channels.ScatteringByteChannel;
*/ */
public class DuplicatedByteBuf extends AbstractDerivedByteBuf { public class DuplicatedByteBuf extends AbstractDerivedByteBuf {
private final ByteBuf buffer; private ByteBuf buffer;
public DuplicatedByteBuf(ByteBuf buffer) { public DuplicatedByteBuf(ByteBuf buffer) {
super(buffer.maxCapacity()); super(buffer.maxCapacity());
init(buffer);
}
DuplicatedByteBuf(int maxCapacity) {
super(maxCapacity);
}
final void init(ByteBuf buffer) {
maxCapacity(buffer.maxCapacity());
if (buffer instanceof DuplicatedByteBuf) { if (buffer instanceof DuplicatedByteBuf) {
this.buffer = ((DuplicatedByteBuf) buffer).buffer; this.buffer = ((DuplicatedByteBuf) buffer).buffer;
} else { } else {
@ -43,6 +51,8 @@ public class DuplicatedByteBuf extends AbstractDerivedByteBuf {
} }
setIndex(buffer.readerIndex(), buffer.writerIndex()); setIndex(buffer.readerIndex(), buffer.writerIndex());
markReaderIndex();
markWriterIndex();
} }
@Override @Override

View File

@ -68,6 +68,16 @@ abstract class PooledByteBuf<T> extends AbstractReferenceCountedByteBuf {
initThread = Thread.currentThread(); initThread = Thread.currentThread();
} }
@Override
public ByteBuf slice(int index, int length) {
return PooledSlicedByteBuf.newInstance(this, index, length);
}
@Override
public ByteBuf duplicate() {
return PooledDuplicatedByteBuf.newInstance(this);
}
@Override @Override
public final int capacity() { public final int capacity() {
return length; return length;

View File

@ -0,0 +1,56 @@
/*
* Copyright 2015 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.Recycler;
final class PooledDuplicatedByteBuf extends DuplicatedByteBuf {
private static final Recycler<PooledDuplicatedByteBuf> RECYCLER = new Recycler<PooledDuplicatedByteBuf>() {
@Override
protected PooledDuplicatedByteBuf newObject(Handle handle) {
return new PooledDuplicatedByteBuf(handle);
}
};
static PooledDuplicatedByteBuf newInstance(ByteBuf buffer) {
PooledDuplicatedByteBuf buf = RECYCLER.get();
buf.init(buffer);
return buf;
}
private final Recycler.Handle recyclerHandle;
private PooledDuplicatedByteBuf(Recycler.Handle recyclerHandle) {
super(0);
this.recyclerHandle = recyclerHandle;
}
@Override
public ByteBuf slice(int index, int length) {
return PooledSlicedByteBuf.newInstance(this, index, length);
}
@Override
public ByteBuf duplicate() {
return newInstance(this);
}
@Override
protected void deallocate() {
RECYCLER.recycle(this, recyclerHandle);
}
}

View File

@ -0,0 +1,56 @@
/*
* Copyright 2015 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.Recycler;
final class PooledSlicedByteBuf extends SlicedByteBuf {
private static final Recycler<PooledSlicedByteBuf> RECYCLER = new Recycler<PooledSlicedByteBuf>() {
@Override
protected PooledSlicedByteBuf newObject(Handle handle) {
return new PooledSlicedByteBuf(handle);
}
};
static PooledSlicedByteBuf newInstance(ByteBuf buffer, int index, int length) {
PooledSlicedByteBuf buf = RECYCLER.get();
buf.init(buffer, index, length);
return buf;
}
private final Recycler.Handle recyclerHandle;
private PooledSlicedByteBuf(Recycler.Handle recyclerHandle) {
super(0);
this.recyclerHandle = recyclerHandle;
}
@Override
public ByteBuf slice(int index, int length) {
return newInstance(this, index, length);
}
@Override
public ByteBuf duplicate() {
return PooledDuplicatedByteBuf.newInstance(this);
}
@Override
protected void deallocate() {
RECYCLER.recycle(this, recyclerHandle);
}
}

View File

@ -32,12 +32,20 @@ import java.nio.channels.ScatteringByteChannel;
*/ */
public class SlicedByteBuf extends AbstractDerivedByteBuf { public class SlicedByteBuf extends AbstractDerivedByteBuf {
private final ByteBuf buffer; private ByteBuf buffer;
private final int adjustment; private int adjustment;
private final int length; private int length;
public SlicedByteBuf(ByteBuf buffer, int index, int length) { public SlicedByteBuf(ByteBuf buffer, int index, int length) {
super(length); super(length);
init(buffer, index, length);
}
SlicedByteBuf(int length) {
super(length);
}
final void init(ByteBuf buffer, int index, int length) {
if (index < 0 || index > buffer.capacity() - length) { if (index < 0 || index > buffer.capacity() - length) {
throw new IndexOutOfBoundsException(buffer + ".slice(" + index + ", " + length + ')'); throw new IndexOutOfBoundsException(buffer + ".slice(" + index + ", " + length + ')');
} }
@ -53,8 +61,9 @@ public class SlicedByteBuf extends AbstractDerivedByteBuf {
adjustment = index; adjustment = index;
} }
this.length = length; this.length = length;
maxCapacity(length);
writerIndex(length); setIndex(0, length);
discardMarks();
} }
@Override @Override