blob: f7fe9fa36bc164c4d6395987ef2230c05f1834c0 [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>
12#include <string>
13#include <vector>
14
15#include "webrtc/base/array_view.h"
16#include "webrtc/base/buffer.h"
17#include "webrtc/base/checks.h"
18#include "webrtc/base/gunit.h"
danilchapa28780e2016-11-18 01:46:23 -080019#include "webrtc/test/gmock.h"
Karl Wiberge2a83ee2015-10-26 19:51:29 +010020
21namespace rtc {
22
23namespace {
kwibergbd431722016-09-05 04:20:54 -070024
danilchapa28780e2016-11-18 01:46:23 -080025using ::testing::ElementsAre;
26using ::testing::IsEmpty;
27
Karl Wiberge2a83ee2015-10-26 19:51:29 +010028template <typename T>
29void Call(ArrayView<T>) {}
kwibergbd431722016-09-05 04:20:54 -070030
Karl Wiberge2a83ee2015-10-26 19:51:29 +010031} // namespace
32
33TEST(ArrayViewTest, TestConstructFromPtrAndArray) {
34 char arr[] = "Arrr!";
35 const char carr[] = "Carrr!";
36 Call<const char>(arr);
37 Call<const char>(carr);
38 Call<char>(arr);
39 // Call<char>(carr); // Compile error, because can't drop const.
40 // Call<int>(arr); // Compile error, because incompatible types.
41 ArrayView<int*> x;
42 EXPECT_EQ(0u, x.size());
43 EXPECT_EQ(nullptr, x.data());
44 ArrayView<char> y = arr;
45 EXPECT_EQ(6u, y.size());
46 EXPECT_EQ(arr, y.data());
47 ArrayView<const char> z(arr + 1, 3);
48 EXPECT_EQ(3u, z.size());
49 EXPECT_EQ(arr + 1, z.data());
50 ArrayView<const char> w(arr, 2);
51 EXPECT_EQ(2u, w.size());
52 EXPECT_EQ(arr, w.data());
53 ArrayView<char> q(arr, 0);
54 EXPECT_EQ(0u, q.size());
55 EXPECT_EQ(nullptr, q.data());
56#if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID)
57 // DCHECK error (nullptr with nonzero size).
58 EXPECT_DEATH(ArrayView<int>(static_cast<int*>(nullptr), 5), "");
59#endif
60 // These are compile errors, because incompatible types.
61 // ArrayView<int> m = arr;
62 // ArrayView<float> n(arr + 2, 2);
63}
64
65TEST(ArrayViewTest, TestCopyConstructor) {
66 char arr[] = "Arrr!";
67 ArrayView<char> x = arr;
68 EXPECT_EQ(6u, x.size());
69 EXPECT_EQ(arr, x.data());
70 ArrayView<char> y = x; // Copy non-const -> non-const.
71 EXPECT_EQ(6u, y.size());
72 EXPECT_EQ(arr, y.data());
73 ArrayView<const char> z = x; // Copy non-const -> const.
74 EXPECT_EQ(6u, z.size());
75 EXPECT_EQ(arr, z.data());
76 ArrayView<const char> w = z; // Copy const -> const.
77 EXPECT_EQ(6u, w.size());
78 EXPECT_EQ(arr, w.data());
79 // ArrayView<char> v = z; // Compile error, because can't drop const.
80}
81
82TEST(ArrayViewTest, TestCopyAssignment) {
83 char arr[] = "Arrr!";
84 ArrayView<char> x(arr);
85 EXPECT_EQ(6u, x.size());
86 EXPECT_EQ(arr, x.data());
87 ArrayView<char> y;
88 y = x; // Copy non-const -> non-const.
89 EXPECT_EQ(6u, y.size());
90 EXPECT_EQ(arr, y.data());
91 ArrayView<const char> z;
92 z = x; // Copy non-const -> const.
93 EXPECT_EQ(6u, z.size());
94 EXPECT_EQ(arr, z.data());
95 ArrayView<const char> w;
96 w = z; // Copy const -> const.
97 EXPECT_EQ(6u, w.size());
98 EXPECT_EQ(arr, w.data());
99 // ArrayView<char> v;
100 // v = z; // Compile error, because can't drop const.
101}
102
103TEST(ArrayViewTest, TestStdVector) {
104 std::vector<int> v;
105 v.push_back(3);
106 v.push_back(11);
107 Call<const int>(v);
108 Call<int>(v);
109 // Call<unsigned int>(v); // Compile error, because incompatible types.
110 ArrayView<int> x = v;
111 EXPECT_EQ(2u, x.size());
112 EXPECT_EQ(v.data(), x.data());
113 ArrayView<const int> y;
114 y = v;
115 EXPECT_EQ(2u, y.size());
116 EXPECT_EQ(v.data(), y.data());
117 // ArrayView<double> d = v; // Compile error, because incompatible types.
118 const std::vector<int> cv;
119 Call<const int>(cv);
120 // Call<int>(cv); // Compile error, because can't drop const.
121 ArrayView<const int> z = cv;
122 EXPECT_EQ(0u, z.size());
123 EXPECT_EQ(nullptr, z.data());
124 // ArrayView<int> w = cv; // Compile error, because can't drop const.
125}
126
127TEST(ArrayViewTest, TestRtcBuffer) {
128 rtc::Buffer b = "so buffer";
129 Call<const uint8_t>(b);
130 Call<uint8_t>(b);
131 // Call<int8_t>(b); // Compile error, because incompatible types.
132 ArrayView<uint8_t> x = b;
133 EXPECT_EQ(10u, x.size());
134 EXPECT_EQ(b.data(), x.data());
135 ArrayView<const uint8_t> y;
136 y = b;
137 EXPECT_EQ(10u, y.size());
138 EXPECT_EQ(b.data(), y.data());
139 // ArrayView<char> d = b; // Compile error, because incompatible types.
140 const rtc::Buffer cb = "very const";
141 Call<const uint8_t>(cb);
142 // Call<uint8_t>(cb); // Compile error, because can't drop const.
143 ArrayView<const uint8_t> z = cb;
144 EXPECT_EQ(11u, z.size());
145 EXPECT_EQ(cb.data(), z.data());
146 // ArrayView<uint8_t> w = cb; // Compile error, because can't drop const.
147}
148
149TEST(ArrayViewTest, TestSwap) {
150 const char arr[] = "Arrr!";
151 const char aye[] = "Aye, Cap'n!";
152 ArrayView<const char> x(arr);
153 EXPECT_EQ(6u, x.size());
154 EXPECT_EQ(arr, x.data());
155 ArrayView<const char> y(aye);
156 EXPECT_EQ(12u, y.size());
157 EXPECT_EQ(aye, y.data());
158 using std::swap;
159 swap(x, y);
160 EXPECT_EQ(12u, x.size());
161 EXPECT_EQ(aye, x.data());
162 EXPECT_EQ(6u, y.size());
163 EXPECT_EQ(arr, y.data());
164 // ArrayView<char> z;
165 // swap(x, z); // Compile error, because can't drop const.
166}
167
168TEST(ArrayViewTest, TestIndexing) {
169 char arr[] = "abcdefg";
170 ArrayView<char> x(arr);
171 const ArrayView<char> y(arr);
172 ArrayView<const char> z(arr);
173 EXPECT_EQ(8u, x.size());
174 EXPECT_EQ(8u, y.size());
175 EXPECT_EQ(8u, z.size());
176 EXPECT_EQ('b', x[1]);
177 EXPECT_EQ('c', y[2]);
178 EXPECT_EQ('d', z[3]);
179 x[3] = 'X';
180 y[2] = 'Y';
181 // z[1] = 'Z'; // Compile error, because z's element type is const char.
182 EXPECT_EQ('b', x[1]);
183 EXPECT_EQ('Y', y[2]);
184 EXPECT_EQ('X', z[3]);
185#if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID)
186 EXPECT_DEATH(z[8], ""); // DCHECK error (index out of bounds).
187#endif
188}
189
190TEST(ArrayViewTest, TestIterationEmpty) {
191 ArrayView<std::vector<std::vector<std::vector<std::string>>>> av;
192 EXPECT_FALSE(av.begin());
193 EXPECT_FALSE(av.cbegin());
194 EXPECT_FALSE(av.end());
195 EXPECT_FALSE(av.cend());
196 for (auto& e : av) {
197 EXPECT_TRUE(false);
198 EXPECT_EQ(42u, e.size()); // Dummy use of e to prevent unused var warning.
199 }
200}
201
202TEST(ArrayViewTest, TestIteration) {
203 char arr[] = "Arrr!";
204 ArrayView<char> av(arr);
205 EXPECT_EQ('A', *av.begin());
206 EXPECT_EQ('A', *av.cbegin());
207 EXPECT_EQ('\0', *(av.end() - 1));
208 EXPECT_EQ('\0', *(av.cend() - 1));
209 char i = 0;
210 for (auto& e : av) {
211 EXPECT_EQ(arr + i, &e);
212 e = 's' + i;
213 ++i;
214 }
215 i = 0;
216 for (auto& e : ArrayView<const char>(av)) {
217 EXPECT_EQ(arr + i, &e);
218 // e = 'q' + i; // Compile error, because e is a const char&.
219 ++i;
220 }
221}
222
kwiberg288886b2015-11-06 01:21:35 -0800223TEST(ArrayViewTest, TestEmpty) {
224 EXPECT_TRUE(ArrayView<int>().empty());
225 const int a[] = {1, 2, 3};
226 EXPECT_FALSE(ArrayView<const int>(a).empty());
227}
228
229TEST(ArrayViewTest, TestCompare) {
230 int a[] = {1, 2, 3};
231 int b[] = {1, 2, 3};
232 EXPECT_EQ(ArrayView<int>(a), ArrayView<int>(a));
233 EXPECT_EQ(ArrayView<int>(), ArrayView<int>());
234 EXPECT_NE(ArrayView<int>(a), ArrayView<int>(b));
235 EXPECT_NE(ArrayView<int>(a), ArrayView<int>());
236 EXPECT_NE(ArrayView<int>(a), ArrayView<int>(a, 2));
237}
238
danilchapa28780e2016-11-18 01:46:23 -0800239TEST(ArrayViewTest, TestSubView) {
240 int a[] = {1, 2, 3};
241 ArrayView<int> av(a);
242
243 EXPECT_EQ(av.subview(0), av);
244
245 EXPECT_THAT(av.subview(1), ElementsAre(2, 3));
246 EXPECT_THAT(av.subview(2), ElementsAre(3));
247 EXPECT_THAT(av.subview(3), IsEmpty());
248 EXPECT_THAT(av.subview(4), IsEmpty());
249
250 EXPECT_THAT(av.subview(1, 0), IsEmpty());
251 EXPECT_THAT(av.subview(1, 1), ElementsAre(2));
252 EXPECT_THAT(av.subview(1, 2), ElementsAre(2, 3));
253 EXPECT_THAT(av.subview(1, 3), ElementsAre(2, 3));
254}
255
Karl Wiberge2a83ee2015-10-26 19:51:29 +0100256} // namespace rtc