/* * Copyright (C) 2023 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 "page_util.h" #include "android-base/stringprintf.h" namespace art { using android::base::StringPrintf; bool GetPageFlagsOrCount(art::File& kpage_file, uint64_t page_frame_number, /*out*/ uint64_t& page_flags_or_count, /*out*/ std::string& error_msg) { return GetPageFlagsOrCounts(kpage_file, ArrayRef(&page_frame_number, 1u), ArrayRef(&page_flags_or_count, 1u), error_msg); } bool GetPageFlagsOrCounts(File& kpage_file, ArrayRef page_frame_numbers, /*out*/ ArrayRef page_flags_or_counts, /*out*/ std::string& error_msg) { static_assert(kPageFlagsEntrySize == kPageCountEntrySize, "entry size check"); CHECK_NE(page_frame_numbers.size(), 0u); CHECK_EQ(page_flags_or_counts.size(), page_frame_numbers.size()); CHECK(page_frame_numbers.data() != nullptr); CHECK(page_flags_or_counts.data() != nullptr); size_t size = page_frame_numbers.size(); size_t i = 0; while (i != size) { size_t start = i; ++i; while (i != size && page_frame_numbers[i] - page_frame_numbers[start] == i - start) { ++i; } // Read 64-bit entries from /proc/kpageflags or /proc/kpagecount. if (!kpage_file.PreadFully(page_flags_or_counts.data() + start, (i - start) * kPageMapEntrySize, page_frame_numbers[start] * kPageFlagsEntrySize)) { error_msg = StringPrintf("Failed to read the page flags or counts from %s, error: %s", kpage_file.GetPath().c_str(), strerror(errno)); return false; } } return true; } bool GetPageFrameNumber(File& page_map_file, size_t virtual_page_index, /*out*/ uint64_t& page_frame_number, /*out*/ std::string& error_msg) { return GetPageFrameNumbers( page_map_file, virtual_page_index, ArrayRef(&page_frame_number, 1u), error_msg); } bool GetPageFrameNumbers(File& page_map_file, size_t virtual_page_index, /*out*/ ArrayRef page_frame_numbers, /*out*/ std::string& error_msg) { CHECK_NE(page_frame_numbers.size(), 0u); CHECK(page_frame_numbers.data() != nullptr); // Read 64-bit entries from /proc/$pid/pagemap to get the physical page frame numbers. if (!page_map_file.PreadFully(page_frame_numbers.data(), page_frame_numbers.size() * kPageMapEntrySize, virtual_page_index * kPageMapEntrySize)) { error_msg = StringPrintf("Failed to read virtual page index entries from %s, error: %s", page_map_file.GetPath().c_str(), strerror(errno)); return false; } // Extract page frame numbers from pagemap entries. for (uint64_t& page_frame_number : page_frame_numbers) { page_frame_number &= kPageFrameNumberMask; } return true; } } // namespace art