| /* |
| * 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 "common_runtime_test.h" |
| |
| #include "art_method-inl.h" |
| #include "class_linker.h" |
| #include "jit_code_cache.h" |
| #include "scoped_thread_state_change.h" |
| #include "thread-inl.h" |
| |
| namespace art { |
| namespace jit { |
| |
| class JitCodeCacheTest : public CommonRuntimeTest { |
| public: |
| }; |
| |
| TEST_F(JitCodeCacheTest, TestCoverage) { |
| std::string error_msg; |
| constexpr size_t kSize = 1 * MB; |
| std::unique_ptr<JitCodeCache> code_cache( |
| JitCodeCache::Create(kSize, &error_msg)); |
| ASSERT_TRUE(code_cache.get() != nullptr) << error_msg; |
| ASSERT_TRUE(code_cache->CodeCachePtr() != nullptr); |
| ASSERT_EQ(code_cache->CodeCacheSize(), 0u); |
| ASSERT_GT(code_cache->CodeCacheRemain(), 0u); |
| ASSERT_TRUE(code_cache->DataCachePtr() != nullptr); |
| ASSERT_EQ(code_cache->DataCacheSize(), 0u); |
| ASSERT_GT(code_cache->DataCacheRemain(), 0u); |
| ASSERT_EQ(code_cache->CodeCacheRemain() + code_cache->DataCacheRemain(), kSize); |
| ASSERT_EQ(code_cache->NumMethods(), 0u); |
| ScopedObjectAccess soa(Thread::Current()); |
| StackHandleScope<1> hs(soa.Self()); |
| uint8_t* const reserved_code = code_cache->ReserveCode(soa.Self(), 4 * KB); |
| ASSERT_TRUE(reserved_code != nullptr); |
| ASSERT_TRUE(code_cache->ContainsCodePtr(reserved_code)); |
| ASSERT_EQ(code_cache->NumMethods(), 1u); |
| ClassLinker* const cl = Runtime::Current()->GetClassLinker(); |
| ArtMethod* method = &cl->AllocArtMethodArray(soa.Self(), 1)->At(0); |
| ASSERT_FALSE(code_cache->ContainsMethod(method)); |
| method->SetEntryPointFromQuickCompiledCode(reserved_code); |
| ASSERT_TRUE(code_cache->ContainsMethod(method)); |
| ASSERT_EQ(code_cache->GetCodeFor(method), reserved_code); |
| // Save the code and then change it. |
| code_cache->SaveCompiledCode(method, reserved_code); |
| method->SetEntryPointFromQuickCompiledCode(nullptr); |
| ASSERT_EQ(code_cache->GetCodeFor(method), reserved_code); |
| const uint8_t data_arr[] = {1, 2, 3, 4, 5}; |
| uint8_t* data_ptr = code_cache->AddDataArray(soa.Self(), data_arr, data_arr + sizeof(data_arr)); |
| ASSERT_TRUE(data_ptr != nullptr); |
| ASSERT_EQ(memcmp(data_ptr, data_arr, sizeof(data_arr)), 0); |
| } |
| |
| TEST_F(JitCodeCacheTest, TestOverflow) { |
| std::string error_msg; |
| constexpr size_t kSize = 1 * MB; |
| std::unique_ptr<JitCodeCache> code_cache( |
| JitCodeCache::Create(kSize, &error_msg)); |
| ASSERT_TRUE(code_cache.get() != nullptr) << error_msg; |
| ASSERT_TRUE(code_cache->CodeCachePtr() != nullptr); |
| size_t code_bytes = 0; |
| size_t data_bytes = 0; |
| constexpr size_t kCodeArrSize = 4 * KB; |
| constexpr size_t kDataArrSize = 4 * KB; |
| uint8_t data_arr[kDataArrSize]; |
| std::fill_n(data_arr, arraysize(data_arr), 53); |
| // Add code and data until we are full. |
| uint8_t* code_ptr = nullptr; |
| uint8_t* data_ptr = nullptr; |
| do { |
| code_ptr = code_cache->ReserveCode(Thread::Current(), kCodeArrSize); |
| data_ptr = code_cache->AddDataArray(Thread::Current(), data_arr, data_arr + kDataArrSize); |
| if (code_ptr != nullptr) { |
| code_bytes += kCodeArrSize; |
| } |
| if (data_ptr != nullptr) { |
| data_bytes += kDataArrSize; |
| } |
| } while (code_ptr != nullptr || data_ptr != nullptr); |
| // Make sure we added a reasonable amount |
| CHECK_GT(code_bytes, 0u); |
| CHECK_LE(code_bytes, kSize); |
| CHECK_GT(data_bytes, 0u); |
| CHECK_LE(data_bytes, kSize); |
| CHECK_GE(code_bytes + data_bytes, kSize * 4 / 5); |
| } |
| |
| } // namespace jit |
| } // namespace art |