blob: 1645f4841f4924035cdf6ea38a0d6fe4ba1f2710 [file] [log] [blame]
// Copyright 2019 Google LLC
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// Tests to ensure that conditional fields work.
#include <stdint.h>
#include <vector>
#include "gtest/gtest.h"
#include "testdata/condition.emb.h"
namespace emboss {
namespace test {
namespace {
TEST(Conditional, WithConditionTrueFieldsAreOk) {
::std::uint8_t buffer[2] = {0, 0};
auto writer = BasicConditionalWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_TRUE(writer.x().Ok());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(Conditional, WithConditionTrueAllFieldsAreReadable) {
::std::uint8_t buffer[2] = {0, 2};
auto writer = BasicConditionalWriter(buffer, sizeof buffer);
EXPECT_EQ(0, writer.x().Read());
EXPECT_EQ(2, writer.xc().Read());
}
TEST(Conditional, WithConditionTrueConditionalFieldIsWritable) {
::std::uint8_t buffer1[2] = {0, 2};
auto writer1 = BasicConditionalWriter(buffer1, sizeof buffer1);
EXPECT_TRUE(writer1.xc().TryToWrite(3));
EXPECT_EQ(3, buffer1[1]);
::std::uint8_t buffer2[2] = {0, 0};
auto writer2 = BasicConditionalWriter(buffer2, sizeof buffer2);
EXPECT_FALSE(writer2.xc().Equals(writer1.xc()));
EXPECT_TRUE(writer2.xc().TryToCopyFrom(writer1.xc()));
EXPECT_TRUE(writer2.xc().Equals(writer1.xc()));
EXPECT_EQ(3, buffer2[1]);
}
TEST(Conditional, WithConditionFalseStructIsOkButConditionalFieldIsNot) {
::std::uint8_t buffer[2] = {1, 2};
auto writer = BasicConditionalWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_TRUE(writer.x().Ok());
EXPECT_FALSE(writer.xc().Ok());
}
#if EMBOSS_CHECK_ABORTS
TEST(Conditional, BasicConditionFalseReadCrashes) {
::std::uint8_t buffer[2] = {1, 2};
auto writer = BasicConditionalWriter(buffer, sizeof buffer);
EXPECT_DEATH(writer.xc().Read(), "");
}
TEST(Conditional, BasicConditionFalseWriteCrashes) {
::std::uint8_t buffer[2] = {1, 2};
auto writer = BasicConditionalWriter(buffer, sizeof buffer);
EXPECT_DEATH(writer.xc().Write(3), "");
}
#endif // EMBOSS_CHECK_ABORTS
TEST(Conditional, BasicConditionTrueSizeIncludesConditionalField) {
::std::uint8_t buffer[2] = {0, 2};
auto writer = BasicConditionalWriter(buffer, sizeof buffer);
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_EQ(2, BasicConditional::MaxSizeInBytes());
EXPECT_EQ(1, BasicConditional::MinSizeInBytes());
EXPECT_EQ(2, writer.MaxSizeInBytes().Read());
EXPECT_EQ(1, writer.MinSizeInBytes().Read());
}
TEST(Conditional, BasicConditionFalseSizeDoesNotIncludeConditionalField) {
::std::uint8_t buffer[2] = {1, 2};
auto writer = BasicConditionalWriter(buffer, sizeof buffer);
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_EQ(2, writer.MaxSizeInBytes().Read());
EXPECT_EQ(1, writer.MinSizeInBytes().Read());
}
TEST(Conditional, WithConditionFalseStructIsOkWhenBufferIsSmall) {
::std::uint8_t buffer[1] = {1};
auto writer = BasicConditionalWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_TRUE(writer.x().Ok());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(Conditional, WithConditionTrueStructIsNotOkWhenBufferIsSmall) {
::std::uint8_t buffer[1] = {0};
auto writer = BasicConditionalWriter(buffer, sizeof buffer);
EXPECT_FALSE(writer.Ok());
EXPECT_TRUE(writer.x().Ok());
EXPECT_TRUE(writer.has_xc().Known());
EXPECT_TRUE(writer.has_xc().Value());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(Conditional, WithNegativeConditionTrueFieldsAreOk) {
::std::uint8_t buffer[2] = {1, 0};
auto writer = NegativeConditionalWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_TRUE(writer.x().Ok());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(Conditional, WithNegativeConditionTrueAllFieldsAreReadable) {
::std::uint8_t buffer[2] = {1, 2};
auto writer = NegativeConditionalWriter(buffer, sizeof buffer);
EXPECT_EQ(1, writer.x().Read());
EXPECT_EQ(2, writer.xc().Read());
}
TEST(Conditional, WithNegativeConditionTrueConditionalFieldIsWritable) {
::std::uint8_t buffer1[2] = {1, 2};
auto writer1 = NegativeConditionalWriter(buffer1, sizeof buffer1);
EXPECT_TRUE(writer1.xc().TryToWrite(3));
EXPECT_EQ(3, buffer1[1]);
::std::uint8_t buffer2[2] = {1, 0};
auto writer2 = NegativeConditionalWriter(buffer2, sizeof buffer2);
EXPECT_FALSE(writer2.xc().Equals(writer1.xc()));
EXPECT_TRUE(writer2.xc().TryToCopyFrom(writer1.xc()));
EXPECT_TRUE(writer2.xc().Equals(writer1.xc()));
EXPECT_EQ(3, buffer2[1]);
}
TEST(Conditional,
WithNegativeConditionFalseStructIsOkButConditionalFieldIsNot) {
::std::uint8_t buffer[2] = {0, 2};
auto writer = NegativeConditionalWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_TRUE(writer.x().Ok());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(Conditional, NegativeConditionFalseReadCrashes) {
::std::uint8_t buffer1[2] = {0, 2};
auto writer1 = NegativeConditionalWriter(buffer1, sizeof buffer1);
#if EMBOSS_CHECK_ABORTS
EXPECT_DEATH(writer1.xc().Read(), "");
#endif // EMBOSS_CHECK_ABORTS
::std::uint8_t buffer2[2] = {0, 0};
auto writer2 = BasicConditionalWriter(buffer2, sizeof buffer2);
EXPECT_TRUE(writer2.xc().CouldWriteValue(2));
EXPECT_EQ(writer2.xc().Read(), 0);
EXPECT_FALSE(writer2.xc().TryToCopyFrom(writer1.xc()));
EXPECT_EQ(writer2.xc().Read(), 0);
}
TEST(Conditional, NegativeConditionFalseWriteCrashes) {
::std::uint8_t buffer1[2] = {0, 2};
auto writer1 = NegativeConditionalWriter(buffer1, sizeof buffer1);
#if EMBOSS_CHECK_ABORTS
EXPECT_DEATH(writer1.xc().Write(3), "");
#endif // EMBOSS_CHECK_ABORTS
::std::uint8_t buffer2[2] = {0, 2};
auto writer2 = NegativeConditionalWriter(buffer2, sizeof buffer2);
EXPECT_FALSE(writer2.xc().TryToCopyFrom(writer1.xc()));
}
TEST(Conditional, NegativeConditionTrueSizeIncludesConditionalField) {
::std::uint8_t buffer[2] = {1, 2};
auto writer = NegativeConditionalWriter(buffer, sizeof buffer);
EXPECT_EQ(2U, writer.SizeInBytes());
}
TEST(Conditional, NegativeConditionFalseSizeDoesNotIncludeConditionalField) {
::std::uint8_t buffer[2] = {0, 2};
auto writer = NegativeConditionalWriter(buffer, sizeof buffer);
EXPECT_EQ(1U, writer.SizeInBytes());
}
TEST(Conditional, WithNegativeConditionFalseStructIsOkWhenBufferIsSmall) {
::std::uint8_t buffer[1] = {0};
auto writer = NegativeConditionalWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_TRUE(writer.x().Ok());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(Conditional, WithNegativeConditionTrueStructIsNotOkWhenBufferIsSmall) {
::std::uint8_t buffer[1] = {1};
auto writer = NegativeConditionalWriter(buffer, sizeof buffer);
EXPECT_FALSE(writer.Ok());
EXPECT_TRUE(writer.x().Ok());
EXPECT_TRUE(writer.has_xc().Known());
EXPECT_TRUE(writer.has_xc().Value());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(Conditional,
SizeIncludesUnconditionalFieldsThatOverlapWithConditionalFields) {
::std::uint8_t buffer[2] = {1, 2};
auto writer = ConditionalAndUnconditionalOverlappingFinalFieldWriter(
buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_EQ(2U, writer.SizeInBytes());
}
TEST(Conditional,
SizeIsConstantWhenUnconditionalFieldsOverlapWithConditionalFields) {
EXPECT_EQ(
2U,
ConditionalAndUnconditionalOverlappingFinalFieldWriter::SizeInBytes());
}
TEST(Conditional, WhenConditionalFieldIsFirstSizeIsConstant) {
EXPECT_EQ(2U, ConditionalBasicConditionalFieldFirstWriter::SizeInBytes());
}
TEST(Conditional, WhenConditionIsFalseDynamicallyPlacedFieldDoesNotAffectSize) {
::std::uint8_t buffer[3] = {1, 0, 10};
auto writer = ConditionalAndDynamicLocationWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_EQ(3U, writer.SizeInBytes());
}
TEST(Conditional, WhenConditionIsTrueDynamicallyPlacedFieldDoesAffectSize) {
::std::uint8_t buffer[4] = {0, 0, 3, 0};
auto writer = ConditionalAndDynamicLocationWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_EQ(4U, writer.SizeInBytes());
}
TEST(Conditional, WhenConditionIsTrueDynamicallyPlacedFieldOutOfRangeIsError) {
::std::uint8_t buffer[3] = {0, 0, 3};
auto writer = ConditionalAndDynamicLocationWriter(buffer, sizeof buffer);
EXPECT_FALSE(writer.Ok());
EXPECT_EQ(4U, writer.SizeInBytes());
}
TEST(Conditional, ConditionUsesMinInt) {
::std::uint8_t buffer[2] = {0, 0};
auto view = MakeConditionUsesMinIntView(buffer, sizeof buffer);
EXPECT_TRUE(view.Ok());
EXPECT_FALSE(view.has_xc().ValueOr(true));
EXPECT_EQ(1U, view.SizeInBytes());
buffer[0] = 0x80;
EXPECT_TRUE(view.Ok());
EXPECT_EQ(-0x80, view.x().Read());
EXPECT_TRUE(view.has_xc().ValueOr(false));
EXPECT_EQ(2U, view.SizeInBytes());
}
TEST(Conditional,
StructWithNestedConditionIsNotOkWhenOuterConditionDoesNotExist) {
::std::uint8_t buffer[3] = {1, 0, 3};
auto writer = NestedConditionalWriter(buffer, sizeof buffer);
ASSERT_FALSE(writer.IntrinsicSizeInBytes().Ok());
ASSERT_FALSE((writer.xc().Ok()));
ASSERT_FALSE(writer.SizeIsKnown());
ASSERT_FALSE(writer.IsComplete());
ASSERT_FALSE(writer.Ok());
ASSERT_TRUE(writer.has_xc().Known());
ASSERT_FALSE(writer.has_xc().Value());
ASSERT_FALSE(writer.has_xcc().Known());
}
TEST(Conditional,
StructWithCorrectNestedConditionIsOkWhenOuterConditionDoesNotExist) {
::std::uint8_t buffer[3] = {1, 0, 3};
auto writer = CorrectNestedConditionalWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.IsComplete());
EXPECT_TRUE(writer.Ok());
EXPECT_TRUE(writer.has_xc().Known());
EXPECT_FALSE(writer.has_xc().Value());
EXPECT_TRUE(writer.has_xcc().Known());
EXPECT_FALSE(writer.has_xcc().Value());
EXPECT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
}
TEST(Conditional, StructWithNestedConditionIsOkWhenOuterConditionExists) {
::std::uint8_t buffer[3] = {0, 1, 3};
auto writer = NestedConditionalWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_TRUE(writer.has_xc().Known());
EXPECT_TRUE(writer.has_xc().Value());
EXPECT_TRUE(writer.has_xcc().Known());
EXPECT_FALSE(writer.has_xcc().Value());
EXPECT_EQ(2U, writer.SizeInBytes());
}
TEST(Conditional, AlwaysMissingFieldDoesNotContributeToStaticSize) {
EXPECT_EQ(0U, OnlyAlwaysFalseConditionWriter::SizeInBytes());
EXPECT_EQ(1U, AlwaysFalseConditionWriter::SizeInBytes());
}
TEST(Conditional, AlwaysMissingFieldDoesNotContributeToSize) {
::std::uint8_t buffer[1] = {0};
auto view = MakeAlwaysFalseConditionDynamicSizeView(buffer, sizeof buffer);
ASSERT_TRUE(view.SizeIsKnown());
EXPECT_EQ(1U, view.SizeInBytes());
}
TEST(Conditional, StructIsOkWithAlwaysMissingField) {
::std::uint8_t buffer[1] = {0};
auto writer = AlwaysFalseConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_EQ(1U, AlwaysFalseConditionView::SizeInBytes());
}
TEST(Conditional, StructIsOkWithOnlyAlwaysMissingField) {
::std::uint8_t buffer[1] = {0};
auto writer = OnlyAlwaysFalseConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(0U, writer.SizeInBytes());
EXPECT_EQ(0U, OnlyAlwaysFalseConditionView::SizeInBytes());
}
TEST(Conditional, ConditionDoesNotBlockStaticSize) {
EXPECT_EQ(3U, ConditionDoesNotContributeToSizeView::SizeInBytes());
}
TEST(Conditional, EqualsHaveAllFields) {
::std::array</**/ ::std::uint8_t, 2> buf_a = {0, 1};
::std::array</**/ ::std::uint8_t, 2> buf_b = {0, 1};
EXPECT_EQ(buf_a, buf_b);
auto a = BasicConditionalWriter(&buf_a);
auto a_const = BasicConditionalWriter(
static_cast</**/ ::std::array</**/ ::std::uint8_t, 2> *>(&buf_a));
auto b = BasicConditionalWriter(&buf_b);
EXPECT_TRUE(a.has_xc().Known());
EXPECT_TRUE(a.has_xc().Value());
EXPECT_TRUE(b.has_xc().Known());
EXPECT_TRUE(b.has_xc().Value());
EXPECT_TRUE(a.Equals(a));
EXPECT_TRUE(a.UncheckedEquals(a));
EXPECT_TRUE(b.Equals(b));
EXPECT_TRUE(b.UncheckedEquals(b));
EXPECT_TRUE(a.Equals(b));
EXPECT_TRUE(a.UncheckedEquals(b));
EXPECT_TRUE(b.Equals(a));
EXPECT_TRUE(b.UncheckedEquals(a));
EXPECT_TRUE(a_const.Equals(b));
EXPECT_TRUE(a_const.UncheckedEquals(b));
EXPECT_TRUE(b.Equals(a_const));
EXPECT_TRUE(b.UncheckedEquals(a_const));
b.xc().Write(b.xc().Read() + 1);
EXPECT_FALSE(a.Equals(b));
EXPECT_FALSE(a.UncheckedEquals(b));
EXPECT_FALSE(b.Equals(a));
EXPECT_FALSE(b.UncheckedEquals(a));
EXPECT_FALSE(a_const.Equals(b));
EXPECT_FALSE(a_const.UncheckedEquals(b));
EXPECT_FALSE(b.Equals(a_const));
EXPECT_FALSE(b.UncheckedEquals(a_const));
}
TEST(Conditional, EqualsOneViewMissingField) {
::std::array</**/ ::std::uint8_t, 2> buf_a = {0, 1};
::std::array</**/ ::std::uint8_t, 2> buf_b = {1, 1};
EXPECT_NE(buf_a, buf_b);
auto a = BasicConditionalWriter(&buf_a);
auto b = BasicConditionalWriter(&buf_b);
EXPECT_TRUE(a.has_xc().Known());
EXPECT_TRUE(a.has_xc().Value());
EXPECT_TRUE(b.has_xc().Known());
EXPECT_FALSE(b.has_xc().Value());
EXPECT_FALSE(a.Equals(b));
EXPECT_FALSE(a.UncheckedEquals(b));
EXPECT_FALSE(b.Equals(a));
EXPECT_FALSE(b.UncheckedEquals(a));
}
TEST(Conditional, EqualsBothFieldsMissing) {
::std::array</**/ ::std::uint8_t, 2> buf_a = {1, 1};
::std::array</**/ ::std::uint8_t, 2> buf_b = {1, 1};
EXPECT_EQ(buf_a, buf_b);
auto a = BasicConditionalWriter(&buf_a);
auto a_const = BasicConditionalWriter(
static_cast</**/ ::std::array</**/ ::std::uint8_t, 2> *>(&buf_a));
auto b = BasicConditionalWriter(&buf_b);
EXPECT_TRUE(a.has_xc().Known());
EXPECT_FALSE(a.has_xc().Value());
EXPECT_TRUE(b.has_xc().Known());
EXPECT_FALSE(b.has_xc().Value());
EXPECT_TRUE(a.Equals(b));
EXPECT_TRUE(a.UncheckedEquals(b));
EXPECT_TRUE(b.Equals(a));
EXPECT_TRUE(b.UncheckedEquals(a));
EXPECT_TRUE(a_const.Equals(b));
EXPECT_TRUE(a_const.UncheckedEquals(b));
EXPECT_TRUE(b.Equals(a_const));
EXPECT_TRUE(b.UncheckedEquals(a_const));
++buf_b[1];
EXPECT_NE(buf_a, buf_b);
EXPECT_TRUE(a.Equals(b));
EXPECT_TRUE(a.UncheckedEquals(b));
EXPECT_TRUE(b.Equals(a));
EXPECT_TRUE(b.UncheckedEquals(a));
EXPECT_TRUE(a_const.Equals(b));
EXPECT_TRUE(a_const.UncheckedEquals(b));
EXPECT_TRUE(b.Equals(a_const));
EXPECT_TRUE(b.UncheckedEquals(a_const));
}
TEST(Conditional, TrueEnumBasedCondition) {
::std::uint8_t buffer[2] = {1};
auto writer = EnumConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_TRUE(writer.has_xc().Value());
EXPECT_EQ(0, writer.xc().Read());
EXPECT_TRUE(writer.has_xc2().Value());
EXPECT_EQ(0, writer.xc2().Read());
}
TEST(Conditional, FalseEnumBasedCondition) {
::std::uint8_t buffer[2] = {0};
auto writer = EnumConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
EXPECT_FALSE(writer.has_xc().Value());
EXPECT_FALSE(writer.xc2().Ok());
EXPECT_FALSE(writer.has_xc2().Value());
}
TEST(Conditional, TrueEnumBasedNegativeCondition) {
::std::uint8_t buffer[2] = {0};
auto writer = NegativeEnumConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_EQ(0, writer.xc().Read());
}
TEST(Conditional, FalseEnumBasedNegativeCondition) {
::std::uint8_t buffer[2] = {1};
auto writer = NegativeEnumConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(LessThanConditional, LessThan) {
::std::uint8_t buffer[2] = {4};
auto writer = LessThanConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(LessThanConditional, Equal) {
::std::uint8_t buffer[2] = {5};
auto writer = LessThanConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(LessThanConditional, GreaterThan) {
::std::uint8_t buffer[2] = {6};
auto writer = LessThanConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(LessThanOrEqualConditional, LessThan) {
::std::uint8_t buffer[2] = {4};
auto writer = LessThanOrEqualConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(LessThanOrEqualConditional, Equal) {
::std::uint8_t buffer[2] = {5};
auto writer = LessThanOrEqualConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(LessThanOrEqualConditional, GreaterThan) {
::std::uint8_t buffer[2] = {6};
auto writer = LessThanOrEqualConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(GreaterThanConditional, LessThan) {
::std::uint8_t buffer[2] = {4};
auto writer = GreaterThanConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(GreaterThanConditional, Equal) {
::std::uint8_t buffer[2] = {5};
auto writer = GreaterThanConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(GreaterThanConditional, GreaterThan) {
::std::uint8_t buffer[2] = {6};
auto writer = GreaterThanConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(GreaterThanOrEqualConditional, LessThan) {
::std::uint8_t buffer[2] = {4};
auto writer = GreaterThanOrEqualConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(1U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(GreaterThanOrEqualConditional, Equal) {
::std::uint8_t buffer[2] = {5};
auto writer = GreaterThanOrEqualConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(GreaterThanOrEqualConditional, GreaterThan) {
::std::uint8_t buffer[2] = {6};
auto writer = GreaterThanOrEqualConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(RangeConditional, ValueTooSmall) {
::std::uint8_t buffer[3] = {1, 9};
auto writer = RangeConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(RangeConditional, ValueTooLarge) {
::std::uint8_t buffer[3] = {11, 12};
auto writer = RangeConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(RangeConditional, ValuesSwapped) {
::std::uint8_t buffer[3] = {8, 7};
auto writer = RangeConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(RangeConditional, True) {
::std::uint8_t buffer[3] = {7, 8};
auto writer = RangeConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(3U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(ReverseRangeConditional, ValueTooSmall) {
::std::uint8_t buffer[3] = {1, 9};
auto writer = ReverseRangeConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(ReverseRangeConditional, ValueTooLarge) {
::std::uint8_t buffer[3] = {11, 12};
auto writer = ReverseRangeConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(ReverseRangeConditional, ValuesSwapped) {
::std::uint8_t buffer[3] = {8, 7};
auto writer = ReverseRangeConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(ReverseRangeConditional, True) {
::std::uint8_t buffer[3] = {7, 8};
auto writer = ReverseRangeConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(3U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(AndConditional, BothFalse) {
::std::uint8_t buffer[3] = {1, 1};
auto writer = AndConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(AndConditional, FirstFalse) {
::std::uint8_t buffer[3] = {1, 5};
auto writer = AndConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(AndConditional, SecondFalse) {
::std::uint8_t buffer[3] = {5, 1};
auto writer = AndConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(AndConditional, BothTrue) {
::std::uint8_t buffer[3] = {5, 5};
auto writer = AndConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(3U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(OrConditional, BothFalse) {
::std::uint8_t buffer[3] = {1, 1};
auto writer = OrConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(2U, writer.SizeInBytes());
EXPECT_FALSE(writer.xc().Ok());
}
TEST(OrConditional, FirstFalse) {
::std::uint8_t buffer[3] = {1, 5};
auto writer = OrConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(3U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(OrConditional, SecondFalse) {
::std::uint8_t buffer[3] = {5, 1};
auto writer = OrConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(3U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(OrConditional, BothTrue) {
::std::uint8_t buffer[3] = {5, 5};
auto writer = OrConditionWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
ASSERT_TRUE(writer.SizeIsKnown());
EXPECT_EQ(3U, writer.SizeInBytes());
EXPECT_TRUE(writer.xc().Ok());
}
TEST(ChoiceConditional, UseX) {
::std::array</**/ ::std::uint8_t, 4> buffer = {1, 5, 0, 10};
auto view = MakeChoiceConditionView(&buffer);
EXPECT_TRUE(view.Ok());
EXPECT_TRUE(view.SizeIsKnown());
EXPECT_EQ(4U, view.SizeInBytes());
EXPECT_TRUE(view.has_xyc().ValueOr(false));
EXPECT_EQ(10, view.xyc().Read());
}
TEST(ChoiceConditional, UseY) {
::std::array</**/ ::std::uint8_t, 4> buffer = {2, 5, 0, 10};
auto view = MakeChoiceConditionView(&buffer);
EXPECT_TRUE(view.Ok());
EXPECT_TRUE(view.SizeIsKnown());
EXPECT_EQ(3U, view.SizeInBytes());
EXPECT_FALSE(view.has_xyc().ValueOr(true));
}
TEST(FlagConditional, True) {
::std::uint8_t buffer[2] = {0x80, 0xff};
auto writer = ContainsContainsBitsWriter(buffer, sizeof buffer);
EXPECT_TRUE(writer.Ok());
EXPECT_TRUE(writer.SizeIsKnown());
EXPECT_TRUE(writer.top().Ok());
}
TEST(WriteToString, MissingFieldsAreNotWritten) {
::std::uint8_t buffer[2] = {0x01, 0x00};
auto reader = BasicConditionalWriter(buffer, 1U);
EXPECT_EQ(
"{\n"
" x: 1 # 0x1\n"
"}",
::emboss::WriteToString(reader, ::emboss::MultilineText()));
EXPECT_EQ("{ x: 1 }", ::emboss::WriteToString(reader));
}
TEST(WriteToString, NotOkFieldsAreNotWritten) {
::std::uint8_t buffer[2] = {0x00, 0x00};
auto reader = BasicConditionalWriter(buffer, 1U);
EXPECT_FALSE(reader.Ok());
EXPECT_EQ(
"{\n"
" x: 0 # 0x0\n"
" # xc: UNREADABLE\n"
"}",
::emboss::WriteToString(
reader, ::emboss::MultilineText().WithAllowPartialOutput(true)));
EXPECT_EQ(
"{ x: 0 }",
::emboss::WriteToString(
reader, ::emboss::TextOutputOptions().WithAllowPartialOutput(true)));
}
TEST(WriteToString, NotOkStructSubFieldsAreNotWritten) {
::std::uint8_t buffer[2] = {0x00, 0x00};
auto reader = ConditionalInlineWriter(buffer, 2U);
EXPECT_FALSE(reader.Ok());
EXPECT_EQ(
"{\n"
" payload_id: 0 # 0x0\n"
" type_0: {\n"
" a: 0 # 0x0\n"
" # b: UNREADABLE\n"
" # c: UNREADABLE\n"
" }\n"
"}",
::emboss::WriteToString(
reader, ::emboss::MultilineText().WithAllowPartialOutput(true)));
EXPECT_EQ(
"{ payload_id: 0, type_0: { a: 0 } }",
::emboss::WriteToString(
reader, ::emboss::TextOutputOptions().WithAllowPartialOutput(true)));
}
TEST(WriteToString, PresentFieldsNotWritten) {
::std::uint8_t buffer[2] = {0x00, 0x01};
auto reader = BasicConditionalWriter(buffer, 2U);
EXPECT_EQ(
"{\n"
" x: 0 # 0x0\n"
" xc: 1 # 0x1\n"
"}",
::emboss::WriteToString(reader, ::emboss::MultilineText()));
EXPECT_EQ("{ x: 0, xc: 1 }", ::emboss::WriteToString(reader));
}
TEST(WriteToString, AlwaysFalseCondition) {
::std::uint8_t buffer[2] = {0x00};
auto reader = MakeAlwaysFalseConditionView(buffer, 1U);
EXPECT_EQ(
"{\n"
" x: 0 # 0x0\n"
"}",
::emboss::WriteToString(reader, ::emboss::MultilineText()));
EXPECT_EQ("{ x: 0 }", ::emboss::WriteToString(reader));
}
TEST(WriteToString, OnlyAlwaysFalseCondition) {
::std::uint8_t buffer[2] = {0x00};
auto reader = MakeOnlyAlwaysFalseConditionView(buffer, 0U);
EXPECT_EQ(
"{\n"
"}",
::emboss::WriteToString(reader, ::emboss::MultilineText()));
EXPECT_EQ("{ }", ::emboss::WriteToString(reader));
}
TEST(WriteToString, EmptyStruct) {
::std::uint8_t buffer[2] = {0x00};
auto reader = MakeEmptyStructView(buffer, 0U);
EXPECT_EQ(
"{\n"
"}",
::emboss::WriteToString(reader, ::emboss::MultilineText()));
EXPECT_EQ("{ }", ::emboss::WriteToString(reader));
}
TEST(ConditionalInline, ConditionalInline) {
::std::uint8_t buffer[4] = {0x00, 0x01, 0x02, 0x03};
auto reader = ConditionalInlineWriter(buffer, 4U);
EXPECT_EQ(1, reader.type_0().a().Read());
EXPECT_TRUE(reader.has_type_1().Known());
EXPECT_FALSE(reader.has_type_1().Value());
}
TEST(ConditionalAnonymous, ConditionalAnonymous) {
::std::array</**/ ::std::uint8_t, 2> buffer = {0x00, 0x98};
auto view = MakeConditionalAnonymousView(&buffer);
EXPECT_TRUE(view.Ok());
EXPECT_FALSE(view.has_low().Value());
EXPECT_FALSE(view.has_mid().Value());
EXPECT_FALSE(view.has_high().Value());
view.x().Write(100);
EXPECT_TRUE(view.has_low().Value());
EXPECT_FALSE(view.has_mid().Value());
EXPECT_TRUE(view.has_high().Value());
EXPECT_EQ(0, view.low().Read());
EXPECT_EQ(1, view.high().Read());
view.low().Write(1);
EXPECT_TRUE(view.has_low().Value());
EXPECT_TRUE(view.has_mid().Value());
EXPECT_TRUE(view.has_high().Value());
EXPECT_EQ(1, view.low().Read());
EXPECT_EQ(3, view.mid().Read());
EXPECT_EQ(1, view.high().Read());
}
TEST(ConditionalOnFlag, ConditionalOnFlag) {
::std::array</**/ ::std::uint8_t, 2> buffer = {0x00, 0x98};
auto view = MakeConditionalOnFlagView(&buffer);
EXPECT_TRUE(view.Ok());
EXPECT_FALSE(view.enabled().Read());
EXPECT_FALSE(view.has_value().Value());
buffer[0] = 1;
EXPECT_TRUE(view.enabled().Read());
EXPECT_TRUE(view.has_value().Value());
EXPECT_EQ(0x98, view.value().Read());
}
} // namespace
} // namespace test
} // namespace emboss