blob: 7a2be4172b3f45b14fb392da491f12ae2c29216a [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
76TEST_F(ValidateProcMemInfo, TestMapsUsage) {
77 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
99class ValidateProcMemInfoWss : public ::testing::Test {
100 protected:
101 void SetUp() override {
102 ASSERT_EQ(0, pm_kernel_create(&ker));
103 ASSERT_EQ(0, pm_process_create(ker, pid, &proc));
104 proc_mem = new ProcMemInfo(pid, true);
105 ASSERT_NE(proc_mem, nullptr);
106 }
107
108 void TearDown() override {
109 delete proc_mem;
110 pm_process_destroy(proc);
111 pm_kernel_destroy(ker);
112 }
113
114 pm_kernel_t* ker;
115 pm_process_t* proc;
116 ProcMemInfo* proc_mem;
117};
118
119TEST_F(ValidateProcMemInfoWss, TestWorkingTestReset) {
120 // Expect reset to succeed
121 EXPECT_TRUE(proc_mem->WssReset());
122}
123
124TEST_F(ValidateProcMemInfoWss, TestWssEquality) {
125 // Read wss using libpagemap
126 pm_memusage_t wss_pagemap;
127 EXPECT_EQ(0, pm_process_workingset(proc, &wss_pagemap, 0));
128
129 // Read wss using libmeminfo
130 MemUsage wss = proc_mem->Wss();
131
132 // compare
133 EXPECT_EQ(wss_pagemap.rss, wss.rss);
134 EXPECT_EQ(wss_pagemap.pss, wss.pss);
135 EXPECT_EQ(wss_pagemap.uss, wss.uss);
136}
137
138class ValidatePageAcct : public ::testing::Test {
139 protected:
140 void SetUp() override {
141 ASSERT_EQ(0, pm_kernel_create(&ker));
142 ASSERT_EQ(0, pm_process_create(ker, pid, &proc));
143 }
144
145 void TearDown() override {
146 pm_process_destroy(proc);
147 pm_kernel_destroy(ker);
148 }
149
150 pm_kernel_t* ker;
151 pm_process_t* proc;
152};
153
154TEST_F(ValidatePageAcct, TestPageFlags) {
155 PageAcct& pi = PageAcct::Instance();
156 pi.InitPageAcct(false);
157
158 uint64_t* pagemap;
159 size_t num_pages;
160 for (size_t i = 0; i < proc->num_maps; i++) {
161 ASSERT_EQ(0, pm_map_pagemap(proc->maps[i], &pagemap, &num_pages));
162 for (size_t j = 0; j < num_pages; j++) {
163 if (!PM_PAGEMAP_PRESENT(pagemap[j])) continue;
164
165 uint64_t pfn = PM_PAGEMAP_PFN(pagemap[j]);
166 uint64_t page_flags_pagemap, page_flags_meminfo;
167
168 ASSERT_EQ(0, pm_kernel_flags(ker, pfn, &page_flags_pagemap));
169 ASSERT_TRUE(pi.PageFlags(pfn, &page_flags_meminfo));
170 // check if page flags equal
171 EXPECT_EQ(page_flags_pagemap, page_flags_meminfo);
172 }
173 free(pagemap);
174 }
175}
176
177TEST_F(ValidatePageAcct, TestPageCounts) {
178 PageAcct& pi = PageAcct::Instance();
179 pi.InitPageAcct(false);
180
181 uint64_t* pagemap;
182 size_t num_pages;
183 for (size_t i = 0; i < proc->num_maps; i++) {
184 ASSERT_EQ(0, pm_map_pagemap(proc->maps[i], &pagemap, &num_pages));
185 for (size_t j = 0; j < num_pages; j++) {
186 uint64_t pfn = PM_PAGEMAP_PFN(pagemap[j]);
187 uint64_t map_count_pagemap, map_count_meminfo;
188
189 ASSERT_EQ(0, pm_kernel_count(ker, pfn, &map_count_pagemap));
190 ASSERT_TRUE(pi.PageMapCount(pfn, &map_count_meminfo));
191 // check if map counts are equal
192 EXPECT_EQ(map_count_pagemap, map_count_meminfo);
193 }
194 free(pagemap);
195 }
196}
197
198TEST_F(ValidatePageAcct, TestPageIdle) {
199 // skip the test if idle page tracking isn't enabled
200 if (pm_kernel_init_page_idle(ker) != 0) {
201 return;
202 }
203
204 PageAcct& pi = PageAcct::Instance();
205 ASSERT_TRUE(pi.InitPageAcct(true));
206
207 uint64_t* pagemap;
208 size_t num_pages;
209 for (size_t i = 0; i < proc->num_maps; i++) {
210 ASSERT_EQ(0, pm_map_pagemap(proc->maps[i], &pagemap, &num_pages));
211 for (size_t j = 0; j < num_pages; j++) {
212 if (!PM_PAGEMAP_PRESENT(pagemap[j])) continue;
213 uint64_t pfn = PM_PAGEMAP_PFN(pagemap[j]);
214
215 ASSERT_EQ(0, pm_kernel_mark_page_idle(ker, &pfn, 1));
216 int idle_status_pagemap = pm_kernel_get_page_idle(ker, pfn);
217 int idle_status_meminfo = pi.IsPageIdle(pfn);
218 EXPECT_EQ(idle_status_pagemap, idle_status_meminfo);
219 }
220 free(pagemap);
221 }
222}
223
224TEST(SysMemInfoParser, TestSysMemInfoFile) {
225 std::string meminfo = R"meminfo(MemTotal: 3019740 kB
226MemFree: 1809728 kB
227MemAvailable: 2546560 kB
228Buffers: 54736 kB
229Cached: 776052 kB
230SwapCached: 0 kB
231Active: 445856 kB
232Inactive: 459092 kB
233Active(anon): 78492 kB
234Inactive(anon): 2240 kB
235Active(file): 367364 kB
236Inactive(file): 456852 kB
237Unevictable: 3096 kB
238Mlocked: 3096 kB
239SwapTotal: 0 kB
240SwapFree: 0 kB
241Dirty: 32 kB
242Writeback: 0 kB
243AnonPages: 74988 kB
244Mapped: 62624 kB
245Shmem: 4020 kB
246Slab: 86464 kB
247SReclaimable: 44432 kB
248SUnreclaim: 42032 kB
249KernelStack: 4880 kB
250PageTables: 2900 kB
251NFS_Unstable: 0 kB
252Bounce: 0 kB
253WritebackTmp: 0 kB
254CommitLimit: 1509868 kB
255Committed_AS: 80296 kB
256VmallocTotal: 263061440 kB
257VmallocUsed: 0 kB
258VmallocChunk: 0 kB
259AnonHugePages: 6144 kB
260ShmemHugePages: 0 kB
261ShmemPmdMapped: 0 kB
262CmaTotal: 131072 kB
263CmaFree: 130380 kB
264HugePages_Total: 0
265HugePages_Free: 0
266HugePages_Rsvd: 0
267HugePages_Surp: 0
268Hugepagesize: 2048 kB)meminfo";
269
270 TemporaryFile tf;
271 ASSERT_TRUE(tf.fd != -1);
272 ASSERT_TRUE(::android::base::WriteStringToFd(meminfo, tf.fd));
273
274 SysMemInfo mi;
275 ASSERT_TRUE(mi.ReadMemInfo(tf.path));
276 EXPECT_EQ(mi.mem_total_kb(), 3019740);
277 EXPECT_EQ(mi.mem_page_tables_kb(), 2900);
278}
279
280TEST(SysMemInfoParser, TestEmptyFile) {
281 TemporaryFile tf;
282 std::string empty_string = "";
283 ASSERT_TRUE(tf.fd != -1);
284 ASSERT_TRUE(::android::base::WriteStringToFd(empty_string, tf.fd));
285
286 SysMemInfo mi;
287 EXPECT_TRUE(mi.ReadMemInfo(tf.path));
288 EXPECT_EQ(mi.mem_total_kb(), 0);
289}
290
291int main(int argc, char** argv) {
292 ::testing::InitGoogleTest(&argc, argv);
293 if (argc <= 1) {
294 cerr << "Pid of a permanently sleeping process must be provided." << endl;
295 exit(EXIT_FAILURE);
296 }
297 ::android::base::InitLogging(argv, android::base::StderrLogger);
298 pid = std::stoi(std::string(argv[1]));
299 return RUN_ALL_TESTS();
300}