blob: 2856c2dcf4ad6bf1f8370627993a8c8bf1785d87 [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 Patilc6497eb2018-11-20 09:31:36 -0800248TEST(TestProcMemInfo, TestMapsEmpty) {
249 ProcMemInfo proc_mem(pid);
250 const std::vector<Vma>& maps = proc_mem.Maps();
251 EXPECT_GT(maps.size(), 0);
252}
253
254TEST(TestProcMemInfo, TestUsageEmpty) {
255 // If we created the object for getting working set,
256 // the usage must be empty
257 ProcMemInfo proc_mem(pid, true);
258 const MemUsage& usage = proc_mem.Usage();
259 EXPECT_EQ(usage.rss, 0);
260 EXPECT_EQ(usage.vss, 0);
261 EXPECT_EQ(usage.pss, 0);
262 EXPECT_EQ(usage.uss, 0);
263 EXPECT_EQ(usage.swap, 0);
264}
265
266TEST(TestProcMemInfoWssReset, TestWssEmpty) {
267 // If we created the object for getting usage,
268 // the working set must be empty
269 ProcMemInfo proc_mem(pid, false);
270 const MemUsage& wss = proc_mem.Wss();
271 EXPECT_EQ(wss.rss, 0);
272 EXPECT_EQ(wss.vss, 0);
273 EXPECT_EQ(wss.pss, 0);
274 EXPECT_EQ(wss.uss, 0);
275 EXPECT_EQ(wss.swap, 0);
276}
277
278TEST(TestProcMemInfoWssReset, TestSwapOffsetsEmpty) {
279 // If we created the object for getting working set,
280 // the swap offsets must be empty
281 ProcMemInfo proc_mem(pid, true);
282 const std::vector<uint16_t>& swap_offsets = proc_mem.SwapOffsets();
283 EXPECT_EQ(swap_offsets.size(), 0);
284}
285
Sandeep Patil549feab2018-11-19 11:38:40 -0800286TEST(ValidateProcMemInfoFlags, TestPageFlags1) {
287 // Create proc object using libpagemap
288 pm_kernel_t* ker;
289 ASSERT_EQ(0, pm_kernel_create(&ker));
290 pm_process_t* proc;
291 ASSERT_EQ(0, pm_process_create(ker, pid, &proc));
292
293 // count swapbacked pages using libpagemap
294 pm_memusage_t proc_usage;
295 pm_memusage_zero(&proc_usage);
296 ASSERT_EQ(0, pm_process_usage_flags(proc, &proc_usage, (1 << KPF_SWAPBACKED),
297 (1 << KPF_SWAPBACKED)));
298
299 // Create ProcMemInfo that counts swapbacked pages
300 ProcMemInfo proc_mem(pid, false, (1 << KPF_SWAPBACKED), (1 << KPF_SWAPBACKED));
301
302 EXPECT_EQ(proc_usage.vss, proc_mem.Usage().vss);
303 EXPECT_EQ(proc_usage.rss, proc_mem.Usage().rss);
304 EXPECT_EQ(proc_usage.pss, proc_mem.Usage().pss);
305 EXPECT_EQ(proc_usage.uss, proc_mem.Usage().uss);
306
307 pm_process_destroy(proc);
308 pm_kernel_destroy(ker);
309}
310
311TEST(ValidateProcMemInfoFlags, TestPageFlags2) {
312 // Create proc object using libpagemap
313 pm_kernel_t* ker;
314 ASSERT_EQ(0, pm_kernel_create(&ker));
315 pm_process_t* proc;
316 ASSERT_EQ(0, pm_process_create(ker, pid, &proc));
317
318 // count non-swapbacked pages using libpagemap
319 pm_memusage_t proc_usage;
320 pm_memusage_zero(&proc_usage);
321 ASSERT_EQ(0, pm_process_usage_flags(proc, &proc_usage, (1 << KPF_SWAPBACKED), 0));
322
323 // Create ProcMemInfo that counts non-swapbacked pages
324 ProcMemInfo proc_mem(pid, false, 0, (1 << KPF_SWAPBACKED));
325
326 EXPECT_EQ(proc_usage.vss, proc_mem.Usage().vss);
327 EXPECT_EQ(proc_usage.rss, proc_mem.Usage().rss);
328 EXPECT_EQ(proc_usage.pss, proc_mem.Usage().pss);
329 EXPECT_EQ(proc_usage.uss, proc_mem.Usage().uss);
330
331 pm_process_destroy(proc);
332 pm_kernel_destroy(ker);
333}
334
Sandeep Patil54d87212018-08-29 17:10:47 -0700335TEST(SysMemInfoParser, TestSysMemInfoFile) {
336 std::string meminfo = R"meminfo(MemTotal: 3019740 kB
337MemFree: 1809728 kB
338MemAvailable: 2546560 kB
339Buffers: 54736 kB
340Cached: 776052 kB
341SwapCached: 0 kB
342Active: 445856 kB
343Inactive: 459092 kB
344Active(anon): 78492 kB
345Inactive(anon): 2240 kB
346Active(file): 367364 kB
347Inactive(file): 456852 kB
348Unevictable: 3096 kB
349Mlocked: 3096 kB
Sandeep Patil2f0b6eb2018-12-11 09:28:38 -0800350SwapTotal: 32768 kB
351SwapFree: 4096 kB
Sandeep Patil54d87212018-08-29 17:10:47 -0700352Dirty: 32 kB
353Writeback: 0 kB
354AnonPages: 74988 kB
355Mapped: 62624 kB
356Shmem: 4020 kB
357Slab: 86464 kB
358SReclaimable: 44432 kB
359SUnreclaim: 42032 kB
360KernelStack: 4880 kB
361PageTables: 2900 kB
362NFS_Unstable: 0 kB
363Bounce: 0 kB
364WritebackTmp: 0 kB
365CommitLimit: 1509868 kB
366Committed_AS: 80296 kB
367VmallocTotal: 263061440 kB
Sandeep Patil2f0b6eb2018-12-11 09:28:38 -0800368VmallocUsed: 65536 kB
Sandeep Patil54d87212018-08-29 17:10:47 -0700369VmallocChunk: 0 kB
370AnonHugePages: 6144 kB
371ShmemHugePages: 0 kB
372ShmemPmdMapped: 0 kB
373CmaTotal: 131072 kB
374CmaFree: 130380 kB
375HugePages_Total: 0
376HugePages_Free: 0
377HugePages_Rsvd: 0
378HugePages_Surp: 0
379Hugepagesize: 2048 kB)meminfo";
380
381 TemporaryFile tf;
382 ASSERT_TRUE(tf.fd != -1);
383 ASSERT_TRUE(::android::base::WriteStringToFd(meminfo, tf.fd));
384
385 SysMemInfo mi;
386 ASSERT_TRUE(mi.ReadMemInfo(tf.path));
387 EXPECT_EQ(mi.mem_total_kb(), 3019740);
Sandeep Patil2f0b6eb2018-12-11 09:28:38 -0800388 EXPECT_EQ(mi.mem_free_kb(), 1809728);
389 EXPECT_EQ(mi.mem_buffers_kb(), 54736);
390 EXPECT_EQ(mi.mem_cached_kb(), 776052);
391 EXPECT_EQ(mi.mem_shmem_kb(), 4020);
392 EXPECT_EQ(mi.mem_slab_kb(), 86464);
393 EXPECT_EQ(mi.mem_slab_reclaimable_kb(), 44432);
394 EXPECT_EQ(mi.mem_slab_unreclaimable_kb(), 42032);
395 EXPECT_EQ(mi.mem_swap_kb(), 32768);
396 EXPECT_EQ(mi.mem_swap_free_kb(), 4096);
397 EXPECT_EQ(mi.mem_mapped_kb(), 62624);
398 EXPECT_EQ(mi.mem_vmalloc_used_kb(), 65536);
Sandeep Patil54d87212018-08-29 17:10:47 -0700399 EXPECT_EQ(mi.mem_page_tables_kb(), 2900);
Sandeep Patil2f0b6eb2018-12-11 09:28:38 -0800400 EXPECT_EQ(mi.mem_kernel_stack_kb(), 4880);
Sandeep Patil54d87212018-08-29 17:10:47 -0700401}
402
403TEST(SysMemInfoParser, TestEmptyFile) {
404 TemporaryFile tf;
405 std::string empty_string = "";
406 ASSERT_TRUE(tf.fd != -1);
407 ASSERT_TRUE(::android::base::WriteStringToFd(empty_string, tf.fd));
408
409 SysMemInfo mi;
410 EXPECT_TRUE(mi.ReadMemInfo(tf.path));
411 EXPECT_EQ(mi.mem_total_kb(), 0);
412}
413
Sandeep Patil2f0b6eb2018-12-11 09:28:38 -0800414TEST(SysMemInfoParser, TestZramTotal) {
Sandeep Patil70fa72d2018-11-09 19:18:29 -0800415 std::string exec_dir = ::android::base::GetExecutableDirectory();
416
417 SysMemInfo mi;
418 std::string zram_mmstat_dir = exec_dir + "/testdata1/";
419 EXPECT_EQ(mi.mem_zram_kb(zram_mmstat_dir), 30504);
420
421 std::string zram_memused_dir = exec_dir + "/testdata2/";
422 EXPECT_EQ(mi.mem_zram_kb(zram_memused_dir), 30504);
423}
424
Sandeep Patil2f0b6eb2018-12-11 09:28:38 -0800425enum {
426 MEMINFO_TOTAL,
427 MEMINFO_FREE,
428 MEMINFO_BUFFERS,
429 MEMINFO_CACHED,
430 MEMINFO_SHMEM,
431 MEMINFO_SLAB,
432 MEMINFO_SLAB_RECLAIMABLE,
433 MEMINFO_SLAB_UNRECLAIMABLE,
434 MEMINFO_SWAP_TOTAL,
435 MEMINFO_SWAP_FREE,
436 MEMINFO_ZRAM_TOTAL,
437 MEMINFO_MAPPED,
438 MEMINFO_VMALLOC_USED,
439 MEMINFO_PAGE_TABLES,
440 MEMINFO_KERNEL_STACK,
441 MEMINFO_COUNT
442};
443
444TEST(SysMemInfoParser, TestZramWithTags) {
445 std::string meminfo = R"meminfo(MemTotal: 3019740 kB
446MemFree: 1809728 kB
447MemAvailable: 2546560 kB
448Buffers: 54736 kB
449Cached: 776052 kB
450SwapCached: 0 kB
451Active: 445856 kB
452Inactive: 459092 kB
453Active(anon): 78492 kB
454Inactive(anon): 2240 kB
455Active(file): 367364 kB
456Inactive(file): 456852 kB
457Unevictable: 3096 kB
458Mlocked: 3096 kB
459SwapTotal: 32768 kB
460SwapFree: 4096 kB
461Dirty: 32 kB
462Writeback: 0 kB
463AnonPages: 74988 kB
464Mapped: 62624 kB
465Shmem: 4020 kB
466Slab: 86464 kB
467SReclaimable: 44432 kB
468SUnreclaim: 42032 kB
469KernelStack: 4880 kB
470PageTables: 2900 kB
471NFS_Unstable: 0 kB
472Bounce: 0 kB
473WritebackTmp: 0 kB
474CommitLimit: 1509868 kB
475Committed_AS: 80296 kB
476VmallocTotal: 263061440 kB
477VmallocUsed: 65536 kB
478VmallocChunk: 0 kB
479AnonHugePages: 6144 kB
480ShmemHugePages: 0 kB
481ShmemPmdMapped: 0 kB
482CmaTotal: 131072 kB
483CmaFree: 130380 kB
484HugePages_Total: 0
485HugePages_Free: 0
486HugePages_Rsvd: 0
487HugePages_Surp: 0
488Hugepagesize: 2048 kB)meminfo";
489
490 TemporaryFile tf;
491 ASSERT_TRUE(tf.fd != -1);
492 ASSERT_TRUE(::android::base::WriteStringToFd(meminfo, tf.fd));
493 std::string file = std::string(tf.path);
494 std::vector<uint64_t> mem(MEMINFO_COUNT);
495 std::vector<std::string> tags(SysMemInfo::kDefaultSysMemInfoTags);
496 auto it = tags.begin();
497 tags.insert(it + MEMINFO_ZRAM_TOTAL, "Zram:");
498 SysMemInfo mi;
499
500 // Read system memory info
501 EXPECT_TRUE(mi.ReadMemInfo(tags, &mem, file));
502
503 EXPECT_EQ(mem[MEMINFO_TOTAL], 3019740);
504 EXPECT_EQ(mem[MEMINFO_FREE], 1809728);
505 EXPECT_EQ(mem[MEMINFO_BUFFERS], 54736);
506 EXPECT_EQ(mem[MEMINFO_CACHED], 776052);
507 EXPECT_EQ(mem[MEMINFO_SHMEM], 4020);
508 EXPECT_EQ(mem[MEMINFO_SLAB], 86464);
509 EXPECT_EQ(mem[MEMINFO_SLAB_RECLAIMABLE], 44432);
510 EXPECT_EQ(mem[MEMINFO_SLAB_UNRECLAIMABLE], 42032);
511 EXPECT_EQ(mem[MEMINFO_SWAP_TOTAL], 32768);
512 EXPECT_EQ(mem[MEMINFO_SWAP_FREE], 4096);
513 EXPECT_EQ(mem[MEMINFO_MAPPED], 62624);
514 EXPECT_EQ(mem[MEMINFO_VMALLOC_USED], 65536);
515 EXPECT_EQ(mem[MEMINFO_PAGE_TABLES], 2900);
516 EXPECT_EQ(mem[MEMINFO_KERNEL_STACK], 4880);
517}
518
Sandeep Patil54d87212018-08-29 17:10:47 -0700519int main(int argc, char** argv) {
520 ::testing::InitGoogleTest(&argc, argv);
521 if (argc <= 1) {
522 cerr << "Pid of a permanently sleeping process must be provided." << endl;
523 exit(EXIT_FAILURE);
524 }
525 ::android::base::InitLogging(argv, android::base::StderrLogger);
526 pid = std::stoi(std::string(argv[1]));
527 return RUN_ALL_TESTS();
528}