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

110 lines
3.9 KiB
Java
Raw Normal View History

2014-10-08 20:20:15 +00:00
// Copyright (c) 2013-2014 Sandstorm Development Group, Inc. and contributors
// Licensed under the MIT License:
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
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-29 19:15:37 +00:00
this.buf.limit(0);
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) {
2020-04-29 07:15:57 +00:00
throw new DecodeException("premature EOF");
2014-09-05 19:52:05 +00:00
}
numRead += res;
}
return numRead;
}
}