/* * Copyright (C) 2016 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. */ #define LOG_TAG "hidl-cache-test" #include <android/hidl/allocator/1.0/IAllocator.h> #include <android/hidl/memory/1.0/IMemory.h> #include <android/hidl/memory/token/1.0/IMemoryToken.h> #include <gtest/gtest.h> #include <hidlcache/MemoryDealer.h> #include <hidlcache/mapping.h> #include <hidlmemory/HidlMemoryToken.h> #include <hidlmemory/mapping.h> #include "HidlMemoryCache.h" #define EXPECT_OK(__ret__) EXPECT_TRUE(isOk(__ret__)) template <typename T> static inline ::testing::AssertionResult isOk(const ::android::hardware::Return<T>& ret) { return ret.isOk() ? (::testing::AssertionSuccess() << ret.description()) : (::testing::AssertionFailure() << ret.description()); } namespace android { namespace hardware { void HidlCacheWhiteBoxTest() { using ::android::hardware::HidlMemoryCache; using ::android::hardware::HidlMemoryToken; using ::android::hidl::allocator::V1_0::IAllocator; using ::android::hidl::memory::V1_0::IMemory; using ::android::hidl::memory::token::V1_0::IMemoryToken; using ::android::hidl::memory::block::V1_0::MemoryBlock; sp<IAllocator> ashmemAllocator; ashmemAllocator = IAllocator::getService("ashmem"); ASSERT_NE(nullptr, ashmemAllocator.get()); ASSERT_TRUE(ashmemAllocator->isRemote()); // allocator is always remote sp<HidlMemory> mem; EXPECT_OK(ashmemAllocator->allocate(1024, [&](bool success, const hidl_memory& _mem) { ASSERT_TRUE(success); mem = HidlMemory::getInstance(_mem); })); sp<IMemoryToken> token = new HidlMemoryToken(mem); MemoryBlock blk = {token, 0x200 /* size */, 0x100 /* offset */}; sp<IMemoryToken> mtoken = blk.token; mtoken->get([&](const hidl_memory& mem) { sp<IMemory> memory = mapMemory(mem); }); sp<HidlMemoryCache> cache = HidlMemoryCache::getInstance(); EXPECT_FALSE(cache->cached(token)); MemoryBlock blk2 = {token, 0x200 /* size */, 0x300 /* offset */}; EXPECT_FALSE(cache->cached(token)); { sp<IMemory> mem1 = cache->fetch(token); EXPECT_TRUE(cache->cached(token)); EXPECT_NE(nullptr, cache->getCachedLocked(token).get()); sp<IMemory> mem2 = cache->fetch(token); EXPECT_TRUE(cache->cached(token)); EXPECT_NE(nullptr, cache->getCachedLocked(token).get()); } EXPECT_FALSE(cache->cached(token)); { sp<IMemory> mem1 = mapMemory(blk); EXPECT_TRUE(cache->cached(token)); EXPECT_NE(nullptr, cache->getCachedLocked(token).get()); uint8_t* data = static_cast<uint8_t*>(static_cast<void*>(mem1->getPointer())); EXPECT_NE(nullptr, data); } { sp<IMemory> mem2 = mapMemory(blk); EXPECT_TRUE(cache->cached(token)); EXPECT_NE(nullptr, cache->getCachedLocked(token).get()); } EXPECT_FALSE(cache->cached(token)); EXPECT_TRUE(cache->lock(token)); EXPECT_TRUE(cache->cached(token)); EXPECT_NE(nullptr, cache->getCachedLocked(token).get()); EXPECT_TRUE(cache->unlock(token)); EXPECT_FALSE(cache->cached(token)); } } // namespace hardware class HidlCacheTest : public ::testing::Test {}; TEST_F(HidlCacheTest, TestAll) { hardware::HidlCacheWhiteBoxTest(); } TEST_F(HidlCacheTest, MemoryDealer) { using ::android::hardware::HidlMemory; using ::android::hardware::hidl_memory; using ::android::hardware::HidlMemoryDealer; using ::android::hidl::allocator::V1_0::IAllocator; using ::android::hidl::memory::block::V1_0::MemoryBlock; sp<IAllocator> ashmemAllocator; ashmemAllocator = IAllocator::getService("ashmem"); sp<HidlMemory> m1; sp<HidlMemory> m2; // test MemoryDealer EXPECT_OK(ashmemAllocator->allocate(2048, [&m1](bool success, const hidl_memory& mem) { ASSERT_TRUE(success); m1 = HidlMemory::getInstance(mem); })); EXPECT_OK(ashmemAllocator->allocate(4096, [&m2](bool success, const hidl_memory& mem) { ASSERT_TRUE(success); m2 = HidlMemory::getInstance(mem); })); sp<HidlMemoryDealer> dealer; // m1 does not statisfy the alignment requirement and should fail. dealer = HidlMemoryDealer::getInstance(*m1); EXPECT_EQ(nullptr, dealer.get()); dealer = HidlMemoryDealer::getInstance(*m2); EXPECT_NE(nullptr, dealer.get()); EXPECT_EQ(dealer->heap()->getSize(), 4096ULL); MemoryBlock blk = dealer->allocate(1024); EXPECT_TRUE(HidlMemoryDealer::isOk(blk)); MemoryBlock blk2 = dealer->allocate(2048); EXPECT_TRUE(HidlMemoryDealer::isOk(blk2)); MemoryBlock blk3 = dealer->allocate(2048); EXPECT_FALSE(HidlMemoryDealer::isOk(blk3)); dealer->deallocate(blk2.offset); blk3 = dealer->allocate(2048); EXPECT_TRUE(HidlMemoryDealer::isOk(blk3)); } int main(int argc, char** argv) { ::testing::InitGoogleTest(&argc, argv); return RUN_ALL_TESTS(); } } // namespace android