1 // Copyright 2013 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.
5 #include "net/quic/iovector.h"
9 #include "base/logging.h"
10 #include "testing/gtest/include/gtest/gtest.h"
18 const char* const test_data
[] = {
19 "test string 1, a medium size one.",
21 "test string 3, a looooooooooooong loooooooooooooooong string"
24 TEST(IOVectorTest
, CopyConstructor
) {
26 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_data
); ++i
) {
27 iov1
.Append(const_cast<char*>(test_data
[i
]), strlen(test_data
[i
]));
30 EXPECT_EQ(iov2
.Size(), iov1
.Size());
31 for (size_t i
= 0; i
< iov2
.Size(); ++i
) {
32 EXPECT_TRUE(iov2
.iovec()[i
].iov_base
== iov1
.iovec()[i
].iov_base
);
33 EXPECT_EQ(iov2
.iovec()[i
].iov_len
, iov1
.iovec()[i
].iov_len
);
35 EXPECT_EQ(iov2
.TotalBufferSize(), iov1
.TotalBufferSize());
38 TEST(IOVectorTest
, AssignmentOperator
) {
40 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_data
); ++i
) {
41 iov1
.Append(const_cast<char*>(test_data
[i
]), strlen(test_data
[i
]));
44 iov2
.Append(const_cast<char*>("ephemeral string"), 16);
45 // The following assignment results in a shallow copy;
46 // both IOVectors point to the same underlying data.
48 EXPECT_EQ(iov2
.Size(), iov1
.Size());
49 for (size_t i
= 0; i
< iov2
.Size(); ++i
) {
50 EXPECT_TRUE(iov2
.iovec()[i
].iov_base
== iov1
.iovec()[i
].iov_base
);
51 EXPECT_EQ(iov2
.iovec()[i
].iov_len
, iov1
.iovec()[i
].iov_len
);
53 EXPECT_EQ(iov2
.TotalBufferSize(), iov1
.TotalBufferSize());
56 TEST(IOVectorTest
, Append
) {
59 const struct iovec
* iov2
= iov
.iovec();
61 ASSERT_EQ(0u, iov
.Size());
62 ASSERT_TRUE(iov2
== NULL
);
63 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_data
); ++i
) {
64 const int str_len
= strlen(test_data
[i
]);
65 const int append_len
= str_len
/ 2;
66 // This should append a new block
67 iov
.Append(const_cast<char*>(test_data
[i
]), append_len
);
69 ASSERT_EQ(i
+ 1, static_cast<size_t>(iov
.Size()));
70 ASSERT_TRUE(iov
.LastBlockEnd() == test_data
[i
] + append_len
);
71 // This should just lengthen the existing block.
72 iov
.Append(const_cast<char*>(test_data
[i
] + append_len
),
73 str_len
- append_len
);
74 length
+= (str_len
- append_len
);
75 ASSERT_EQ(i
+ 1, static_cast<size_t>(iov
.Size()));
76 ASSERT_TRUE(iov
.LastBlockEnd() == test_data
[i
] + str_len
);
80 ASSERT_TRUE(iov2
!= NULL
);
81 for (size_t i
= 0; i
< iov
.Size(); ++i
) {
82 ASSERT_TRUE(test_data
[i
] == iov2
[i
].iov_base
);
83 ASSERT_EQ(strlen(test_data
[i
]), iov2
[i
].iov_len
);
87 TEST(IOVectorTest
, AppendIovec
) {
89 const struct iovec test_iov
[] = {
90 {const_cast<char*>("foo"), 3},
91 {const_cast<char*>("bar"), 3},
92 {const_cast<char*>("buzzzz"), 6}
94 iov
.AppendIovec(test_iov
, ARRAYSIZE_UNSAFE(test_iov
));
95 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_iov
); ++i
) {
96 EXPECT_EQ(test_iov
[i
].iov_base
, iov
.iovec()[i
].iov_base
);
97 EXPECT_EQ(test_iov
[i
].iov_len
, iov
.iovec()[i
].iov_len
);
100 // Test AppendIovecAtMostBytes.
102 // Stop in the middle of a block.
103 EXPECT_EQ(5u, iov
.AppendIovecAtMostBytes(test_iov
, ARRAYSIZE_UNSAFE(test_iov
),
105 EXPECT_EQ(5u, iov
.TotalBufferSize());
106 iov
.Append(static_cast<char*>(test_iov
[1].iov_base
) + 2, 1);
107 // Make sure the boundary case, where max_bytes == size of block also works.
108 EXPECT_EQ(6u, iov
.AppendIovecAtMostBytes(&test_iov
[2], 1, 6));
109 ASSERT_LE(ARRAYSIZE_UNSAFE(test_iov
), static_cast<size_t>(iov
.Size()));
110 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_iov
); ++i
) {
111 EXPECT_EQ(test_iov
[i
].iov_base
, iov
.iovec()[i
].iov_base
);
112 EXPECT_EQ(test_iov
[i
].iov_len
, iov
.iovec()[i
].iov_len
);
116 TEST(IOVectorTest
, ConsumeHalfBlocks
) {
120 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_data
); ++i
) {
121 const int str_len
= strlen(test_data
[i
]);
122 iov
.Append(const_cast<char*>(test_data
[i
]), str_len
);
125 const char* endp
= iov
.LastBlockEnd();
126 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_data
); ++i
) {
127 const struct iovec
* iov2
= iov
.iovec();
128 const size_t str_len
= strlen(test_data
[i
]);
129 size_t tmp
= str_len
/ 2;
131 ASSERT_TRUE(iov2
!= NULL
);
132 ASSERT_TRUE(iov2
[0].iov_base
== test_data
[i
]);
133 ASSERT_EQ(str_len
, iov2
[0].iov_len
);
135 // Consume half of the first block.
136 size_t consumed
= iov
.Consume(tmp
);
137 ASSERT_EQ(tmp
, consumed
);
138 ASSERT_EQ(ARRAYSIZE_UNSAFE(test_data
) - i
, static_cast<size_t>(iov
.Size()));
140 ASSERT_TRUE(iov2
!= NULL
);
141 ASSERT_TRUE(iov2
[0].iov_base
== test_data
[i
] + tmp
);
142 ASSERT_EQ(iov2
[0].iov_len
, str_len
- tmp
);
144 // Consume the rest of the first block
145 consumed
= iov
.Consume(str_len
- tmp
);
146 ASSERT_EQ(str_len
- tmp
, consumed
);
147 ASSERT_EQ(ARRAYSIZE_UNSAFE(test_data
) - i
- 1,
148 static_cast<size_t>(iov
.Size()));
150 if (iov
.Size() > 0) {
151 ASSERT_TRUE(iov2
!= NULL
);
152 ASSERT_TRUE(iov
.LastBlockEnd() == endp
);
154 ASSERT_TRUE(iov2
== NULL
);
155 ASSERT_TRUE(iov
.LastBlockEnd() == NULL
);
160 TEST(IOVectorTest
, ConsumeTwoAndHalfBlocks
) {
164 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_data
); ++i
) {
165 const int str_len
= strlen(test_data
[i
]);
166 iov
.Append(const_cast<char*>(test_data
[i
]), str_len
);
169 const size_t last_len
= strlen(test_data
[ARRAYSIZE_UNSAFE(test_data
) - 1]);
170 const size_t half_len
= last_len
/ 2;
172 const char* endp
= iov
.LastBlockEnd();
173 size_t consumed
= iov
.Consume(length
- half_len
);
174 ASSERT_EQ(length
- half_len
, consumed
);
175 const struct iovec
* iov2
= iov
.iovec();
176 ASSERT_TRUE(iov2
!= NULL
);
177 ASSERT_EQ(1u, iov
.Size());
178 ASSERT_TRUE(iov2
[0].iov_base
==
179 test_data
[ARRAYSIZE_UNSAFE(test_data
) - 1] + last_len
- half_len
);
180 ASSERT_EQ(half_len
, iov2
[0].iov_len
);
181 ASSERT_TRUE(iov
.LastBlockEnd() == endp
);
183 consumed
= iov
.Consume(half_len
);
184 ASSERT_EQ(half_len
, consumed
);
186 ASSERT_EQ(0u, iov
.Size());
187 ASSERT_TRUE(iov2
== NULL
);
188 ASSERT_TRUE(iov
.LastBlockEnd() == NULL
);
191 TEST(IOVectorTest
, ConsumeTooMuch
) {
195 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_data
); ++i
) {
196 const int str_len
= strlen(test_data
[i
]);
197 iov
.Append(const_cast<char*>(test_data
[i
]), str_len
);
202 consumed
= iov
.Consume(length
);
203 // TODO(rtenneti): enable when chromium supports EXPECT_DFATAL.
206 {consumed = iov.Consume(length + 1);},
207 "Attempting to consume 1 non-existent bytes.");
209 ASSERT_EQ(length
, consumed
);
210 const struct iovec
* iov2
= iov
.iovec();
211 ASSERT_EQ(0u, iov
.Size());
212 ASSERT_TRUE(iov2
== NULL
);
213 ASSERT_TRUE(iov
.LastBlockEnd() == NULL
);
216 TEST(IOVectorTest
, Clear
) {
220 for (size_t i
= 0; i
< ARRAYSIZE_UNSAFE(test_data
); ++i
) {
221 const int str_len
= strlen(test_data
[i
]);
222 iov
.Append(const_cast<char*>(test_data
[i
]), str_len
);
225 const struct iovec
* iov2
= iov
.iovec();
226 ASSERT_TRUE(iov2
!= NULL
);
227 ASSERT_EQ(ARRAYSIZE_UNSAFE(test_data
), static_cast<size_t>(iov
.Size()));
231 ASSERT_EQ(0u, iov
.Size());
232 ASSERT_TRUE(iov2
== NULL
);
235 TEST(IOVectorTest
, Capacity
) {
237 // Note: IOVector merges adjacent Appends() into a single iov.
238 // Therefore, if we expect final size of iov to be 3, we must insure
239 // that the items we are appending are not adjacent. To achieve that
240 // we use use an array (a[1] provides a buffer between a[0] and b[0],
241 // and makes them non-adjacent).
242 char a
[2], b
[2], c
[2];
243 iov
.Append(&a
[0], 1);
244 iov
.Append(&b
[0], 1);
245 iov
.Append(&c
[0], 1);
246 ASSERT_EQ(3u, iov
.Size());
247 size_t capacity
= iov
.Capacity();
248 EXPECT_LE(iov
.Size(), capacity
);
250 // The capacity should not have changed.
251 EXPECT_EQ(capacity
, iov
.Capacity());
254 TEST(IOVectorTest
, Swap
) {
256 // See IOVector merge comment above.
257 char a
[2], b
[2], c
[2], d
[2], e
[2];
258 iov1
.Append(&a
[0], 1);
259 iov1
.Append(&b
[0], 1);
261 iov2
.Append(&c
[0], 1);
262 iov2
.Append(&d
[0], 1);
263 iov2
.Append(&e
[0], 1);
266 ASSERT_EQ(3u, iov1
.Size());
267 EXPECT_EQ(&c
[0], iov1
.iovec()[0].iov_base
);
268 EXPECT_EQ(1u, iov1
.iovec()[0].iov_len
);
269 EXPECT_EQ(&d
[0], iov1
.iovec()[1].iov_base
);
270 EXPECT_EQ(1u, iov1
.iovec()[1].iov_len
);
271 EXPECT_EQ(&e
[0], iov1
.iovec()[2].iov_base
);
272 EXPECT_EQ(1u, iov1
.iovec()[2].iov_len
);
274 ASSERT_EQ(2u, iov2
.Size());
275 EXPECT_EQ(&a
[0], iov2
.iovec()[0].iov_base
);
276 EXPECT_EQ(1u, iov2
.iovec()[0].iov_len
);
277 EXPECT_EQ(&b
[0], iov2
.iovec()[1].iov_base
);
278 EXPECT_EQ(1u, iov2
.iovec()[1].iov_len
);