-
Notifications
You must be signed in to change notification settings - Fork 107
/
Copy pathpage_util.cc
100 lines (86 loc) · 3.8 KB
/
page_util.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
/*
* 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<const uint64_t>(&page_frame_number, 1u),
ArrayRef<uint64_t>(&page_flags_or_count, 1u),
error_msg);
}
bool GetPageFlagsOrCounts(File& kpage_file,
ArrayRef<const uint64_t> page_frame_numbers,
/*out*/ ArrayRef<uint64_t> 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<uint64_t>(&page_frame_number, 1u), error_msg);
}
bool GetPageFrameNumbers(File& page_map_file,
size_t virtual_page_index,
/*out*/ ArrayRef<uint64_t> 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