| // Copyright (c) 2010, Google Inc. |
| // All rights reserved. |
| // |
| // Redistribution and use in source and binary forms, with or without |
| // modification, are permitted provided that the following conditions are |
| // met: |
| // |
| // * Redistributions of source code must retain the above copyright |
| // notice, this list of conditions and the following disclaimer. |
| // * Redistributions in binary form must reproduce the above |
| // copyright notice, this list of conditions and the following disclaimer |
| // in the documentation and/or other materials provided with the |
| // distribution. |
| // * Neither the name of Google Inc. nor the names of its |
| // contributors may be used to endorse or promote products derived from |
| // this software without specific prior written permission. |
| // |
| // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS |
| // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT |
| // LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR |
| // A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT |
| // OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
| // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT |
| // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, |
| // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY |
| // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
| // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
| // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
| |
| #include <ios> |
| #include <string> |
| #include <vector> |
| |
| #include "breakpad_googletest_includes.h" |
| #include "common/using_std_string.h" |
| #include "processor/binarystream.h" |
| |
| namespace { |
| using std::ios_base; |
| using std::vector; |
| using google_breakpad::binarystream; |
| |
| |
| class BinaryStreamBasicTest : public ::testing::Test { |
| protected: |
| binarystream stream; |
| }; |
| |
| TEST_F(BinaryStreamBasicTest, ReadU8) { |
| uint8_t u8 = 0; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u8; |
| ASSERT_TRUE(stream.eof()); |
| EXPECT_EQ(0U, u8); |
| stream.rewind(); |
| stream.clear(); |
| stream << (uint8_t)1; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u8; |
| EXPECT_EQ(1, u8); |
| EXPECT_FALSE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadU16) { |
| uint16_t u16 = 0; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u16; |
| ASSERT_TRUE(stream.eof()); |
| EXPECT_EQ(0U, u16); |
| stream.rewind(); |
| stream.clear(); |
| stream << (uint16_t)1; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u16; |
| EXPECT_EQ(1, u16); |
| EXPECT_FALSE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadU32) { |
| uint32_t u32 = 0; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u32; |
| ASSERT_TRUE(stream.eof()); |
| EXPECT_EQ(0U, u32); |
| stream.rewind(); |
| stream.clear(); |
| stream << (uint32_t)1; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u32; |
| EXPECT_EQ(1U, u32); |
| EXPECT_FALSE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadU64) { |
| uint64_t u64 = 0; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u64; |
| ASSERT_TRUE(stream.eof()); |
| EXPECT_EQ(0U, u64); |
| stream.rewind(); |
| stream.clear(); |
| stream << (uint64_t)1; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u64; |
| EXPECT_EQ(1U, u64); |
| EXPECT_FALSE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadString) { |
| string s(""); |
| ASSERT_FALSE(stream.eof()); |
| stream >> s; |
| ASSERT_TRUE(stream.eof()); |
| EXPECT_EQ("", s); |
| // write an empty string to the stream, read it back |
| s = "abcd"; |
| stream.rewind(); |
| stream.clear(); |
| stream << string(""); |
| stream >> s; |
| EXPECT_EQ("", s); |
| EXPECT_FALSE(stream.eof()); |
| stream.rewind(); |
| stream.clear(); |
| stream << string("test"); |
| ASSERT_FALSE(stream.eof()); |
| stream >> s; |
| EXPECT_EQ("test", s); |
| EXPECT_FALSE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadEmptyString) { |
| string s("abc"); |
| stream << string(""); |
| stream >> s; |
| EXPECT_EQ("", s); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadMultiU8) { |
| const uint8_t ea = 0, eb = 100, ec = 200, ed = 0xFF; |
| uint8_t a, b, c, d, e; |
| stream << ea << eb << ec << ed; |
| stream >> a >> b >> c >> d; |
| ASSERT_FALSE(stream.eof()); |
| EXPECT_EQ(ea, a); |
| EXPECT_EQ(eb, b); |
| EXPECT_EQ(ec, c); |
| EXPECT_EQ(ed, d); |
| ASSERT_FALSE(stream.eof()); |
| e = 0; |
| stream >> e; |
| EXPECT_EQ(0U, e); |
| ASSERT_TRUE(stream.eof()); |
| // try reading all at once, including one past eof |
| stream.rewind(); |
| stream.clear(); |
| ASSERT_FALSE(stream.eof()); |
| a = b = c = d = e = 0; |
| stream << ea << eb << ec << ed; |
| stream >> a >> b >> c >> d >> e; |
| EXPECT_EQ(ea, a); |
| EXPECT_EQ(eb, b); |
| EXPECT_EQ(ec, c); |
| EXPECT_EQ(ed, d); |
| EXPECT_EQ(0U, e); |
| EXPECT_TRUE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadMultiU16) { |
| const uint16_t ea = 0, eb = 0x100, ec = 0x8000, ed = 0xFFFF; |
| uint16_t a, b, c, d, e; |
| stream << ea << eb << ec << ed; |
| stream >> a >> b >> c >> d; |
| ASSERT_FALSE(stream.eof()); |
| EXPECT_EQ(ea, a); |
| EXPECT_EQ(eb, b); |
| EXPECT_EQ(ec, c); |
| EXPECT_EQ(ed, d); |
| ASSERT_FALSE(stream.eof()); |
| e = 0; |
| stream >> e; |
| EXPECT_EQ(0U, e); |
| EXPECT_TRUE(stream.eof()); |
| // try reading all at once, including one past eof |
| stream.rewind(); |
| stream.clear(); |
| ASSERT_FALSE(stream.eof()); |
| a = b = c = d = e = 0; |
| stream << ea << eb << ec << ed; |
| stream >> a >> b >> c >> d >> e; |
| EXPECT_EQ(ea, a); |
| EXPECT_EQ(eb, b); |
| EXPECT_EQ(ec, c); |
| EXPECT_EQ(ed, d); |
| EXPECT_EQ(0U, e); |
| EXPECT_TRUE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadMultiU32) { |
| const uint32_t ea = 0, eb = 0x10000, ec = 0x8000000, ed = 0xFFFFFFFF; |
| uint32_t a, b, c, d, e; |
| stream << ea << eb << ec << ed; |
| stream >> a >> b >> c >> d; |
| ASSERT_FALSE(stream.eof()); |
| EXPECT_EQ(ea, a); |
| EXPECT_EQ(eb, b); |
| EXPECT_EQ(ec, c); |
| EXPECT_EQ(ed, d); |
| ASSERT_FALSE(stream.eof()); |
| e = 0; |
| stream >> e; |
| EXPECT_EQ(0U, e); |
| EXPECT_TRUE(stream.eof()); |
| // try reading all at once, including one past eof |
| stream.rewind(); |
| stream.clear(); |
| ASSERT_FALSE(stream.eof()); |
| a = b = c = d = e = 0; |
| stream << ea << eb << ec << ed; |
| stream >> a >> b >> c >> d >> e; |
| EXPECT_EQ(ea, a); |
| EXPECT_EQ(eb, b); |
| EXPECT_EQ(ec, c); |
| EXPECT_EQ(ed, d); |
| EXPECT_EQ(0U, e); |
| EXPECT_TRUE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadMultiU64) { |
| const uint64_t ea = 0, eb = 0x10000, ec = 0x100000000ULL, |
| ed = 0xFFFFFFFFFFFFFFFFULL; |
| uint64_t a, b, c, d, e; |
| stream << ea << eb << ec << ed; |
| stream >> a >> b >> c >> d; |
| ASSERT_FALSE(stream.eof()); |
| EXPECT_EQ(ea, a); |
| EXPECT_EQ(eb, b); |
| EXPECT_EQ(ec, c); |
| EXPECT_EQ(ed, d); |
| ASSERT_FALSE(stream.eof()); |
| e = 0; |
| stream >> e; |
| EXPECT_EQ(0U, e); |
| EXPECT_TRUE(stream.eof()); |
| // try reading all at once, including one past eof |
| stream.rewind(); |
| stream.clear(); |
| ASSERT_FALSE(stream.eof()); |
| a = b = c = d = e = 0; |
| stream << ea << eb << ec << ed; |
| stream >> a >> b >> c >> d >> e; |
| EXPECT_EQ(ea, a); |
| EXPECT_EQ(eb, b); |
| EXPECT_EQ(ec, c); |
| EXPECT_EQ(ed, d); |
| EXPECT_EQ(0U, e); |
| EXPECT_TRUE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadMixed) { |
| const uint8_t e8 = 0x10; |
| const uint16_t e16 = 0x2020; |
| const uint32_t e32 = 0x30303030; |
| const uint64_t e64 = 0x4040404040404040ULL; |
| const string es = "test"; |
| uint8_t u8 = 0; |
| uint16_t u16 = 0; |
| uint32_t u32 = 0; |
| uint64_t u64 = 0; |
| string s("test"); |
| stream << e8 << e16 << e32 << e64 << es; |
| stream >> u8 >> u16 >> u32 >> u64 >> s; |
| EXPECT_FALSE(stream.eof()); |
| EXPECT_EQ(e8, u8); |
| EXPECT_EQ(e16, u16); |
| EXPECT_EQ(e32, u32); |
| EXPECT_EQ(e64, u64); |
| EXPECT_EQ(es, s); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadStringMissing) { |
| // ensure that reading a string where only the length is present fails |
| uint16_t u16 = 8; |
| stream << u16; |
| stream.rewind(); |
| string s(""); |
| stream >> s; |
| EXPECT_EQ("", s); |
| EXPECT_TRUE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, ReadStringTruncated) { |
| // ensure that reading a string where not all the data is present fails |
| uint16_t u16 = 8; |
| stream << u16; |
| stream << (uint8_t)'t' << (uint8_t)'e' << (uint8_t)'s' << (uint8_t)'t'; |
| stream.rewind(); |
| string s(""); |
| stream >> s; |
| EXPECT_EQ("", s); |
| EXPECT_TRUE(stream.eof()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, StreamByteLength) { |
| // Test that the stream buffer contains the right amount of data |
| stream << (uint8_t)0 << (uint16_t)1 << (uint32_t)2 << (uint64_t)3 |
| << string("test"); |
| string s = stream.str(); |
| EXPECT_EQ(21U, s.length()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, AppendStreamResultsByteLength) { |
| // Test that appending the str() results from two streams |
| // gives the right byte length |
| binarystream stream2; |
| stream << (uint8_t)0 << (uint16_t)1; |
| stream2 << (uint32_t)0 << (uint64_t)2 |
| << string("test"); |
| string s = stream.str(); |
| string s2 = stream2.str(); |
| s.append(s2); |
| EXPECT_EQ(21U, s.length()); |
| } |
| |
| TEST_F(BinaryStreamBasicTest, StreamSetStr) { |
| const string es("test"); |
| stream << es; |
| binarystream stream2; |
| stream2.str(stream.str()); |
| string s; |
| stream2 >> s; |
| EXPECT_FALSE(stream2.eof()); |
| EXPECT_EQ("test", s); |
| s = ""; |
| stream2.str(stream.str()); |
| stream2.rewind(); |
| stream2 >> s; |
| EXPECT_FALSE(stream2.eof()); |
| EXPECT_EQ("test", s); |
| } |
| |
| class BinaryStreamU8Test : public ::testing::Test { |
| protected: |
| binarystream stream; |
| |
| void SetUp() { |
| stream << (uint8_t)1; |
| } |
| }; |
| |
| TEST_F(BinaryStreamU8Test, ReadU16) { |
| uint16_t u16 = 0; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u16; |
| ASSERT_TRUE(stream.eof()); |
| EXPECT_EQ(0U, u16); |
| } |
| |
| TEST_F(BinaryStreamU8Test, ReadU32) { |
| uint32_t u32 = 0; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u32; |
| ASSERT_TRUE(stream.eof()); |
| EXPECT_EQ(0U, u32); |
| } |
| |
| TEST_F(BinaryStreamU8Test, ReadU64) { |
| uint64_t u64 = 0; |
| ASSERT_FALSE(stream.eof()); |
| stream >> u64; |
| ASSERT_TRUE(stream.eof()); |
| EXPECT_EQ(0U, u64); |
| } |
| |
| TEST_F(BinaryStreamU8Test, ReadString) { |
| string s(""); |
| ASSERT_FALSE(stream.eof()); |
| stream >> s; |
| ASSERT_TRUE(stream.eof()); |
| EXPECT_EQ("", s); |
| } |
| |
| |
| TEST(BinaryStreamTest, InitWithData) { |
| const char *data = "abcd"; |
| binarystream stream(data); |
| uint8_t a, b, c, d; |
| stream >> a >> b >> c >> d; |
| ASSERT_FALSE(stream.eof()); |
| EXPECT_EQ('a', a); |
| EXPECT_EQ('b', b); |
| EXPECT_EQ('c', c); |
| EXPECT_EQ('d', d); |
| } |
| |
| TEST(BinaryStreamTest, InitWithDataLeadingNull) { |
| const char *data = "\0abcd"; |
| binarystream stream(data, 5); |
| uint8_t z, a, b, c, d; |
| stream >> z >> a >> b >> c >> d; |
| ASSERT_FALSE(stream.eof()); |
| EXPECT_EQ(0U, z); |
| EXPECT_EQ('a', a); |
| EXPECT_EQ('b', b); |
| EXPECT_EQ('c', c); |
| EXPECT_EQ('d', d); |
| } |
| |
| TEST(BinaryStreamTest, InitWithDataVector) { |
| vector<char> data; |
| data.push_back('a'); |
| data.push_back('b'); |
| data.push_back('c'); |
| data.push_back('d'); |
| data.push_back('e'); |
| data.resize(4); |
| binarystream stream(&data[0], data.size()); |
| uint8_t a, b, c, d; |
| stream >> a >> b >> c >> d; |
| ASSERT_FALSE(stream.eof()); |
| EXPECT_EQ('a', a); |
| EXPECT_EQ('b', b); |
| EXPECT_EQ('c', c); |
| EXPECT_EQ('d', d); |
| } |
| |
| } // namespace |
| |
| int main(int argc, char *argv[]) { |
| ::testing::InitGoogleTest(&argc, argv); |
| return RUN_ALL_TESTS(); |
| } |