capnproto-java-rpc/runtime/src/main/java/org/capnproto/BufferedInputStreamWrapper.java

88 lines
2.7 KiB
Java
Raw Normal View History

2014-09-05 19:52:05 +00:00
package org.capnproto;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.channels.ReadableByteChannel;
public final class BufferedInputStreamWrapper implements BufferedInputStream {
private final ReadableByteChannel inner;
private final ByteBuffer buf;
public BufferedInputStreamWrapper(ReadableByteChannel chan) {
this.inner = chan;
this.buf = ByteBuffer.allocate(8192);
this.buf.limit(0);
2014-09-05 19:52:05 +00:00
}
public final int read(ByteBuffer dst) throws IOException {
int numBytes = dst.remaining();
if (numBytes < this.buf.remaining()) {
2014-09-05 19:52:05 +00:00
//# Serve from the current buffer.
ByteBuffer slice = this.buf.slice();
slice.limit(numBytes);
dst.put(slice);
this.buf.position(this.buf.position() + numBytes);
return numBytes;
} else {
//# Copy current available into destination.
int fromFirstBuffer = this.buf.remaining();
2014-09-25 18:40:44 +00:00
{
ByteBuffer slice = this.buf.slice();
slice.limit(fromFirstBuffer);
dst.put(slice);
}
2014-09-05 19:52:05 +00:00
numBytes -= fromFirstBuffer;
if (numBytes <= this.buf.capacity()) {
//# Read the next buffer-full.
this.buf.clear();
2014-09-05 19:52:05 +00:00
int n = readAtLeast(this.inner, this.buf, numBytes);
2014-09-25 18:40:44 +00:00
this.buf.rewind();
ByteBuffer slice = this.buf.slice();
slice.limit(numBytes);
dst.put(slice);
2014-09-05 19:52:05 +00:00
this.buf.limit(n);
2014-09-05 19:52:05 +00:00
this.buf.position(numBytes);
return fromFirstBuffer + numBytes;
} else {
//# Forward large read to the underlying stream.
this.buf.clear();
2014-09-25 18:40:44 +00:00
return fromFirstBuffer + readAtLeast(this.inner, dst, numBytes);
2014-09-05 19:52:05 +00:00
}
}
}
2014-09-27 14:02:07 +00:00
public final ByteBuffer getReadBuffer() throws IOException {
if (this.buf.remaining() == 0) {
this.buf.clear();
int n = readAtLeast(this.inner, this.buf, 1);
2014-09-27 14:02:07 +00:00
this.buf.rewind();
this.buf.limit(n);
2014-09-27 14:02:07 +00:00
}
2014-09-05 19:52:05 +00:00
return this.buf;
}
public final void close() throws IOException {
this.inner.close();
}
public final boolean isOpen() {
return this.inner.isOpen();
}
public static int readAtLeast(ReadableByteChannel reader, ByteBuffer buf, int minBytes) throws IOException {
int numRead = 0;
while (numRead < minBytes) {
int res = reader.read(buf);
if (res < 0) {
throw new Error("premature EOF");
}
numRead += res;
}
return numRead;
}
}