blob: 694ed0b0085e82b6f47d09645fd7f9d4c793e8a9 [file] [log] [blame]
Karl Wiberge2a83ee2015-10-26 19:51:29 +01001/*
2 * Copyright 2015 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 <algorithm>
Alessio Bazzica858c4d72018-05-14 16:33:58 +020012#include <array>
Karl Wiberge2a83ee2015-10-26 19:51:29 +010013#include <string>
kwiberg529662a2017-09-04 05:43:17 -070014#include <utility>
Karl Wiberge2a83ee2015-10-26 19:51:29 +010015#include <vector>
16
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020017#include "api/array_view.h"
18#include "rtc_base/buffer.h"
19#include "rtc_base/checks.h"
20#include "rtc_base/gunit.h"
21#include "test/gmock.h"
Karl Wiberge2a83ee2015-10-26 19:51:29 +010022
23namespace rtc {
24
25namespace {
kwibergbd431722016-09-05 04:20:54 -070026
danilchapa28780e2016-11-18 01:46:23 -080027using ::testing::ElementsAre;
28using ::testing::IsEmpty;
29
Karl Wiberge2a83ee2015-10-26 19:51:29 +010030template <typename T>
31void Call(ArrayView<T>) {}
kwibergbd431722016-09-05 04:20:54 -070032
Karl Wiberge2a83ee2015-10-26 19:51:29 +010033} // namespace
34
35TEST(ArrayViewTest, TestConstructFromPtrAndArray) {
36 char arr[] = "Arrr!";
37 const char carr[] = "Carrr!";
38 Call<const char>(arr);
39 Call<const char>(carr);
40 Call<char>(arr);
41 // Call<char>(carr); // Compile error, because can't drop const.
42 // Call<int>(arr); // Compile error, because incompatible types.
43 ArrayView<int*> x;
44 EXPECT_EQ(0u, x.size());
45 EXPECT_EQ(nullptr, x.data());
46 ArrayView<char> y = arr;
47 EXPECT_EQ(6u, y.size());
48 EXPECT_EQ(arr, y.data());
kwibergbfc7f022017-03-02 12:33:50 -080049 ArrayView<char, 6> yf = arr;
50 static_assert(yf.size() == 6, "");
51 EXPECT_EQ(arr, yf.data());
Karl Wiberge2a83ee2015-10-26 19:51:29 +010052 ArrayView<const char> z(arr + 1, 3);
53 EXPECT_EQ(3u, z.size());
54 EXPECT_EQ(arr + 1, z.data());
kwibergbfc7f022017-03-02 12:33:50 -080055 ArrayView<const char, 3> zf(arr + 1, 3);
56 static_assert(zf.size() == 3, "");
57 EXPECT_EQ(arr + 1, zf.data());
Karl Wiberge2a83ee2015-10-26 19:51:29 +010058 ArrayView<const char> w(arr, 2);
59 EXPECT_EQ(2u, w.size());
60 EXPECT_EQ(arr, w.data());
kwibergbfc7f022017-03-02 12:33:50 -080061 ArrayView<const char, 2> wf(arr, 2);
62 static_assert(wf.size() == 2, "");
63 EXPECT_EQ(arr, wf.data());
Karl Wiberge2a83ee2015-10-26 19:51:29 +010064 ArrayView<char> q(arr, 0);
65 EXPECT_EQ(0u, q.size());
66 EXPECT_EQ(nullptr, q.data());
kwibergbfc7f022017-03-02 12:33:50 -080067 ArrayView<char, 0> qf(arr, 0);
68 static_assert(qf.size() == 0, "");
69 EXPECT_EQ(nullptr, qf.data());
Karl Wiberge2a83ee2015-10-26 19:51:29 +010070#if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID)
71 // DCHECK error (nullptr with nonzero size).
72 EXPECT_DEATH(ArrayView<int>(static_cast<int*>(nullptr), 5), "");
73#endif
74 // These are compile errors, because incompatible types.
75 // ArrayView<int> m = arr;
76 // ArrayView<float> n(arr + 2, 2);
77}
78
kwibergbfc7f022017-03-02 12:33:50 -080079TEST(ArrayViewTest, TestCopyConstructorVariable) {
Karl Wiberge2a83ee2015-10-26 19:51:29 +010080 char arr[] = "Arrr!";
81 ArrayView<char> x = arr;
82 EXPECT_EQ(6u, x.size());
83 EXPECT_EQ(arr, x.data());
84 ArrayView<char> y = x; // Copy non-const -> non-const.
85 EXPECT_EQ(6u, y.size());
86 EXPECT_EQ(arr, y.data());
87 ArrayView<const char> z = x; // Copy non-const -> const.
88 EXPECT_EQ(6u, z.size());
89 EXPECT_EQ(arr, z.data());
90 ArrayView<const char> w = z; // Copy const -> const.
91 EXPECT_EQ(6u, w.size());
92 EXPECT_EQ(arr, w.data());
93 // ArrayView<char> v = z; // Compile error, because can't drop const.
94}
95
kwibergbfc7f022017-03-02 12:33:50 -080096TEST(ArrayViewTest, TestCopyConstructorFixed) {
97 char arr[] = "Arrr!";
98 ArrayView<char, 6> x = arr;
99 static_assert(x.size() == 6, "");
100 EXPECT_EQ(arr, x.data());
101
102 // Copy fixed -> fixed.
103 ArrayView<char, 6> y = x; // Copy non-const -> non-const.
104 static_assert(y.size() == 6, "");
105 EXPECT_EQ(arr, y.data());
106 ArrayView<const char, 6> z = x; // Copy non-const -> const.
107 static_assert(z.size() == 6, "");
108 EXPECT_EQ(arr, z.data());
109 ArrayView<const char, 6> w = z; // Copy const -> const.
110 static_assert(w.size() == 6, "");
111 EXPECT_EQ(arr, w.data());
112 // ArrayView<char, 6> v = z; // Compile error, because can't drop const.
113
114 // Copy fixed -> variable.
115 ArrayView<char> yv = x; // Copy non-const -> non-const.
116 EXPECT_EQ(6u, yv.size());
117 EXPECT_EQ(arr, yv.data());
118 ArrayView<const char> zv = x; // Copy non-const -> const.
119 EXPECT_EQ(6u, zv.size());
120 EXPECT_EQ(arr, zv.data());
121 ArrayView<const char> wv = z; // Copy const -> const.
122 EXPECT_EQ(6u, wv.size());
123 EXPECT_EQ(arr, wv.data());
124 // ArrayView<char> vv = z; // Compile error, because can't drop const.
125}
126
127TEST(ArrayViewTest, TestCopyAssignmentVariable) {
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100128 char arr[] = "Arrr!";
129 ArrayView<char> x(arr);
130 EXPECT_EQ(6u, x.size());
131 EXPECT_EQ(arr, x.data());
132 ArrayView<char> y;
133 y = x; // Copy non-const -> non-const.
134 EXPECT_EQ(6u, y.size());
135 EXPECT_EQ(arr, y.data());
136 ArrayView<const char> z;
137 z = x; // Copy non-const -> const.
138 EXPECT_EQ(6u, z.size());
139 EXPECT_EQ(arr, z.data());
140 ArrayView<const char> w;
141 w = z; // Copy const -> const.
142 EXPECT_EQ(6u, w.size());
143 EXPECT_EQ(arr, w.data());
144 // ArrayView<char> v;
145 // v = z; // Compile error, because can't drop const.
146}
147
kwibergbfc7f022017-03-02 12:33:50 -0800148TEST(ArrayViewTest, TestCopyAssignmentFixed) {
149 char arr[] = "Arrr!";
150 char init[] = "Init!";
151 ArrayView<char, 6> x(arr);
152 EXPECT_EQ(arr, x.data());
153
154 // Copy fixed -> fixed.
155 ArrayView<char, 6> y(init);
156 y = x; // Copy non-const -> non-const.
157 EXPECT_EQ(arr, y.data());
158 ArrayView<const char, 6> z(init);
159 z = x; // Copy non-const -> const.
160 EXPECT_EQ(arr, z.data());
161 ArrayView<const char, 6> w(init);
162 w = z; // Copy const -> const.
163 EXPECT_EQ(arr, w.data());
164 // ArrayView<char, 6> v(init);
165 // v = z; // Compile error, because can't drop const.
166
167 // Copy fixed -> variable.
168 ArrayView<char> yv;
169 yv = x; // Copy non-const -> non-const.
170 EXPECT_EQ(6u, yv.size());
171 EXPECT_EQ(arr, yv.data());
172 ArrayView<const char> zv;
173 zv = x; // Copy non-const -> const.
174 EXPECT_EQ(6u, zv.size());
175 EXPECT_EQ(arr, zv.data());
176 ArrayView<const char> wv;
177 wv = z; // Copy const -> const.
178 EXPECT_EQ(6u, wv.size());
179 EXPECT_EQ(arr, wv.data());
180 // ArrayView<char> v;
181 // v = z; // Compile error, because can't drop const.
182}
183
Alessio Bazzica858c4d72018-05-14 16:33:58 +0200184TEST(ArrayViewTest, TestStdArray) {
185 constexpr size_t size = 5;
186 std::array<float, size> arr{};
187 // Fixed size view.
188 rtc::ArrayView<float, size> arr_view_fixed(arr);
189 EXPECT_EQ(arr.data(), arr_view_fixed.data());
190 static_assert(size == arr_view_fixed.size(), "");
191 // Variable size view.
192 rtc::ArrayView<float> arr_view(arr);
193 EXPECT_EQ(arr.data(), arr_view.data());
194 EXPECT_EQ(size, arr_view.size());
195}
196
Alessio Bazzica28a325b2018-05-15 14:57:51 +0200197TEST(ArrayViewTest, TestConstStdArray) {
198 constexpr size_t size = 5;
199
200 constexpr std::array<float, size> constexpr_arr{};
201 rtc::ArrayView<const float, size> constexpr_arr_view(constexpr_arr);
202 EXPECT_EQ(constexpr_arr.data(), constexpr_arr_view.data());
203 static_assert(constexpr_arr.size() == constexpr_arr_view.size(), "");
204
205 const std::array<float, size> const_arr{};
206 rtc::ArrayView<const float, size> const_arr_view(const_arr);
207 EXPECT_EQ(const_arr.data(), const_arr_view.data());
208 static_assert(const_arr.size() == const_arr_view.size(), "");
209
210 std::array<float, size> non_const_arr{};
211 rtc::ArrayView<const float, size> non_const_arr_view(non_const_arr);
212 EXPECT_EQ(non_const_arr.data(), non_const_arr_view.data());
213 static_assert(non_const_arr.size() == non_const_arr_view.size(), "");
214}
215
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100216TEST(ArrayViewTest, TestStdVector) {
217 std::vector<int> v;
218 v.push_back(3);
219 v.push_back(11);
220 Call<const int>(v);
221 Call<int>(v);
222 // Call<unsigned int>(v); // Compile error, because incompatible types.
223 ArrayView<int> x = v;
224 EXPECT_EQ(2u, x.size());
225 EXPECT_EQ(v.data(), x.data());
226 ArrayView<const int> y;
227 y = v;
228 EXPECT_EQ(2u, y.size());
229 EXPECT_EQ(v.data(), y.data());
230 // ArrayView<double> d = v; // Compile error, because incompatible types.
231 const std::vector<int> cv;
232 Call<const int>(cv);
233 // Call<int>(cv); // Compile error, because can't drop const.
234 ArrayView<const int> z = cv;
235 EXPECT_EQ(0u, z.size());
236 EXPECT_EQ(nullptr, z.data());
237 // ArrayView<int> w = cv; // Compile error, because can't drop const.
238}
239
240TEST(ArrayViewTest, TestRtcBuffer) {
241 rtc::Buffer b = "so buffer";
242 Call<const uint8_t>(b);
243 Call<uint8_t>(b);
244 // Call<int8_t>(b); // Compile error, because incompatible types.
245 ArrayView<uint8_t> x = b;
246 EXPECT_EQ(10u, x.size());
247 EXPECT_EQ(b.data(), x.data());
248 ArrayView<const uint8_t> y;
249 y = b;
250 EXPECT_EQ(10u, y.size());
251 EXPECT_EQ(b.data(), y.data());
252 // ArrayView<char> d = b; // Compile error, because incompatible types.
253 const rtc::Buffer cb = "very const";
254 Call<const uint8_t>(cb);
255 // Call<uint8_t>(cb); // Compile error, because can't drop const.
256 ArrayView<const uint8_t> z = cb;
257 EXPECT_EQ(11u, z.size());
258 EXPECT_EQ(cb.data(), z.data());
259 // ArrayView<uint8_t> w = cb; // Compile error, because can't drop const.
260}
261
kwibergbfc7f022017-03-02 12:33:50 -0800262TEST(ArrayViewTest, TestSwapVariable) {
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100263 const char arr[] = "Arrr!";
264 const char aye[] = "Aye, Cap'n!";
265 ArrayView<const char> x(arr);
266 EXPECT_EQ(6u, x.size());
267 EXPECT_EQ(arr, x.data());
268 ArrayView<const char> y(aye);
269 EXPECT_EQ(12u, y.size());
270 EXPECT_EQ(aye, y.data());
271 using std::swap;
272 swap(x, y);
273 EXPECT_EQ(12u, x.size());
274 EXPECT_EQ(aye, x.data());
275 EXPECT_EQ(6u, y.size());
276 EXPECT_EQ(arr, y.data());
277 // ArrayView<char> z;
278 // swap(x, z); // Compile error, because can't drop const.
279}
280
kwibergbfc7f022017-03-02 12:33:50 -0800281TEST(FixArrayViewTest, TestSwapFixed) {
282 const char arr[] = "Arr!";
283 char aye[] = "Aye!";
284 ArrayView<const char, 5> x(arr);
285 EXPECT_EQ(arr, x.data());
286 ArrayView<const char, 5> y(aye);
287 EXPECT_EQ(aye, y.data());
288 using std::swap;
289 swap(x, y);
290 EXPECT_EQ(aye, x.data());
291 EXPECT_EQ(arr, y.data());
292 // ArrayView<char, 5> z(aye);
293 // swap(x, z); // Compile error, because can't drop const.
294 // ArrayView<const char, 4> w(aye, 4);
295 // swap(x, w); // Compile error, because different sizes.
296}
297
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100298TEST(ArrayViewTest, TestIndexing) {
299 char arr[] = "abcdefg";
300 ArrayView<char> x(arr);
301 const ArrayView<char> y(arr);
kwibergbfc7f022017-03-02 12:33:50 -0800302 ArrayView<const char, 8> z(arr);
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100303 EXPECT_EQ(8u, x.size());
304 EXPECT_EQ(8u, y.size());
305 EXPECT_EQ(8u, z.size());
306 EXPECT_EQ('b', x[1]);
307 EXPECT_EQ('c', y[2]);
308 EXPECT_EQ('d', z[3]);
309 x[3] = 'X';
310 y[2] = 'Y';
311 // z[1] = 'Z'; // Compile error, because z's element type is const char.
312 EXPECT_EQ('b', x[1]);
313 EXPECT_EQ('Y', y[2]);
314 EXPECT_EQ('X', z[3]);
315#if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID)
316 EXPECT_DEATH(z[8], ""); // DCHECK error (index out of bounds).
317#endif
318}
319
320TEST(ArrayViewTest, TestIterationEmpty) {
kwibergbfc7f022017-03-02 12:33:50 -0800321 // Variable-size.
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100322 ArrayView<std::vector<std::vector<std::vector<std::string>>>> av;
kwibergbfc7f022017-03-02 12:33:50 -0800323 EXPECT_EQ(av.begin(), av.end());
324 EXPECT_EQ(av.cbegin(), av.cend());
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100325 for (auto& e : av) {
326 EXPECT_TRUE(false);
327 EXPECT_EQ(42u, e.size()); // Dummy use of e to prevent unused var warning.
328 }
kwibergbfc7f022017-03-02 12:33:50 -0800329
330 // Fixed-size.
331 ArrayView<std::vector<std::vector<std::vector<std::string>>>, 0> af;
332 EXPECT_EQ(af.begin(), af.end());
333 EXPECT_EQ(af.cbegin(), af.cend());
334 for (auto& e : af) {
335 EXPECT_TRUE(false);
336 EXPECT_EQ(42u, e.size()); // Dummy use of e to prevent unused var warning.
337 }
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100338}
339
kwibergbfc7f022017-03-02 12:33:50 -0800340TEST(ArrayViewTest, TestIterationVariable) {
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100341 char arr[] = "Arrr!";
342 ArrayView<char> av(arr);
343 EXPECT_EQ('A', *av.begin());
344 EXPECT_EQ('A', *av.cbegin());
345 EXPECT_EQ('\0', *(av.end() - 1));
346 EXPECT_EQ('\0', *(av.cend() - 1));
347 char i = 0;
348 for (auto& e : av) {
349 EXPECT_EQ(arr + i, &e);
350 e = 's' + i;
351 ++i;
352 }
353 i = 0;
354 for (auto& e : ArrayView<const char>(av)) {
355 EXPECT_EQ(arr + i, &e);
356 // e = 'q' + i; // Compile error, because e is a const char&.
357 ++i;
358 }
359}
360
kwibergbfc7f022017-03-02 12:33:50 -0800361TEST(ArrayViewTest, TestIterationFixed) {
362 char arr[] = "Arrr!";
363 ArrayView<char, 6> av(arr);
364 EXPECT_EQ('A', *av.begin());
365 EXPECT_EQ('A', *av.cbegin());
366 EXPECT_EQ('\0', *(av.end() - 1));
367 EXPECT_EQ('\0', *(av.cend() - 1));
368 char i = 0;
369 for (auto& e : av) {
370 EXPECT_EQ(arr + i, &e);
371 e = 's' + i;
372 ++i;
373 }
374 i = 0;
375 for (auto& e : ArrayView<const char, 6>(av)) {
376 EXPECT_EQ(arr + i, &e);
377 // e = 'q' + i; // Compile error, because e is a const char&.
378 ++i;
379 }
380}
381
kwiberg288886b2015-11-06 01:21:35 -0800382TEST(ArrayViewTest, TestEmpty) {
383 EXPECT_TRUE(ArrayView<int>().empty());
384 const int a[] = {1, 2, 3};
385 EXPECT_FALSE(ArrayView<const int>(a).empty());
kwibergbfc7f022017-03-02 12:33:50 -0800386
387 static_assert(ArrayView<int, 0>::empty(), "");
388 static_assert(!ArrayView<int, 3>::empty(), "");
kwiberg288886b2015-11-06 01:21:35 -0800389}
390
391TEST(ArrayViewTest, TestCompare) {
392 int a[] = {1, 2, 3};
393 int b[] = {1, 2, 3};
kwibergbfc7f022017-03-02 12:33:50 -0800394
kwiberg288886b2015-11-06 01:21:35 -0800395 EXPECT_EQ(ArrayView<int>(a), ArrayView<int>(a));
kwibergbfc7f022017-03-02 12:33:50 -0800396 EXPECT_EQ((ArrayView<int, 3>(a)), (ArrayView<int, 3>(a)));
397 EXPECT_EQ(ArrayView<int>(a), (ArrayView<int, 3>(a)));
kwiberg288886b2015-11-06 01:21:35 -0800398 EXPECT_EQ(ArrayView<int>(), ArrayView<int>());
kwibergbfc7f022017-03-02 12:33:50 -0800399 EXPECT_EQ(ArrayView<int>(), ArrayView<int>(a, 0));
400 EXPECT_EQ(ArrayView<int>(a, 0), ArrayView<int>(b, 0));
401 EXPECT_EQ((ArrayView<int, 0>(a, 0)), ArrayView<int>());
402
kwiberg288886b2015-11-06 01:21:35 -0800403 EXPECT_NE(ArrayView<int>(a), ArrayView<int>(b));
kwibergbfc7f022017-03-02 12:33:50 -0800404 EXPECT_NE((ArrayView<int, 3>(a)), (ArrayView<int, 3>(b)));
405 EXPECT_NE((ArrayView<int, 3>(a)), ArrayView<int>(b));
kwiberg288886b2015-11-06 01:21:35 -0800406 EXPECT_NE(ArrayView<int>(a), ArrayView<int>());
407 EXPECT_NE(ArrayView<int>(a), ArrayView<int>(a, 2));
kwibergbfc7f022017-03-02 12:33:50 -0800408 EXPECT_NE((ArrayView<int, 3>(a)), (ArrayView<int, 2>(a, 2)));
kwiberg288886b2015-11-06 01:21:35 -0800409}
410
kwibergbfc7f022017-03-02 12:33:50 -0800411TEST(ArrayViewTest, TestSubViewVariable) {
danilchapa28780e2016-11-18 01:46:23 -0800412 int a[] = {1, 2, 3};
413 ArrayView<int> av(a);
414
415 EXPECT_EQ(av.subview(0), av);
416
417 EXPECT_THAT(av.subview(1), ElementsAre(2, 3));
418 EXPECT_THAT(av.subview(2), ElementsAre(3));
419 EXPECT_THAT(av.subview(3), IsEmpty());
420 EXPECT_THAT(av.subview(4), IsEmpty());
421
422 EXPECT_THAT(av.subview(1, 0), IsEmpty());
423 EXPECT_THAT(av.subview(1, 1), ElementsAre(2));
424 EXPECT_THAT(av.subview(1, 2), ElementsAre(2, 3));
425 EXPECT_THAT(av.subview(1, 3), ElementsAre(2, 3));
426}
427
kwibergbfc7f022017-03-02 12:33:50 -0800428TEST(ArrayViewTest, TestSubViewFixed) {
429 int a[] = {1, 2, 3};
430 ArrayView<int, 3> av(a);
431
432 EXPECT_EQ(av.subview(0), av);
433
434 EXPECT_THAT(av.subview(1), ElementsAre(2, 3));
435 EXPECT_THAT(av.subview(2), ElementsAre(3));
436 EXPECT_THAT(av.subview(3), IsEmpty());
437 EXPECT_THAT(av.subview(4), IsEmpty());
438
439 EXPECT_THAT(av.subview(1, 0), IsEmpty());
440 EXPECT_THAT(av.subview(1, 1), ElementsAre(2));
441 EXPECT_THAT(av.subview(1, 2), ElementsAre(2, 3));
442 EXPECT_THAT(av.subview(1, 3), ElementsAre(2, 3));
443}
444
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100445} // namespace rtc