blob: e28c3524943cbb3ea1921c2a56a95e66d2f0f871 [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
Sandeep Patilf1291992018-11-19 15:25:18 -0800145 EXPECT_TRUE(ProcMemInfo::ResetWorkingSet(pid));
Sandeep Patil54d87212018-08-29 17:10:47 -0700146}
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
Sandeep Patil549feab2018-11-19 11:38:40 -0800248TEST(ValidateProcMemInfoFlags, TestPageFlags1) {
249 // Create proc object using libpagemap
250 pm_kernel_t* ker;
251 ASSERT_EQ(0, pm_kernel_create(&ker));
252 pm_process_t* proc;
253 ASSERT_EQ(0, pm_process_create(ker, pid, &proc));
254
255 // count swapbacked pages using libpagemap
256 pm_memusage_t proc_usage;
257 pm_memusage_zero(&proc_usage);
258 ASSERT_EQ(0, pm_process_usage_flags(proc, &proc_usage, (1 << KPF_SWAPBACKED),
259 (1 << KPF_SWAPBACKED)));
260
261 // Create ProcMemInfo that counts swapbacked pages
262 ProcMemInfo proc_mem(pid, false, (1 << KPF_SWAPBACKED), (1 << KPF_SWAPBACKED));
263
264 EXPECT_EQ(proc_usage.vss, proc_mem.Usage().vss);
265 EXPECT_EQ(proc_usage.rss, proc_mem.Usage().rss);
266 EXPECT_EQ(proc_usage.pss, proc_mem.Usage().pss);
267 EXPECT_EQ(proc_usage.uss, proc_mem.Usage().uss);
268
269 pm_process_destroy(proc);
270 pm_kernel_destroy(ker);
271}
272
273TEST(ValidateProcMemInfoFlags, TestPageFlags2) {
274 // Create proc object using libpagemap
275 pm_kernel_t* ker;
276 ASSERT_EQ(0, pm_kernel_create(&ker));
277 pm_process_t* proc;
278 ASSERT_EQ(0, pm_process_create(ker, pid, &proc));
279
280 // count non-swapbacked pages using libpagemap
281 pm_memusage_t proc_usage;
282 pm_memusage_zero(&proc_usage);
283 ASSERT_EQ(0, pm_process_usage_flags(proc, &proc_usage, (1 << KPF_SWAPBACKED), 0));
284
285 // Create ProcMemInfo that counts non-swapbacked pages
286 ProcMemInfo proc_mem(pid, false, 0, (1 << KPF_SWAPBACKED));
287
288 EXPECT_EQ(proc_usage.vss, proc_mem.Usage().vss);
289 EXPECT_EQ(proc_usage.rss, proc_mem.Usage().rss);
290 EXPECT_EQ(proc_usage.pss, proc_mem.Usage().pss);
291 EXPECT_EQ(proc_usage.uss, proc_mem.Usage().uss);
292
293 pm_process_destroy(proc);
294 pm_kernel_destroy(ker);
295}
296
Sandeep Patil54d87212018-08-29 17:10:47 -0700297TEST(SysMemInfoParser, TestSysMemInfoFile) {
298 std::string meminfo = R"meminfo(MemTotal: 3019740 kB
299MemFree: 1809728 kB
300MemAvailable: 2546560 kB
301Buffers: 54736 kB
302Cached: 776052 kB
303SwapCached: 0 kB
304Active: 445856 kB
305Inactive: 459092 kB
306Active(anon): 78492 kB
307Inactive(anon): 2240 kB
308Active(file): 367364 kB
309Inactive(file): 456852 kB
310Unevictable: 3096 kB
311Mlocked: 3096 kB
312SwapTotal: 0 kB
313SwapFree: 0 kB
314Dirty: 32 kB
315Writeback: 0 kB
316AnonPages: 74988 kB
317Mapped: 62624 kB
318Shmem: 4020 kB
319Slab: 86464 kB
320SReclaimable: 44432 kB
321SUnreclaim: 42032 kB
322KernelStack: 4880 kB
323PageTables: 2900 kB
324NFS_Unstable: 0 kB
325Bounce: 0 kB
326WritebackTmp: 0 kB
327CommitLimit: 1509868 kB
328Committed_AS: 80296 kB
329VmallocTotal: 263061440 kB
330VmallocUsed: 0 kB
331VmallocChunk: 0 kB
332AnonHugePages: 6144 kB
333ShmemHugePages: 0 kB
334ShmemPmdMapped: 0 kB
335CmaTotal: 131072 kB
336CmaFree: 130380 kB
337HugePages_Total: 0
338HugePages_Free: 0
339HugePages_Rsvd: 0
340HugePages_Surp: 0
341Hugepagesize: 2048 kB)meminfo";
342
343 TemporaryFile tf;
344 ASSERT_TRUE(tf.fd != -1);
345 ASSERT_TRUE(::android::base::WriteStringToFd(meminfo, tf.fd));
346
347 SysMemInfo mi;
348 ASSERT_TRUE(mi.ReadMemInfo(tf.path));
349 EXPECT_EQ(mi.mem_total_kb(), 3019740);
350 EXPECT_EQ(mi.mem_page_tables_kb(), 2900);
351}
352
353TEST(SysMemInfoParser, TestEmptyFile) {
354 TemporaryFile tf;
355 std::string empty_string = "";
356 ASSERT_TRUE(tf.fd != -1);
357 ASSERT_TRUE(::android::base::WriteStringToFd(empty_string, tf.fd));
358
359 SysMemInfo mi;
360 EXPECT_TRUE(mi.ReadMemInfo(tf.path));
361 EXPECT_EQ(mi.mem_total_kb(), 0);
362}
363
Sandeep Patil70fa72d2018-11-09 19:18:29 -0800364TEST(SysMemInfoParse, TestZramTotal) {
365 std::string exec_dir = ::android::base::GetExecutableDirectory();
366
367 SysMemInfo mi;
368 std::string zram_mmstat_dir = exec_dir + "/testdata1/";
369 EXPECT_EQ(mi.mem_zram_kb(zram_mmstat_dir), 30504);
370
371 std::string zram_memused_dir = exec_dir + "/testdata2/";
372 EXPECT_EQ(mi.mem_zram_kb(zram_memused_dir), 30504);
373}
374
Sandeep Patil54d87212018-08-29 17:10:47 -0700375int main(int argc, char** argv) {
376 ::testing::InitGoogleTest(&argc, argv);
377 if (argc <= 1) {
378 cerr << "Pid of a permanently sleeping process must be provided." << endl;
379 exit(EXIT_FAILURE);
380 }
381 ::android::base::InitLogging(argv, android::base::StderrLogger);
382 pid = std::stoi(std::string(argv[1]));
383 return RUN_ALL_TESTS();
384}