1 // Copyright (c) 2006-2008 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
7 #include "base/basictypes.h"
8 #include "base/pickle.h"
9 #include "base/scoped_ptr.h"
10 #include "base/string16.h"
11 #include "testing/gtest/include/gtest/gtest.h"
15 const int testint
= 2093847192;
16 const std::string
teststr("Hello world"); // note non-aligned string length
17 const std::wstring
testwstr(L
"Hello, world");
18 const char testdata
[] = "AAA\0BBB\0";
19 const int testdatalen
= arraysize(testdata
) - 1;
20 const bool testbool1
= false;
21 const bool testbool2
= true;
23 // checks that the result
24 void VerifyResult(const Pickle
& pickle
) {
28 EXPECT_TRUE(pickle
.ReadInt(&iter
, &outint
));
29 EXPECT_EQ(testint
, outint
);
32 EXPECT_TRUE(pickle
.ReadString(&iter
, &outstr
));
33 EXPECT_EQ(teststr
, outstr
);
36 EXPECT_TRUE(pickle
.ReadWString(&iter
, &outwstr
));
37 EXPECT_EQ(testwstr
, outwstr
);
40 EXPECT_TRUE(pickle
.ReadBool(&iter
, &outbool
));
41 EXPECT_EQ(testbool1
, outbool
);
42 EXPECT_TRUE(pickle
.ReadBool(&iter
, &outbool
));
43 EXPECT_EQ(testbool2
, outbool
);
47 EXPECT_TRUE(pickle
.ReadData(&iter
, &outdata
, &outdatalen
));
48 EXPECT_EQ(testdatalen
, outdatalen
);
49 EXPECT_EQ(memcmp(testdata
, outdata
, outdatalen
), 0);
51 EXPECT_TRUE(pickle
.ReadData(&iter
, &outdata
, &outdatalen
));
52 EXPECT_EQ(testdatalen
, outdatalen
);
53 EXPECT_EQ(memcmp(testdata
, outdata
, outdatalen
), 0);
55 // reads past the end should fail
56 EXPECT_FALSE(pickle
.ReadInt(&iter
, &outint
));
61 TEST(PickleTest
, EncodeDecode
) {
64 EXPECT_TRUE(pickle
.WriteInt(testint
));
65 EXPECT_TRUE(pickle
.WriteString(teststr
));
66 EXPECT_TRUE(pickle
.WriteWString(testwstr
));
67 EXPECT_TRUE(pickle
.WriteBool(testbool1
));
68 EXPECT_TRUE(pickle
.WriteBool(testbool2
));
69 EXPECT_TRUE(pickle
.WriteData(testdata
, testdatalen
));
71 // Over allocate BeginWriteData so we can test TrimWriteData.
72 char* dest
= pickle
.BeginWriteData(testdatalen
+ 100);
74 memcpy(dest
, testdata
, testdatalen
);
76 pickle
.TrimWriteData(testdatalen
);
80 // test copy constructor
81 Pickle
pickle2(pickle
);
82 VerifyResult(pickle2
);
87 VerifyResult(pickle3
);
90 TEST(PickleTest
, ZeroLenStr
) {
92 EXPECT_TRUE(pickle
.WriteString(""));
96 EXPECT_TRUE(pickle
.ReadString(&iter
, &outstr
));
97 EXPECT_EQ("", outstr
);
100 TEST(PickleTest
, ZeroLenWStr
) {
102 EXPECT_TRUE(pickle
.WriteWString(L
""));
106 EXPECT_TRUE(pickle
.ReadString(&iter
, &outstr
));
107 EXPECT_EQ("", outstr
);
110 TEST(PickleTest
, BadLenStr
) {
112 EXPECT_TRUE(pickle
.WriteInt(-2));
116 EXPECT_FALSE(pickle
.ReadString(&iter
, &outstr
));
119 TEST(PickleTest
, BadLenWStr
) {
121 EXPECT_TRUE(pickle
.WriteInt(-1));
124 std::wstring woutstr
;
125 EXPECT_FALSE(pickle
.ReadWString(&iter
, &woutstr
));
128 TEST(PickleTest
, FindNext
) {
130 EXPECT_TRUE(pickle
.WriteInt(1));
131 EXPECT_TRUE(pickle
.WriteString("Domo"));
133 const char* start
= reinterpret_cast<const char*>(pickle
.data());
134 const char* end
= start
+ pickle
.size();
136 EXPECT_TRUE(end
== Pickle::FindNext(pickle
.header_size_
, start
, end
));
137 EXPECT_TRUE(NULL
== Pickle::FindNext(pickle
.header_size_
, start
, end
- 1));
138 EXPECT_TRUE(end
== Pickle::FindNext(pickle
.header_size_
, start
, end
+ 1));
141 TEST(PickleTest
, IteratorHasRoom
) {
143 EXPECT_TRUE(pickle
.WriteInt(1));
144 EXPECT_TRUE(pickle
.WriteInt(2));
146 const void* iter
= 0;
147 EXPECT_FALSE(pickle
.IteratorHasRoomFor(iter
, 1));
148 iter
= pickle
.payload();
149 EXPECT_TRUE(pickle
.IteratorHasRoomFor(iter
, 0));
150 EXPECT_TRUE(pickle
.IteratorHasRoomFor(iter
, 1));
151 EXPECT_FALSE(pickle
.IteratorHasRoomFor(iter
, -1));
152 EXPECT_TRUE(pickle
.IteratorHasRoomFor(iter
, sizeof(int) * 2));
153 EXPECT_FALSE(pickle
.IteratorHasRoomFor(iter
, (sizeof(int) * 2) + 1));
156 TEST(PickleTest
, Resize
) {
157 size_t unit
= Pickle::kPayloadUnit
;
158 scoped_array
<char> data(new char[unit
]);
159 char* data_ptr
= data
.get();
160 for (size_t i
= 0; i
< unit
; i
++)
163 // construct a message that will be exactly the size of one payload unit,
164 // note that any data will have a 4-byte header indicating the size
165 const size_t payload_size_after_header
= unit
- sizeof(uint32
);
167 pickle
.WriteData(data_ptr
,
168 static_cast<int>(payload_size_after_header
- sizeof(uint32
)));
169 size_t cur_payload
= payload_size_after_header
;
171 // note: we assume 'unit' is a power of 2
172 EXPECT_EQ(unit
, pickle
.capacity());
173 EXPECT_EQ(pickle
.payload_size(), payload_size_after_header
);
175 // fill out a full page (noting data header)
176 pickle
.WriteData(data_ptr
, static_cast<int>(unit
- sizeof(uint32
)));
178 EXPECT_EQ(unit
* 2, pickle
.capacity());
179 EXPECT_EQ(cur_payload
, pickle
.payload_size());
181 // one more byte should double the capacity
182 pickle
.WriteData(data_ptr
, 1);
184 EXPECT_EQ(unit
* 4, pickle
.capacity());
185 EXPECT_EQ(cur_payload
, pickle
.payload_size());
190 struct CustomHeader
: Pickle::Header
{
196 TEST(PickleTest
, HeaderPadding
) {
197 const uint32 kMagic
= 0x12345678;
199 Pickle
pickle(sizeof(CustomHeader
));
200 pickle
.WriteInt(kMagic
);
202 // this should not overwrite the 'int' payload
203 pickle
.headerT
<CustomHeader
>()->blah
= 10;
207 ASSERT_TRUE(pickle
.ReadInt(&iter
, &result
));
209 EXPECT_EQ(static_cast<uint32
>(result
), kMagic
);
212 TEST(PickleTest
, EqualsOperator
) {
216 Pickle
copy_refs_source_buffer(static_cast<const char*>(source
.data()),
219 copy
= copy_refs_source_buffer
;
220 ASSERT_EQ(source
.size(), copy
.size());
223 TEST(PickleTest
, EvilLengths
) {
225 std::string
str(100000, 'A');
226 source
.WriteData(str
.c_str(), 100000);
227 // ReadString16 used to have its read buffer length calculation wrong leading
228 // to out-of-bounds reading.
231 EXPECT_FALSE(source
.ReadString16(&iter
, &str16
));
233 // And check we didn't break ReadString16.
234 str16
= (wchar_t) 'A';
236 str16_pickle
.WriteString16(str16
);
238 EXPECT_TRUE(str16_pickle
.ReadString16(&iter
, &str16
));
239 EXPECT_EQ(1U, str16
.length());
241 // Check we don't fail in a length check with large WStrings.
243 big_len
.WriteInt(1 << 30);
246 EXPECT_FALSE(big_len
.ReadWString(&iter
, &wstr
));
249 // Check we can write zero bytes of data and 'data' can be NULL.
250 TEST(PickleTest
, ZeroLength
) {
252 EXPECT_TRUE(pickle
.WriteData(NULL
, 0));
257 EXPECT_TRUE(pickle
.ReadData(&iter
, &outdata
, &outdatalen
));
258 EXPECT_EQ(0, outdatalen
);
259 // We can't assert that outdata is NULL.