blob: 86eb722c2adb555104a1c818eabfc4cfd378a3d2 [file] [log] [blame]
henrike@webrtc.orgf0488722014-05-13 18:00:26 +00001/*
2 * Copyright 2004 The WebRTC Project Authors. All rights reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#include "webrtc/base/gunit.h"
12#include "webrtc/base/stream.h"
kjellander@webrtc.org34ac7762014-09-19 13:47:47 +000013#include "webrtc/test/testsupport/gtest_disable.h"
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000014
15namespace rtc {
16
17///////////////////////////////////////////////////////////////////////////////
18// TestStream
19///////////////////////////////////////////////////////////////////////////////
20
21class TestStream : public StreamInterface {
22 public:
23 TestStream() : pos_(0) { }
24
25 virtual StreamState GetState() const { return SS_OPEN; }
26 virtual StreamResult Read(void* buffer, size_t buffer_len,
27 size_t* read, int* error) {
28 unsigned char* uc_buffer = static_cast<unsigned char*>(buffer);
29 for (size_t i = 0; i < buffer_len; ++i) {
30 uc_buffer[i] = static_cast<unsigned char>(pos_++);
31 }
32 if (read)
33 *read = buffer_len;
34 return SR_SUCCESS;
35 }
36 virtual StreamResult Write(const void* data, size_t data_len,
37 size_t* written, int* error) {
38 if (error)
39 *error = -1;
40 return SR_ERROR;
41 }
42 virtual void Close() { }
43 virtual bool SetPosition(size_t position) {
44 pos_ = position;
45 return true;
46 }
47 virtual bool GetPosition(size_t* position) const {
48 if (position) *position = pos_;
49 return true;
50 }
51 virtual bool GetSize(size_t* size) const {
52 return false;
53 }
54 virtual bool GetAvailable(size_t* size) const {
55 return false;
56 }
57
58 private:
59 size_t pos_;
60};
61
62bool VerifyTestBuffer(unsigned char* buffer, size_t len,
63 unsigned char value) {
64 bool passed = true;
65 for (size_t i = 0; i < len; ++i) {
66 if (buffer[i] != value++) {
67 passed = false;
68 break;
69 }
70 }
71 // Ensure that we don't pass again without re-writing
72 memset(buffer, 0, len);
73 return passed;
74}
75
76void SeekTest(StreamInterface* stream, const unsigned char value) {
77 size_t bytes;
78 unsigned char buffer[13] = { 0 };
79 const size_t kBufSize = sizeof(buffer);
80
81 EXPECT_EQ(stream->Read(buffer, kBufSize, &bytes, NULL), SR_SUCCESS);
82 EXPECT_EQ(bytes, kBufSize);
83 EXPECT_TRUE(VerifyTestBuffer(buffer, kBufSize, value));
84 EXPECT_TRUE(stream->GetPosition(&bytes));
85 EXPECT_EQ(13U, bytes);
86
87 EXPECT_TRUE(stream->SetPosition(7));
88
89 EXPECT_EQ(stream->Read(buffer, kBufSize, &bytes, NULL), SR_SUCCESS);
90 EXPECT_EQ(bytes, kBufSize);
91 EXPECT_TRUE(VerifyTestBuffer(buffer, kBufSize, value + 7));
92 EXPECT_TRUE(stream->GetPosition(&bytes));
93 EXPECT_EQ(20U, bytes);
94}
95
henrike@webrtc.orgf0488722014-05-13 18:00:26 +000096TEST(FifoBufferTest, TestAll) {
97 const size_t kSize = 16;
98 const char in[kSize * 2 + 1] = "0123456789ABCDEFGHIJKLMNOPQRSTUV";
99 char out[kSize * 2];
100 void* p;
101 const void* q;
102 size_t bytes;
103 FifoBuffer buf(kSize);
104 StreamInterface* stream = &buf;
105
106 // Test assumptions about base state
107 EXPECT_EQ(SS_OPEN, stream->GetState());
108 EXPECT_EQ(SR_BLOCK, stream->Read(out, kSize, &bytes, NULL));
109 EXPECT_TRUE(NULL != stream->GetReadData(&bytes));
110 EXPECT_EQ((size_t)0, bytes);
111 stream->ConsumeReadData(0);
112 EXPECT_TRUE(NULL != stream->GetWriteBuffer(&bytes));
113 EXPECT_EQ(kSize, bytes);
114 stream->ConsumeWriteBuffer(0);
115
116 // Try a full write
117 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize, &bytes, NULL));
118 EXPECT_EQ(kSize, bytes);
119
120 // Try a write that should block
121 EXPECT_EQ(SR_BLOCK, stream->Write(in, kSize, &bytes, NULL));
122
123 // Try a full read
124 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize, &bytes, NULL));
125 EXPECT_EQ(kSize, bytes);
126 EXPECT_EQ(0, memcmp(in, out, kSize));
127
128 // Try a read that should block
129 EXPECT_EQ(SR_BLOCK, stream->Read(out, kSize, &bytes, NULL));
130
131 // Try a too-big write
132 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize * 2, &bytes, NULL));
133 EXPECT_EQ(bytes, kSize);
134
135 // Try a too-big read
136 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize * 2, &bytes, NULL));
137 EXPECT_EQ(kSize, bytes);
138 EXPECT_EQ(0, memcmp(in, out, kSize));
139
140 // Try some small writes and reads
141 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize / 2, &bytes, NULL));
142 EXPECT_EQ(kSize / 2, bytes);
143 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize / 2, &bytes, NULL));
144 EXPECT_EQ(kSize / 2, bytes);
145 EXPECT_EQ(0, memcmp(in, out, kSize / 2));
146 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize / 2, &bytes, NULL));
147 EXPECT_EQ(kSize / 2, bytes);
148 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize / 2, &bytes, NULL));
149 EXPECT_EQ(kSize / 2, bytes);
150 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize / 2, &bytes, NULL));
151 EXPECT_EQ(kSize / 2, bytes);
152 EXPECT_EQ(0, memcmp(in, out, kSize / 2));
153 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize / 2, &bytes, NULL));
154 EXPECT_EQ(kSize / 2, bytes);
155 EXPECT_EQ(0, memcmp(in, out, kSize / 2));
156
157 // Try wraparound reads and writes in the following pattern
158 // WWWWWWWWWWWW.... 0123456789AB....
159 // RRRRRRRRXXXX.... ........89AB....
160 // WWWW....XXXXWWWW 4567....89AB0123
161 // XXXX....RRRRXXXX 4567........0123
162 // XXXXWWWWWWWWXXXX 4567012345670123
163 // RRRRXXXXXXXXRRRR ....01234567....
164 // ....RRRRRRRR.... ................
165 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize * 3 / 4, &bytes, NULL));
166 EXPECT_EQ(kSize * 3 / 4, bytes);
167 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize / 2, &bytes, NULL));
168 EXPECT_EQ(kSize / 2, bytes);
169 EXPECT_EQ(0, memcmp(in, out, kSize / 2));
170 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize / 2, &bytes, NULL));
171 EXPECT_EQ(kSize / 2, bytes);
172 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize / 4, &bytes, NULL));
173 EXPECT_EQ(kSize / 4 , bytes);
174 EXPECT_EQ(0, memcmp(in + kSize / 2, out, kSize / 4));
175 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize / 2, &bytes, NULL));
176 EXPECT_EQ(kSize / 2, bytes);
177 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize / 2, &bytes, NULL));
178 EXPECT_EQ(kSize / 2 , bytes);
179 EXPECT_EQ(0, memcmp(in, out, kSize / 2));
180 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize / 2, &bytes, NULL));
181 EXPECT_EQ(kSize / 2 , bytes);
182 EXPECT_EQ(0, memcmp(in, out, kSize / 2));
183
184 // Use GetWriteBuffer to reset the read_position for the next tests
185 stream->GetWriteBuffer(&bytes);
186 stream->ConsumeWriteBuffer(0);
187
188 // Try using GetReadData to do a full read
189 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize, &bytes, NULL));
190 q = stream->GetReadData(&bytes);
191 EXPECT_TRUE(NULL != q);
192 EXPECT_EQ(kSize, bytes);
193 EXPECT_EQ(0, memcmp(q, in, kSize));
194 stream->ConsumeReadData(kSize);
195 EXPECT_EQ(SR_BLOCK, stream->Read(out, kSize, &bytes, NULL));
196
197 // Try using GetReadData to do some small reads
198 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize, &bytes, NULL));
199 q = stream->GetReadData(&bytes);
200 EXPECT_TRUE(NULL != q);
201 EXPECT_EQ(kSize, bytes);
202 EXPECT_EQ(0, memcmp(q, in, kSize / 2));
203 stream->ConsumeReadData(kSize / 2);
204 q = stream->GetReadData(&bytes);
205 EXPECT_TRUE(NULL != q);
206 EXPECT_EQ(kSize / 2, bytes);
207 EXPECT_EQ(0, memcmp(q, in + kSize / 2, kSize / 2));
208 stream->ConsumeReadData(kSize / 2);
209 EXPECT_EQ(SR_BLOCK, stream->Read(out, kSize, &bytes, NULL));
210
211 // Try using GetReadData in a wraparound case
212 // WWWWWWWWWWWWWWWW 0123456789ABCDEF
213 // RRRRRRRRRRRRXXXX ............CDEF
214 // WWWWWWWW....XXXX 01234567....CDEF
215 // ............RRRR 01234567........
216 // RRRRRRRR........ ................
217 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize, &bytes, NULL));
218 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize * 3 / 4, &bytes, NULL));
219 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize / 2, &bytes, NULL));
220 q = stream->GetReadData(&bytes);
221 EXPECT_TRUE(NULL != q);
222 EXPECT_EQ(kSize / 4, bytes);
223 EXPECT_EQ(0, memcmp(q, in + kSize * 3 / 4, kSize / 4));
224 stream->ConsumeReadData(kSize / 4);
225 q = stream->GetReadData(&bytes);
226 EXPECT_TRUE(NULL != q);
227 EXPECT_EQ(kSize / 2, bytes);
228 EXPECT_EQ(0, memcmp(q, in, kSize / 2));
229 stream->ConsumeReadData(kSize / 2);
230
231 // Use GetWriteBuffer to reset the read_position for the next tests
232 stream->GetWriteBuffer(&bytes);
233 stream->ConsumeWriteBuffer(0);
234
235 // Try using GetWriteBuffer to do a full write
236 p = stream->GetWriteBuffer(&bytes);
237 EXPECT_TRUE(NULL != p);
238 EXPECT_EQ(kSize, bytes);
239 memcpy(p, in, kSize);
240 stream->ConsumeWriteBuffer(kSize);
241 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize, &bytes, NULL));
242 EXPECT_EQ(kSize, bytes);
243 EXPECT_EQ(0, memcmp(in, out, kSize));
244
245 // Try using GetWriteBuffer to do some small writes
246 p = stream->GetWriteBuffer(&bytes);
247 EXPECT_TRUE(NULL != p);
248 EXPECT_EQ(kSize, bytes);
249 memcpy(p, in, kSize / 2);
250 stream->ConsumeWriteBuffer(kSize / 2);
251 p = stream->GetWriteBuffer(&bytes);
252 EXPECT_TRUE(NULL != p);
253 EXPECT_EQ(kSize / 2, bytes);
254 memcpy(p, in + kSize / 2, kSize / 2);
255 stream->ConsumeWriteBuffer(kSize / 2);
256 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize, &bytes, NULL));
257 EXPECT_EQ(kSize, bytes);
258 EXPECT_EQ(0, memcmp(in, out, kSize));
259
260 // Try using GetWriteBuffer in a wraparound case
261 // WWWWWWWWWWWW.... 0123456789AB....
262 // RRRRRRRRXXXX.... ........89AB....
263 // ........XXXXWWWW ........89AB0123
264 // WWWW....XXXXXXXX 4567....89AB0123
265 // RRRR....RRRRRRRR ................
266 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize * 3 / 4, &bytes, NULL));
267 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize / 2, &bytes, NULL));
268 p = stream->GetWriteBuffer(&bytes);
269 EXPECT_TRUE(NULL != p);
270 EXPECT_EQ(kSize / 4, bytes);
271 memcpy(p, in, kSize / 4);
272 stream->ConsumeWriteBuffer(kSize / 4);
273 p = stream->GetWriteBuffer(&bytes);
274 EXPECT_TRUE(NULL != p);
275 EXPECT_EQ(kSize / 2, bytes);
276 memcpy(p, in + kSize / 4, kSize / 4);
277 stream->ConsumeWriteBuffer(kSize / 4);
278 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize * 3 / 4, &bytes, NULL));
279 EXPECT_EQ(kSize * 3 / 4, bytes);
280 EXPECT_EQ(0, memcmp(in + kSize / 2, out, kSize / 4));
281 EXPECT_EQ(0, memcmp(in, out + kSize / 4, kSize / 4));
282
283 // Check that the stream is now empty
284 EXPECT_EQ(SR_BLOCK, stream->Read(out, kSize, &bytes, NULL));
285
286 // Try growing the buffer
287 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize, &bytes, NULL));
288 EXPECT_EQ(kSize, bytes);
289 EXPECT_TRUE(buf.SetCapacity(kSize * 2));
290 EXPECT_EQ(SR_SUCCESS, stream->Write(in + kSize, kSize, &bytes, NULL));
291 EXPECT_EQ(kSize, bytes);
292 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize * 2, &bytes, NULL));
293 EXPECT_EQ(kSize * 2, bytes);
294 EXPECT_EQ(0, memcmp(in, out, kSize * 2));
295
296 // Try shrinking the buffer
297 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize, &bytes, NULL));
298 EXPECT_EQ(kSize, bytes);
299 EXPECT_TRUE(buf.SetCapacity(kSize));
300 EXPECT_EQ(SR_BLOCK, stream->Write(in, kSize, &bytes, NULL));
301 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize, &bytes, NULL));
302 EXPECT_EQ(kSize, bytes);
303 EXPECT_EQ(0, memcmp(in, out, kSize));
304
305 // Write to the stream, close it, read the remaining bytes
306 EXPECT_EQ(SR_SUCCESS, stream->Write(in, kSize / 2, &bytes, NULL));
307 stream->Close();
308 EXPECT_EQ(SS_CLOSED, stream->GetState());
309 EXPECT_EQ(SR_EOS, stream->Write(in, kSize / 2, &bytes, NULL));
310 EXPECT_EQ(SR_SUCCESS, stream->Read(out, kSize / 2, &bytes, NULL));
311 EXPECT_EQ(0, memcmp(in, out, kSize / 2));
312 EXPECT_EQ(SR_EOS, stream->Read(out, kSize / 2, &bytes, NULL));
313}
314
315TEST(FifoBufferTest, FullBufferCheck) {
316 FifoBuffer buff(10);
317 buff.ConsumeWriteBuffer(10);
318
319 size_t free;
320 EXPECT_TRUE(buff.GetWriteBuffer(&free) != NULL);
321 EXPECT_EQ(0U, free);
322}
323
324TEST(FifoBufferTest, WriteOffsetAndReadOffset) {
325 const size_t kSize = 16;
326 const char in[kSize * 2 + 1] = "0123456789ABCDEFGHIJKLMNOPQRSTUV";
327 char out[kSize * 2];
328 FifoBuffer buf(kSize);
329
330 // Write 14 bytes.
331 EXPECT_EQ(SR_SUCCESS, buf.Write(in, 14, NULL, NULL));
332
333 // Make sure data is in |buf|.
334 size_t buffered;
335 EXPECT_TRUE(buf.GetBuffered(&buffered));
336 EXPECT_EQ(14u, buffered);
337
338 // Read 10 bytes.
339 buf.ConsumeReadData(10);
340
341 // There should be now 12 bytes of available space.
342 size_t remaining;
343 EXPECT_TRUE(buf.GetWriteRemaining(&remaining));
344 EXPECT_EQ(12u, remaining);
345
346 // Write at offset 12, this should fail.
347 EXPECT_EQ(SR_BLOCK, buf.WriteOffset(in, 10, 12, NULL));
348
349 // Write 8 bytes at offset 4, this wraps around the buffer.
350 EXPECT_EQ(SR_SUCCESS, buf.WriteOffset(in, 8, 4, NULL));
351
352 // Number of available space remains the same until we call
353 // ConsumeWriteBuffer().
354 EXPECT_TRUE(buf.GetWriteRemaining(&remaining));
355 EXPECT_EQ(12u, remaining);
356 buf.ConsumeWriteBuffer(12);
357
358 // There's 4 bytes bypassed and 4 bytes no read so skip them and verify the
359 // 8 bytes written.
360 size_t read;
361 EXPECT_EQ(SR_SUCCESS, buf.ReadOffset(out, 8, 8, &read));
362 EXPECT_EQ(8u, read);
363 EXPECT_EQ(0, memcmp(out, in, 8));
364
365 // There should still be 16 bytes available for reading.
366 EXPECT_TRUE(buf.GetBuffered(&buffered));
367 EXPECT_EQ(16u, buffered);
368
369 // Read at offset 16, this should fail since we don't have that much data.
370 EXPECT_EQ(SR_BLOCK, buf.ReadOffset(out, 10, 16, NULL));
371}
372
henrike@webrtc.orgf0488722014-05-13 18:00:26 +0000373} // namespace rtc