blob: f973694e2cccc7416f4a77a2702d0b29f0ee0726 [file] [log] [blame]
Sandeep Patil54d87212018-08-29 17:10:47 -07001/*
2 * Copyright (C) 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#include <sys/types.h>
18#include <sys/wait.h>
19#include <unistd.h>
20
21#include <gtest/gtest.h>
22
23#include <string>
24#include <vector>
25
26#include <meminfo/pageacct.h>
27#include <meminfo/procmeminfo.h>
28#include <meminfo/sysmeminfo.h>
29#include <pagemap/pagemap.h>
30
31#include <android-base/file.h>
32#include <android-base/logging.h>
Sandeep Patil54d87212018-08-29 17:10:47 -070033
34using namespace std;
35using namespace android::meminfo;
36
37pid_t pid = -1;
38
39class ValidateProcMemInfo : public ::testing::Test {
40 protected:
41 void SetUp() override {
42 ASSERT_EQ(0, pm_kernel_create(&ker));
43 ASSERT_EQ(0, pm_process_create(ker, pid, &proc));
44 proc_mem = new ProcMemInfo(pid);
45 ASSERT_NE(proc_mem, nullptr);
46 }
47
48 void TearDown() override {
49 delete proc_mem;
50 pm_process_destroy(proc);
51 pm_kernel_destroy(ker);
52 }
53
54 pm_kernel_t* ker;
55 pm_process_t* proc;
56 ProcMemInfo* proc_mem;
57};
58
59TEST_F(ValidateProcMemInfo, TestMapsSize) {
60 const std::vector<Vma>& maps = proc_mem->Maps();
61 ASSERT_FALSE(maps.empty()) << "Process " << getpid() << " maps are empty";
62}
63
64TEST_F(ValidateProcMemInfo, TestMapsEquality) {
65 const std::vector<Vma>& maps = proc_mem->Maps();
66 ASSERT_EQ(proc->num_maps, maps.size());
67
68 for (size_t i = 0; i < maps.size(); ++i) {
69 EXPECT_EQ(proc->maps[i]->start, maps[i].start);
70 EXPECT_EQ(proc->maps[i]->end, maps[i].end);
71 EXPECT_EQ(proc->maps[i]->offset, maps[i].offset);
72 EXPECT_EQ(std::string(proc->maps[i]->name), maps[i].name);
73 }
74}
75
Sandeep Patil2259fdf2018-11-09 16:42:45 -080076TEST_F(ValidateProcMemInfo, TestMaps) {
Sandeep Patil54d87212018-08-29 17:10:47 -070077 const std::vector<Vma>& maps = proc_mem->Maps();
78 ASSERT_FALSE(maps.empty());
79 ASSERT_EQ(proc->num_maps, maps.size());
80
81 pm_memusage_t map_usage, proc_usage;
82 pm_memusage_zero(&map_usage);
83 pm_memusage_zero(&proc_usage);
84 for (size_t i = 0; i < maps.size(); i++) {
85 ASSERT_EQ(0, pm_map_usage(proc->maps[i], &map_usage));
86 EXPECT_EQ(map_usage.vss, maps[i].usage.vss) << "VSS mismatch for map: " << maps[i].name;
87 EXPECT_EQ(map_usage.rss, maps[i].usage.rss) << "RSS mismatch for map: " << maps[i].name;
88 EXPECT_EQ(map_usage.pss, maps[i].usage.pss) << "PSS mismatch for map: " << maps[i].name;
89 EXPECT_EQ(map_usage.uss, maps[i].usage.uss) << "USS mismatch for map: " << maps[i].name;
90 pm_memusage_add(&proc_usage, &map_usage);
91 }
92
93 EXPECT_EQ(proc_usage.vss, proc_mem->Usage().vss);
94 EXPECT_EQ(proc_usage.rss, proc_mem->Usage().rss);
95 EXPECT_EQ(proc_usage.pss, proc_mem->Usage().pss);
96 EXPECT_EQ(proc_usage.uss, proc_mem->Usage().uss);
97}
98
Sandeep Patil2259fdf2018-11-09 16:42:45 -080099TEST_F(ValidateProcMemInfo, TestSwapUsage) {
100 const std::vector<Vma>& maps = proc_mem->Maps();
101 ASSERT_FALSE(maps.empty());
102 ASSERT_EQ(proc->num_maps, maps.size());
103
104 pm_memusage_t map_usage, proc_usage;
105 pm_memusage_zero(&map_usage);
106 pm_memusage_zero(&proc_usage);
107 for (size_t i = 0; i < maps.size(); i++) {
108 ASSERT_EQ(0, pm_map_usage(proc->maps[i], &map_usage));
109 EXPECT_EQ(map_usage.swap, maps[i].usage.swap) << "SWAP mismatch for map: " << maps[i].name;
110 pm_memusage_add(&proc_usage, &map_usage);
111 }
112
113 EXPECT_EQ(proc_usage.swap, proc_mem->Usage().swap);
114}
115
116TEST_F(ValidateProcMemInfo, TestSwapOffsets) {
117 const MemUsage& proc_usage = proc_mem->Usage();
118 const std::vector<uint16_t>& swap_offsets = proc_mem->SwapOffsets();
119
120 EXPECT_EQ(proc_usage.swap / getpagesize(), swap_offsets.size());
121}
122
Sandeep Patil54d87212018-08-29 17:10:47 -0700123class ValidateProcMemInfoWss : public ::testing::Test {
124 protected:
125 void SetUp() override {
126 ASSERT_EQ(0, pm_kernel_create(&ker));
127 ASSERT_EQ(0, pm_process_create(ker, pid, &proc));
128 proc_mem = new ProcMemInfo(pid, true);
129 ASSERT_NE(proc_mem, nullptr);
130 }
131
132 void TearDown() override {
133 delete proc_mem;
134 pm_process_destroy(proc);
135 pm_kernel_destroy(ker);
136 }
137
138 pm_kernel_t* ker;
139 pm_process_t* proc;
140 ProcMemInfo* proc_mem;
141};
142
143TEST_F(ValidateProcMemInfoWss, TestWorkingTestReset) {
144 // Expect reset to succeed
145 EXPECT_TRUE(proc_mem->WssReset());
146}
147
148TEST_F(ValidateProcMemInfoWss, TestWssEquality) {
149 // Read wss using libpagemap
150 pm_memusage_t wss_pagemap;
151 EXPECT_EQ(0, pm_process_workingset(proc, &wss_pagemap, 0));
152
153 // Read wss using libmeminfo
154 MemUsage wss = proc_mem->Wss();
155
156 // compare
157 EXPECT_EQ(wss_pagemap.rss, wss.rss);
158 EXPECT_EQ(wss_pagemap.pss, wss.pss);
159 EXPECT_EQ(wss_pagemap.uss, wss.uss);
160}
161
162class ValidatePageAcct : public ::testing::Test {
163 protected:
164 void SetUp() override {
165 ASSERT_EQ(0, pm_kernel_create(&ker));
166 ASSERT_EQ(0, pm_process_create(ker, pid, &proc));
167 }
168
169 void TearDown() override {
170 pm_process_destroy(proc);
171 pm_kernel_destroy(ker);
172 }
173
174 pm_kernel_t* ker;
175 pm_process_t* proc;
176};
177
178TEST_F(ValidatePageAcct, TestPageFlags) {
179 PageAcct& pi = PageAcct::Instance();
180 pi.InitPageAcct(false);
181
182 uint64_t* pagemap;
183 size_t num_pages;
184 for (size_t i = 0; i < proc->num_maps; i++) {
185 ASSERT_EQ(0, pm_map_pagemap(proc->maps[i], &pagemap, &num_pages));
186 for (size_t j = 0; j < num_pages; j++) {
187 if (!PM_PAGEMAP_PRESENT(pagemap[j])) continue;
188
189 uint64_t pfn = PM_PAGEMAP_PFN(pagemap[j]);
190 uint64_t page_flags_pagemap, page_flags_meminfo;
191
192 ASSERT_EQ(0, pm_kernel_flags(ker, pfn, &page_flags_pagemap));
193 ASSERT_TRUE(pi.PageFlags(pfn, &page_flags_meminfo));
194 // check if page flags equal
195 EXPECT_EQ(page_flags_pagemap, page_flags_meminfo);
196 }
197 free(pagemap);
198 }
199}
200
201TEST_F(ValidatePageAcct, TestPageCounts) {
202 PageAcct& pi = PageAcct::Instance();
203 pi.InitPageAcct(false);
204
205 uint64_t* pagemap;
206 size_t num_pages;
207 for (size_t i = 0; i < proc->num_maps; i++) {
208 ASSERT_EQ(0, pm_map_pagemap(proc->maps[i], &pagemap, &num_pages));
209 for (size_t j = 0; j < num_pages; j++) {
210 uint64_t pfn = PM_PAGEMAP_PFN(pagemap[j]);
211 uint64_t map_count_pagemap, map_count_meminfo;
212
213 ASSERT_EQ(0, pm_kernel_count(ker, pfn, &map_count_pagemap));
214 ASSERT_TRUE(pi.PageMapCount(pfn, &map_count_meminfo));
215 // check if map counts are equal
216 EXPECT_EQ(map_count_pagemap, map_count_meminfo);
217 }
218 free(pagemap);
219 }
220}
221
222TEST_F(ValidatePageAcct, TestPageIdle) {
223 // skip the test if idle page tracking isn't enabled
224 if (pm_kernel_init_page_idle(ker) != 0) {
225 return;
226 }
227
228 PageAcct& pi = PageAcct::Instance();
229 ASSERT_TRUE(pi.InitPageAcct(true));
230
231 uint64_t* pagemap;
232 size_t num_pages;
233 for (size_t i = 0; i < proc->num_maps; i++) {
234 ASSERT_EQ(0, pm_map_pagemap(proc->maps[i], &pagemap, &num_pages));
235 for (size_t j = 0; j < num_pages; j++) {
236 if (!PM_PAGEMAP_PRESENT(pagemap[j])) continue;
237 uint64_t pfn = PM_PAGEMAP_PFN(pagemap[j]);
238
239 ASSERT_EQ(0, pm_kernel_mark_page_idle(ker, &pfn, 1));
240 int idle_status_pagemap = pm_kernel_get_page_idle(ker, pfn);
241 int idle_status_meminfo = pi.IsPageIdle(pfn);
242 EXPECT_EQ(idle_status_pagemap, idle_status_meminfo);
243 }
244 free(pagemap);
245 }
246}
247
248TEST(SysMemInfoParser, TestSysMemInfoFile) {
249 std::string meminfo = R"meminfo(MemTotal: 3019740 kB
250MemFree: 1809728 kB
251MemAvailable: 2546560 kB
252Buffers: 54736 kB
253Cached: 776052 kB
254SwapCached: 0 kB
255Active: 445856 kB
256Inactive: 459092 kB
257Active(anon): 78492 kB
258Inactive(anon): 2240 kB
259Active(file): 367364 kB
260Inactive(file): 456852 kB
261Unevictable: 3096 kB
262Mlocked: 3096 kB
263SwapTotal: 0 kB
264SwapFree: 0 kB
265Dirty: 32 kB
266Writeback: 0 kB
267AnonPages: 74988 kB
268Mapped: 62624 kB
269Shmem: 4020 kB
270Slab: 86464 kB
271SReclaimable: 44432 kB
272SUnreclaim: 42032 kB
273KernelStack: 4880 kB
274PageTables: 2900 kB
275NFS_Unstable: 0 kB
276Bounce: 0 kB
277WritebackTmp: 0 kB
278CommitLimit: 1509868 kB
279Committed_AS: 80296 kB
280VmallocTotal: 263061440 kB
281VmallocUsed: 0 kB
282VmallocChunk: 0 kB
283AnonHugePages: 6144 kB
284ShmemHugePages: 0 kB
285ShmemPmdMapped: 0 kB
286CmaTotal: 131072 kB
287CmaFree: 130380 kB
288HugePages_Total: 0
289HugePages_Free: 0
290HugePages_Rsvd: 0
291HugePages_Surp: 0
292Hugepagesize: 2048 kB)meminfo";
293
294 TemporaryFile tf;
295 ASSERT_TRUE(tf.fd != -1);
296 ASSERT_TRUE(::android::base::WriteStringToFd(meminfo, tf.fd));
297
298 SysMemInfo mi;
299 ASSERT_TRUE(mi.ReadMemInfo(tf.path));
300 EXPECT_EQ(mi.mem_total_kb(), 3019740);
301 EXPECT_EQ(mi.mem_page_tables_kb(), 2900);
302}
303
304TEST(SysMemInfoParser, TestEmptyFile) {
305 TemporaryFile tf;
306 std::string empty_string = "";
307 ASSERT_TRUE(tf.fd != -1);
308 ASSERT_TRUE(::android::base::WriteStringToFd(empty_string, tf.fd));
309
310 SysMemInfo mi;
311 EXPECT_TRUE(mi.ReadMemInfo(tf.path));
312 EXPECT_EQ(mi.mem_total_kb(), 0);
313}
314
315int main(int argc, char** argv) {
316 ::testing::InitGoogleTest(&argc, argv);
317 if (argc <= 1) {
318 cerr << "Pid of a permanently sleeping process must be provided." << endl;
319 exit(EXIT_FAILURE);
320 }
321 ::android::base::InitLogging(argv, android::base::StderrLogger);
322 pid = std::stoi(std::string(argv[1]));
323 return RUN_ALL_TESTS();
324}