Karl Wiberg | e2a83ee | 2015-10-26 19:51:29 +0100 | [diff] [blame] | 1 | /* |
| 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" |
danilchap | a28780e | 2016-11-18 01:46:23 -0800 | [diff] [blame^] | 19 | #include "webrtc/test/gmock.h" |
Karl Wiberg | e2a83ee | 2015-10-26 19:51:29 +0100 | [diff] [blame] | 20 | |
| 21 | namespace rtc { |
| 22 | |
| 23 | namespace { |
kwiberg | bd43172 | 2016-09-05 04:20:54 -0700 | [diff] [blame] | 24 | |
danilchap | a28780e | 2016-11-18 01:46:23 -0800 | [diff] [blame^] | 25 | using ::testing::ElementsAre; |
| 26 | using ::testing::IsEmpty; |
| 27 | |
Karl Wiberg | e2a83ee | 2015-10-26 19:51:29 +0100 | [diff] [blame] | 28 | template <typename T> |
| 29 | void Call(ArrayView<T>) {} |
kwiberg | bd43172 | 2016-09-05 04:20:54 -0700 | [diff] [blame] | 30 | |
Karl Wiberg | e2a83ee | 2015-10-26 19:51:29 +0100 | [diff] [blame] | 31 | } // namespace |
| 32 | |
| 33 | TEST(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 | |
| 65 | TEST(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 | |
| 82 | TEST(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 | |
| 103 | TEST(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 | |
| 127 | TEST(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 | |
| 149 | TEST(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 | |
| 168 | TEST(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 | |
| 190 | TEST(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 | |
| 202 | TEST(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 | |
kwiberg | 288886b | 2015-11-06 01:21:35 -0800 | [diff] [blame] | 223 | TEST(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 | |
| 229 | TEST(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 | |
danilchap | a28780e | 2016-11-18 01:46:23 -0800 | [diff] [blame^] | 239 | TEST(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 Wiberg | e2a83ee | 2015-10-26 19:51:29 +0100 | [diff] [blame] | 256 | } // namespace rtc |