forked from netty/netty
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Directly use memory addresses for gathering writes to reduce gc press…
…ure. Part of [netty#2239] This also does factor out some logic of ChannelOutboundBuffer. Mainly we not need nioBuffers() for many transports and also not need to copy from heap to direct buffer. So this functionality was moved to NioSocketChannelOutboundBuffer. Also introduce a EpollChannelOutboundBuffer which makes use of memory addresses for all the writes to reduce GC pressure
- Loading branch information
Norman Maurer
committed
Feb 21, 2014
1 parent
bea8107
commit 60b830b
Showing
20 changed files
with
1,020 additions
and
392 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
202 changes: 202 additions & 0 deletions
202
transport-native-epoll/src/main/java/io/netty/channel/epoll/EpollChannelOutboundBuffer.java
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,202 @@ | ||
/* | ||
* Copyright 2014 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.channel.epoll; | ||
|
||
import io.netty.buffer.ByteBuf; | ||
import io.netty.channel.ChannelOutboundBuffer; | ||
import io.netty.util.Recycler; | ||
|
||
import java.nio.ByteBuffer; | ||
import java.util.Arrays; | ||
|
||
/** | ||
* Special {@link ChannelOutboundBuffer} implementation which allows to obtain an array of {@link AddressEntry} | ||
* and so doing gathering writes without the need to create a {@link ByteBuffer} internally. This reduce | ||
* GC pressure a lot. | ||
*/ | ||
final class EpollChannelOutboundBuffer extends ChannelOutboundBuffer { | ||
private AddressEntry[] addresses; | ||
private int addressCount; | ||
private long addressSize; | ||
private static final Recycler<EpollChannelOutboundBuffer> RECYCLER = new Recycler<EpollChannelOutboundBuffer>() { | ||
@Override | ||
protected EpollChannelOutboundBuffer newObject(Handle<EpollChannelOutboundBuffer> handle) { | ||
return new EpollChannelOutboundBuffer(handle); | ||
} | ||
}; | ||
|
||
/** | ||
* Get a new instance of this {@link EpollChannelOutboundBuffer} and attach it the given {@link EpollSocketChannel} | ||
*/ | ||
static EpollChannelOutboundBuffer newInstance(EpollSocketChannel channel) { | ||
EpollChannelOutboundBuffer buffer = RECYCLER.get(); | ||
buffer.channel = channel; | ||
return buffer; | ||
} | ||
|
||
private EpollChannelOutboundBuffer(Recycler.Handle<? extends ChannelOutboundBuffer> handle) { | ||
super(handle); | ||
addresses = new AddressEntry[INITIAL_CAPACITY]; | ||
} | ||
|
||
/** | ||
* Check if the message is a {@link ByteBuf} and if so if it has a memoryAddress. If not it will convert this | ||
* {@link ByteBuf} to be able to operate on the memoryAddress directly for maximal performance. | ||
*/ | ||
@Override | ||
protected Object beforeAdd(Object msg) { | ||
if (msg instanceof ByteBuf) { | ||
ByteBuf buf = (ByteBuf) msg; | ||
if (!buf.hasMemoryAddress()) { | ||
ByteBuf direct = copyToDirectByteBuf(buf); | ||
return direct; | ||
} | ||
} | ||
return msg; | ||
} | ||
|
||
/** | ||
* Returns an array of {@link AddressEntry}'s if the currently pending messages are made of {@link ByteBuf} only. | ||
* {@code null} is returned otherwise. If this method returns a non-null array, {@link #addressCount()} and | ||
* {@link #addressSize()} ()} will return the number of {@link AddressEntry}'s in the returned array and the total | ||
* number of readable bytes of the NIO buffers respectively. | ||
* <p> | ||
* Note that the returned array is reused and thus should not escape | ||
* {@link io.netty.channel.AbstractChannel#doWrite(ChannelOutboundBuffer)}. | ||
* Refer to {@link EpollSocketChannel#doWrite(ChannelOutboundBuffer)} for an example. | ||
* </p> | ||
*/ | ||
AddressEntry[] memoryAddresses() { | ||
long addressSize = 0; | ||
int addressCount = 0; | ||
final Entry[] buffer = entries(); | ||
final int mask = buffer.length - 1; | ||
AddressEntry[] addresses = this.addresses; | ||
Object m; | ||
int unflushed = unflushed(); | ||
int flushed = flushed(); | ||
while (flushed != unflushed && (m = buffer[flushed].msg()) != null) { | ||
if (!(m instanceof ByteBuf)) { | ||
this.addressCount = 0; | ||
this.addressSize = 0; | ||
return null; | ||
} | ||
|
||
AddressEntry entry = (AddressEntry) buffer[flushed]; | ||
|
||
// Check if the entry was cancelled. if so we just skip it. | ||
if (!entry.isCancelled()) { | ||
ByteBuf buf = (ByteBuf) m; | ||
final int readerIndex = buf.readerIndex(); | ||
final int readableBytes = buf.writerIndex() - readerIndex; | ||
|
||
if (readableBytes > 0) { | ||
addressSize += readableBytes; | ||
// See if there is enough space to at least store one more entry. | ||
int neededSpace = addressCount + 1; | ||
if (neededSpace > addresses.length) { | ||
this.addresses = addresses = | ||
expandAddressesArray(addresses, neededSpace, addressCount); | ||
} | ||
entry.memoryAddress = buf.memoryAddress(); | ||
entry.readerIndex = buf.readerIndex(); | ||
entry.writerIndex = buf.writerIndex(); | ||
|
||
addresses[addressCount ++] = entry; | ||
} | ||
} | ||
|
||
flushed = flushed + 1 & mask; | ||
} | ||
this.addressCount = addressCount; | ||
this.addressSize = addressSize; | ||
|
||
return addresses; | ||
} | ||
|
||
private static AddressEntry[] expandAddressesArray(AddressEntry[] array, int neededSpace, int size) { | ||
int newCapacity = array.length; | ||
do { | ||
// double capacity until it is big enough | ||
// See https://github.com/netty/netty/issues/1890 | ||
newCapacity <<= 1; | ||
|
||
if (newCapacity < 0) { | ||
throw new IllegalStateException(); | ||
} | ||
|
||
} while (neededSpace > newCapacity); | ||
|
||
AddressEntry[] newArray = new AddressEntry[newCapacity]; | ||
System.arraycopy(array, 0, newArray, 0, size); | ||
|
||
return newArray; | ||
} | ||
|
||
/** | ||
* Return the number of {@link AddressEntry}'s which can be written. | ||
*/ | ||
int addressCount() { | ||
return addressCount; | ||
} | ||
|
||
/** | ||
* Return the number of bytes that can be written via gathering writes. | ||
*/ | ||
long addressSize() { | ||
return addressSize; | ||
} | ||
|
||
@Override | ||
public void recycle() { | ||
if (addresses.length > INITIAL_CAPACITY) { | ||
addresses = new AddressEntry[INITIAL_CAPACITY]; | ||
} else { | ||
// null out the nio buffers array so the can be GC'ed | ||
// https://github.com/netty/netty/issues/1763 | ||
Arrays.fill(addresses, null); | ||
} | ||
super.recycle(); | ||
} | ||
|
||
@Override | ||
protected AddressEntry newEntry() { | ||
return new AddressEntry(); | ||
} | ||
|
||
static final class AddressEntry extends Entry { | ||
// These fields will be accessed via JNI directly so be carefully when touch them! | ||
long memoryAddress; | ||
int readerIndex; | ||
int writerIndex; | ||
|
||
@Override | ||
public void clear() { | ||
memoryAddress = -1; | ||
readerIndex = 0; | ||
writerIndex = 0; | ||
super.clear(); | ||
} | ||
|
||
@Override | ||
public int cancel() { | ||
memoryAddress = -1; | ||
readerIndex = 0; | ||
writerIndex = 0; | ||
return super.cancel(); | ||
} | ||
} | ||
} |
Oops, something went wrong.