Support reading all segments from a single ByteBuffer, with tests
Sometimes it is desirable to read all segments from a single ByteBuffer, e.g. if reading messages from a memory-mapped file. Here we add such support, and add a test to check that the ByteBuffer read behaves the same as the ReadableByteChannel read.
This commit is contained in:
parent
ba0c120c77
commit
d2634a21d0
2 changed files with 163 additions and 0 deletions
|
@ -107,6 +107,46 @@ public final class Serialize {
|
|||
return new MessageReader(segmentSlices, options);
|
||||
}
|
||||
|
||||
public static MessageReader read(ByteBuffer bb) throws IOException {
|
||||
return read(bb, ReaderOptions.DEFAULT_READER_OPTIONS);
|
||||
}
|
||||
|
||||
public static MessageReader read(ByteBuffer bb, ReaderOptions options) throws IOException {
|
||||
bb.order(ByteOrder.LITTLE_ENDIAN);
|
||||
|
||||
int segmentCount = 1 + bb.getInt();
|
||||
if (segmentCount > 512) {
|
||||
throw new IOException("too many segments");
|
||||
}
|
||||
|
||||
ByteBuffer[] segmentSlices = new ByteBuffer[segmentCount];
|
||||
|
||||
int segmentSizesBase = bb.position();
|
||||
int segmentSizesSize = segmentCount * 4;
|
||||
|
||||
int align = Constants.BYTES_PER_WORD - 1;
|
||||
int segmentBase = (segmentSizesBase + segmentSizesSize + align) & ~align;
|
||||
|
||||
int totalWords = 0;
|
||||
|
||||
for (int ii = 0; ii < segmentCount; ++ii) {
|
||||
int segmentSize = bb.getInt(segmentSizesBase + ii * 4);
|
||||
|
||||
bb.position(segmentBase + totalWords * Constants.BYTES_PER_WORD);
|
||||
segmentSlices[ii] = bb.slice();
|
||||
segmentSlices[ii].limit(segmentSize * Constants.BYTES_PER_WORD);
|
||||
segmentSlices[ii].order(ByteOrder.LITTLE_ENDIAN);
|
||||
|
||||
totalWords += segmentSize;
|
||||
}
|
||||
|
||||
if (totalWords > options.traversalLimitInWords) {
|
||||
throw new DecodeException("Message size exceeds traversal limit.");
|
||||
}
|
||||
|
||||
return new MessageReader(segmentSlices, options);
|
||||
}
|
||||
|
||||
public static void write(WritableByteChannel outputChannel,
|
||||
MessageBuilder message) throws IOException {
|
||||
ByteBuffer[] segments = message.getSegmentsForOutput();
|
||||
|
|
123
runtime/src/test/scala/org/capnproto/SerializeSuite.scala
Normal file
123
runtime/src/test/scala/org/capnproto/SerializeSuite.scala
Normal file
|
@ -0,0 +1,123 @@
|
|||
// 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.
|
||||
|
||||
package org.capnproto
|
||||
|
||||
import java.nio.ByteBuffer
|
||||
|
||||
import org.scalatest.FunSuite
|
||||
import org.scalatest.Matchers._
|
||||
|
||||
class SerializeSuite extends FunSuite {
|
||||
/**
|
||||
* @param exampleSegmentCount number of segments
|
||||
* @param exampleBytes byte array containing `segmentCount` segments; segment `i` contains `i` words each set to `i`
|
||||
*/
|
||||
def expectSerializesTo(exampleSegmentCount: Int, exampleBytes: Array[Byte]): Unit = {
|
||||
def checkSegmentContents(arena: ReaderArena): Unit = {
|
||||
arena.segments should have length exampleSegmentCount
|
||||
for (i <- 0 until exampleSegmentCount) {
|
||||
val segment = arena.segments.get(i)
|
||||
val segmentWords = segment.buffer.asLongBuffer()
|
||||
|
||||
segmentWords.capacity should equal (i)
|
||||
segmentWords.rewind()
|
||||
while (segmentWords.hasRemaining) {
|
||||
segmentWords.get should equal (i)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// ----
|
||||
// read via ReadableByteChannel
|
||||
{
|
||||
val messageReader = Serialize.read(new ArrayInputStream(ByteBuffer.wrap(exampleBytes)))
|
||||
checkSegmentContents(messageReader.arena)
|
||||
}
|
||||
|
||||
// ------
|
||||
// read via ByteBuffer
|
||||
{
|
||||
val messageReader = Serialize.read(ByteBuffer.wrap(exampleBytes))
|
||||
checkSegmentContents(messageReader.arena)
|
||||
}
|
||||
}
|
||||
|
||||
test("SegmentReading") {
|
||||
// When transmitting over a stream, the following should be sent. All integers are unsigned and little-endian.
|
||||
// - (4 bytes) The number of segments, minus one (since there is always at least one segment).
|
||||
// - (N * 4 bytes) The size of each segment, in words.
|
||||
// - (0 or 4 bytes) Padding up to the next word boundary.
|
||||
// - The content of each segment, in order.
|
||||
|
||||
expectSerializesTo(1, Array[Byte](
|
||||
0, 0, 0, 0, // 1 segment
|
||||
0, 0, 0, 0 // Segment 0 contains 0 bytes
|
||||
// No padding
|
||||
// Segment 0 (empty)
|
||||
))
|
||||
|
||||
expectSerializesTo(2, Array[Byte](
|
||||
1, 0, 0, 0, // 2 segments
|
||||
0, 0, 0, 0, // Segment 0 contains 0 words
|
||||
1, 0, 0, 0, // Segment 1 contains 1 words
|
||||
// Padding
|
||||
0, 0, 0, 0,
|
||||
// Segment 0 (empty)
|
||||
// Segment 1
|
||||
1, 0, 0, 0, 0, 0, 0, 0
|
||||
))
|
||||
|
||||
expectSerializesTo(3, Array[Byte](
|
||||
2, 0, 0, 0, // 3 segments
|
||||
0, 0, 0, 0, // Segment 0 contains 0 words
|
||||
1, 0, 0, 0, // Segment 1 contains 1 words
|
||||
2, 0, 0, 0, // Segment 2 contains 2 words
|
||||
// No padding
|
||||
// Segment 0 (empty)
|
||||
// Segment 1
|
||||
1, 0, 0, 0, 0, 0, 0, 0,
|
||||
// Segment 2
|
||||
2, 0, 0, 0, 0, 0, 0, 0,
|
||||
2, 0, 0, 0, 0, 0, 0, 0
|
||||
))
|
||||
|
||||
expectSerializesTo(4, Array[Byte](
|
||||
3, 0, 0, 0, // 4 segments
|
||||
0, 0, 0, 0, // Segment 0 contains 0 words
|
||||
1, 0, 0, 0, // Segment 1 contains 1 words
|
||||
2, 0, 0, 0, // Segment 2 contains 2 words
|
||||
3, 0, 0, 0, // Segment 3 contains 3 words
|
||||
// Padding
|
||||
0, 0, 0, 0,
|
||||
// Segment 0 (empty)
|
||||
// Segment 1
|
||||
1, 0, 0, 0, 0, 0, 0, 0,
|
||||
// Segment 2
|
||||
2, 0, 0, 0, 0, 0, 0, 0,
|
||||
2, 0, 0, 0, 0, 0, 0, 0,
|
||||
// Segment 3
|
||||
3, 0, 0, 0, 0, 0, 0, 0,
|
||||
3, 0, 0, 0, 0, 0, 0, 0,
|
||||
3, 0, 0, 0, 0, 0, 0, 0
|
||||
))
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue