blob: 16640a18be447a63e46eda21c45ad3ca7a36ae7f [file] [log] [blame]
Christopher Ferrisf6f691b2017-09-25 19:23:07 -07001/*
2 * Copyright (C) 2017 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 <elf.h>
18#include <stdint.h>
19#include <sys/mman.h>
20
21#include <memory>
22#include <set>
23#include <string>
24
25#include <gtest/gtest.h>
26
27#include <unwindstack/Elf.h>
28#include <unwindstack/Maps.h>
29#include <unwindstack/Memory.h>
30#include <unwindstack/Regs.h>
31#include <unwindstack/Unwinder.h>
32
33#include "ElfFake.h"
34#include "MemoryFake.h"
35#include "RegsFake.h"
36
37namespace unwindstack {
38
39class MapsFake : public Maps {
40 public:
41 MapsFake() = default;
42 virtual ~MapsFake() = default;
43
44 bool Parse() { return true; }
45
46 void FakeClear() { maps_.clear(); }
47
48 void FakeAddMapInfo(const MapInfo& map_info) { maps_.push_back(map_info); }
49};
50
51class UnwinderTest : public ::testing::Test {
52 protected:
53 static void SetUpTestCase() {
54 maps_.FakeClear();
55 MapInfo info;
56 info.name = "/system/fake/libc.so";
57 info.start = 0x1000;
58 info.end = 0x8000;
59 info.offset = 0;
60 info.flags = PROT_READ | PROT_WRITE;
61 ElfFake* elf = new ElfFake(nullptr);
62 info.elf = elf;
63 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
64 info.elf_offset = 0;
65 maps_.FakeAddMapInfo(info);
66
67 info.name = "[stack]";
68 info.start = 0x10000;
69 info.end = 0x12000;
70 info.flags = PROT_READ | PROT_WRITE;
71 info.elf = nullptr;
72 maps_.FakeAddMapInfo(info);
73
74 info.name = "/dev/fake_device";
75 info.start = 0x13000;
76 info.end = 0x15000;
77 info.flags = PROT_READ | PROT_WRITE | MAPS_FLAGS_DEVICE_MAP;
78 info.elf = nullptr;
79 maps_.FakeAddMapInfo(info);
80
81 info.name = "/system/fake/libunwind.so";
82 info.start = 0x20000;
83 info.end = 0x22000;
84 info.flags = PROT_READ | PROT_WRITE;
85 elf = new ElfFake(nullptr);
86 info.elf = elf;
87 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
88 maps_.FakeAddMapInfo(info);
89
90 info.name = "/fake/libanother.so";
91 info.start = 0x23000;
92 info.end = 0x24000;
93 info.flags = PROT_READ | PROT_WRITE;
94 elf = new ElfFake(nullptr);
95 info.elf = elf;
96 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
97 maps_.FakeAddMapInfo(info);
Christopher Ferrisc5a3baa2017-10-17 18:42:03 -070098
99 info.name = "/fake/fake.apk";
100 info.start = 0x43000;
101 info.end = 0x44000;
102 info.offset = 0x1d000;
103 info.flags = PROT_READ | PROT_WRITE;
104 elf = new ElfFake(nullptr);
105 info.elf = elf;
106 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
107 maps_.FakeAddMapInfo(info);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700108 }
109
110 void SetUp() override {
111 ElfInterfaceFake::FakeClear();
112 regs_.FakeSetMachineType(EM_ARM);
113 }
114
115 static MapsFake maps_;
116 static RegsFake regs_;
117 static std::shared_ptr<Memory> process_memory_;
118};
119
120MapsFake UnwinderTest::maps_;
121RegsFake UnwinderTest::regs_(5, 0);
122std::shared_ptr<Memory> UnwinderTest::process_memory_(nullptr);
123
124TEST_F(UnwinderTest, multiple_frames) {
125 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
126 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
127 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
128
129 regs_.FakeSetPc(0x1000);
130 regs_.FakeSetSp(0x10000);
131 ElfInterfaceFake::FakePushStepData(StepData(0x1102, 0x10010, false));
132 ElfInterfaceFake::FakePushStepData(StepData(0x1202, 0x10020, false));
133 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
134
135 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
136 unwinder.Unwind();
137
138 ASSERT_EQ(3U, unwinder.NumFrames());
139
140 auto* frame = &unwinder.frames()[0];
141 EXPECT_EQ(0U, frame->num);
142 EXPECT_EQ(0U, frame->rel_pc);
143 EXPECT_EQ(0x1000U, frame->pc);
144 EXPECT_EQ(0x10000U, frame->sp);
145 EXPECT_EQ("Frame0", frame->function_name);
146 EXPECT_EQ(0U, frame->function_offset);
147 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
148 EXPECT_EQ(0U, frame->map_offset);
149 EXPECT_EQ(0x1000U, frame->map_start);
150 EXPECT_EQ(0x8000U, frame->map_end);
151 EXPECT_EQ(0U, frame->map_load_bias);
152 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
153
154 frame = &unwinder.frames()[1];
155 EXPECT_EQ(1U, frame->num);
156 EXPECT_EQ(0x100U, frame->rel_pc);
157 EXPECT_EQ(0x1100U, frame->pc);
158 EXPECT_EQ(0x10010U, frame->sp);
159 EXPECT_EQ("Frame1", frame->function_name);
160 EXPECT_EQ(1U, frame->function_offset);
161 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
162 EXPECT_EQ(0U, frame->map_offset);
163 EXPECT_EQ(0x1000U, frame->map_start);
164 EXPECT_EQ(0x8000U, frame->map_end);
165 EXPECT_EQ(0U, frame->map_load_bias);
166 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
167
168 frame = &unwinder.frames()[2];
169 EXPECT_EQ(2U, frame->num);
170 EXPECT_EQ(0x200U, frame->rel_pc);
171 EXPECT_EQ(0x1200U, frame->pc);
172 EXPECT_EQ(0x10020U, frame->sp);
173 EXPECT_EQ("Frame2", frame->function_name);
174 EXPECT_EQ(2U, frame->function_offset);
175 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
176 EXPECT_EQ(0U, frame->map_offset);
177 EXPECT_EQ(0x1000U, frame->map_start);
178 EXPECT_EQ(0x8000U, frame->map_end);
179 EXPECT_EQ(0U, frame->map_load_bias);
180 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
181}
182
Christopher Ferrisc5a3baa2017-10-17 18:42:03 -0700183TEST_F(UnwinderTest, non_zero_map_offset) {
184 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
185
186 regs_.FakeSetPc(0x43000);
187 regs_.FakeSetSp(0x10000);
188 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
189
190 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
191 unwinder.Unwind();
192
193 ASSERT_EQ(1U, unwinder.NumFrames());
194
195 auto* frame = &unwinder.frames()[0];
196 EXPECT_EQ(0U, frame->num);
197 EXPECT_EQ(0U, frame->rel_pc);
198 EXPECT_EQ(0x43000U, frame->pc);
199 EXPECT_EQ(0x10000U, frame->sp);
200 EXPECT_EQ("Frame0", frame->function_name);
201 EXPECT_EQ(0U, frame->function_offset);
202 EXPECT_EQ("/fake/fake.apk", frame->map_name);
203 EXPECT_EQ(0x1d000U, frame->map_offset);
204 EXPECT_EQ(0x43000U, frame->map_start);
205 EXPECT_EQ(0x44000U, frame->map_end);
206 EXPECT_EQ(0U, frame->map_load_bias);
207 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
208}
209
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700210// Verify that no attempt to continue after the step indicates it is done.
211TEST_F(UnwinderTest, no_frames_after_finished) {
212 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
213 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
214 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
215 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame3", 3));
216 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame4", 4));
217
218 regs_.FakeSetPc(0x1000);
219 regs_.FakeSetSp(0x10000);
220 ElfInterfaceFake::FakePushStepData(StepData(0x1000, 0x10000, true));
221 ElfInterfaceFake::FakePushStepData(StepData(0x1102, 0x10010, false));
222 ElfInterfaceFake::FakePushStepData(StepData(0x1202, 0x10020, false));
223
224 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
225 unwinder.Unwind();
226
227 ASSERT_EQ(1U, unwinder.NumFrames());
228
229 auto* frame = &unwinder.frames()[0];
230 EXPECT_EQ(0U, frame->num);
231 EXPECT_EQ(0U, frame->rel_pc);
232 EXPECT_EQ(0x1000U, frame->pc);
233 EXPECT_EQ(0x10000U, frame->sp);
234 EXPECT_EQ("Frame0", frame->function_name);
235 EXPECT_EQ(0U, frame->function_offset);
236 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
237 EXPECT_EQ(0U, frame->map_offset);
238 EXPECT_EQ(0x1000U, frame->map_start);
239 EXPECT_EQ(0x8000U, frame->map_end);
240 EXPECT_EQ(0U, frame->map_load_bias);
241 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
242}
243
244// Verify the maximum frames to save.
245TEST_F(UnwinderTest, max_frames) {
246 for (size_t i = 0; i < 30; i++) {
247 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame" + std::to_string(i), i));
248 ElfInterfaceFake::FakePushStepData(StepData(0x1102 + i * 0x100, 0x10010 + i * 0x10, false));
249 }
250
251 regs_.FakeSetPc(0x1000);
252 regs_.FakeSetSp(0x10000);
253
254 Unwinder unwinder(20, &maps_, &regs_, process_memory_);
255 unwinder.Unwind();
256
257 ASSERT_EQ(20U, unwinder.NumFrames());
258
259 for (size_t i = 0; i < 20; i++) {
260 auto* frame = &unwinder.frames()[i];
261 EXPECT_EQ(i, frame->num);
262 EXPECT_EQ(i * 0x100, frame->rel_pc) << "Failed at frame " << i;
263 EXPECT_EQ(0x1000 + i * 0x100, frame->pc) << "Failed at frame " << i;
264 EXPECT_EQ(0x10000 + 0x10 * i, frame->sp) << "Failed at frame " << i;
265 EXPECT_EQ("Frame" + std::to_string(i), frame->function_name) << "Failed at frame " << i;
266 EXPECT_EQ(i, frame->function_offset) << "Failed at frame " << i;
267 EXPECT_EQ("/system/fake/libc.so", frame->map_name) << "Failed at frame " << i;
268 EXPECT_EQ(0U, frame->map_offset) << "Failed at frame " << i;
269 EXPECT_EQ(0x1000U, frame->map_start) << "Failed at frame " << i;
270 EXPECT_EQ(0x8000U, frame->map_end) << "Failed at frame " << i;
271 EXPECT_EQ(0U, frame->map_load_bias) << "Failed at frame " << i;
272 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags) << "Failed at frame " << i;
273 }
274}
275
276// Verify that initial map names frames are removed.
277TEST_F(UnwinderTest, verify_frames_skipped) {
278 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
279 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
280 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
281
282 regs_.FakeSetPc(0x20000);
283 regs_.FakeSetSp(0x10000);
284 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10010, false));
285 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
286 ElfInterfaceFake::FakePushStepData(StepData(0x20002, 0x10030, false));
287 ElfInterfaceFake::FakePushStepData(StepData(0x21002, 0x10040, false));
288 ElfInterfaceFake::FakePushStepData(StepData(0x1002, 0x10050, false));
289 ElfInterfaceFake::FakePushStepData(StepData(0x21002, 0x10060, false));
290 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10070, false));
291 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
292
293 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
294 std::set<std::string> skip_set{"libunwind.so", "libanother.so"};
295 unwinder.Unwind(&skip_set);
296
297 ASSERT_EQ(3U, unwinder.NumFrames());
298
299 auto* frame = &unwinder.frames()[0];
300 EXPECT_EQ(0U, frame->num);
301 EXPECT_EQ(0U, frame->rel_pc);
302 EXPECT_EQ(0x1000U, frame->pc);
303 EXPECT_EQ(0x10050U, frame->sp);
304 EXPECT_EQ("Frame0", frame->function_name);
305 EXPECT_EQ(0U, frame->function_offset);
306 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
307 EXPECT_EQ(0U, frame->map_offset);
308 EXPECT_EQ(0x1000U, frame->map_start);
309 EXPECT_EQ(0x8000U, frame->map_end);
310 EXPECT_EQ(0U, frame->map_load_bias);
311 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
312
313 frame = &unwinder.frames()[1];
314 EXPECT_EQ(1U, frame->num);
315 EXPECT_EQ(0x1000U, frame->rel_pc);
316 EXPECT_EQ(0x21000U, frame->pc);
317 EXPECT_EQ(0x10060U, frame->sp);
318 EXPECT_EQ("Frame1", frame->function_name);
319 EXPECT_EQ(1U, frame->function_offset);
320 EXPECT_EQ("/system/fake/libunwind.so", frame->map_name);
321 EXPECT_EQ(0U, frame->map_offset);
322 EXPECT_EQ(0x20000U, frame->map_start);
323 EXPECT_EQ(0x22000U, frame->map_end);
324 EXPECT_EQ(0U, frame->map_load_bias);
325 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
326
327 frame = &unwinder.frames()[2];
328 EXPECT_EQ(2U, frame->num);
329 EXPECT_EQ(0U, frame->rel_pc);
330 EXPECT_EQ(0x23000U, frame->pc);
331 EXPECT_EQ(0x10070U, frame->sp);
332 EXPECT_EQ("Frame2", frame->function_name);
333 EXPECT_EQ(2U, frame->function_offset);
334 EXPECT_EQ("/fake/libanother.so", frame->map_name);
335 EXPECT_EQ(0U, frame->map_offset);
336 EXPECT_EQ(0x23000U, frame->map_start);
337 EXPECT_EQ(0x24000U, frame->map_end);
338 EXPECT_EQ(0U, frame->map_load_bias);
339 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
340}
341
342// Verify SP in a non-existant map is okay.
343TEST_F(UnwinderTest, sp_not_in_map) {
344 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
345 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
346
347 regs_.FakeSetPc(0x1000);
348 regs_.FakeSetSp(0x53000);
349 ElfInterfaceFake::FakePushStepData(StepData(0x21002, 0x50020, false));
350 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
351
352 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
353 unwinder.Unwind();
354
355 ASSERT_EQ(2U, unwinder.NumFrames());
356
357 auto* frame = &unwinder.frames()[0];
358 EXPECT_EQ(0U, frame->num);
359 EXPECT_EQ(0U, frame->rel_pc);
360 EXPECT_EQ(0x1000U, frame->pc);
361 EXPECT_EQ(0x53000U, frame->sp);
362 EXPECT_EQ("Frame0", frame->function_name);
363 EXPECT_EQ(0U, frame->function_offset);
364 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
365 EXPECT_EQ(0U, frame->map_offset);
366 EXPECT_EQ(0x1000U, frame->map_start);
367 EXPECT_EQ(0x8000U, frame->map_end);
368 EXPECT_EQ(0U, frame->map_load_bias);
369 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
370
371 frame = &unwinder.frames()[1];
372 EXPECT_EQ(1U, frame->num);
373 EXPECT_EQ(0x1000U, frame->rel_pc);
374 EXPECT_EQ(0x21000U, frame->pc);
375 EXPECT_EQ(0x50020U, frame->sp);
376 EXPECT_EQ("Frame1", frame->function_name);
377 EXPECT_EQ(1U, frame->function_offset);
378 EXPECT_EQ("/system/fake/libunwind.so", frame->map_name);
379 EXPECT_EQ(0U, frame->map_offset);
380 EXPECT_EQ(0x20000U, frame->map_start);
381 EXPECT_EQ(0x22000U, frame->map_end);
382 EXPECT_EQ(0U, frame->map_load_bias);
383 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
384}
385
386// Verify PC in a device stops the unwind.
387TEST_F(UnwinderTest, pc_in_device_stops_unwind) {
388 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
389 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
390 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
391
392 regs_.FakeSetPc(0x13000);
393 regs_.FakeSetSp(0x10000);
394 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10010, false));
395 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
396 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
397
398 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
399 unwinder.Unwind();
400
401 ASSERT_EQ(1U, unwinder.NumFrames());
402}
403
404// Verify SP in a device stops the unwind.
405TEST_F(UnwinderTest, sp_in_device_stops_unwind) {
406 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
407 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
408 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
409
410 regs_.FakeSetPc(0x1000);
411 regs_.FakeSetSp(0x13000);
412 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10010, false));
413 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
414 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
415
416 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
417 unwinder.Unwind();
418
419 ASSERT_EQ(1U, unwinder.NumFrames());
420}
421
422// Verify a no map info frame gets a frame.
423TEST_F(UnwinderTest, pc_without_map) {
424 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
425
426 regs_.FakeSetPc(0x41000);
427 regs_.FakeSetSp(0x13000);
428
429 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
430 unwinder.Unwind();
431
432 ASSERT_EQ(1U, unwinder.NumFrames());
433
434 auto* frame = &unwinder.frames()[0];
435 EXPECT_EQ(0U, frame->num);
436 EXPECT_EQ(0x41000U, frame->rel_pc);
437 EXPECT_EQ(0x41000U, frame->pc);
438 EXPECT_EQ(0x13000U, frame->sp);
439 EXPECT_EQ("", frame->function_name);
440 EXPECT_EQ(0U, frame->function_offset);
441 EXPECT_EQ("", frame->map_name);
442 EXPECT_EQ(0U, frame->map_offset);
443 EXPECT_EQ(0U, frame->map_start);
444 EXPECT_EQ(0U, frame->map_end);
445 EXPECT_EQ(0U, frame->map_load_bias);
446 EXPECT_EQ(0, frame->map_flags);
447}
448
449// Verify that a speculative frame is added.
450TEST_F(UnwinderTest, speculative_frame) {
451 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
452 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
453
454 // Fake as if code called a nullptr function.
455 regs_.FakeSetPc(0);
456 regs_.FakeSetSp(0x10000);
457 regs_.FakeSetReturnAddress(0x1202);
458 regs_.FakeSetReturnAddressValid(true);
459
460 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
461 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
462
463 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
464 unwinder.Unwind();
465
466 ASSERT_EQ(3U, unwinder.NumFrames());
467
468 auto* frame = &unwinder.frames()[0];
469 EXPECT_EQ(0U, frame->num);
470 EXPECT_EQ(0U, frame->rel_pc);
471 EXPECT_EQ(0U, frame->pc);
472 EXPECT_EQ(0x10000U, frame->sp);
473 EXPECT_EQ("", frame->function_name);
474 EXPECT_EQ(0U, frame->function_offset);
475 EXPECT_EQ("", frame->map_name);
476 EXPECT_EQ(0U, frame->map_offset);
477 EXPECT_EQ(0U, frame->map_start);
478 EXPECT_EQ(0U, frame->map_end);
479 EXPECT_EQ(0U, frame->map_load_bias);
480 EXPECT_EQ(0, frame->map_flags);
481
482 frame = &unwinder.frames()[1];
483 EXPECT_EQ(1U, frame->num);
484 EXPECT_EQ(0x200U, frame->rel_pc);
485 EXPECT_EQ(0x1200U, frame->pc);
486 EXPECT_EQ(0x10000U, frame->sp);
487 EXPECT_EQ("Frame0", frame->function_name);
488 EXPECT_EQ(0U, frame->function_offset);
489 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
490 EXPECT_EQ(0U, frame->map_offset);
491 EXPECT_EQ(0x1000U, frame->map_start);
492 EXPECT_EQ(0x8000U, frame->map_end);
493 EXPECT_EQ(0U, frame->map_load_bias);
494 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
495
496 frame = &unwinder.frames()[2];
497 EXPECT_EQ(2U, frame->num);
498 EXPECT_EQ(0x100U, frame->rel_pc);
499 EXPECT_EQ(0x23100U, frame->pc);
500 EXPECT_EQ(0x10020U, frame->sp);
501 EXPECT_EQ("Frame1", frame->function_name);
502 EXPECT_EQ(1U, frame->function_offset);
503 EXPECT_EQ("/fake/libanother.so", frame->map_name);
504 EXPECT_EQ(0U, frame->map_offset);
505 EXPECT_EQ(0x23000U, frame->map_start);
506 EXPECT_EQ(0x24000U, frame->map_end);
507 EXPECT_EQ(0U, frame->map_load_bias);
508 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
509}
510
511// Verify that a speculative frame is added then removed because no other
512// frames are added.
513TEST_F(UnwinderTest, speculative_frame_removed) {
514 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
515 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
516
517 // Fake as if code called a nullptr function.
518 regs_.FakeSetPc(0);
519 regs_.FakeSetSp(0x10000);
520 regs_.FakeSetReturnAddress(0x1202);
521 regs_.FakeSetReturnAddressValid(true);
522
523 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
524 unwinder.Unwind();
525
526 ASSERT_EQ(1U, unwinder.NumFrames());
527
528 auto* frame = &unwinder.frames()[0];
529 EXPECT_EQ(0U, frame->num);
530 EXPECT_EQ(0U, frame->rel_pc);
531 EXPECT_EQ(0U, frame->pc);
532 EXPECT_EQ(0x10000U, frame->sp);
533 EXPECT_EQ("", frame->function_name);
534 EXPECT_EQ(0U, frame->function_offset);
535 EXPECT_EQ("", frame->map_name);
536 EXPECT_EQ(0U, frame->map_offset);
537 EXPECT_EQ(0U, frame->map_start);
538 EXPECT_EQ(0U, frame->map_end);
539 EXPECT_EQ(0U, frame->map_load_bias);
540 EXPECT_EQ(0, frame->map_flags);
541}
542
543// Verify format frame code.
544TEST_F(UnwinderTest, format_frame_static) {
545 FrameData frame;
546 frame.num = 1;
547 frame.rel_pc = 0x1000;
548 frame.pc = 0x4000;
549 frame.sp = 0x1000;
550 frame.function_name = "function";
551 frame.function_offset = 100;
552 frame.map_name = "/fake/libfake.so";
553 frame.map_offset = 0x2000;
554 frame.map_start = 0x3000;
555 frame.map_end = 0x6000;
556 frame.map_flags = PROT_READ;
557
558 EXPECT_EQ(" #01 pc 0000000000001000 (offset 0x2000) /fake/libfake.so (function+100)",
559 Unwinder::FormatFrame(frame, false));
560 EXPECT_EQ(" #01 pc 00001000 (offset 0x2000) /fake/libfake.so (function+100)",
561 Unwinder::FormatFrame(frame, true));
562
563 frame.map_offset = 0;
564 EXPECT_EQ(" #01 pc 0000000000001000 /fake/libfake.so (function+100)",
565 Unwinder::FormatFrame(frame, false));
566 EXPECT_EQ(" #01 pc 00001000 /fake/libfake.so (function+100)",
567 Unwinder::FormatFrame(frame, true));
568
569 frame.function_offset = 0;
570 EXPECT_EQ(" #01 pc 0000000000001000 /fake/libfake.so (function)",
571 Unwinder::FormatFrame(frame, false));
572 EXPECT_EQ(" #01 pc 00001000 /fake/libfake.so (function)", Unwinder::FormatFrame(frame, true));
573
574 frame.function_name = "";
575 EXPECT_EQ(" #01 pc 0000000000001000 /fake/libfake.so", Unwinder::FormatFrame(frame, false));
576 EXPECT_EQ(" #01 pc 00001000 /fake/libfake.so", Unwinder::FormatFrame(frame, true));
577
578 frame.map_name = "";
579 EXPECT_EQ(" #01 pc 0000000000001000 <anonymous:3000>", Unwinder::FormatFrame(frame, false));
580 EXPECT_EQ(" #01 pc 00001000 <anonymous:3000>", Unwinder::FormatFrame(frame, true));
581
582 frame.map_start = 0;
583 frame.map_end = 0;
584 EXPECT_EQ(" #01 pc 0000000000001000 <unknown>", Unwinder::FormatFrame(frame, false));
585 EXPECT_EQ(" #01 pc 00001000 <unknown>", Unwinder::FormatFrame(frame, true));
586}
587
588// Verify format frame code.
589TEST_F(UnwinderTest, format_frame) {
590 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 10));
591
592 regs_.FakeSetPc(0x2300);
593 regs_.FakeSetSp(0x10000);
594
595 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
596 unwinder.Unwind();
597
598 ASSERT_EQ(1U, unwinder.NumFrames());
599
600 regs_.FakeSetMachineType(EM_ARM);
601 EXPECT_EQ(" #00 pc 00001300 /system/fake/libc.so (Frame0+10)", unwinder.FormatFrame(0));
602 regs_.FakeSetMachineType(EM_386);
603 EXPECT_EQ(" #00 pc 00001300 /system/fake/libc.so (Frame0+10)", unwinder.FormatFrame(0));
604
605 regs_.FakeSetMachineType(EM_AARCH64);
606 EXPECT_EQ(" #00 pc 0000000000001300 /system/fake/libc.so (Frame0+10)", unwinder.FormatFrame(0));
607 regs_.FakeSetMachineType(EM_X86_64);
608 EXPECT_EQ(" #00 pc 0000000000001300 /system/fake/libc.so (Frame0+10)", unwinder.FormatFrame(0));
609
610 EXPECT_EQ("", unwinder.FormatFrame(1));
611}
612
613} // namespace unwindstack