|  | /* | 
|  | *  Copyright 2015 The WebRTC Project Authors. All rights reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  |  | 
|  | #include "api/array_view.h" | 
|  |  | 
|  | #include <stdint.h> | 
|  |  | 
|  | #include <array> | 
|  | #include <cstddef> | 
|  | #include <string> | 
|  | #include <utility> | 
|  | #include <vector> | 
|  |  | 
|  | #include "rtc_base/buffer.h" | 
|  | #include "rtc_base/checks.h" | 
|  | #include "test/gmock.h" | 
|  | #include "test/gtest.h" | 
|  |  | 
|  | namespace webrtc { | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | using ::testing::ElementsAre; | 
|  | using ::testing::IsEmpty; | 
|  |  | 
|  | template <typename T> | 
|  | size_t Call(ArrayView<T> av) { | 
|  | return av.size(); | 
|  | } | 
|  |  | 
|  | template <typename T, size_t N> | 
|  | void CallFixed(ArrayView<T, N> /* av */) {} | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | TEST(ArrayViewDeathTest, TestConstructFromPtrAndArray) { | 
|  | char arr[] = "Arrr!"; | 
|  | const char carr[] = "Carrr!"; | 
|  | EXPECT_EQ(6u, Call<const char>(arr)); | 
|  | EXPECT_EQ(7u, Call<const char>(carr)); | 
|  | EXPECT_EQ(6u, Call<char>(arr)); | 
|  | // Call<char>(carr);  // Compile error, because can't drop const. | 
|  | // Call<int>(arr);  // Compile error, because incompatible types. | 
|  | ArrayView<int*> x; | 
|  | EXPECT_EQ(0u, x.size()); | 
|  | EXPECT_EQ(nullptr, x.data()); | 
|  | ArrayView<char> y = arr; | 
|  | EXPECT_EQ(6u, y.size()); | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | ArrayView<char, 6> yf = arr; | 
|  | static_assert(yf.size() == 6, ""); | 
|  | EXPECT_EQ(arr, yf.data()); | 
|  | ArrayView<const char> z(arr + 1, 3); | 
|  | EXPECT_EQ(3u, z.size()); | 
|  | EXPECT_EQ(arr + 1, z.data()); | 
|  | ArrayView<const char, 3> zf(arr + 1, 3); | 
|  | static_assert(zf.size() == 3, ""); | 
|  | EXPECT_EQ(arr + 1, zf.data()); | 
|  | ArrayView<const char> w(arr, 2); | 
|  | EXPECT_EQ(2u, w.size()); | 
|  | EXPECT_EQ(arr, w.data()); | 
|  | ArrayView<const char, 2> wf(arr, 2); | 
|  | static_assert(wf.size() == 2, ""); | 
|  | EXPECT_EQ(arr, wf.data()); | 
|  | ArrayView<char> q(arr, 0); | 
|  | EXPECT_EQ(0u, q.size()); | 
|  | EXPECT_EQ(nullptr, q.data()); | 
|  | ArrayView<char, 0> qf(arr, 0); | 
|  | static_assert(qf.size() == 0, ""); | 
|  | EXPECT_EQ(nullptr, qf.data()); | 
|  | #if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID) | 
|  | // DCHECK error (nullptr with nonzero size). | 
|  | EXPECT_DEATH(ArrayView<int>(static_cast<int*>(nullptr), 5), ""); | 
|  | #endif | 
|  | // These are compile errors, because incompatible types. | 
|  | // ArrayView<int> m = arr; | 
|  | // ArrayView<float> n(arr + 2, 2); | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestCopyConstructorVariableLvalue) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char> x = arr; | 
|  | EXPECT_EQ(6u, x.size()); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  | ArrayView<char> y = x;  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(6u, y.size()); | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | ArrayView<const char> z = x;  // Copy non-const -> const. | 
|  | EXPECT_EQ(6u, z.size()); | 
|  | EXPECT_EQ(arr, z.data()); | 
|  | ArrayView<const char> w = z;  // Copy const -> const. | 
|  | EXPECT_EQ(6u, w.size()); | 
|  | EXPECT_EQ(arr, w.data()); | 
|  | // ArrayView<char> v = z;  // Compile error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestCopyConstructorVariableRvalue) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char> x = arr; | 
|  | EXPECT_EQ(6u, x.size()); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  | ArrayView<char> y = std::move(x);  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(6u, y.size()); | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | ArrayView<const char> z = std::move(x);  // Copy non-const -> const. | 
|  | EXPECT_EQ(6u, z.size()); | 
|  | EXPECT_EQ(arr, z.data()); | 
|  | ArrayView<const char> w = std::move(z);  // Copy const -> const. | 
|  | EXPECT_EQ(6u, w.size()); | 
|  | EXPECT_EQ(arr, w.data()); | 
|  | // ArrayView<char> v = std::move(z);  // Error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestCopyConstructorFixedLvalue) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char, 6> x = arr; | 
|  | static_assert(x.size() == 6, ""); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  |  | 
|  | // Copy fixed -> fixed. | 
|  | ArrayView<char, 6> y = x;  // Copy non-const -> non-const. | 
|  | static_assert(y.size() == 6, ""); | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | ArrayView<const char, 6> z = x;  // Copy non-const -> const. | 
|  | static_assert(z.size() == 6, ""); | 
|  | EXPECT_EQ(arr, z.data()); | 
|  | ArrayView<const char, 6> w = z;  // Copy const -> const. | 
|  | static_assert(w.size() == 6, ""); | 
|  | EXPECT_EQ(arr, w.data()); | 
|  | // ArrayView<char, 6> v = z;  // Compile error, because can't drop const. | 
|  |  | 
|  | // Copy fixed -> variable. | 
|  | ArrayView<char> yv = x;  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(6u, yv.size()); | 
|  | EXPECT_EQ(arr, yv.data()); | 
|  | ArrayView<const char> zv = x;  // Copy non-const -> const. | 
|  | EXPECT_EQ(6u, zv.size()); | 
|  | EXPECT_EQ(arr, zv.data()); | 
|  | ArrayView<const char> wv = z;  // Copy const -> const. | 
|  | EXPECT_EQ(6u, wv.size()); | 
|  | EXPECT_EQ(arr, wv.data()); | 
|  | // ArrayView<char> vv = z;  // Compile error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestCopyConstructorFixedRvalue) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char, 6> x = arr; | 
|  | static_assert(x.size() == 6, ""); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  |  | 
|  | // Copy fixed -> fixed. | 
|  | ArrayView<char, 6> y = std::move(x);  // Copy non-const -> non-const. | 
|  | static_assert(y.size() == 6, ""); | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | ArrayView<const char, 6> z = std::move(x);  // Copy non-const -> const. | 
|  | static_assert(z.size() == 6, ""); | 
|  | EXPECT_EQ(arr, z.data()); | 
|  | ArrayView<const char, 6> w = std::move(z);  // Copy const -> const. | 
|  | static_assert(w.size() == 6, ""); | 
|  | EXPECT_EQ(arr, w.data()); | 
|  | // ArrayView<char, 6> v = std::move(z);  // Error, because can't drop const. | 
|  |  | 
|  | // Copy fixed -> variable. | 
|  | ArrayView<char> yv = std::move(x);  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(6u, yv.size()); | 
|  | EXPECT_EQ(arr, yv.data()); | 
|  | ArrayView<const char> zv = std::move(x);  // Copy non-const -> const. | 
|  | EXPECT_EQ(6u, zv.size()); | 
|  | EXPECT_EQ(arr, zv.data()); | 
|  | ArrayView<const char> wv = std::move(z);  // Copy const -> const. | 
|  | EXPECT_EQ(6u, wv.size()); | 
|  | EXPECT_EQ(arr, wv.data()); | 
|  | // ArrayView<char> vv = std::move(z);  // Error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestCopyAssignmentVariableLvalue) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char> x(arr); | 
|  | EXPECT_EQ(6u, x.size()); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  | ArrayView<char> y; | 
|  | y = x;  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(6u, y.size()); | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | ArrayView<const char> z; | 
|  | z = x;  // Copy non-const -> const. | 
|  | EXPECT_EQ(6u, z.size()); | 
|  | EXPECT_EQ(arr, z.data()); | 
|  | ArrayView<const char> w; | 
|  | w = z;  // Copy const -> const. | 
|  | EXPECT_EQ(6u, w.size()); | 
|  | EXPECT_EQ(arr, w.data()); | 
|  | // ArrayView<char> v; | 
|  | // v = z;  // Compile error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestCopyAssignmentVariableRvalue) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char> x(arr); | 
|  | EXPECT_EQ(6u, x.size()); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  | ArrayView<char> y; | 
|  | y = std::move(x);  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(6u, y.size()); | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | ArrayView<const char> z; | 
|  | z = std::move(x);  // Copy non-const -> const. | 
|  | EXPECT_EQ(6u, z.size()); | 
|  | EXPECT_EQ(arr, z.data()); | 
|  | ArrayView<const char> w; | 
|  | w = std::move(z);  // Copy const -> const. | 
|  | EXPECT_EQ(6u, w.size()); | 
|  | EXPECT_EQ(arr, w.data()); | 
|  | // ArrayView<char> v; | 
|  | // v = std::move(z);  // Compile error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestCopyAssignmentFixedLvalue) { | 
|  | char arr[] = "Arrr!"; | 
|  | char init[] = "Init!"; | 
|  | ArrayView<char, 6> x(arr); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  |  | 
|  | // Copy fixed -> fixed. | 
|  | ArrayView<char, 6> y(init); | 
|  | y = x;  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | ArrayView<const char, 6> z(init); | 
|  | z = x;  // Copy non-const -> const. | 
|  | EXPECT_EQ(arr, z.data()); | 
|  | ArrayView<const char, 6> w(init); | 
|  | w = z;  // Copy const -> const. | 
|  | EXPECT_EQ(arr, w.data()); | 
|  | // ArrayView<char, 6> v(init); | 
|  | // v = z;  // Compile error, because can't drop const. | 
|  |  | 
|  | // Copy fixed -> variable. | 
|  | ArrayView<char> yv; | 
|  | yv = x;  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(6u, yv.size()); | 
|  | EXPECT_EQ(arr, yv.data()); | 
|  | ArrayView<const char> zv; | 
|  | zv = x;  // Copy non-const -> const. | 
|  | EXPECT_EQ(6u, zv.size()); | 
|  | EXPECT_EQ(arr, zv.data()); | 
|  | ArrayView<const char> wv; | 
|  | wv = z;  // Copy const -> const. | 
|  | EXPECT_EQ(6u, wv.size()); | 
|  | EXPECT_EQ(arr, wv.data()); | 
|  | // ArrayView<char> v; | 
|  | // v = z;  // Compile error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestCopyAssignmentFixedRvalue) { | 
|  | char arr[] = "Arrr!"; | 
|  | char init[] = "Init!"; | 
|  | ArrayView<char, 6> x(arr); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  |  | 
|  | // Copy fixed -> fixed. | 
|  | ArrayView<char, 6> y(init); | 
|  | y = std::move(x);  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | ArrayView<const char, 6> z(init); | 
|  | z = std::move(x);  // Copy non-const -> const. | 
|  | EXPECT_EQ(arr, z.data()); | 
|  | ArrayView<const char, 6> w(init); | 
|  | w = std::move(z);  // Copy const -> const. | 
|  | EXPECT_EQ(arr, w.data()); | 
|  | // ArrayView<char, 6> v(init); | 
|  | // v = std::move(z);  // Compile error, because can't drop const. | 
|  |  | 
|  | // Copy fixed -> variable. | 
|  | ArrayView<char> yv; | 
|  | yv = std::move(x);  // Copy non-const -> non-const. | 
|  | EXPECT_EQ(6u, yv.size()); | 
|  | EXPECT_EQ(arr, yv.data()); | 
|  | ArrayView<const char> zv; | 
|  | zv = std::move(x);  // Copy non-const -> const. | 
|  | EXPECT_EQ(6u, zv.size()); | 
|  | EXPECT_EQ(arr, zv.data()); | 
|  | ArrayView<const char> wv; | 
|  | wv = std::move(z);  // Copy const -> const. | 
|  | EXPECT_EQ(6u, wv.size()); | 
|  | EXPECT_EQ(arr, wv.data()); | 
|  | // ArrayView<char> v; | 
|  | // v = std::move(z);  // Compile error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestStdArray) { | 
|  | EXPECT_EQ(4u, Call<const int>(std::array<int, 4>{1, 2, 3, 4})); | 
|  | CallFixed<const int, 3>(std::array<int, 3>{2, 3, 4}); | 
|  | constexpr size_t size = 5; | 
|  | std::array<float, size> arr{}; | 
|  | // Fixed size view. | 
|  | ArrayView<float, size> arr_view_fixed(arr); | 
|  | EXPECT_EQ(arr.data(), arr_view_fixed.data()); | 
|  | static_assert(size == arr_view_fixed.size(), ""); | 
|  | // Variable size view. | 
|  | ArrayView<float> arr_view(arr); | 
|  | EXPECT_EQ(arr.data(), arr_view.data()); | 
|  | EXPECT_EQ(size, arr_view.size()); | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestConstStdArray) { | 
|  | constexpr size_t size = 5; | 
|  |  | 
|  | constexpr std::array<float, size> constexpr_arr{}; | 
|  | ArrayView<const float, size> constexpr_arr_view(constexpr_arr); | 
|  | EXPECT_EQ(constexpr_arr.data(), constexpr_arr_view.data()); | 
|  | static_assert(constexpr_arr.size() == constexpr_arr_view.size(), ""); | 
|  |  | 
|  | const std::array<float, size> const_arr{}; | 
|  | ArrayView<const float, size> const_arr_view(const_arr); | 
|  | EXPECT_EQ(const_arr.data(), const_arr_view.data()); | 
|  | static_assert(const_arr.size() == const_arr_view.size(), ""); | 
|  |  | 
|  | std::array<float, size> non_const_arr{}; | 
|  | ArrayView<const float, size> non_const_arr_view(non_const_arr); | 
|  | EXPECT_EQ(non_const_arr.data(), non_const_arr_view.data()); | 
|  | static_assert(non_const_arr.size() == non_const_arr_view.size(), ""); | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestStdVector) { | 
|  | EXPECT_EQ(3u, Call<const int>(std::vector<int>{4, 5, 6})); | 
|  | std::vector<int> v; | 
|  | v.push_back(3); | 
|  | v.push_back(11); | 
|  | EXPECT_EQ(2u, Call<const int>(v)); | 
|  | EXPECT_EQ(2u, Call<int>(v)); | 
|  | // Call<unsigned int>(v);  // Compile error, because incompatible types. | 
|  | ArrayView<int> x = v; | 
|  | EXPECT_EQ(2u, x.size()); | 
|  | EXPECT_EQ(v.data(), x.data()); | 
|  | ArrayView<const int> y; | 
|  | y = v; | 
|  | EXPECT_EQ(2u, y.size()); | 
|  | EXPECT_EQ(v.data(), y.data()); | 
|  | // ArrayView<double> d = v;  // Compile error, because incompatible types. | 
|  | const std::vector<int> cv; | 
|  | EXPECT_EQ(0u, Call<const int>(cv)); | 
|  | // Call<int>(cv);  // Compile error, because can't drop const. | 
|  | ArrayView<const int> z = cv; | 
|  | EXPECT_EQ(0u, z.size()); | 
|  | EXPECT_EQ(nullptr, z.data()); | 
|  | // ArrayView<int> w = cv;  // Compile error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestRtcBuffer) { | 
|  | Buffer b = "so buffer"; | 
|  | EXPECT_EQ(10u, Call<const uint8_t>(b)); | 
|  | EXPECT_EQ(10u, Call<uint8_t>(b)); | 
|  | // Call<int8_t>(b);  // Compile error, because incompatible types. | 
|  | ArrayView<uint8_t> x = b; | 
|  | EXPECT_EQ(10u, x.size()); | 
|  | EXPECT_EQ(b.data(), x.data()); | 
|  | ArrayView<const uint8_t> y; | 
|  | y = b; | 
|  | EXPECT_EQ(10u, y.size()); | 
|  | EXPECT_EQ(b.data(), y.data()); | 
|  | // ArrayView<char> d = b;  // Compile error, because incompatible types. | 
|  | const Buffer cb = "very const"; | 
|  | EXPECT_EQ(11u, Call<const uint8_t>(cb)); | 
|  | // Call<uint8_t>(cb);  // Compile error, because can't drop const. | 
|  | ArrayView<const uint8_t> z = cb; | 
|  | EXPECT_EQ(11u, z.size()); | 
|  | EXPECT_EQ(cb.data(), z.data()); | 
|  | // ArrayView<uint8_t> w = cb;  // Compile error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestSwapVariable) { | 
|  | const char arr[] = "Arrr!"; | 
|  | const char aye[] = "Aye, Cap'n!"; | 
|  | ArrayView<const char> x(arr); | 
|  | EXPECT_EQ(6u, x.size()); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  | ArrayView<const char> y(aye); | 
|  | EXPECT_EQ(12u, y.size()); | 
|  | EXPECT_EQ(aye, y.data()); | 
|  | using std::swap; | 
|  | swap(x, y); | 
|  | EXPECT_EQ(12u, x.size()); | 
|  | EXPECT_EQ(aye, x.data()); | 
|  | EXPECT_EQ(6u, y.size()); | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | // ArrayView<char> z; | 
|  | // swap(x, z);  // Compile error, because can't drop const. | 
|  | } | 
|  |  | 
|  | TEST(FixArrayViewTest, TestSwapFixed) { | 
|  | const char arr[] = "Arr!"; | 
|  | char aye[] = "Aye!"; | 
|  | ArrayView<const char, 5> x(arr); | 
|  | EXPECT_EQ(arr, x.data()); | 
|  | ArrayView<const char, 5> y(aye); | 
|  | EXPECT_EQ(aye, y.data()); | 
|  | using std::swap; | 
|  | swap(x, y); | 
|  | EXPECT_EQ(aye, x.data()); | 
|  | EXPECT_EQ(arr, y.data()); | 
|  | // ArrayView<char, 5> z(aye); | 
|  | // swap(x, z);  // Compile error, because can't drop const. | 
|  | // ArrayView<const char, 4> w(aye, 4); | 
|  | // swap(x, w);  // Compile error, because different sizes. | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewDeathTest, TestIndexing) { | 
|  | char arr[] = "abcdefg"; | 
|  | ArrayView<char> x(arr); | 
|  | const ArrayView<char> y(arr); | 
|  | ArrayView<const char, 8> z(arr); | 
|  | EXPECT_EQ(8u, x.size()); | 
|  | EXPECT_EQ(8u, y.size()); | 
|  | EXPECT_EQ(8u, z.size()); | 
|  | EXPECT_EQ('b', x[1]); | 
|  | EXPECT_EQ('c', y[2]); | 
|  | EXPECT_EQ('d', z[3]); | 
|  | x[3] = 'X'; | 
|  | y[2] = 'Y'; | 
|  | // z[1] = 'Z';  // Compile error, because z's element type is const char. | 
|  | EXPECT_EQ('b', x[1]); | 
|  | EXPECT_EQ('Y', y[2]); | 
|  | EXPECT_EQ('X', z[3]); | 
|  | #if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID) | 
|  | EXPECT_DEATH(z[8], "");  // DCHECK error (index out of bounds). | 
|  | #endif | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestIterationEmpty) { | 
|  | // Variable-size. | 
|  | ArrayView<std::vector<std::vector<std::vector<std::string>>>> av; | 
|  | EXPECT_EQ(av.begin(), av.end()); | 
|  | EXPECT_EQ(av.cbegin(), av.cend()); | 
|  | for (auto& e : av) { | 
|  | EXPECT_TRUE(false); | 
|  | EXPECT_EQ(42u, e.size());  // Dummy use of e to prevent unused var warning. | 
|  | } | 
|  |  | 
|  | // Fixed-size. | 
|  | ArrayView<std::vector<std::vector<std::vector<std::string>>>, 0> af; | 
|  | EXPECT_EQ(af.begin(), af.end()); | 
|  | EXPECT_EQ(af.cbegin(), af.cend()); | 
|  | for (auto& e : af) { | 
|  | EXPECT_TRUE(false); | 
|  | EXPECT_EQ(42u, e.size());  // Dummy use of e to prevent unused var warning. | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestReverseIterationEmpty) { | 
|  | // Variable-size. | 
|  | ArrayView<std::vector<std::vector<std::vector<std::string>>>> av; | 
|  | EXPECT_EQ(av.rbegin(), av.rend()); | 
|  | EXPECT_EQ(av.crbegin(), av.crend()); | 
|  | EXPECT_TRUE(av.empty()); | 
|  |  | 
|  | // Fixed-size. | 
|  | ArrayView<std::vector<std::vector<std::vector<std::string>>>, 0> af; | 
|  | EXPECT_EQ(af.begin(), af.end()); | 
|  | EXPECT_EQ(af.cbegin(), af.cend()); | 
|  | EXPECT_TRUE(af.empty()); | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestIterationVariable) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char> av(arr); | 
|  | EXPECT_EQ('A', *av.begin()); | 
|  | EXPECT_EQ('A', *av.cbegin()); | 
|  | EXPECT_EQ('\0', *(av.end() - 1)); | 
|  | EXPECT_EQ('\0', *(av.cend() - 1)); | 
|  | char i = 0; | 
|  | for (auto& e : av) { | 
|  | EXPECT_EQ(arr + i, &e); | 
|  | e = 's' + i; | 
|  | ++i; | 
|  | } | 
|  | i = 0; | 
|  | for (auto& e : ArrayView<const char>(av)) { | 
|  | EXPECT_EQ(arr + i, &e); | 
|  | // e = 'q' + i;  // Compile error, because e is a const char&. | 
|  | ++i; | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestReverseIterationVariable) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char> av(arr); | 
|  | EXPECT_EQ('\0', *av.rbegin()); | 
|  | EXPECT_EQ('\0', *av.crbegin()); | 
|  | EXPECT_EQ('A', *(av.rend() - 1)); | 
|  | EXPECT_EQ('A', *(av.crend() - 1)); | 
|  |  | 
|  | const char* cit = av.cend() - 1; | 
|  | for (auto crit = av.crbegin(); crit != av.crend(); ++crit, --cit) { | 
|  | EXPECT_EQ(*cit, *crit); | 
|  | } | 
|  |  | 
|  | char* it = av.end() - 1; | 
|  | for (auto rit = av.rbegin(); rit != av.rend(); ++rit, --it) { | 
|  | EXPECT_EQ(*it, *rit); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestIterationFixed) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char, 6> av(arr); | 
|  | EXPECT_EQ('A', *av.begin()); | 
|  | EXPECT_EQ('A', *av.cbegin()); | 
|  | EXPECT_EQ('\0', *(av.end() - 1)); | 
|  | EXPECT_EQ('\0', *(av.cend() - 1)); | 
|  | char i = 0; | 
|  | for (auto& e : av) { | 
|  | EXPECT_EQ(arr + i, &e); | 
|  | e = 's' + i; | 
|  | ++i; | 
|  | } | 
|  | i = 0; | 
|  | for (auto& e : ArrayView<const char, 6>(av)) { | 
|  | EXPECT_EQ(arr + i, &e); | 
|  | // e = 'q' + i;  // Compile error, because e is a const char&. | 
|  | ++i; | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestReverseIterationFixed) { | 
|  | char arr[] = "Arrr!"; | 
|  | ArrayView<char, 6> av(arr); | 
|  | EXPECT_EQ('\0', *av.rbegin()); | 
|  | EXPECT_EQ('\0', *av.crbegin()); | 
|  | EXPECT_EQ('A', *(av.rend() - 1)); | 
|  | EXPECT_EQ('A', *(av.crend() - 1)); | 
|  |  | 
|  | const char* cit = av.cend() - 1; | 
|  | for (auto crit = av.crbegin(); crit != av.crend(); ++crit, --cit) { | 
|  | EXPECT_EQ(*cit, *crit); | 
|  | } | 
|  |  | 
|  | char* it = av.end() - 1; | 
|  | for (auto rit = av.rbegin(); rit != av.rend(); ++rit, --it) { | 
|  | EXPECT_EQ(*it, *rit); | 
|  | } | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestEmpty) { | 
|  | EXPECT_TRUE(ArrayView<int>().empty()); | 
|  | const int a[] = {1, 2, 3}; | 
|  | EXPECT_FALSE(ArrayView<const int>(a).empty()); | 
|  |  | 
|  | static_assert(ArrayView<int, 0>::empty(), ""); | 
|  | static_assert(!ArrayView<int, 3>::empty(), ""); | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestCompare) { | 
|  | int a[] = {1, 2, 3}; | 
|  | int b[] = {1, 2, 3}; | 
|  |  | 
|  | EXPECT_EQ(ArrayView<int>(a), ArrayView<int>(a)); | 
|  | EXPECT_EQ((ArrayView<int, 3>(a)), (ArrayView<int, 3>(a))); | 
|  | EXPECT_EQ(ArrayView<int>(a), (ArrayView<int, 3>(a))); | 
|  | EXPECT_EQ(ArrayView<int>(), ArrayView<int>()); | 
|  | EXPECT_EQ(ArrayView<int>(), ArrayView<int>(a, 0)); | 
|  | EXPECT_EQ(ArrayView<int>(a, 0), ArrayView<int>(b, 0)); | 
|  | EXPECT_EQ((ArrayView<int, 0>(a, 0)), ArrayView<int>()); | 
|  |  | 
|  | EXPECT_NE(ArrayView<int>(a), ArrayView<int>(b)); | 
|  | EXPECT_NE((ArrayView<int, 3>(a)), (ArrayView<int, 3>(b))); | 
|  | EXPECT_NE((ArrayView<int, 3>(a)), ArrayView<int>(b)); | 
|  | EXPECT_NE(ArrayView<int>(a), ArrayView<int>()); | 
|  | EXPECT_NE(ArrayView<int>(a), ArrayView<int>(a, 2)); | 
|  | EXPECT_NE((ArrayView<int, 3>(a)), (ArrayView<int, 2>(a, 2))); | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestSubViewVariable) { | 
|  | int a[] = {1, 2, 3}; | 
|  | ArrayView<int> av(a); | 
|  |  | 
|  | EXPECT_EQ(av.subview(0), av); | 
|  |  | 
|  | EXPECT_THAT(av.subview(1), ElementsAre(2, 3)); | 
|  | EXPECT_THAT(av.subview(2), ElementsAre(3)); | 
|  | EXPECT_THAT(av.subview(3), IsEmpty()); | 
|  | EXPECT_THAT(av.subview(4), IsEmpty()); | 
|  |  | 
|  | EXPECT_THAT(av.subview(1, 0), IsEmpty()); | 
|  | EXPECT_THAT(av.subview(1, 1), ElementsAre(2)); | 
|  | EXPECT_THAT(av.subview(1, 2), ElementsAre(2, 3)); | 
|  | EXPECT_THAT(av.subview(1, 3), ElementsAre(2, 3)); | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestSubViewFixed) { | 
|  | int a[] = {1, 2, 3}; | 
|  | ArrayView<int, 3> av(a); | 
|  |  | 
|  | EXPECT_EQ(av.subview(0), av); | 
|  |  | 
|  | EXPECT_THAT(av.subview(1), ElementsAre(2, 3)); | 
|  | EXPECT_THAT(av.subview(2), ElementsAre(3)); | 
|  | EXPECT_THAT(av.subview(3), IsEmpty()); | 
|  | EXPECT_THAT(av.subview(4), IsEmpty()); | 
|  |  | 
|  | EXPECT_THAT(av.subview(1, 0), IsEmpty()); | 
|  | EXPECT_THAT(av.subview(1, 1), ElementsAre(2)); | 
|  | EXPECT_THAT(av.subview(1, 2), ElementsAre(2, 3)); | 
|  | EXPECT_THAT(av.subview(1, 3), ElementsAre(2, 3)); | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestReinterpretCastFixedSize) { | 
|  | uint8_t bytes[] = {1, 2, 3}; | 
|  | ArrayView<uint8_t, 3> uint8_av(bytes); | 
|  | ArrayView<int8_t, 3> int8_av = reinterpret_array_view<int8_t>(uint8_av); | 
|  | EXPECT_EQ(int8_av.size(), uint8_av.size()); | 
|  | EXPECT_EQ(int8_av[0], 1); | 
|  | EXPECT_EQ(int8_av[1], 2); | 
|  | EXPECT_EQ(int8_av[2], 3); | 
|  | } | 
|  |  | 
|  | TEST(ArrayViewTest, TestReinterpretCastVariableSize) { | 
|  | std::vector<int8_t> v = {1, 2, 3}; | 
|  | ArrayView<int8_t> int8_av(v); | 
|  | ArrayView<uint8_t> uint8_av = reinterpret_array_view<uint8_t>(int8_av); | 
|  | EXPECT_EQ(int8_av.size(), uint8_av.size()); | 
|  | EXPECT_EQ(uint8_av[0], 1); | 
|  | EXPECT_EQ(uint8_av[1], 2); | 
|  | EXPECT_EQ(uint8_av[2], 3); | 
|  | } | 
|  | }  // namespace webrtc |