/* * Copyright (C) 2018 The Android Open Source Project * * 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 * * http://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. */ #include <algorithm> #include <cstdint> #include <utility> #include <gtest/gtest.h> #include "netdutils/MemBlock.h" #include "netdutils/Slice.h" namespace android { namespace netdutils { namespace { constexpr unsigned DNS_PACKET_SIZE = 512; constexpr int ARBITRARY_VALUE = 0x55; MemBlock makeArbitraryMemBlock(size_t len) { MemBlock result(len); // Do some fictional work before returning. for (Slice slice = result.get(); !slice.empty(); slice = drop(slice, 1)) { slice.base()[0] = ARBITRARY_VALUE; } return result; } void checkAllZeros(Slice slice) { for (; !slice.empty(); slice = drop(slice, 1)) { EXPECT_EQ(0U, slice.base()[0]); } } void checkArbitraryMemBlock(const MemBlock& block, size_t expectedSize) { Slice slice = block.get(); EXPECT_EQ(expectedSize, slice.size()); EXPECT_NE(nullptr, slice.base()); for (; !slice.empty(); slice = drop(slice, 1)) { EXPECT_EQ(ARBITRARY_VALUE, slice.base()[0]); } } void checkHelloMello(Slice dest, Slice src) { EXPECT_EQ('h', dest.base()[0]); EXPECT_EQ('e', dest.base()[1]); EXPECT_EQ('l', dest.base()[2]); EXPECT_EQ('l', dest.base()[3]); EXPECT_EQ('o', dest.base()[4]); src.base()[0] = 'm'; EXPECT_EQ('h', dest.base()[0]); } } // namespace TEST(MemBlockTest, Empty) { MemBlock empty; EXPECT_TRUE(empty.get().empty()); EXPECT_EQ(nullptr, empty.get().base()); } TEST(MemBlockTest, ExplicitZero) { MemBlock zero(0); EXPECT_TRUE(zero.get().empty()); EXPECT_EQ(nullptr, zero.get().base()); } TEST(MemBlockTest, BasicAllocation) { MemBlock dnsPacket(DNS_PACKET_SIZE); Slice slice = dnsPacket.get(); EXPECT_EQ(DNS_PACKET_SIZE, slice.size()); // Verify the space is '\0'-initialized. ASSERT_NO_FATAL_FAILURE(checkAllZeros(slice)); EXPECT_NE(nullptr, slice.base()); } TEST(MemBlockTest, MoveConstruction) { MemBlock block(makeArbitraryMemBlock(DNS_PACKET_SIZE)); ASSERT_NO_FATAL_FAILURE(checkArbitraryMemBlock(block, DNS_PACKET_SIZE)); } TEST(MemBlockTest, MoveAssignmentOrConstruction) { MemBlock block = makeArbitraryMemBlock(DNS_PACKET_SIZE); ASSERT_NO_FATAL_FAILURE(checkArbitraryMemBlock(block, DNS_PACKET_SIZE)); } TEST(MemBlockTest, StdMoveAssignment) { constexpr unsigned SIZE = 10; MemBlock block; EXPECT_TRUE(block.get().empty()); EXPECT_EQ(nullptr, block.get().base()); { MemBlock block2 = makeArbitraryMemBlock(SIZE); EXPECT_EQ(SIZE, block2.get().size()); // More fictional work. for (unsigned i = 0; i < SIZE; i++) { block2.get().base()[i] = i; } block = std::move(block2); } EXPECT_EQ(SIZE, block.get().size()); for (unsigned i = 0; i < SIZE; i++) { EXPECT_EQ(i, block.get().base()[i]); } } TEST(MemBlockTest, ConstructionFromSlice) { uint8_t data[] = {'h', 'e', 'l', 'l', 'o'}; Slice dataSlice(Slice(data, sizeof(data) / sizeof(data[0]))); MemBlock dataCopy(dataSlice); ASSERT_NO_FATAL_FAILURE(checkHelloMello(dataCopy.get(), dataSlice)); } TEST(MemBlockTest, ImplicitCastToSlice) { uint8_t data[] = {'h', 'e', 'l', 'l', 'o'}; Slice dataSlice(Slice(data, sizeof(data) / sizeof(data[0]))); MemBlock dataCopy(dataSlice.size()); // NOTE: no explicit MemBlock::get(). // Verify the space is '\0'-initialized. ASSERT_NO_FATAL_FAILURE(checkAllZeros(dataCopy)); copy(dataCopy, dataSlice); ASSERT_NO_FATAL_FAILURE(checkHelloMello(dataCopy, dataSlice)); } } // namespace netdutils } // namespace android