blob: 8dc884f640cb7db3662710ca98698370fa9cae65 [file] [log] [blame]
Christopher Ferris09385e72017-04-05 13:25:04 -07001/*
2 * Copyright (C) 2016 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
Christopher Ferris60521c72017-08-18 15:10:53 -070017#include <inttypes.h>
Christopher Ferris09385e72017-04-05 13:25:04 -070018#include <sys/mman.h>
19
20#include <android-base/file.h>
Christopher Ferris60521c72017-08-18 15:10:53 -070021#include <android-base/stringprintf.h>
Christopher Ferris09385e72017-04-05 13:25:04 -070022#include <android-base/test_utils.h>
23#include <gtest/gtest.h>
24
Christopher Ferrisd226a512017-07-14 10:37:19 -070025#include <unwindstack/Maps.h>
26
27namespace unwindstack {
Christopher Ferris09385e72017-04-05 13:25:04 -070028
Christopher Ferris60521c72017-08-18 15:10:53 -070029static void VerifyLine(std::string line, MapInfo* info) {
30 BufferMaps maps(line.c_str());
31
32 if (info == nullptr) {
33 ASSERT_FALSE(maps.Parse()) << "Failed on: " + line;
34 } else {
35 ASSERT_TRUE(maps.Parse()) << "Failed on: " + line;
36 MapInfo* element = maps.Get(0);
37 ASSERT_TRUE(element != nullptr) << "Failed on: " + line;
Christopher Ferrisbe788d82017-11-27 14:50:38 -080038 info->start = element->start;
39 info->end = element->end;
40 info->offset = element->offset;
41 info->flags = element->flags;
42 info->name = element->name;
43 info->elf_offset = element->elf_offset;
Christopher Ferris60521c72017-08-18 15:10:53 -070044 }
45}
46
47TEST(MapsTest, verify_parse_line) {
48 MapInfo info;
49
50 VerifyLine("01-02 rwxp 03 04:05 06\n", &info);
51 EXPECT_EQ(1U, info.start);
52 EXPECT_EQ(2U, info.end);
53 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, info.flags);
54 EXPECT_EQ(3U, info.offset);
55 EXPECT_EQ("", info.name);
56
57 VerifyLine("0a-0b ---s 0c 0d:0e 06 /fake/name\n", &info);
58 EXPECT_EQ(0xaU, info.start);
59 EXPECT_EQ(0xbU, info.end);
60 EXPECT_EQ(0U, info.flags);
61 EXPECT_EQ(0xcU, info.offset);
62 EXPECT_EQ("/fake/name", info.name);
63
64 VerifyLine("01-02 rwxp 03 04:05 06 /fake/name/again\n", &info);
65 EXPECT_EQ(1U, info.start);
66 EXPECT_EQ(2U, info.end);
67 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, info.flags);
68 EXPECT_EQ(3U, info.offset);
69 EXPECT_EQ("/fake/name/again", info.name);
70
71 VerifyLine("-00 rwxp 00 00:00 0\n", nullptr);
72 VerifyLine("00- rwxp 00 00:00 0\n", nullptr);
73 VerifyLine("00-00 rwxp 00 :00 0\n", nullptr);
74 VerifyLine("00-00 rwxp 00 00:00 \n", nullptr);
75 VerifyLine("x-00 rwxp 00 00:00 0\n", nullptr);
76 VerifyLine("00 -00 rwxp 00 00:00 0\n", nullptr);
77 VerifyLine("00-x rwxp 00 00:00 0\n", nullptr);
78 VerifyLine("00-x rwxp 00 00:00 0\n", nullptr);
79 VerifyLine("00-00x rwxp 00 00:00 0\n", nullptr);
80 VerifyLine("00-00 rwxp0 00 00:00 0\n", nullptr);
81 VerifyLine("00-00 rwxp0 00 00:00 0\n", nullptr);
82 VerifyLine("00-00 rwp 00 00:00 0\n", nullptr);
83 VerifyLine("00-00 rwxp 0000:00 0\n", nullptr);
84 VerifyLine("00-00 rwxp 00 00 :00 0\n", nullptr);
85 VerifyLine("00-00 rwxp 00 00: 00 0\n", nullptr);
86 VerifyLine("00-00 rwxp 00 00:000\n", nullptr);
87 VerifyLine("00-00 rwxp 00 00:00 0/fake\n", nullptr);
88 VerifyLine("00-00 xxxx 00 00:00 0 /fake\n", nullptr);
89 VerifyLine("00-00 ywxp 00 00:00 0 /fake\n", nullptr);
90 VerifyLine("00-00 ryxp 00 00:00 0 /fake\n", nullptr);
91 VerifyLine("00-00 rwyp 00 00:00 0 /fake\n", nullptr);
92 VerifyLine("00-00 rwx- 00 00:00 0 /fake\n", nullptr);
93 VerifyLine("0\n", nullptr);
94 VerifyLine("00\n", nullptr);
95 VerifyLine("00-\n", nullptr);
96 VerifyLine("00-0\n", nullptr);
97 VerifyLine("00-00\n", nullptr);
98 VerifyLine("00-00 \n", nullptr);
99 VerifyLine("00-00 -\n", nullptr);
100 VerifyLine("00-00 r\n", nullptr);
101 VerifyLine("00-00 --\n", nullptr);
102 VerifyLine("00-00 rw\n", nullptr);
103 VerifyLine("00-00 ---\n", nullptr);
104 VerifyLine("00-00 rwx\n", nullptr);
105 VerifyLine("00-00 ---s\n", nullptr);
106 VerifyLine("00-00 ---p\n", nullptr);
107 VerifyLine("00-00 ---s 0\n", nullptr);
108 VerifyLine("00-00 ---p 0 \n", nullptr);
109 VerifyLine("00-00 ---p 0 0\n", nullptr);
110 VerifyLine("00-00 ---p 0 0:\n", nullptr);
111 VerifyLine("00-00 ---p 0 0:0\n", nullptr);
112 VerifyLine("00-00 ---p 0 0:0 \n", nullptr);
113
114 // Line to verify that the parser will detect a completely malformed line
115 // properly.
116 VerifyLine("7ffff7dda000-7ffff7dfd7ffff7ff3000-7ffff7ff4000 ---p 0000f000 fc:02 44171565\n",
117 nullptr);
118}
119
120TEST(MapsTest, verify_large_values) {
121 MapInfo info;
122#if defined(__LP64__)
123 VerifyLine("fabcdef012345678-f12345678abcdef8 rwxp f0b0d0f010305070 00:00 0\n", &info);
124 EXPECT_EQ(0xfabcdef012345678UL, info.start);
125 EXPECT_EQ(0xf12345678abcdef8UL, info.end);
126 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, info.flags);
127 EXPECT_EQ(0xf0b0d0f010305070UL, info.offset);
128#else
129 VerifyLine("f2345678-fabcdef8 rwxp f0305070 00:00 0\n", &info);
130 EXPECT_EQ(0xf2345678UL, info.start);
131 EXPECT_EQ(0xfabcdef8UL, info.end);
132 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, info.flags);
133 EXPECT_EQ(0xf0305070UL, info.offset);
134#endif
135}
136
Christopher Ferris09385e72017-04-05 13:25:04 -0700137TEST(MapsTest, parse_permissions) {
138 BufferMaps maps(
Christopher Ferris60521c72017-08-18 15:10:53 -0700139 "1000-2000 ---s 00000000 00:00 0\n"
140 "2000-3000 r--s 00000000 00:00 0\n"
141 "3000-4000 -w-s 00000000 00:00 0\n"
142 "4000-5000 --xp 00000000 00:00 0\n"
143 "5000-6000 rwxp 00000000 00:00 0\n");
Christopher Ferris09385e72017-04-05 13:25:04 -0700144
145 ASSERT_TRUE(maps.Parse());
146 ASSERT_EQ(5U, maps.Total());
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800147
148 MapInfo* info = maps.Get(0);
149 ASSERT_TRUE(info != nullptr);
150 EXPECT_EQ(PROT_NONE, info->flags);
151 EXPECT_EQ(0x1000U, info->start);
152 EXPECT_EQ(0x2000U, info->end);
153 EXPECT_EQ(0U, info->offset);
154 EXPECT_EQ("", info->name);
155
156 info = maps.Get(1);
157 ASSERT_TRUE(info != nullptr);
158 EXPECT_EQ(PROT_READ, info->flags);
159 EXPECT_EQ(0x2000U, info->start);
160 EXPECT_EQ(0x3000U, info->end);
161 EXPECT_EQ(0U, info->offset);
162 EXPECT_EQ("", info->name);
163
164 info = maps.Get(2);
165 ASSERT_TRUE(info != nullptr);
166 EXPECT_EQ(PROT_WRITE, info->flags);
167 EXPECT_EQ(0x3000U, info->start);
168 EXPECT_EQ(0x4000U, info->end);
169 EXPECT_EQ(0U, info->offset);
170 EXPECT_EQ("", info->name);
171
172 info = maps.Get(3);
173 ASSERT_TRUE(info != nullptr);
174 EXPECT_EQ(PROT_EXEC, info->flags);
175 EXPECT_EQ(0x4000U, info->start);
176 EXPECT_EQ(0x5000U, info->end);
177 EXPECT_EQ(0U, info->offset);
178 EXPECT_EQ("", info->name);
179
180 info = maps.Get(4);
181 ASSERT_TRUE(info != nullptr);
182 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, info->flags);
183 EXPECT_EQ(0x5000U, info->start);
184 EXPECT_EQ(0x6000U, info->end);
185 EXPECT_EQ(0U, info->offset);
186 EXPECT_EQ("", info->name);
187
188 ASSERT_TRUE(maps.Get(5) == nullptr);
Christopher Ferris09385e72017-04-05 13:25:04 -0700189}
190
191TEST(MapsTest, parse_name) {
192 BufferMaps maps(
Christopher Ferris60521c72017-08-18 15:10:53 -0700193 "7b29b000-7b29e000 rw-p 00000000 00:00 0\n"
194 "7b29e000-7b29f000 rw-p 00000000 00:00 0 /system/lib/fake.so\n"
195 "7b29f000-7b2a0000 rw-p 00000000 00:00 0");
Christopher Ferris09385e72017-04-05 13:25:04 -0700196
197 ASSERT_TRUE(maps.Parse());
198 ASSERT_EQ(3U, maps.Total());
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800199
200 MapInfo* info = maps.Get(0);
201 ASSERT_TRUE(info != nullptr);
202 EXPECT_EQ("", info->name);
203 EXPECT_EQ(0x7b29b000U, info->start);
204 EXPECT_EQ(0x7b29e000U, info->end);
205 EXPECT_EQ(0U, info->offset);
206 EXPECT_EQ(PROT_READ | PROT_WRITE, info->flags);
207
208 info = maps.Get(1);
209 ASSERT_TRUE(info != nullptr);
210 EXPECT_EQ("/system/lib/fake.so", info->name);
211 EXPECT_EQ(0x7b29e000U, info->start);
212 EXPECT_EQ(0x7b29f000U, info->end);
213 EXPECT_EQ(0U, info->offset);
214 EXPECT_EQ(PROT_READ | PROT_WRITE, info->flags);
215
216 info = maps.Get(2);
217 ASSERT_TRUE(info != nullptr);
218 EXPECT_EQ("", info->name);
219 EXPECT_EQ(0x7b29f000U, info->start);
220 EXPECT_EQ(0x7b2a0000U, info->end);
221 EXPECT_EQ(0U, info->offset);
222 EXPECT_EQ(PROT_READ | PROT_WRITE, info->flags);
223
224 ASSERT_TRUE(maps.Get(3) == nullptr);
Christopher Ferris09385e72017-04-05 13:25:04 -0700225}
226
227TEST(MapsTest, parse_offset) {
228 BufferMaps maps(
229 "a000-e000 rw-p 00000000 00:00 0 /system/lib/fake.so\n"
230 "e000-f000 rw-p 00a12345 00:00 0 /system/lib/fake.so\n");
231
232 ASSERT_TRUE(maps.Parse());
233 ASSERT_EQ(2U, maps.Total());
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800234
235 MapInfo* info = maps.Get(0);
236 ASSERT_TRUE(info != nullptr);
237 EXPECT_EQ(0U, info->offset);
238 EXPECT_EQ(0xa000U, info->start);
239 EXPECT_EQ(0xe000U, info->end);
240 EXPECT_EQ(PROT_READ | PROT_WRITE, info->flags);
241 EXPECT_EQ("/system/lib/fake.so", info->name);
242
243 info = maps.Get(1);
244 ASSERT_TRUE(info != nullptr);
245 EXPECT_EQ(0xa12345U, info->offset);
246 EXPECT_EQ(0xe000U, info->start);
247 EXPECT_EQ(0xf000U, info->end);
248 EXPECT_EQ(PROT_READ | PROT_WRITE, info->flags);
249 EXPECT_EQ("/system/lib/fake.so", info->name);
250
251 ASSERT_TRUE(maps.Get(2) == nullptr);
252}
253
254TEST(MapsTest, iterate) {
255 BufferMaps maps(
256 "a000-e000 rw-p 00000000 00:00 0 /system/lib/fake.so\n"
257 "e000-f000 rw-p 00a12345 00:00 0 /system/lib/fake.so\n");
258
259 ASSERT_TRUE(maps.Parse());
260 ASSERT_EQ(2U, maps.Total());
261
262 Maps::iterator it = maps.begin();
263 EXPECT_EQ(0xa000U, (*it)->start);
264 EXPECT_EQ(0xe000U, (*it)->end);
Christopher Ferris09385e72017-04-05 13:25:04 -0700265 ++it;
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800266 EXPECT_EQ(0xe000U, (*it)->start);
267 EXPECT_EQ(0xf000U, (*it)->end);
Christopher Ferris09385e72017-04-05 13:25:04 -0700268 ++it;
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800269 EXPECT_EQ(maps.end(), it);
270}
271
272TEST(MapsTest, const_iterate) {
273 BufferMaps maps(
274 "a000-e000 rw-p 00000000 00:00 0 /system/lib/fake.so\n"
275 "e000-f000 rw-p 00a12345 00:00 0 /system/lib/fake.so\n");
276
277 ASSERT_TRUE(maps.Parse());
278 ASSERT_EQ(2U, maps.Total());
279
280 Maps::const_iterator it = maps.begin();
281 EXPECT_EQ(0xa000U, (*it)->start);
282 EXPECT_EQ(0xe000U, (*it)->end);
283 ++it;
284 EXPECT_EQ(0xe000U, (*it)->start);
285 EXPECT_EQ(0xf000U, (*it)->end);
286 ++it;
287 EXPECT_EQ(maps.end(), it);
Christopher Ferris09385e72017-04-05 13:25:04 -0700288}
289
290TEST(MapsTest, device) {
291 BufferMaps maps(
292 "a000-e000 rw-p 00000000 00:00 0 /dev/\n"
293 "f000-f100 rw-p 00000000 00:00 0 /dev/does_not_exist\n"
294 "f100-f200 rw-p 00000000 00:00 0 /dev/ashmem/does_not_exist\n"
295 "f200-f300 rw-p 00000000 00:00 0 /devsomething/does_not_exist\n");
296
297 ASSERT_TRUE(maps.Parse());
298 ASSERT_EQ(4U, maps.Total());
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800299
300 MapInfo* info = maps.Get(0);
301 ASSERT_TRUE(info != nullptr);
302 EXPECT_TRUE(info->flags & 0x8000);
303 EXPECT_EQ("/dev/", info->name);
304
305 info = maps.Get(1);
306 EXPECT_TRUE(info->flags & 0x8000);
307 EXPECT_EQ("/dev/does_not_exist", info->name);
308
309 info = maps.Get(2);
310 EXPECT_FALSE(info->flags & 0x8000);
311 EXPECT_EQ("/dev/ashmem/does_not_exist", info->name);
312
313 info = maps.Get(3);
314 EXPECT_FALSE(info->flags & 0x8000);
315 EXPECT_EQ("/devsomething/does_not_exist", info->name);
Christopher Ferris09385e72017-04-05 13:25:04 -0700316}
317
318TEST(MapsTest, file_smoke) {
319 TemporaryFile tf;
320 ASSERT_TRUE(tf.fd != -1);
321
322 ASSERT_TRUE(
Christopher Ferris60521c72017-08-18 15:10:53 -0700323 android::base::WriteStringToFile("7b29b000-7b29e000 r-xp a0000000 00:00 0 /fake.so\n"
324 "7b2b0000-7b2e0000 r-xp b0000000 00:00 0 /fake2.so\n"
325 "7b2e0000-7b2f0000 r-xp c0000000 00:00 0 /fake3.so\n",
Christopher Ferris09385e72017-04-05 13:25:04 -0700326 tf.path, 0660, getuid(), getgid()));
327
328 FileMaps maps(tf.path);
329
330 ASSERT_TRUE(maps.Parse());
331 ASSERT_EQ(3U, maps.Total());
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800332
333 MapInfo* info = maps.Get(0);
334 ASSERT_TRUE(info != nullptr);
335 EXPECT_EQ(0x7b29b000U, info->start);
336 EXPECT_EQ(0x7b29e000U, info->end);
337 EXPECT_EQ(0xa0000000U, info->offset);
338 EXPECT_EQ(PROT_READ | PROT_EXEC, info->flags);
339 EXPECT_EQ("/fake.so", info->name);
340
341 info = maps.Get(1);
342 ASSERT_TRUE(info != nullptr);
343 EXPECT_EQ(0x7b2b0000U, info->start);
344 EXPECT_EQ(0x7b2e0000U, info->end);
345 EXPECT_EQ(0xb0000000U, info->offset);
346 EXPECT_EQ(PROT_READ | PROT_EXEC, info->flags);
347 EXPECT_EQ("/fake2.so", info->name);
348
349 info = maps.Get(2);
350 ASSERT_TRUE(info != nullptr);
351 EXPECT_EQ(0x7b2e0000U, info->start);
352 EXPECT_EQ(0x7b2f0000U, info->end);
353 EXPECT_EQ(0xc0000000U, info->offset);
354 EXPECT_EQ(PROT_READ | PROT_EXEC, info->flags);
355 EXPECT_EQ("/fake3.so", info->name);
356
357 ASSERT_TRUE(maps.Get(3) == nullptr);
Christopher Ferris09385e72017-04-05 13:25:04 -0700358}
359
Christopher Ferris60521c72017-08-18 15:10:53 -0700360TEST(MapsTest, file_no_map_name) {
361 TemporaryFile tf;
362 ASSERT_TRUE(tf.fd != -1);
363
364 ASSERT_TRUE(
365 android::base::WriteStringToFile("7b29b000-7b29e000 r-xp a0000000 00:00 0\n"
366 "7b2b0000-7b2e0000 r-xp b0000000 00:00 0 /fake2.so\n"
367 "7b2e0000-7b2f0000 r-xp c0000000 00:00 0 \n",
368 tf.path, 0660, getuid(), getgid()));
369
370 FileMaps maps(tf.path);
371
372 ASSERT_TRUE(maps.Parse());
373 ASSERT_EQ(3U, maps.Total());
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800374
375 MapInfo* info = maps.Get(0);
376 ASSERT_TRUE(info != nullptr);
377 EXPECT_EQ(0x7b29b000U, info->start);
378 EXPECT_EQ(0x7b29e000U, info->end);
379 EXPECT_EQ(0xa0000000U, info->offset);
380 EXPECT_EQ(PROT_READ | PROT_EXEC, info->flags);
381 EXPECT_EQ("", info->name);
382
383 info = maps.Get(1);
384 ASSERT_TRUE(info != nullptr);
385 EXPECT_EQ(0x7b2b0000U, info->start);
386 EXPECT_EQ(0x7b2e0000U, info->end);
387 EXPECT_EQ(0xb0000000U, info->offset);
388 EXPECT_EQ(PROT_READ | PROT_EXEC, info->flags);
389 EXPECT_EQ("/fake2.so", info->name);
390
391 info = maps.Get(2);
392 ASSERT_TRUE(info != nullptr);
393 EXPECT_EQ(0x7b2e0000U, info->start);
394 EXPECT_EQ(0x7b2f0000U, info->end);
395 EXPECT_EQ(0xc0000000U, info->offset);
396 EXPECT_EQ(PROT_READ | PROT_EXEC, info->flags);
397 EXPECT_EQ("", info->name);
398
399 ASSERT_TRUE(maps.Get(3) == nullptr);
Christopher Ferris60521c72017-08-18 15:10:53 -0700400}
401
402// Verify that a file that crosses a buffer is parsed correctly.
403static std::string CreateEntry(size_t index) {
404 return android::base::StringPrintf("%08zx-%08zx rwxp 0000 00:00 0\n", index * 4096,
405 (index + 1) * 4096);
406}
407
408TEST(MapsTest, file_buffer_cross) {
409 constexpr size_t kBufferSize = 2048;
410 TemporaryFile tf;
411 ASSERT_TRUE(tf.fd != -1);
412
413 // Compute how many to add in the first buffer.
414 size_t entry_len = CreateEntry(0).size();
415 size_t index;
416 std::string file_data;
417 for (index = 0; index < kBufferSize / entry_len; index++) {
418 file_data += CreateEntry(index);
419 }
420 // Add a long name to make sure that the first buffer does not contain a
421 // complete line.
422 // Remove the last newline.
423 size_t extra = 0;
424 size_t leftover = kBufferSize % entry_len;
425 size_t overlap1_index = 0;
426 std::string overlap1_name;
427 if (leftover == 0) {
428 // Exact match, add a long name to cross over the value.
429 overlap1_name = "/fake/name/is/long/on/purpose";
430 file_data.erase(file_data.size() - 1);
431 file_data += ' ' + overlap1_name + '\n';
432 extra = entry_len + overlap1_name.size() + 1;
433 overlap1_index = index;
434 }
435
436 // Compute how many need to go in to hit the buffer boundary exactly.
437 size_t bytes_left_in_buffer = kBufferSize - extra;
438 size_t entries_to_add = bytes_left_in_buffer / entry_len + index;
439 for (; index < entries_to_add; index++) {
440 file_data += CreateEntry(index);
441 }
442
443 // Now figure out how many bytes to add to get exactly to the buffer boundary.
444 leftover = bytes_left_in_buffer % entry_len;
445 std::string overlap2_name;
446 size_t overlap2_index = 0;
447 if (leftover != 0) {
448 file_data.erase(file_data.size() - 1);
449 file_data += ' ';
450 overlap2_name = std::string(leftover - 1, 'x');
451 file_data += overlap2_name + '\n';
452 overlap2_index = index - 1;
453 }
454
455 // Now add a few entries on the next page.
456 for (size_t start = index; index < start + 10; index++) {
457 file_data += CreateEntry(index);
458 }
459
460 ASSERT_TRUE(android::base::WriteStringToFile(file_data, tf.path, 0660, getuid(), getgid()));
461
462 FileMaps maps(tf.path);
463 ASSERT_TRUE(maps.Parse());
464 EXPECT_EQ(index, maps.Total());
465 // Verify all of the maps.
466 for (size_t i = 0; i < index; i++) {
467 MapInfo* info = maps.Get(i);
468 ASSERT_TRUE(info != nullptr) << "Failed verifying index " + std::to_string(i);
469 EXPECT_EQ(i * 4096, info->start) << "Failed verifying index " + std::to_string(i);
470 EXPECT_EQ((i + 1) * 4096, info->end) << "Failed verifying index " + std::to_string(i);
471 EXPECT_EQ(0U, info->offset) << "Failed verifying index " + std::to_string(i);
472 if (overlap1_index != 0 && i == overlap1_index) {
473 EXPECT_EQ(overlap1_name, info->name) << "Failed verifying overlap1 name " + std::to_string(i);
474 } else if (overlap2_index != 0 && i == overlap2_index) {
475 EXPECT_EQ(overlap2_name, info->name) << "Failed verifying overlap2 name " + std::to_string(i);
476 } else {
477 EXPECT_EQ("", info->name) << "Failed verifying index " + std::to_string(i);
478 }
479 }
480}
481
482TEST(MapsTest, file_should_fail) {
483 TemporaryFile tf;
484 ASSERT_TRUE(tf.fd != -1);
485
486 ASSERT_TRUE(android::base::WriteStringToFile(
487 "7ffff7dda000-7ffff7dfd7ffff7ff3000-7ffff7ff4000 ---p 0000f000 fc:02 44171565\n", tf.path,
488 0660, getuid(), getgid()));
489
490 FileMaps maps(tf.path);
491
492 ASSERT_FALSE(maps.Parse());
493}
494
495// Create a maps file that is extremely large.
496TEST(MapsTest, large_file) {
497 TemporaryFile tf;
498 ASSERT_TRUE(tf.fd != -1);
499
500 std::string file_data;
501 uint64_t start = 0x700000;
502 for (size_t i = 0; i < 5000; i++) {
503 file_data +=
504 android::base::StringPrintf("%" PRIx64 "-%" PRIx64 " r-xp 1000 00:0 0 /fake%zu.so\n",
505 start + i * 4096, start + (i + 1) * 4096, i);
506 }
507
508 ASSERT_TRUE(android::base::WriteStringToFile(file_data, tf.path, 0660, getuid(), getgid()));
509
510 FileMaps maps(tf.path);
511
512 ASSERT_TRUE(maps.Parse());
513 ASSERT_EQ(5000U, maps.Total());
514 for (size_t i = 0; i < 5000; i++) {
515 MapInfo* info = maps.Get(i);
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800516 EXPECT_EQ(start + i * 4096, info->start) << "Failed at map " + std::to_string(i);
517 EXPECT_EQ(start + (i + 1) * 4096, info->end) << "Failed at map " + std::to_string(i);
Christopher Ferris60521c72017-08-18 15:10:53 -0700518 std::string name = "/fake" + std::to_string(i) + ".so";
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800519 EXPECT_EQ(name, info->name) << "Failed at map " + std::to_string(i);
Christopher Ferris60521c72017-08-18 15:10:53 -0700520 }
521}
522
Christopher Ferris09385e72017-04-05 13:25:04 -0700523TEST(MapsTest, find) {
524 BufferMaps maps(
525 "1000-2000 r--p 00000010 00:00 0 /system/lib/fake1.so\n"
526 "3000-4000 -w-p 00000020 00:00 0 /system/lib/fake2.so\n"
527 "6000-8000 --xp 00000030 00:00 0 /system/lib/fake3.so\n"
528 "a000-b000 rw-p 00000040 00:00 0 /system/lib/fake4.so\n"
529 "e000-f000 rwxp 00000050 00:00 0 /system/lib/fake5.so\n");
530 ASSERT_TRUE(maps.Parse());
531 ASSERT_EQ(5U, maps.Total());
532
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800533 EXPECT_TRUE(maps.Find(0x500) == nullptr);
534 EXPECT_TRUE(maps.Find(0x2000) == nullptr);
535 EXPECT_TRUE(maps.Find(0x5010) == nullptr);
536 EXPECT_TRUE(maps.Find(0x9a00) == nullptr);
537 EXPECT_TRUE(maps.Find(0xf000) == nullptr);
538 EXPECT_TRUE(maps.Find(0xf010) == nullptr);
Christopher Ferris09385e72017-04-05 13:25:04 -0700539
540 MapInfo* info = maps.Find(0x1000);
541 ASSERT_TRUE(info != nullptr);
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800542 EXPECT_EQ(0x1000U, info->start);
543 EXPECT_EQ(0x2000U, info->end);
544 EXPECT_EQ(0x10U, info->offset);
545 EXPECT_EQ(PROT_READ, info->flags);
546 EXPECT_EQ("/system/lib/fake1.so", info->name);
Christopher Ferris09385e72017-04-05 13:25:04 -0700547
548 info = maps.Find(0x3020);
549 ASSERT_TRUE(info != nullptr);
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800550 EXPECT_EQ(0x3000U, info->start);
551 EXPECT_EQ(0x4000U, info->end);
552 EXPECT_EQ(0x20U, info->offset);
553 EXPECT_EQ(PROT_WRITE, info->flags);
554 EXPECT_EQ("/system/lib/fake2.so", info->name);
Christopher Ferris09385e72017-04-05 13:25:04 -0700555
556 info = maps.Find(0x6020);
557 ASSERT_TRUE(info != nullptr);
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800558 EXPECT_EQ(0x6000U, info->start);
559 EXPECT_EQ(0x8000U, info->end);
560 EXPECT_EQ(0x30U, info->offset);
561 EXPECT_EQ(PROT_EXEC, info->flags);
562 EXPECT_EQ("/system/lib/fake3.so", info->name);
Christopher Ferris09385e72017-04-05 13:25:04 -0700563
564 info = maps.Find(0xafff);
565 ASSERT_TRUE(info != nullptr);
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800566 EXPECT_EQ(0xa000U, info->start);
567 EXPECT_EQ(0xb000U, info->end);
568 EXPECT_EQ(0x40U, info->offset);
569 EXPECT_EQ(PROT_READ | PROT_WRITE, info->flags);
570 EXPECT_EQ("/system/lib/fake4.so", info->name);
Christopher Ferris09385e72017-04-05 13:25:04 -0700571
572 info = maps.Find(0xe500);
573 ASSERT_TRUE(info != nullptr);
Christopher Ferrisbe788d82017-11-27 14:50:38 -0800574 EXPECT_EQ(0xe000U, info->start);
575 EXPECT_EQ(0xf000U, info->end);
576 EXPECT_EQ(0x50U, info->offset);
577 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, info->flags);
578 EXPECT_EQ("/system/lib/fake5.so", info->name);
Christopher Ferris09385e72017-04-05 13:25:04 -0700579}
Christopher Ferrisd226a512017-07-14 10:37:19 -0700580
581} // namespace unwindstack