/* * Copyright (C) 2017 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 "src/base/test/vm_test_utils.h" #include "perfetto/base/build_config.h" #include "perfetto/ext/base/utils.h" #include #include #include #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) #include #include #include #else // PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) #include #include #endif // PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) #include "perfetto/base/build_config.h" #include "perfetto/base/logging.h" #include "perfetto/ext/base/utils.h" namespace perfetto { namespace base { namespace vm_test_utils { bool IsMapped(void* start, size_t size) { const size_t page_size = GetSysPageSize(); #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) int retries = 5; size_t number_of_entries = 4000; // Just a guess. PSAPI_WORKING_SET_INFORMATION* ws_info = nullptr; std::vector buffer; for (;;) { size_t buffer_size = sizeof(PSAPI_WORKING_SET_INFORMATION) + (number_of_entries * sizeof(PSAPI_WORKING_SET_BLOCK)); buffer.resize(buffer_size); ws_info = reinterpret_cast(&buffer[0]); // On success, |buffer_| is populated with info about the working set of // |process|. On ERROR_BAD_LENGTH failure, increase the size of the // buffer and try again. if (QueryWorkingSet(GetCurrentProcess(), &buffer[0], buffer_size)) break; // Success PERFETTO_CHECK(GetLastError() == ERROR_BAD_LENGTH); number_of_entries = ws_info->NumberOfEntries; // Maybe some entries are being added right now. Increase the buffer to // take that into account. Increasing by 10% should generally be enough. number_of_entries = static_cast(double(number_of_entries) * 1.1); PERFETTO_CHECK(--retries > 0); // If we're looping, eventually fail. } void* end = reinterpret_cast(start) + size; // Now scan the working-set information looking for the addresses. unsigned pages_found = 0; for (unsigned i = 0; i < ws_info->NumberOfEntries; ++i) { void* address = reinterpret_cast( ws_info->WorkingSetInfo[i].VirtualPage * page_size); if (address >= start && address < end) ++pages_found; } if (pages_found * page_size == size) return true; return false; #elif PERFETTO_BUILDFLAG(PERFETTO_OS_FUCHSIA) // Fuchsia doesn't yet support paging (b/119503290). ignore_result(page_size); return true; #elif PERFETTO_BUILDFLAG(PERFETTO_OS_NACL) // mincore isn't available on NaCL. ignore_result(page_size); return true; #else #if PERFETTO_BUILDFLAG(PERFETTO_OS_APPLE) using PageState = char; static constexpr PageState kIncoreMask = MINCORE_INCORE; #else using PageState = unsigned char; static constexpr PageState kIncoreMask = 1; #endif const size_t num_pages = (size + page_size - 1) / page_size; std::unique_ptr page_states(new PageState[num_pages]); memset(page_states.get(), 0, num_pages * sizeof(PageState)); int res = mincore(start, size, page_states.get()); // Linux returns ENOMEM when an unmapped memory range is passed. // MacOS instead returns 0 but leaves the page_states empty. if (res == -1 && errno == ENOMEM) return false; PERFETTO_CHECK(res == 0); for (size_t i = 0; i < num_pages; i++) { if (!(page_states[i] & kIncoreMask)) return false; } return true; #endif } } // namespace vm_test_utils } // namespace base } // namespace perfetto