| // Copyright 2022 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 "quiche/common/quiche_data_reader.h" | 
 |  | 
 | #include <cstdint> | 
 |  | 
 | #include "quiche/common/platform/api/quiche_test.h" | 
 | #include "quiche/common/quiche_endian.h" | 
 |  | 
 | namespace quiche { | 
 |  | 
 | // TODO(b/214573190): Test Endianness::HOST_BYTE_ORDER. | 
 | // TODO(b/214573190): Test ReadUInt8, ReadUInt24, ReadUInt64, ReadBytesToUInt64, | 
 | // ReadStringPiece8, ReadStringPiece, ReadTag, etc. | 
 |  | 
 | TEST(QuicheDataReaderTest, ReadUInt16) { | 
 |   // Data in network byte order. | 
 |   const uint16_t kData[] = { | 
 |       QuicheEndian::HostToNet16(1), | 
 |       QuicheEndian::HostToNet16(1 << 15), | 
 |   }; | 
 |  | 
 |   QuicheDataReader reader(reinterpret_cast<const char*>(kData), sizeof(kData)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |  | 
 |   uint16_t uint16_val; | 
 |   EXPECT_TRUE(reader.ReadUInt16(&uint16_val)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |   EXPECT_EQ(1, uint16_val); | 
 |  | 
 |   EXPECT_TRUE(reader.ReadUInt16(&uint16_val)); | 
 |   EXPECT_TRUE(reader.IsDoneReading()); | 
 |   EXPECT_EQ(1 << 15, uint16_val); | 
 | } | 
 |  | 
 | TEST(QuicheDataReaderTest, ReadUInt32) { | 
 |   // Data in network byte order. | 
 |   const uint32_t kData[] = { | 
 |       QuicheEndian::HostToNet32(1), | 
 |       QuicheEndian::HostToNet32(0x80000000), | 
 |   }; | 
 |  | 
 |   QuicheDataReader reader(reinterpret_cast<const char*>(kData), | 
 |                           ABSL_ARRAYSIZE(kData) * sizeof(uint32_t)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |  | 
 |   uint32_t uint32_val; | 
 |   EXPECT_TRUE(reader.ReadUInt32(&uint32_val)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |   EXPECT_EQ(1u, uint32_val); | 
 |  | 
 |   EXPECT_TRUE(reader.ReadUInt32(&uint32_val)); | 
 |   EXPECT_TRUE(reader.IsDoneReading()); | 
 |   EXPECT_EQ(1u << 31, uint32_val); | 
 | } | 
 |  | 
 | TEST(QuicheDataReaderTest, ReadStringPiece16) { | 
 |   // Data in network byte order. | 
 |   const char kData[] = { | 
 |       0x00, 0x02,  // uint16_t(2) | 
 |       0x48, 0x69,  // "Hi" | 
 |       0x00, 0x10,  // uint16_t(16) | 
 |       0x54, 0x65, 0x73, 0x74, 0x69, 0x6e, 0x67, 0x2c, | 
 |       0x20, 0x31, 0x2c, 0x20, 0x32, 0x2c, 0x20, 0x33,  // "Testing, 1, 2, 3" | 
 |   }; | 
 |  | 
 |   QuicheDataReader reader(kData, ABSL_ARRAYSIZE(kData)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |  | 
 |   absl::string_view stringpiece_val; | 
 |   EXPECT_TRUE(reader.ReadStringPiece16(&stringpiece_val)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |   EXPECT_EQ(0, stringpiece_val.compare("Hi")); | 
 |  | 
 |   EXPECT_TRUE(reader.ReadStringPiece16(&stringpiece_val)); | 
 |   EXPECT_TRUE(reader.IsDoneReading()); | 
 |   EXPECT_EQ(0, stringpiece_val.compare("Testing, 1, 2, 3")); | 
 | } | 
 |  | 
 | TEST(QuicheDataReaderTest, ReadUInt16WithBufferTooSmall) { | 
 |   // Data in network byte order. | 
 |   const char kData[] = { | 
 |       0x00,  // part of a uint16_t | 
 |   }; | 
 |  | 
 |   QuicheDataReader reader(kData, ABSL_ARRAYSIZE(kData)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |  | 
 |   uint16_t uint16_val; | 
 |   EXPECT_FALSE(reader.ReadUInt16(&uint16_val)); | 
 | } | 
 |  | 
 | TEST(QuicheDataReaderTest, ReadUInt32WithBufferTooSmall) { | 
 |   // Data in network byte order. | 
 |   const char kData[] = { | 
 |       0x00, 0x00, 0x00,  // part of a uint32_t | 
 |   }; | 
 |  | 
 |   QuicheDataReader reader(kData, ABSL_ARRAYSIZE(kData)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |  | 
 |   uint32_t uint32_val; | 
 |   EXPECT_FALSE(reader.ReadUInt32(&uint32_val)); | 
 |  | 
 |   // Also make sure that trying to read a uint16_t, which technically could | 
 |   // work, fails immediately due to previously encountered failed read. | 
 |   uint16_t uint16_val; | 
 |   EXPECT_FALSE(reader.ReadUInt16(&uint16_val)); | 
 | } | 
 |  | 
 | // Tests ReadStringPiece16() with a buffer too small to fit the entire string. | 
 | TEST(QuicheDataReaderTest, ReadStringPiece16WithBufferTooSmall) { | 
 |   // Data in network byte order. | 
 |   const char kData[] = { | 
 |       0x00, 0x03,  // uint16_t(3) | 
 |       0x48, 0x69,  // "Hi" | 
 |   }; | 
 |  | 
 |   QuicheDataReader reader(kData, ABSL_ARRAYSIZE(kData)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |  | 
 |   absl::string_view stringpiece_val; | 
 |   EXPECT_FALSE(reader.ReadStringPiece16(&stringpiece_val)); | 
 |  | 
 |   // Also make sure that trying to read a uint16_t, which technically could | 
 |   // work, fails immediately due to previously encountered failed read. | 
 |   uint16_t uint16_val; | 
 |   EXPECT_FALSE(reader.ReadUInt16(&uint16_val)); | 
 | } | 
 |  | 
 | // Tests ReadStringPiece16() with a buffer too small even to fit the length. | 
 | TEST(QuicheDataReaderTest, ReadStringPiece16WithBufferWayTooSmall) { | 
 |   // Data in network byte order. | 
 |   const char kData[] = { | 
 |       0x00,  // part of a uint16_t | 
 |   }; | 
 |  | 
 |   QuicheDataReader reader(kData, ABSL_ARRAYSIZE(kData)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |  | 
 |   absl::string_view stringpiece_val; | 
 |   EXPECT_FALSE(reader.ReadStringPiece16(&stringpiece_val)); | 
 |  | 
 |   // Also make sure that trying to read a uint16_t, which technically could | 
 |   // work, fails immediately due to previously encountered failed read. | 
 |   uint16_t uint16_val; | 
 |   EXPECT_FALSE(reader.ReadUInt16(&uint16_val)); | 
 | } | 
 |  | 
 | TEST(QuicheDataReaderTest, ReadBytes) { | 
 |   // Data in network byte order. | 
 |   const char kData[] = { | 
 |       0x66, 0x6f, 0x6f,  // "foo" | 
 |       0x48, 0x69,        // "Hi" | 
 |   }; | 
 |  | 
 |   QuicheDataReader reader(kData, ABSL_ARRAYSIZE(kData)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |  | 
 |   char dest1[3] = {}; | 
 |   EXPECT_TRUE(reader.ReadBytes(&dest1, ABSL_ARRAYSIZE(dest1))); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |   EXPECT_EQ("foo", absl::string_view(dest1, ABSL_ARRAYSIZE(dest1))); | 
 |  | 
 |   char dest2[2] = {}; | 
 |   EXPECT_TRUE(reader.ReadBytes(&dest2, ABSL_ARRAYSIZE(dest2))); | 
 |   EXPECT_TRUE(reader.IsDoneReading()); | 
 |   EXPECT_EQ("Hi", absl::string_view(dest2, ABSL_ARRAYSIZE(dest2))); | 
 | } | 
 |  | 
 | TEST(QuicheDataReaderTest, ReadBytesWithBufferTooSmall) { | 
 |   // Data in network byte order. | 
 |   const char kData[] = { | 
 |       0x01, | 
 |   }; | 
 |  | 
 |   QuicheDataReader reader(kData, ABSL_ARRAYSIZE(kData)); | 
 |   EXPECT_FALSE(reader.IsDoneReading()); | 
 |  | 
 |   char dest[ABSL_ARRAYSIZE(kData) + 2] = {}; | 
 |   EXPECT_FALSE(reader.ReadBytes(&dest, ABSL_ARRAYSIZE(kData) + 1)); | 
 |   EXPECT_STREQ("", dest); | 
 | } | 
 |  | 
 | }  // namespace quiche |