|  | // Copyright (c) 2013 The Chromium Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | #include "quic/core/quic_utils.h" | 
|  |  | 
|  | #include <string> | 
|  |  | 
|  | #include "absl/base/macros.h" | 
|  | #include "absl/strings/string_view.h" | 
|  | #include "quic/core/crypto/crypto_protocol.h" | 
|  | #include "quic/core/quic_connection_id.h" | 
|  | #include "quic/core/quic_types.h" | 
|  | #include "quic/platform/api/quic_test.h" | 
|  | #include "quic/test_tools/quic_test_utils.h" | 
|  |  | 
|  | namespace quic { | 
|  | namespace test { | 
|  | namespace { | 
|  |  | 
|  | class QuicUtilsTest : public QuicTest {}; | 
|  |  | 
|  | TEST_F(QuicUtilsTest, DetermineAddressChangeType) { | 
|  | const std::string kIPv4String1 = "1.2.3.4"; | 
|  | const std::string kIPv4String2 = "1.2.3.5"; | 
|  | const std::string kIPv4String3 = "1.1.3.5"; | 
|  | const std::string kIPv6String1 = "2001:700:300:1800::f"; | 
|  | const std::string kIPv6String2 = "2001:700:300:1800:1:1:1:f"; | 
|  | QuicSocketAddress old_address; | 
|  | QuicSocketAddress new_address; | 
|  | QuicIpAddress address; | 
|  |  | 
|  | EXPECT_EQ(NO_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  | ASSERT_TRUE(address.FromString(kIPv4String1)); | 
|  | old_address = QuicSocketAddress(address, 1234); | 
|  | EXPECT_EQ(NO_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  | new_address = QuicSocketAddress(address, 1234); | 
|  | EXPECT_EQ(NO_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  |  | 
|  | new_address = QuicSocketAddress(address, 5678); | 
|  | EXPECT_EQ(PORT_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  | ASSERT_TRUE(address.FromString(kIPv6String1)); | 
|  | old_address = QuicSocketAddress(address, 1234); | 
|  | new_address = QuicSocketAddress(address, 5678); | 
|  | EXPECT_EQ(PORT_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  |  | 
|  | ASSERT_TRUE(address.FromString(kIPv4String1)); | 
|  | old_address = QuicSocketAddress(address, 1234); | 
|  | ASSERT_TRUE(address.FromString(kIPv6String1)); | 
|  | new_address = QuicSocketAddress(address, 1234); | 
|  | EXPECT_EQ(IPV4_TO_IPV6_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  |  | 
|  | old_address = QuicSocketAddress(address, 1234); | 
|  | ASSERT_TRUE(address.FromString(kIPv4String1)); | 
|  | new_address = QuicSocketAddress(address, 1234); | 
|  | EXPECT_EQ(IPV6_TO_IPV4_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  |  | 
|  | ASSERT_TRUE(address.FromString(kIPv6String2)); | 
|  | new_address = QuicSocketAddress(address, 1234); | 
|  | EXPECT_EQ(IPV6_TO_IPV6_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  |  | 
|  | ASSERT_TRUE(address.FromString(kIPv4String1)); | 
|  | old_address = QuicSocketAddress(address, 1234); | 
|  | ASSERT_TRUE(address.FromString(kIPv4String2)); | 
|  | new_address = QuicSocketAddress(address, 1234); | 
|  | EXPECT_EQ(IPV4_SUBNET_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  | ASSERT_TRUE(address.FromString(kIPv4String3)); | 
|  | new_address = QuicSocketAddress(address, 1234); | 
|  | EXPECT_EQ(IPV4_TO_IPV4_CHANGE, | 
|  | QuicUtils::DetermineAddressChangeType(old_address, new_address)); | 
|  | } | 
|  |  | 
|  | QuicUint128 IncrementalHashReference(const void* data, size_t len) { | 
|  | // The two constants are defined as part of the hash algorithm. | 
|  | // see http://www.isthe.com/chongo/tech/comp/fnv/ | 
|  | // hash = 144066263297769815596495629667062367629 | 
|  | QuicUint128 hash = MakeQuicUint128(UINT64_C(7809847782465536322), | 
|  | UINT64_C(7113472399480571277)); | 
|  | // kPrime = 309485009821345068724781371 | 
|  | const QuicUint128 kPrime = MakeQuicUint128(16777216, 315); | 
|  | const uint8_t* octets = reinterpret_cast<const uint8_t*>(data); | 
|  | for (size_t i = 0; i < len; ++i) { | 
|  | hash = hash ^ MakeQuicUint128(0, octets[i]); | 
|  | hash = hash * kPrime; | 
|  | } | 
|  | return hash; | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, ReferenceTest) { | 
|  | std::vector<uint8_t> data(32); | 
|  | for (size_t i = 0; i < data.size(); ++i) { | 
|  | data[i] = i % 255; | 
|  | } | 
|  | EXPECT_EQ(IncrementalHashReference(data.data(), data.size()), | 
|  | QuicUtils::FNV1a_128_Hash(absl::string_view( | 
|  | reinterpret_cast<const char*>(data.data()), data.size()))); | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, IsUnackable) { | 
|  | for (size_t i = FIRST_PACKET_STATE; i <= LAST_PACKET_STATE; ++i) { | 
|  | if (i == NEVER_SENT || i == ACKED || i == UNACKABLE) { | 
|  | EXPECT_FALSE(QuicUtils::IsAckable(static_cast<SentPacketState>(i))); | 
|  | } else { | 
|  | EXPECT_TRUE(QuicUtils::IsAckable(static_cast<SentPacketState>(i))); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, RetransmissionTypeToPacketState) { | 
|  | for (size_t i = FIRST_TRANSMISSION_TYPE; i <= LAST_TRANSMISSION_TYPE; ++i) { | 
|  | if (i == NOT_RETRANSMISSION) { | 
|  | continue; | 
|  | } | 
|  | SentPacketState state = QuicUtils::RetransmissionTypeToPacketState( | 
|  | static_cast<TransmissionType>(i)); | 
|  | if (i == HANDSHAKE_RETRANSMISSION) { | 
|  | EXPECT_EQ(HANDSHAKE_RETRANSMITTED, state); | 
|  | } else if (i == LOSS_RETRANSMISSION) { | 
|  | EXPECT_EQ(LOST, state); | 
|  | } else if (i == ALL_ZERO_RTT_RETRANSMISSION) { | 
|  | EXPECT_EQ(UNACKABLE, state); | 
|  | } else if (i == TLP_RETRANSMISSION) { | 
|  | EXPECT_EQ(TLP_RETRANSMITTED, state); | 
|  | } else if (i == RTO_RETRANSMISSION) { | 
|  | EXPECT_EQ(RTO_RETRANSMITTED, state); | 
|  | } else if (i == PTO_RETRANSMISSION) { | 
|  | EXPECT_EQ(PTO_RETRANSMITTED, state); | 
|  | } else if (i == PROBING_RETRANSMISSION) { | 
|  | EXPECT_EQ(PROBE_RETRANSMITTED, state); | 
|  | } else if (i == PATH_RETRANSMISSION) { | 
|  | EXPECT_EQ(NOT_CONTRIBUTING_RTT, state); | 
|  | } else if (i == ALL_INITIAL_RETRANSMISSION) { | 
|  | EXPECT_EQ(UNACKABLE, state); | 
|  | } else { | 
|  | DCHECK(false) | 
|  | << "No corresponding packet state according to transmission type: " | 
|  | << i; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, IsIetfPacketHeader) { | 
|  | // IETF QUIC short header | 
|  | uint8_t first_byte = 0; | 
|  | EXPECT_TRUE(QuicUtils::IsIetfPacketHeader(first_byte)); | 
|  | EXPECT_TRUE(QuicUtils::IsIetfPacketShortHeader(first_byte)); | 
|  |  | 
|  | // IETF QUIC long header | 
|  | first_byte |= (FLAGS_LONG_HEADER | FLAGS_DEMULTIPLEXING_BIT); | 
|  | EXPECT_TRUE(QuicUtils::IsIetfPacketHeader(first_byte)); | 
|  | EXPECT_FALSE(QuicUtils::IsIetfPacketShortHeader(first_byte)); | 
|  |  | 
|  | // IETF QUIC long header, version negotiation. | 
|  | first_byte = 0; | 
|  | first_byte |= FLAGS_LONG_HEADER; | 
|  | EXPECT_TRUE(QuicUtils::IsIetfPacketHeader(first_byte)); | 
|  | EXPECT_FALSE(QuicUtils::IsIetfPacketShortHeader(first_byte)); | 
|  |  | 
|  | // GQUIC | 
|  | first_byte = 0; | 
|  | first_byte |= PACKET_PUBLIC_FLAGS_8BYTE_CONNECTION_ID; | 
|  | EXPECT_FALSE(QuicUtils::IsIetfPacketHeader(first_byte)); | 
|  | EXPECT_FALSE(QuicUtils::IsIetfPacketShortHeader(first_byte)); | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, ReplacementConnectionIdIsDeterministic) { | 
|  | // Verify that two equal connection IDs get the same replacement. | 
|  | QuicConnectionId connection_id64a = TestConnectionId(33); | 
|  | QuicConnectionId connection_id64b = TestConnectionId(33); | 
|  | EXPECT_EQ(connection_id64a, connection_id64b); | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id64a), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id64b)); | 
|  | QuicConnectionId connection_id72a = TestConnectionIdNineBytesLong(42); | 
|  | QuicConnectionId connection_id72b = TestConnectionIdNineBytesLong(42); | 
|  | EXPECT_EQ(connection_id72a, connection_id72b); | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id72a), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id72b)); | 
|  | // Test variant with custom length. | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id64a, 7), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id64b, 7)); | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id64a, 9), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id64b, 9)); | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id64a, 16), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id64b, 16)); | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id72a, 7), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id72b, 7)); | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id72a, 9), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id72b, 9)); | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id72a, 16), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id72b, 16)); | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id72a, 32), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id72b, 32)); | 
|  | EXPECT_EQ(QuicUtils::CreateReplacementConnectionId(connection_id72a, 255), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id72b, 255)); | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, ReplacementConnectionIdLengthIsCorrect) { | 
|  | // Verify that all lengths get replaced by kQuicDefaultConnectionIdLength. | 
|  | const char connection_id_bytes[255] = {}; | 
|  | for (uint8_t i = 0; i < sizeof(connection_id_bytes) - 1; ++i) { | 
|  | QuicConnectionId connection_id(connection_id_bytes, i); | 
|  | QuicConnectionId replacement_connection_id = | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id); | 
|  | EXPECT_EQ(kQuicDefaultConnectionIdLength, | 
|  | replacement_connection_id.length()); | 
|  | // Test variant with custom length. | 
|  | QuicConnectionId replacement_connection_id7 = | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id, 7); | 
|  | EXPECT_EQ(7, replacement_connection_id7.length()); | 
|  | QuicConnectionId replacement_connection_id9 = | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id, 9); | 
|  | EXPECT_EQ(9, replacement_connection_id9.length()); | 
|  | QuicConnectionId replacement_connection_id16 = | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id, 16); | 
|  | EXPECT_EQ(16, replacement_connection_id16.length()); | 
|  | QuicConnectionId replacement_connection_id32 = | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id, 32); | 
|  | EXPECT_EQ(32, replacement_connection_id32.length()); | 
|  | QuicConnectionId replacement_connection_id255 = | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id, 255); | 
|  | EXPECT_EQ(255, replacement_connection_id255.length()); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, ReplacementConnectionIdHasEntropy) { | 
|  | // Make sure all these test connection IDs have different replacements. | 
|  | for (uint64_t i = 0; i < 256; ++i) { | 
|  | QuicConnectionId connection_id_i = TestConnectionId(i); | 
|  | EXPECT_NE(connection_id_i, | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id_i)); | 
|  | for (uint64_t j = i + 1; j <= 256; ++j) { | 
|  | QuicConnectionId connection_id_j = TestConnectionId(j); | 
|  | EXPECT_NE(connection_id_i, connection_id_j); | 
|  | EXPECT_NE(QuicUtils::CreateReplacementConnectionId(connection_id_i), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id_j)); | 
|  | // Test variant with custom length. | 
|  | EXPECT_NE(QuicUtils::CreateReplacementConnectionId(connection_id_i, 7), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id_j, 7)); | 
|  | EXPECT_NE(QuicUtils::CreateReplacementConnectionId(connection_id_i, 9), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id_j, 9)); | 
|  | EXPECT_NE(QuicUtils::CreateReplacementConnectionId(connection_id_i, 16), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id_j, 16)); | 
|  | EXPECT_NE(QuicUtils::CreateReplacementConnectionId(connection_id_i, 32), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id_j, 32)); | 
|  | EXPECT_NE(QuicUtils::CreateReplacementConnectionId(connection_id_i, 255), | 
|  | QuicUtils::CreateReplacementConnectionId(connection_id_j, 255)); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, RandomConnectionId) { | 
|  | MockRandom random(33); | 
|  | QuicConnectionId connection_id = QuicUtils::CreateRandomConnectionId(&random); | 
|  | EXPECT_EQ(connection_id.length(), sizeof(uint64_t)); | 
|  | char connection_id_bytes[sizeof(uint64_t)]; | 
|  | random.RandBytes(connection_id_bytes, ABSL_ARRAYSIZE(connection_id_bytes)); | 
|  | EXPECT_EQ(connection_id, | 
|  | QuicConnectionId(static_cast<char*>(connection_id_bytes), | 
|  | ABSL_ARRAYSIZE(connection_id_bytes))); | 
|  | EXPECT_NE(connection_id, EmptyQuicConnectionId()); | 
|  | EXPECT_NE(connection_id, TestConnectionId()); | 
|  | EXPECT_NE(connection_id, TestConnectionId(1)); | 
|  | EXPECT_NE(connection_id, TestConnectionIdNineBytesLong(1)); | 
|  | EXPECT_EQ(QuicUtils::CreateRandomConnectionId().length(), | 
|  | kQuicDefaultConnectionIdLength); | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, RandomConnectionIdVariableLength) { | 
|  | MockRandom random(1337); | 
|  | const uint8_t connection_id_length = 9; | 
|  | QuicConnectionId connection_id = | 
|  | QuicUtils::CreateRandomConnectionId(connection_id_length, &random); | 
|  | EXPECT_EQ(connection_id.length(), connection_id_length); | 
|  | char connection_id_bytes[connection_id_length]; | 
|  | random.RandBytes(connection_id_bytes, ABSL_ARRAYSIZE(connection_id_bytes)); | 
|  | EXPECT_EQ(connection_id, | 
|  | QuicConnectionId(static_cast<char*>(connection_id_bytes), | 
|  | ABSL_ARRAYSIZE(connection_id_bytes))); | 
|  | EXPECT_NE(connection_id, EmptyQuicConnectionId()); | 
|  | EXPECT_NE(connection_id, TestConnectionId()); | 
|  | EXPECT_NE(connection_id, TestConnectionId(1)); | 
|  | EXPECT_NE(connection_id, TestConnectionIdNineBytesLong(1)); | 
|  | EXPECT_EQ(QuicUtils::CreateRandomConnectionId(connection_id_length).length(), | 
|  | connection_id_length); | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, VariableLengthConnectionId) { | 
|  | EXPECT_FALSE(VersionAllowsVariableLengthConnectionIds(QUIC_VERSION_43)); | 
|  | EXPECT_TRUE(QuicUtils::IsConnectionIdValidForVersion( | 
|  | QuicUtils::CreateZeroConnectionId(QUIC_VERSION_43), QUIC_VERSION_43)); | 
|  | EXPECT_TRUE(QuicUtils::IsConnectionIdValidForVersion( | 
|  | QuicUtils::CreateZeroConnectionId(QUIC_VERSION_50), QUIC_VERSION_50)); | 
|  | EXPECT_NE(QuicUtils::CreateZeroConnectionId(QUIC_VERSION_43), | 
|  | EmptyQuicConnectionId()); | 
|  | EXPECT_EQ(QuicUtils::CreateZeroConnectionId(QUIC_VERSION_50), | 
|  | EmptyQuicConnectionId()); | 
|  | EXPECT_FALSE(QuicUtils::IsConnectionIdValidForVersion(EmptyQuicConnectionId(), | 
|  | QUIC_VERSION_43)); | 
|  | } | 
|  |  | 
|  | TEST_F(QuicUtilsTest, StatelessResetToken) { | 
|  | QuicConnectionId connection_id1a = test::TestConnectionId(1); | 
|  | QuicConnectionId connection_id1b = test::TestConnectionId(1); | 
|  | QuicConnectionId connection_id2 = test::TestConnectionId(2); | 
|  | QuicUint128 token1a = QuicUtils::GenerateStatelessResetToken(connection_id1a); | 
|  | QuicUint128 token1b = QuicUtils::GenerateStatelessResetToken(connection_id1b); | 
|  | QuicUint128 token2 = QuicUtils::GenerateStatelessResetToken(connection_id2); | 
|  | EXPECT_EQ(token1a, token1b); | 
|  | EXPECT_NE(token1a, token2); | 
|  | } | 
|  |  | 
|  | enum class TestEnumClassBit : uint8_t { | 
|  | BIT_ZERO = 0, | 
|  | BIT_ONE, | 
|  | BIT_TWO, | 
|  | }; | 
|  |  | 
|  | enum TestEnumBit { | 
|  | TEST_BIT_0 = 0, | 
|  | TEST_BIT_1, | 
|  | TEST_BIT_2, | 
|  | }; | 
|  |  | 
|  | TEST(QuicBitMaskTest, EnumClass) { | 
|  | BitMask64 mask(TestEnumClassBit::BIT_ZERO, TestEnumClassBit::BIT_TWO); | 
|  | EXPECT_TRUE(mask.IsSet(TestEnumClassBit::BIT_ZERO)); | 
|  | EXPECT_FALSE(mask.IsSet(TestEnumClassBit::BIT_ONE)); | 
|  | EXPECT_TRUE(mask.IsSet(TestEnumClassBit::BIT_TWO)); | 
|  |  | 
|  | mask.ClearAll(); | 
|  | EXPECT_FALSE(mask.IsSet(TestEnumClassBit::BIT_ZERO)); | 
|  | EXPECT_FALSE(mask.IsSet(TestEnumClassBit::BIT_ONE)); | 
|  | EXPECT_FALSE(mask.IsSet(TestEnumClassBit::BIT_TWO)); | 
|  | } | 
|  |  | 
|  | TEST(QuicBitMaskTest, Enum) { | 
|  | BitMask64 mask(TEST_BIT_1, TEST_BIT_2); | 
|  | EXPECT_FALSE(mask.IsSet(TEST_BIT_0)); | 
|  | EXPECT_TRUE(mask.IsSet(TEST_BIT_1)); | 
|  | EXPECT_TRUE(mask.IsSet(TEST_BIT_2)); | 
|  |  | 
|  | mask.ClearAll(); | 
|  | EXPECT_FALSE(mask.IsSet(TEST_BIT_0)); | 
|  | EXPECT_FALSE(mask.IsSet(TEST_BIT_1)); | 
|  | EXPECT_FALSE(mask.IsSet(TEST_BIT_2)); | 
|  | } | 
|  |  | 
|  | TEST(QuicBitMaskTest, Integer) { | 
|  | BitMask64 mask(1, 3); | 
|  | mask.Set(3); | 
|  | mask.Set(5, 7, 9); | 
|  | EXPECT_FALSE(mask.IsSet(0)); | 
|  | EXPECT_TRUE(mask.IsSet(1)); | 
|  | EXPECT_FALSE(mask.IsSet(2)); | 
|  | EXPECT_TRUE(mask.IsSet(3)); | 
|  | EXPECT_FALSE(mask.IsSet(4)); | 
|  | EXPECT_TRUE(mask.IsSet(5)); | 
|  | EXPECT_FALSE(mask.IsSet(6)); | 
|  | EXPECT_TRUE(mask.IsSet(7)); | 
|  | EXPECT_FALSE(mask.IsSet(8)); | 
|  | EXPECT_TRUE(mask.IsSet(9)); | 
|  | } | 
|  |  | 
|  | TEST(QuicBitMaskTest, NumBits) { | 
|  | EXPECT_EQ(64u, BitMask64::NumBits()); | 
|  | EXPECT_EQ(32u, BitMask<uint32_t>::NumBits()); | 
|  | } | 
|  |  | 
|  | TEST(QuicBitMaskTest, Constructor) { | 
|  | BitMask64 empty_mask; | 
|  | for (size_t bit = 0; bit < empty_mask.NumBits(); ++bit) { | 
|  | EXPECT_FALSE(empty_mask.IsSet(bit)); | 
|  | } | 
|  |  | 
|  | BitMask64 mask(1, 3); | 
|  | BitMask64 mask2 = mask; | 
|  | BitMask64 mask3(mask2); | 
|  |  | 
|  | for (size_t bit = 0; bit < mask.NumBits(); ++bit) { | 
|  | EXPECT_EQ(mask.IsSet(bit), mask2.IsSet(bit)); | 
|  | EXPECT_EQ(mask.IsSet(bit), mask3.IsSet(bit)); | 
|  | } | 
|  |  | 
|  | EXPECT_TRUE(std::is_trivially_copyable<BitMask64>::value); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  | }  // namespace test | 
|  | }  // namespace quic |