/* * Copyright (C) 2015 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include <gtest/gtest.h> #include <utils/FatVector.h> #include <tests/common/TestUtils.h> using namespace android; using namespace android::uirenderer; template <class VectorType> static bool allocationIsInternal(VectorType& v) { // allocation array (from &v[0] to &v[0] + v.capacity) is // located within the vector object itself return (char*)(&v) <= (char*)(&v[0]) && (char*)(&v + 1) >= (char*)(&v[0] + v.capacity()); } TEST(FatVector, baseline) { // Verify allocation behavior FatVector contrasts against - allocations are always external std::vector<int> v; for (int i = 0; i < 50; i++) { v.push_back(i); EXPECT_FALSE(allocationIsInternal(v)); } } TEST(FatVector, simpleAllocate) { FatVector<int, 4> v; EXPECT_EQ(4u, v.capacity()); // can insert 4 items into internal buffer for (int i = 0; i < 4; i++) { v.push_back(i); EXPECT_TRUE(allocationIsInternal(v)); } // then will fall back to external allocation for (int i = 5; i < 50; i++) { v.push_back(i); EXPECT_FALSE(allocationIsInternal(v)); } } TEST(FatVector, preSizeConstructor) { { FatVector<int, 4> v(32); EXPECT_EQ(32u, v.capacity()); EXPECT_EQ(32u, v.size()); EXPECT_FALSE(allocationIsInternal(v)); } { FatVector<int, 4> v(4); EXPECT_EQ(4u, v.capacity()); EXPECT_EQ(4u, v.size()); EXPECT_TRUE(allocationIsInternal(v)); } { FatVector<int, 4> v(2); EXPECT_EQ(4u, v.capacity()); EXPECT_EQ(2u, v.size()); EXPECT_TRUE(allocationIsInternal(v)); } } TEST(FatVector, shrink) { FatVector<int, 10> v; EXPECT_TRUE(allocationIsInternal(v)); // push into external alloc v.resize(11); EXPECT_FALSE(allocationIsInternal(v)); // shrinking back to internal alloc succeeds // note that shrinking further will succeed, but is a waste v.resize(10); v.shrink_to_fit(); EXPECT_TRUE(allocationIsInternal(v)); } TEST(FatVector, destructorInternal) { int count = 0; { // push 1 into external allocation, verify destruction happens once FatVector<TestUtils::SignalingDtor, 0> v; v.emplace_back(&count); EXPECT_FALSE(allocationIsInternal(v)); EXPECT_EQ(0, count) << "Destruction shouldn't have happened yet"; } EXPECT_EQ(1, count) << "Destruction should happen exactly once"; } TEST(FatVector, destructorExternal) { int count = 0; { // push 10 into internal allocation, verify 10 destructors called FatVector<TestUtils::SignalingDtor, 10> v; for (int i = 0; i < 10; i++) { v.emplace_back(&count); EXPECT_TRUE(allocationIsInternal(v)); } EXPECT_EQ(0, count) << "Destruction shouldn't have happened yet"; } EXPECT_EQ(10, count) << "Destruction should happen exactly once"; }