|  | // 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 |