blob: 4d7362e3ff600fbe9ebd46a8a47ccc6435eddc63 [file] [log] [blame]
// 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.
#include "message.h"
#include "test-util.h"
#include <kj/array.h>
#include <kj/vector.h>
#include <kj/debug.h>
#include <kj/compat/gtest.h>
namespace capnp {
namespace _ { // private
namespace {
TEST(Message, MallocBuilderWithFirstSegment) {
word scratch[16];
memset(scratch, 0, sizeof(scratch));
MallocMessageBuilder builder(kj::arrayPtr(scratch, 16), AllocationStrategy::FIXED_SIZE);
kj::ArrayPtr<word> segment = builder.allocateSegment(1);
EXPECT_EQ(scratch, segment.begin());
EXPECT_EQ(16u, segment.size());
segment = builder.allocateSegment(1);
EXPECT_NE(scratch, segment.begin());
EXPECT_EQ(16u, segment.size());
segment = builder.allocateSegment(1);
EXPECT_NE(scratch, segment.begin());
EXPECT_EQ(16u, segment.size());
}
class TestInitMessageBuilder: public MessageBuilder {
public:
TestInitMessageBuilder(kj::ArrayPtr<SegmentInit> segments): MessageBuilder(segments) {}
kj::ArrayPtr<word> allocateSegment(uint minimumSize) override {
auto array = kj::heapArray<word>(minimumSize);
memset(array.begin(), 0, array.asBytes().size());
allocations.add(kj::mv(array));
return allocations.back();
}
kj::Vector<kj::Array<word>> allocations;
};
TEST(Message, MessageBuilderInit) {
MallocMessageBuilder builder(2048);
initTestMessage(builder.getRoot<TestAllTypes>());
// Pull the segments out and make a segment init table out of them.
//
// We const_cast for simplicity of implementing the test, but you shouldn't do that at home. :)
auto segs = builder.getSegmentsForOutput();
ASSERT_EQ(1, segs.size());
auto segInits = KJ_MAP(seg, segs) -> MessageBuilder::SegmentInit {
return { kj::arrayPtr(const_cast<word*>(seg.begin()), seg.size()), seg.size() };
};
// Init a new builder from the old segments.
TestInitMessageBuilder builder2(segInits);
checkTestMessage(builder2.getRoot<TestAllTypes>());
// Verify that they're really using the same underlying memory.
builder2.getRoot<TestAllTypes>().setInt64Field(123321);
EXPECT_EQ(123321, builder.getRoot<TestAllTypes>().getInt64Field());
// Force builder2 to allocate new space.
EXPECT_EQ(0, builder2.allocations.size());
builder2.getRoot<TestAllTypes>().setTextField("foobarbaz");
EXPECT_EQ(1, builder2.allocations.size());
}
TEST(Message, MessageBuilderInitMultiSegment) {
// Same as previous test, but with a message containing many segments.
MallocMessageBuilder builder(1, AllocationStrategy::FIXED_SIZE);
initTestMessage(builder.getRoot<TestAllTypes>());
// Pull the segments out and make a segment init table out of them.
//
// We const_cast for simplicity of implementing the test, but you shouldn't do that at home. :)
auto segs = builder.getSegmentsForOutput();
ASSERT_NE(1, segs.size());
auto segInits = KJ_MAP(seg, segs) -> MessageBuilder::SegmentInit {
return { kj::arrayPtr(const_cast<word*>(seg.begin()), seg.size()), seg.size() };
};
// Init a new builder from the old segments.
TestInitMessageBuilder builder2(segInits);
checkTestMessage(builder2.getRoot<TestAllTypes>());
// Verify that they're really using the same underlying memory.
builder2.getRoot<TestAllTypes>().setInt64Field(123321);
EXPECT_EQ(123321, builder.getRoot<TestAllTypes>().getInt64Field());
// Force builder2 to allocate new space.
EXPECT_EQ(0, builder2.allocations.size());
builder2.getRoot<TestAllTypes>().setTextField("foobarbaz");
EXPECT_EQ(1, builder2.allocations.size());
}
TEST(Message, MessageBuilderInitSpaceAvailable) {
word buffer[2048];
memset(buffer, 0, sizeof(buffer));
MallocMessageBuilder builder(buffer);
initTestMessage(builder.getRoot<TestAllTypes>());
// Find out how much space in `buffer` was used in order to use in initializing the new message.
auto segs = builder.getSegmentsForOutput();
ASSERT_EQ(1, segs.size());
KJ_ASSERT(segs[0].begin() == buffer);
MessageBuilder::SegmentInit init = { kj::ArrayPtr<word>(buffer), segs[0].size() };
// Init a new builder from the old segments.
TestInitMessageBuilder builder2(kj::arrayPtr(&init, 1));
checkTestMessage(builder2.getRoot<TestAllTypes>());
// Verify that they're really using the same underlying memory.
builder2.getRoot<TestAllTypes>().setInt64Field(123321);
EXPECT_EQ(123321, builder.getRoot<TestAllTypes>().getInt64Field());
// Ask builder2 to allocate new space. It should go into the free space at the end of the
// segment.
EXPECT_EQ(0, builder2.allocations.size());
builder2.getRoot<TestAllTypes>().setTextField("foobarbaz");
EXPECT_EQ(0, builder2.allocations.size());
EXPECT_EQ(kj::implicitCast<void*>(buffer + segs[0].size()),
kj::implicitCast<void*>(builder2.getRoot<TestAllTypes>().getTextField().begin()));
}
TEST(Message, ReadWriteDataStruct) {
MallocMessageBuilder builder;
auto root = builder.getRoot<TestAllTypes>();
root.setUInt32Field(123);
root.setFloat64Field(1.5);
root.setTextField("foo");
auto copy = readDataStruct<TestAllTypes>(writeDataStruct(root));
EXPECT_EQ(123, copy.getUInt32Field());
EXPECT_EQ(1.5, copy.getFloat64Field());
EXPECT_FALSE(copy.hasTextField());
checkTestMessageAllZero(readDataStruct<TestAllTypes>(nullptr));
checkTestMessageAllZero(defaultValue<TestAllTypes>());
}
KJ_TEST("clone()") {
MallocMessageBuilder builder(2048);
initTestMessage(builder.getRoot<TestAllTypes>());
auto copy = clone(builder.getRoot<TestAllTypes>().asReader());
checkTestMessage(*copy);
}
#if !CAPNP_ALLOW_UNALIGNED
KJ_TEST("disallow unaligned") {
union {
char buffer[16];
word align;
};
memset(buffer, 0, sizeof(buffer));
auto unaligned = kj::arrayPtr(reinterpret_cast<word*>(buffer + 1), 1);
kj::ArrayPtr<const word> segments[1] = {unaligned};
SegmentArrayMessageReader message(segments);
KJ_EXPECT_THROW_RECOVERABLE_MESSAGE("unaligned", message.getRoot<TestAllTypes>());
}
#endif
// TODO(test): More tests.
} // namespace
} // namespace _ (private)
} // namespace capnp