io.rsocket.buffer.Tuple3ByteBuf Maven / Gradle / Ivy
package io.rsocket.buffer;
import io.netty.buffer.ByteBuf;
import io.netty.buffer.ByteBufAllocator;
import io.netty.buffer.Unpooled;
import io.netty.util.ReferenceCountUtil;
import java.io.IOException;
import java.io.OutputStream;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;
import java.nio.channels.GatheringByteChannel;
import java.nio.charset.Charset;
class Tuple3ByteBuf extends AbstractTupleByteBuf {
private static final long ONE_MASK = 0x100000000L;
private static final long TWO_MASK = 0x200000000L;
private static final long THREE_MASK = 0x400000000L;
private static final long MASK = 0x700000000L;
private final ByteBuf one;
private final ByteBuf two;
private final ByteBuf three;
private final int oneReadIndex;
private final int twoReadIndex;
private final int threeReadIndex;
private final int oneReadableBytes;
private final int twoReadableBytes;
private final int threeReadableBytes;
private final int twoRelativeIndex;
private final int threeRelativeIndex;
private boolean freed;
Tuple3ByteBuf(ByteBufAllocator allocator, ByteBuf one, ByteBuf two, ByteBuf three) {
super(allocator, one.readableBytes() + two.readableBytes() + three.readableBytes());
this.one = one;
this.two = two;
this.three = three;
this.oneReadIndex = one.readerIndex();
this.twoReadIndex = two.readerIndex();
this.threeReadIndex = three.readerIndex();
this.oneReadableBytes = one.readableBytes();
this.twoReadableBytes = two.readableBytes();
this.threeReadableBytes = three.readableBytes();
this.twoRelativeIndex = oneReadableBytes;
this.threeRelativeIndex = twoRelativeIndex + twoReadableBytes;
this.freed = false;
}
@Override
public boolean isDirect() {
return one.isDirect() && two.isDirect() && three.isDirect();
}
public long calculateRelativeIndex(int index) {
checkIndex(index, 0);
long relativeIndex;
long mask;
if (index >= threeRelativeIndex) {
relativeIndex = threeReadIndex + (index - twoReadableBytes - oneReadableBytes);
mask = THREE_MASK;
} else if (index >= twoRelativeIndex) {
relativeIndex = twoReadIndex + (index - oneReadableBytes);
mask = TWO_MASK;
} else {
relativeIndex = oneReadIndex + index;
mask = ONE_MASK;
}
return relativeIndex | mask;
}
public ByteBuf getPart(int index) {
long ri = calculateRelativeIndex(index);
switch ((int) ((ri & MASK) >>> 32L)) {
case 0x1:
return one;
case 0x2:
return two;
case 0x4:
return three;
default:
throw new IllegalStateException();
}
}
@Override
public int nioBufferCount() {
return one.nioBufferCount() + two.nioBufferCount() + three.nioBufferCount();
}
@Override
public ByteBuffer nioBuffer() {
ByteBuffer[] oneBuffers = one.nioBuffers();
ByteBuffer[] twoBuffers = two.nioBuffers();
ByteBuffer[] threeBuffers = three.nioBuffers();
ByteBuffer merged =
BufferUtil.allocateDirectAligned(capacity, DEFAULT_DIRECT_MEMORY_CACHE_ALIGNMENT)
.order(order());
for (ByteBuffer b : oneBuffers) {
merged.put(b);
}
for (ByteBuffer b : twoBuffers) {
merged.put(b);
}
for (ByteBuffer b : threeBuffers) {
merged.put(b);
}
merged.flip();
return merged;
}
@Override
public ByteBuffer[] nioBuffers(int index, int length) {
if (length == 0) {
return new ByteBuffer[] {EMPTY_NIO_BUFFER};
}
long ri = calculateRelativeIndex(index);
index = (int) (ri & Integer.MAX_VALUE);
switch ((int) ((ri & MASK) >>> 32L)) {
case 0x1:
{
ByteBuffer[] oneBuffer;
ByteBuffer[] twoBuffer;
ByteBuffer[] threeBuffer;
int l = Math.min(oneReadableBytes - index, length);
oneBuffer = one.nioBuffers(index, l);
length -= l;
if (length != 0) {
l = Math.min(twoReadableBytes, length);
twoBuffer = two.nioBuffers(twoReadIndex, l);
length -= l;
if (length != 0) {
threeBuffer = three.nioBuffers(threeReadIndex, length);
ByteBuffer[] results =
new ByteBuffer[oneBuffer.length + twoBuffer.length + threeBuffer.length];
System.arraycopy(oneBuffer, 0, results, 0, oneBuffer.length);
System.arraycopy(twoBuffer, 0, results, oneBuffer.length, twoBuffer.length);
System.arraycopy(threeBuffer, 0, results, twoBuffer.length, threeBuffer.length);
return results;
} else {
ByteBuffer[] results = new ByteBuffer[oneBuffer.length + twoBuffer.length];
System.arraycopy(oneBuffer, 0, results, 0, oneBuffer.length);
System.arraycopy(twoBuffer, 0, results, oneBuffer.length, twoBuffer.length);
return results;
}
} else {
return oneBuffer;
}
}
case 0x2:
{
ByteBuffer[] twoBuffer;
ByteBuffer[] threeBuffer;
int l = Math.min(twoReadableBytes - index, length);
twoBuffer = two.nioBuffers(index, length);
length -= l;
if (length != 0) {
threeBuffer = three.nioBuffers(threeReadIndex, length);
ByteBuffer[] results = new ByteBuffer[twoBuffer.length + threeBuffer.length];
System.arraycopy(twoBuffer, 0, results, 0, twoBuffer.length);
System.arraycopy(threeBuffer, 0, results, threeBuffer.length, twoBuffer.length);
return results;
} else {
return twoBuffer;
}
}
case 0x4:
return three.nioBuffers(index, length);
default:
throw new IllegalStateException();
}
}
@Override
public ByteBuf getBytes(int index, ByteBuf dst, int dstIndex, int length) {
checkDstIndex(index, length, dstIndex, dst.capacity());
long ri = calculateRelativeIndex(index);
index = (int) (ri & Integer.MAX_VALUE);
switch ((int) ((ri & MASK) >>> 32L)) {
case 0x1:
{
int l = Math.min(oneReadableBytes - index, length);
one.getBytes(index, dst, dstIndex, l);
length -= l;
dstIndex += l;
if (length != 0) {
l = Math.min(twoReadableBytes, length);
two.getBytes(twoReadIndex, dst, dstIndex, l);
length -= l;
dstIndex += l;
if (length != 0) {
three.getBytes(threeReadIndex, dst, dstIndex, length);
}
}
break;
}
case 0x2:
{
int l = Math.min(twoReadableBytes - index, length);
two.getBytes(index, dst, dstIndex, l);
length -= l;
dstIndex += l;
if (length != 0) {
three.getBytes(threeReadIndex, dst, dstIndex, length);
}
break;
}
case 0x4:
{
three.getBytes(index, dst, dstIndex, length);
break;
}
default:
throw new IllegalStateException();
}
return this;
}
@Override
public ByteBuf getBytes(int index, byte[] dst, int dstIndex, int length) {
ByteBuf dstBuf = Unpooled.wrappedBuffer(dst);
int min = Math.min(dst.length, capacity);
return getBytes(0, dstBuf, index, min);
}
@Override
public ByteBuf getBytes(int index, ByteBuffer dst) {
ByteBuf dstBuf = Unpooled.wrappedBuffer(dst);
int min = Math.min(dst.limit(), capacity);
return getBytes(0, dstBuf, index, min);
}
@Override
public ByteBuf getBytes(int index, final OutputStream out, int length) throws IOException {
checkIndex(index, length);
long ri = calculateRelativeIndex(index);
index = (int) (ri & Integer.MAX_VALUE);
switch ((int) ((ri & MASK) >>> 32L)) {
case 0x1:
{
int l = Math.min(oneReadableBytes - index, length);
one.getBytes(index, out, l);
length -= l;
if (length != 0) {
l = Math.min(twoReadableBytes, length);
two.getBytes(twoReadIndex, out, l);
length -= l;
if (length != 0) {
three.getBytes(threeReadIndex, out, length);
}
}
break;
}
case 0x2:
{
int l = Math.min(twoReadableBytes - index, length);
two.getBytes(index, out, l);
length -= l;
if (length != 0) {
three.getBytes(threeReadIndex, out, length);
}
break;
}
case 0x4:
{
three.getBytes(index, out, length);
break;
}
default:
throw new IllegalStateException();
}
return this;
}
@Override
public int getBytes(int index, GatheringByteChannel out, int length) throws IOException {
checkIndex(index, length);
int read = 0;
long ri = calculateRelativeIndex(index);
index = (int) (ri & Integer.MAX_VALUE);
switch ((int) ((ri & MASK) >>> 32L)) {
case 0x1:
{
int l = Math.min(oneReadableBytes - index, length);
read += one.getBytes(index, out, l);
length -= l;
if (length != 0) {
l = Math.min(twoReadableBytes, length);
read += two.getBytes(twoReadIndex, out, l);
length -= l;
if (length != 0) {
read += three.getBytes(threeReadIndex, out, length);
}
}
break;
}
case 0x2:
{
int l = Math.min(twoReadableBytes - index, length);
read += two.getBytes(index, out, l);
length -= l;
if (length != 0) {
read += three.getBytes(threeReadIndex, out, length);
}
break;
}
case 0x4:
{
read += three.getBytes(index, out, length);
break;
}
default:
throw new IllegalStateException();
}
return read;
}
@Override
public int getBytes(int index, FileChannel out, long position, int length) throws IOException {
checkIndex(index, length);
int read = 0;
long ri = calculateRelativeIndex(index);
index = (int) (ri & Integer.MAX_VALUE);
switch ((int) ((ri & MASK) >>> 32L)) {
case 0x1:
{
int l = Math.min(oneReadableBytes - index, length);
read += one.getBytes(index, out, position, l);
length -= l;
position += l;
if (length != 0) {
l = Math.min(twoReadableBytes, length);
read += two.getBytes(twoReadIndex, out, position, l);
length -= l;
position += l;
if (length != 0) {
read += three.getBytes(threeReadIndex, out, position, length);
}
}
break;
}
case 0x2:
{
int l = Math.min(twoReadableBytes - index, length);
read += two.getBytes(index, out, position, l);
length -= l;
position += l;
if (length != 0) {
read += three.getBytes(threeReadIndex, out, position, length);
}
break;
}
case 0x4:
{
read += three.getBytes(index, out, position, length);
break;
}
default:
throw new IllegalStateException();
}
return read;
}
@Override
public ByteBuf copy(int index, int length) {
checkIndex(index, length);
ByteBuf buffer = allocator.buffer(length);
if (index == 0 && length == capacity) {
buffer.writeBytes(one, oneReadIndex, oneReadableBytes);
buffer.writeBytes(two, twoReadIndex, twoReadableBytes);
buffer.writeBytes(three, threeReadIndex, threeReadableBytes);
return buffer;
}
long ri = calculateRelativeIndex(index);
index = (int) (ri & Integer.MAX_VALUE);
switch ((int) ((ri & MASK) >>> 32L)) {
case 0x1:
{
int l = Math.min(oneReadableBytes - index, length);
buffer.writeBytes(one, index, l);
length -= l;
if (length != 0) {
l = Math.min(twoReadableBytes, length);
buffer.writeBytes(two, twoReadIndex, l);
length -= l;
if (length != 0) {
buffer.writeBytes(three, threeReadIndex, length);
}
}
return buffer;
}
case 0x2:
{
int l = Math.min(twoReadableBytes - index, length);
buffer.writeBytes(two, index, l);
length -= l;
if (length != 0) {
buffer.writeBytes(three, threeReadIndex, length);
}
return buffer;
}
case 0x4:
{
buffer.writeBytes(three, index, length);
return buffer;
}
default:
throw new IllegalStateException();
}
}
@Override
public ByteBuf retainedSlice() {
return new Tuple3ByteBuf(
allocator,
one.retainedSlice(oneReadIndex, oneReadableBytes),
two.retainedSlice(twoReadIndex, twoReadableBytes),
three.retainedSlice(threeReadIndex, threeReadableBytes));
}
@Override
public ByteBuf slice(final int readIndex, int length) {
checkIndex(readIndex, length);
if (readIndex == 0 && length == capacity) {
return new Tuple3ByteBuf(
allocator,
one.slice(oneReadIndex, oneReadableBytes),
two.slice(twoReadIndex, twoReadableBytes),
three.slice(threeReadIndex, threeReadableBytes));
}
long ri = calculateRelativeIndex(readIndex);
int index = (int) (ri & Integer.MAX_VALUE);
switch ((int) ((ri & MASK) >>> 32L)) {
case 0x1:
{
ByteBuf oneSlice;
ByteBuf twoSlice;
ByteBuf threeSlice;
int l = Math.min(oneReadableBytes - index, length);
oneSlice = one.slice(index, l);
length -= l;
if (length != 0) {
l = Math.min(twoReadableBytes, length);
twoSlice = two.slice(twoReadIndex, l);
length -= l;
if (length != 0) {
threeSlice = three.slice(threeReadIndex, length);
return new Tuple3ByteBuf(allocator, oneSlice, twoSlice, threeSlice);
} else {
return new Tuple2ByteBuf(allocator, oneSlice, twoSlice);
}
} else {
return oneSlice;
}
}
case 0x2:
{
ByteBuf twoSlice;
ByteBuf threeSlice;
int l = Math.min(twoReadableBytes - index, length);
twoSlice = two.slice(index, l);
length -= l;
if (length != 0) {
threeSlice = three.slice(threeReadIndex, length);
return new Tuple2ByteBuf(allocator, twoSlice, threeSlice);
} else {
return twoSlice;
}
}
case 0x4:
{
return three.slice(index, length);
}
default:
throw new IllegalStateException();
}
}
@Override
protected void deallocate() {
if (freed) {
return;
}
freed = true;
ReferenceCountUtil.safeRelease(one);
ReferenceCountUtil.safeRelease(two);
ReferenceCountUtil.safeRelease(three);
}
@Override
public String toString(Charset charset) {
StringBuilder builder = new StringBuilder(3);
builder.append(one.toString(charset));
builder.append(two.toString(charset));
builder.append(three.toString(charset));
return builder.toString();
}
@Override
public String toString(int index, int length, Charset charset) {
// TODO - make this smarter
return toString(charset).substring(index, length);
}
@Override
public String toString() {
return "Tuple3ByteBuf{"
+ "capacity="
+ capacity
+ ", one="
+ one
+ ", two="
+ two
+ ", three="
+ three
+ ", allocator="
+ allocator
+ ", oneReadIndex="
+ oneReadIndex
+ ", twoReadIndex="
+ twoReadIndex
+ ", threeReadIndex="
+ threeReadIndex
+ ", oneReadableBytes="
+ oneReadableBytes
+ ", twoReadableBytes="
+ twoReadableBytes
+ ", threeReadableBytes="
+ threeReadableBytes
+ ", twoRelativeIndex="
+ twoRelativeIndex
+ ", threeRelativeIndex="
+ threeRelativeIndex
+ '}';
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy