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);
|
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,
|
public static void write(WritableByteChannel outputChannel,
|
||||||
MessageBuilder message) throws IOException {
|
MessageBuilder message) throws IOException {
|
||||||
ByteBuffer[] segments = message.getSegmentsForOutput();
|
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