blob: 6e9d41df0b49eda371d7113ed3fba22d252b10eb [file] [log] [blame]
[email protected]ce208f872012-03-07 20:42:561// Copyright (c) 2012 The Chromium Authors. All rights reserved.
license.botbf09a502008-08-24 00:55:552// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
[email protected]d26cd5272008-08-07 13:40:164
5#include <string>
initial.commitd7cae122008-07-26 21:49:386
7#include "base/basictypes.h"
[email protected]3b63f8f42011-03-28 01:54:158#include "base/memory/scoped_ptr.h"
initial.commitd7cae122008-07-26 21:49:389#include "base/pickle.h"
[email protected]d529cb02013-06-10 19:06:5710#include "base/strings/string16.h"
initial.commitd7cae122008-07-26 21:49:3811#include "testing/gtest/include/gtest/gtest.h"
12
13namespace {
14
15const int testint = 2093847192;
[email protected]e64ff5e2009-07-28 21:00:0316const std::string teststr("Hello world"); // note non-aligned string length
initial.commitd7cae122008-07-26 21:49:3817const std::wstring testwstr(L"Hello, world");
18const char testdata[] = "AAA\0BBB\0";
19const int testdatalen = arraysize(testdata) - 1;
20const bool testbool1 = false;
21const bool testbool2 = true;
[email protected]6d81b482011-02-22 19:47:1922const uint16 testuint16 = 32123;
[email protected]b1f61b032012-11-28 15:40:5823const float testfloat = 3.1415926935f;
initial.commitd7cae122008-07-26 21:49:3824
25// checks that the result
26void VerifyResult(const Pickle& pickle) {
[email protected]ce208f872012-03-07 20:42:5627 PickleIterator iter(pickle);
initial.commitd7cae122008-07-26 21:49:3828
29 int outint;
30 EXPECT_TRUE(pickle.ReadInt(&iter, &outint));
31 EXPECT_EQ(testint, outint);
32
33 std::string outstr;
34 EXPECT_TRUE(pickle.ReadString(&iter, &outstr));
35 EXPECT_EQ(teststr, outstr);
36
37 std::wstring outwstr;
38 EXPECT_TRUE(pickle.ReadWString(&iter, &outwstr));
39 EXPECT_EQ(testwstr, outwstr);
40
41 bool outbool;
42 EXPECT_TRUE(pickle.ReadBool(&iter, &outbool));
[email protected]18853692013-01-03 12:30:5243 EXPECT_FALSE(outbool);
initial.commitd7cae122008-07-26 21:49:3844 EXPECT_TRUE(pickle.ReadBool(&iter, &outbool));
[email protected]18853692013-01-03 12:30:5245 EXPECT_TRUE(outbool);
initial.commitd7cae122008-07-26 21:49:3846
[email protected]6d81b482011-02-22 19:47:1947 uint16 outuint16;
48 EXPECT_TRUE(pickle.ReadUInt16(&iter, &outuint16));
49 EXPECT_EQ(testuint16, outuint16);
50
[email protected]b1f61b032012-11-28 15:40:5851 float outfloat;
52 EXPECT_TRUE(pickle.ReadFloat(&iter, &outfloat));
53 EXPECT_EQ(testfloat, outfloat);
54
initial.commitd7cae122008-07-26 21:49:3855 const char* outdata;
56 int outdatalen;
57 EXPECT_TRUE(pickle.ReadData(&iter, &outdata, &outdatalen));
58 EXPECT_EQ(testdatalen, outdatalen);
59 EXPECT_EQ(memcmp(testdata, outdata, outdatalen), 0);
60
initial.commitd7cae122008-07-26 21:49:3861 // reads past the end should fail
62 EXPECT_FALSE(pickle.ReadInt(&iter, &outint));
63}
64
65} // namespace
66
67TEST(PickleTest, EncodeDecode) {
68 Pickle pickle;
69
70 EXPECT_TRUE(pickle.WriteInt(testint));
71 EXPECT_TRUE(pickle.WriteString(teststr));
72 EXPECT_TRUE(pickle.WriteWString(testwstr));
73 EXPECT_TRUE(pickle.WriteBool(testbool1));
74 EXPECT_TRUE(pickle.WriteBool(testbool2));
[email protected]6d81b482011-02-22 19:47:1975 EXPECT_TRUE(pickle.WriteUInt16(testuint16));
[email protected]b1f61b032012-11-28 15:40:5876 EXPECT_TRUE(pickle.WriteFloat(testfloat));
initial.commitd7cae122008-07-26 21:49:3877 EXPECT_TRUE(pickle.WriteData(testdata, testdatalen));
initial.commitd7cae122008-07-26 21:49:3878 VerifyResult(pickle);
79
80 // test copy constructor
81 Pickle pickle2(pickle);
82 VerifyResult(pickle2);
83
84 // test operator=
85 Pickle pickle3;
86 pickle3 = pickle;
87 VerifyResult(pickle3);
88}
89
[email protected]d87f8e6f2010-11-15 19:31:2390// Tests that we can handle really small buffers.
91TEST(PickleTest, SmallBuffer) {
[email protected]604eb052013-01-18 14:21:5892 scoped_ptr<char[]> buffer(new char[1]);
[email protected]d87f8e6f2010-11-15 19:31:2393
94 // We should not touch the buffer.
95 Pickle pickle(buffer.get(), 1);
96
[email protected]ce208f872012-03-07 20:42:5697 PickleIterator iter(pickle);
[email protected]d87f8e6f2010-11-15 19:31:2398 int data;
99 EXPECT_FALSE(pickle.ReadInt(&iter, &data));
100}
101
102// Tests that we can handle improper headers.
103TEST(PickleTest, BigSize) {
104 int buffer[] = { 0x56035200, 25, 40, 50 };
105
106 Pickle pickle(reinterpret_cast<char*>(buffer), sizeof(buffer));
107
[email protected]ce208f872012-03-07 20:42:56108 PickleIterator iter(pickle);
[email protected]d87f8e6f2010-11-15 19:31:23109 int data;
110 EXPECT_FALSE(pickle.ReadInt(&iter, &data));
111}
112
113TEST(PickleTest, UnalignedSize) {
114 int buffer[] = { 10, 25, 40, 50 };
115
116 Pickle pickle(reinterpret_cast<char*>(buffer), sizeof(buffer));
117
[email protected]ce208f872012-03-07 20:42:56118 PickleIterator iter(pickle);
[email protected]d87f8e6f2010-11-15 19:31:23119 int data;
120 EXPECT_FALSE(pickle.ReadInt(&iter, &data));
121}
122
initial.commitd7cae122008-07-26 21:49:38123TEST(PickleTest, ZeroLenStr) {
124 Pickle pickle;
[email protected]007b3f82013-04-09 08:46:45125 EXPECT_TRUE(pickle.WriteString(std::string()));
initial.commitd7cae122008-07-26 21:49:38126
[email protected]ce208f872012-03-07 20:42:56127 PickleIterator iter(pickle);
initial.commitd7cae122008-07-26 21:49:38128 std::string outstr;
129 EXPECT_TRUE(pickle.ReadString(&iter, &outstr));
130 EXPECT_EQ("", outstr);
131}
132
133TEST(PickleTest, ZeroLenWStr) {
134 Pickle pickle;
[email protected]007b3f82013-04-09 08:46:45135 EXPECT_TRUE(pickle.WriteWString(std::wstring()));
initial.commitd7cae122008-07-26 21:49:38136
[email protected]ce208f872012-03-07 20:42:56137 PickleIterator iter(pickle);
initial.commitd7cae122008-07-26 21:49:38138 std::string outstr;
139 EXPECT_TRUE(pickle.ReadString(&iter, &outstr));
140 EXPECT_EQ("", outstr);
141}
142
143TEST(PickleTest, BadLenStr) {
144 Pickle pickle;
145 EXPECT_TRUE(pickle.WriteInt(-2));
146
[email protected]ce208f872012-03-07 20:42:56147 PickleIterator iter(pickle);
initial.commitd7cae122008-07-26 21:49:38148 std::string outstr;
149 EXPECT_FALSE(pickle.ReadString(&iter, &outstr));
150}
151
152TEST(PickleTest, BadLenWStr) {
153 Pickle pickle;
154 EXPECT_TRUE(pickle.WriteInt(-1));
155
[email protected]ce208f872012-03-07 20:42:56156 PickleIterator iter(pickle);
initial.commitd7cae122008-07-26 21:49:38157 std::wstring woutstr;
158 EXPECT_FALSE(pickle.ReadWString(&iter, &woutstr));
159}
160
161TEST(PickleTest, FindNext) {
162 Pickle pickle;
163 EXPECT_TRUE(pickle.WriteInt(1));
164 EXPECT_TRUE(pickle.WriteString("Domo"));
165
166 const char* start = reinterpret_cast<const char*>(pickle.data());
167 const char* end = start + pickle.size();
168
169 EXPECT_TRUE(end == Pickle::FindNext(pickle.header_size_, start, end));
170 EXPECT_TRUE(NULL == Pickle::FindNext(pickle.header_size_, start, end - 1));
171 EXPECT_TRUE(end == Pickle::FindNext(pickle.header_size_, start, end + 1));
172}
173
[email protected]137d2372011-01-26 13:02:27174TEST(PickleTest, FindNextWithIncompleteHeader) {
175 size_t header_size = sizeof(Pickle::Header);
[email protected]604eb052013-01-18 14:21:58176 scoped_ptr<char[]> buffer(new char[header_size - 1]);
[email protected]137d2372011-01-26 13:02:27177 memset(buffer.get(), 0x1, header_size - 1);
178
179 const char* start = buffer.get();
180 const char* end = start + header_size - 1;
181
182 EXPECT_TRUE(NULL == Pickle::FindNext(header_size, start, end));
183}
184
[email protected]9dbfa9842013-11-01 09:43:45185#if defined(COMPILER_MSVC)
186#pragma warning(push)
187#pragma warning(disable: 4146)
188#endif
[email protected]33a38dd2013-11-01 09:06:26189TEST(PickleTest, FindNextOverflow) {
190 size_t header_size = sizeof(Pickle::Header);
191 size_t header_size2 = 2 * header_size;
192 size_t payload_received = 100;
193 scoped_ptr<char[]> buffer(new char[header_size2 + payload_received]);
194 const char* start = buffer.get();
195 Pickle::Header* header = reinterpret_cast<Pickle::Header*>(buffer.get());
196 const char* end = start + header_size2 + payload_received;
197 // It is impossible to construct an overflow test otherwise.
198 if (sizeof(size_t) > sizeof(header->payload_size) ||
199 sizeof(uintptr_t) > sizeof(header->payload_size))
200 return;
201
202 header->payload_size = -(reinterpret_cast<uintptr_t>(start) + header_size2);
203 EXPECT_TRUE(NULL == Pickle::FindNext(header_size2, start, end));
204
205 header->payload_size = -header_size2;
206 EXPECT_TRUE(NULL == Pickle::FindNext(header_size2, start, end));
207
208 header->payload_size = 0;
209 end = start + header_size;
210 EXPECT_TRUE(NULL == Pickle::FindNext(header_size2, start, end));
211}
[email protected]9dbfa9842013-11-01 09:43:45212#if defined(COMPILER_MSVC)
213#pragma warning(pop)
214#endif
[email protected]33a38dd2013-11-01 09:06:26215
[email protected]ce208f872012-03-07 20:42:56216TEST(PickleTest, GetReadPointerAndAdvance) {
initial.commitd7cae122008-07-26 21:49:38217 Pickle pickle;
[email protected]ce208f872012-03-07 20:42:56218
219 PickleIterator iter(pickle);
220 EXPECT_FALSE(iter.GetReadPointerAndAdvance(1));
221
initial.commitd7cae122008-07-26 21:49:38222 EXPECT_TRUE(pickle.WriteInt(1));
223 EXPECT_TRUE(pickle.WriteInt(2));
[email protected]ce208f872012-03-07 20:42:56224 int bytes = sizeof(int) * 2;
initial.commitd7cae122008-07-26 21:49:38225
[email protected]ce208f872012-03-07 20:42:56226 EXPECT_TRUE(PickleIterator(pickle).GetReadPointerAndAdvance(0));
227 EXPECT_TRUE(PickleIterator(pickle).GetReadPointerAndAdvance(1));
228 EXPECT_FALSE(PickleIterator(pickle).GetReadPointerAndAdvance(-1));
229 EXPECT_TRUE(PickleIterator(pickle).GetReadPointerAndAdvance(bytes));
230 EXPECT_FALSE(PickleIterator(pickle).GetReadPointerAndAdvance(bytes + 1));
231 EXPECT_FALSE(PickleIterator(pickle).GetReadPointerAndAdvance(INT_MAX));
232 EXPECT_FALSE(PickleIterator(pickle).GetReadPointerAndAdvance(INT_MIN));
initial.commitd7cae122008-07-26 21:49:38233}
234
235TEST(PickleTest, Resize) {
[email protected]44a1cbfa2008-08-15 01:05:11236 size_t unit = Pickle::kPayloadUnit;
[email protected]604eb052013-01-18 14:21:58237 scoped_ptr<char[]> data(new char[unit]);
initial.commitd7cae122008-07-26 21:49:38238 char* data_ptr = data.get();
[email protected]44a1cbfa2008-08-15 01:05:11239 for (size_t i = 0; i < unit; i++)
initial.commitd7cae122008-07-26 21:49:38240 data_ptr[i] = 'G';
241
242 // construct a message that will be exactly the size of one payload unit,
243 // note that any data will have a 4-byte header indicating the size
[email protected]44a1cbfa2008-08-15 01:05:11244 const size_t payload_size_after_header = unit - sizeof(uint32);
initial.commitd7cae122008-07-26 21:49:38245 Pickle pickle;
[email protected]1ac79482008-08-15 01:40:41246 pickle.WriteData(data_ptr,
247 static_cast<int>(payload_size_after_header - sizeof(uint32)));
[email protected]44a1cbfa2008-08-15 01:05:11248 size_t cur_payload = payload_size_after_header;
initial.commitd7cae122008-07-26 21:49:38249
[email protected]2de46262009-03-16 20:21:57250 // note: we assume 'unit' is a power of 2
[email protected]d1b319fc2013-10-31 04:03:02251 EXPECT_EQ(unit, pickle.capacity_after_header());
initial.commitd7cae122008-07-26 21:49:38252 EXPECT_EQ(pickle.payload_size(), payload_size_after_header);
253
254 // fill out a full page (noting data header)
[email protected]1ac79482008-08-15 01:40:41255 pickle.WriteData(data_ptr, static_cast<int>(unit - sizeof(uint32)));
initial.commitd7cae122008-07-26 21:49:38256 cur_payload += unit;
[email protected]d1b319fc2013-10-31 04:03:02257 EXPECT_EQ(unit * 2, pickle.capacity_after_header());
initial.commitd7cae122008-07-26 21:49:38258 EXPECT_EQ(cur_payload, pickle.payload_size());
259
[email protected]2de46262009-03-16 20:21:57260 // one more byte should double the capacity
initial.commitd7cae122008-07-26 21:49:38261 pickle.WriteData(data_ptr, 1);
262 cur_payload += 5;
[email protected]d1b319fc2013-10-31 04:03:02263 EXPECT_EQ(unit * 4, pickle.capacity_after_header());
initial.commitd7cae122008-07-26 21:49:38264 EXPECT_EQ(cur_payload, pickle.payload_size());
265}
266
[email protected]d26cd5272008-08-07 13:40:16267namespace {
initial.commitd7cae122008-07-26 21:49:38268
[email protected]d26cd5272008-08-07 13:40:16269struct CustomHeader : Pickle::Header {
270 int blah;
271};
272
273} // namespace
274
275TEST(PickleTest, HeaderPadding) {
initial.commitd7cae122008-07-26 21:49:38276 const uint32 kMagic = 0x12345678;
277
278 Pickle pickle(sizeof(CustomHeader));
279 pickle.WriteInt(kMagic);
280
281 // this should not overwrite the 'int' payload
282 pickle.headerT<CustomHeader>()->blah = 10;
283
[email protected]ce208f872012-03-07 20:42:56284 PickleIterator iter(pickle);
initial.commitd7cae122008-07-26 21:49:38285 int result;
286 ASSERT_TRUE(pickle.ReadInt(&iter, &result));
287
[email protected]44a1cbfa2008-08-15 01:05:11288 EXPECT_EQ(static_cast<uint32>(result), kMagic);
initial.commitd7cae122008-07-26 21:49:38289}
290
291TEST(PickleTest, EqualsOperator) {
292 Pickle source;
293 source.WriteInt(1);
294
295 Pickle copy_refs_source_buffer(static_cast<const char*>(source.data()),
296 source.size());
297 Pickle copy;
298 copy = copy_refs_source_buffer;
299 ASSERT_EQ(source.size(), copy.size());
[email protected]d26cd5272008-08-07 13:40:16300}
[email protected]87665562009-06-25 16:54:02301
302TEST(PickleTest, EvilLengths) {
303 Pickle source;
[email protected]b235357d2009-06-25 17:23:49304 std::string str(100000, 'A');
[email protected]ce208f872012-03-07 20:42:56305 EXPECT_TRUE(source.WriteData(str.c_str(), 100000));
[email protected]87665562009-06-25 16:54:02306 // ReadString16 used to have its read buffer length calculation wrong leading
307 // to out-of-bounds reading.
[email protected]ce208f872012-03-07 20:42:56308 PickleIterator iter(source);
[email protected]87665562009-06-25 16:54:02309 string16 str16;
310 EXPECT_FALSE(source.ReadString16(&iter, &str16));
311
312 // And check we didn't break ReadString16.
313 str16 = (wchar_t) 'A';
314 Pickle str16_pickle;
[email protected]ce208f872012-03-07 20:42:56315 EXPECT_TRUE(str16_pickle.WriteString16(str16));
316 iter = PickleIterator(str16_pickle);
[email protected]87665562009-06-25 16:54:02317 EXPECT_TRUE(str16_pickle.ReadString16(&iter, &str16));
318 EXPECT_EQ(1U, str16.length());
319
[email protected]ce208f872012-03-07 20:42:56320 // Check we don't fail in a length check with invalid String16 size.
321 // (1<<31) * sizeof(char16) == 0, so this is particularly evil.
322 Pickle bad_len;
323 EXPECT_TRUE(bad_len.WriteInt(1 << 31));
324 iter = PickleIterator(bad_len);
325 EXPECT_FALSE(bad_len.ReadString16(&iter, &str16));
326
[email protected]87665562009-06-25 16:54:02327 // Check we don't fail in a length check with large WStrings.
328 Pickle big_len;
[email protected]ce208f872012-03-07 20:42:56329 EXPECT_TRUE(big_len.WriteInt(1 << 30));
330 iter = PickleIterator(big_len);
[email protected]87665562009-06-25 16:54:02331 std::wstring wstr;
332 EXPECT_FALSE(big_len.ReadWString(&iter, &wstr));
333}
334
[email protected]e64ff5e2009-07-28 21:00:03335// Check we can write zero bytes of data and 'data' can be NULL.
336TEST(PickleTest, ZeroLength) {
337 Pickle pickle;
338 EXPECT_TRUE(pickle.WriteData(NULL, 0));
339
[email protected]ce208f872012-03-07 20:42:56340 PickleIterator iter(pickle);
[email protected]e64ff5e2009-07-28 21:00:03341 const char* outdata;
342 int outdatalen;
343 EXPECT_TRUE(pickle.ReadData(&iter, &outdata, &outdatalen));
344 EXPECT_EQ(0, outdatalen);
345 // We can't assert that outdata is NULL.
346}
347
[email protected]26d2f472010-03-30 23:52:24348// Check that ReadBytes works properly with an iterator initialized to NULL.
349TEST(PickleTest, ReadBytes) {
350 Pickle pickle;
351 int data = 0x7abcd;
352 EXPECT_TRUE(pickle.WriteBytes(&data, sizeof(data)));
353
[email protected]ce208f872012-03-07 20:42:56354 PickleIterator iter(pickle);
355 const char* outdata_char = NULL;
[email protected]26d2f472010-03-30 23:52:24356 EXPECT_TRUE(pickle.ReadBytes(&iter, &outdata_char, sizeof(data)));
357
358 int outdata;
359 memcpy(&outdata, outdata_char, sizeof(outdata));
360 EXPECT_EQ(data, outdata);
361}