/* * Copyright (C) 2014 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 "monitor_pool.h" #include "common_runtime_test.h" #include "scoped_thread_state_change.h" #include "thread-inl.h" namespace art { class MonitorPoolTest : public CommonRuntimeTest {}; class RandGen { public: explicit RandGen(uint32_t seed) : val_(seed) {} uint32_t next() { val_ = val_ * 48271 % 2147483647 + 13; return val_; } uint32_t val_; }; static void VerifyMonitor(Monitor* mon, Thread* self) { // Check whether the monitor id is correct. EXPECT_EQ(MonitorPool::MonitorIdFromMonitor(mon), mon->GetMonitorId()); // Check whether the monitor id agrees with the compuation. EXPECT_EQ(MonitorPool::ComputeMonitorId(mon, self), mon->GetMonitorId()); // Check whether we can use the monitor ID to get the monitor. EXPECT_EQ(mon, MonitorPool::MonitorFromMonitorId(mon->GetMonitorId())); } TEST_F(MonitorPoolTest, MonitorPoolTest) { std::vector monitors; RandGen r(0x1234); // 1) Create and release monitors without increasing the storage. // Number of max alive monitors before resize. // Note: for correct testing, make sure this is corresponding to monitor-pool's initial size. const size_t kMaxUsage = 28; Thread* self = Thread::Current(); ScopedObjectAccess soa(self); // Allocate and release monitors. for (size_t i = 0; i < 1000 ; i++) { bool alloc; if (monitors.size() == 0) { alloc = true; } else if (monitors.size() == kMaxUsage) { alloc = false; } else { // Random decision. alloc = r.next() % 2 == 0; } if (alloc) { Monitor* mon = MonitorPool::CreateMonitor(self, self, nullptr, static_cast(i)); monitors.push_back(mon); VerifyMonitor(mon, self); } else { // Release a random monitor. size_t index = r.next() % monitors.size(); Monitor* mon = monitors[index]; monitors.erase(monitors.begin() + index); // Recheck the monitor. VerifyMonitor(mon, self); MonitorPool::ReleaseMonitor(self, mon); } } // Loop some time. for (size_t i = 0; i < 10; ++i) { // 2.1) Create enough monitors to require new chunks. size_t target_size = monitors.size() + 2*kMaxUsage; while (monitors.size() < target_size) { Monitor* mon = MonitorPool::CreateMonitor(self, self, nullptr, static_cast(-monitors.size())); monitors.push_back(mon); VerifyMonitor(mon, self); } // 2.2) Verify all monitors. for (Monitor* mon : monitors) { VerifyMonitor(mon, self); } // 2.3) Release a number of monitors randomly. for (size_t j = 0; j < kMaxUsage; j++) { // Release a random monitor. size_t index = r.next() % monitors.size(); Monitor* mon = monitors[index]; monitors.erase(monitors.begin() + index); MonitorPool::ReleaseMonitor(self, mon); } } // Check and release all remaining monitors. for (Monitor* mon : monitors) { VerifyMonitor(mon, self); MonitorPool::ReleaseMonitor(self, mon); } } } // namespace art