blob: 4d0366ca50b61e60c587c18b83c733145bf3fe56 [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);
98 }
99
100 void SetUp() override {
101 ElfInterfaceFake::FakeClear();
102 regs_.FakeSetMachineType(EM_ARM);
103 }
104
105 static MapsFake maps_;
106 static RegsFake regs_;
107 static std::shared_ptr<Memory> process_memory_;
108};
109
110MapsFake UnwinderTest::maps_;
111RegsFake UnwinderTest::regs_(5, 0);
112std::shared_ptr<Memory> UnwinderTest::process_memory_(nullptr);
113
114TEST_F(UnwinderTest, multiple_frames) {
115 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
116 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
117 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
118
119 regs_.FakeSetPc(0x1000);
120 regs_.FakeSetSp(0x10000);
121 ElfInterfaceFake::FakePushStepData(StepData(0x1102, 0x10010, false));
122 ElfInterfaceFake::FakePushStepData(StepData(0x1202, 0x10020, false));
123 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
124
125 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
126 unwinder.Unwind();
127
128 ASSERT_EQ(3U, unwinder.NumFrames());
129
130 auto* frame = &unwinder.frames()[0];
131 EXPECT_EQ(0U, frame->num);
132 EXPECT_EQ(0U, frame->rel_pc);
133 EXPECT_EQ(0x1000U, frame->pc);
134 EXPECT_EQ(0x10000U, frame->sp);
135 EXPECT_EQ("Frame0", frame->function_name);
136 EXPECT_EQ(0U, frame->function_offset);
137 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
138 EXPECT_EQ(0U, frame->map_offset);
139 EXPECT_EQ(0x1000U, frame->map_start);
140 EXPECT_EQ(0x8000U, frame->map_end);
141 EXPECT_EQ(0U, frame->map_load_bias);
142 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
143
144 frame = &unwinder.frames()[1];
145 EXPECT_EQ(1U, frame->num);
146 EXPECT_EQ(0x100U, frame->rel_pc);
147 EXPECT_EQ(0x1100U, frame->pc);
148 EXPECT_EQ(0x10010U, frame->sp);
149 EXPECT_EQ("Frame1", frame->function_name);
150 EXPECT_EQ(1U, frame->function_offset);
151 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
152 EXPECT_EQ(0U, frame->map_offset);
153 EXPECT_EQ(0x1000U, frame->map_start);
154 EXPECT_EQ(0x8000U, frame->map_end);
155 EXPECT_EQ(0U, frame->map_load_bias);
156 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
157
158 frame = &unwinder.frames()[2];
159 EXPECT_EQ(2U, frame->num);
160 EXPECT_EQ(0x200U, frame->rel_pc);
161 EXPECT_EQ(0x1200U, frame->pc);
162 EXPECT_EQ(0x10020U, frame->sp);
163 EXPECT_EQ("Frame2", frame->function_name);
164 EXPECT_EQ(2U, frame->function_offset);
165 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
166 EXPECT_EQ(0U, frame->map_offset);
167 EXPECT_EQ(0x1000U, frame->map_start);
168 EXPECT_EQ(0x8000U, frame->map_end);
169 EXPECT_EQ(0U, frame->map_load_bias);
170 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
171}
172
173// Verify that no attempt to continue after the step indicates it is done.
174TEST_F(UnwinderTest, no_frames_after_finished) {
175 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
176 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
177 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
178 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame3", 3));
179 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame4", 4));
180
181 regs_.FakeSetPc(0x1000);
182 regs_.FakeSetSp(0x10000);
183 ElfInterfaceFake::FakePushStepData(StepData(0x1000, 0x10000, true));
184 ElfInterfaceFake::FakePushStepData(StepData(0x1102, 0x10010, false));
185 ElfInterfaceFake::FakePushStepData(StepData(0x1202, 0x10020, false));
186
187 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
188 unwinder.Unwind();
189
190 ASSERT_EQ(1U, unwinder.NumFrames());
191
192 auto* frame = &unwinder.frames()[0];
193 EXPECT_EQ(0U, frame->num);
194 EXPECT_EQ(0U, frame->rel_pc);
195 EXPECT_EQ(0x1000U, frame->pc);
196 EXPECT_EQ(0x10000U, frame->sp);
197 EXPECT_EQ("Frame0", frame->function_name);
198 EXPECT_EQ(0U, frame->function_offset);
199 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
200 EXPECT_EQ(0U, frame->map_offset);
201 EXPECT_EQ(0x1000U, frame->map_start);
202 EXPECT_EQ(0x8000U, frame->map_end);
203 EXPECT_EQ(0U, frame->map_load_bias);
204 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
205}
206
207// Verify the maximum frames to save.
208TEST_F(UnwinderTest, max_frames) {
209 for (size_t i = 0; i < 30; i++) {
210 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame" + std::to_string(i), i));
211 ElfInterfaceFake::FakePushStepData(StepData(0x1102 + i * 0x100, 0x10010 + i * 0x10, false));
212 }
213
214 regs_.FakeSetPc(0x1000);
215 regs_.FakeSetSp(0x10000);
216
217 Unwinder unwinder(20, &maps_, &regs_, process_memory_);
218 unwinder.Unwind();
219
220 ASSERT_EQ(20U, unwinder.NumFrames());
221
222 for (size_t i = 0; i < 20; i++) {
223 auto* frame = &unwinder.frames()[i];
224 EXPECT_EQ(i, frame->num);
225 EXPECT_EQ(i * 0x100, frame->rel_pc) << "Failed at frame " << i;
226 EXPECT_EQ(0x1000 + i * 0x100, frame->pc) << "Failed at frame " << i;
227 EXPECT_EQ(0x10000 + 0x10 * i, frame->sp) << "Failed at frame " << i;
228 EXPECT_EQ("Frame" + std::to_string(i), frame->function_name) << "Failed at frame " << i;
229 EXPECT_EQ(i, frame->function_offset) << "Failed at frame " << i;
230 EXPECT_EQ("/system/fake/libc.so", frame->map_name) << "Failed at frame " << i;
231 EXPECT_EQ(0U, frame->map_offset) << "Failed at frame " << i;
232 EXPECT_EQ(0x1000U, frame->map_start) << "Failed at frame " << i;
233 EXPECT_EQ(0x8000U, frame->map_end) << "Failed at frame " << i;
234 EXPECT_EQ(0U, frame->map_load_bias) << "Failed at frame " << i;
235 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags) << "Failed at frame " << i;
236 }
237}
238
239// Verify that initial map names frames are removed.
240TEST_F(UnwinderTest, verify_frames_skipped) {
241 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
242 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
243 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
244
245 regs_.FakeSetPc(0x20000);
246 regs_.FakeSetSp(0x10000);
247 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10010, false));
248 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
249 ElfInterfaceFake::FakePushStepData(StepData(0x20002, 0x10030, false));
250 ElfInterfaceFake::FakePushStepData(StepData(0x21002, 0x10040, false));
251 ElfInterfaceFake::FakePushStepData(StepData(0x1002, 0x10050, false));
252 ElfInterfaceFake::FakePushStepData(StepData(0x21002, 0x10060, false));
253 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10070, false));
254 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
255
256 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
257 std::set<std::string> skip_set{"libunwind.so", "libanother.so"};
258 unwinder.Unwind(&skip_set);
259
260 ASSERT_EQ(3U, unwinder.NumFrames());
261
262 auto* frame = &unwinder.frames()[0];
263 EXPECT_EQ(0U, frame->num);
264 EXPECT_EQ(0U, frame->rel_pc);
265 EXPECT_EQ(0x1000U, frame->pc);
266 EXPECT_EQ(0x10050U, frame->sp);
267 EXPECT_EQ("Frame0", frame->function_name);
268 EXPECT_EQ(0U, frame->function_offset);
269 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
270 EXPECT_EQ(0U, frame->map_offset);
271 EXPECT_EQ(0x1000U, frame->map_start);
272 EXPECT_EQ(0x8000U, frame->map_end);
273 EXPECT_EQ(0U, frame->map_load_bias);
274 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
275
276 frame = &unwinder.frames()[1];
277 EXPECT_EQ(1U, frame->num);
278 EXPECT_EQ(0x1000U, frame->rel_pc);
279 EXPECT_EQ(0x21000U, frame->pc);
280 EXPECT_EQ(0x10060U, frame->sp);
281 EXPECT_EQ("Frame1", frame->function_name);
282 EXPECT_EQ(1U, frame->function_offset);
283 EXPECT_EQ("/system/fake/libunwind.so", frame->map_name);
284 EXPECT_EQ(0U, frame->map_offset);
285 EXPECT_EQ(0x20000U, frame->map_start);
286 EXPECT_EQ(0x22000U, frame->map_end);
287 EXPECT_EQ(0U, frame->map_load_bias);
288 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
289
290 frame = &unwinder.frames()[2];
291 EXPECT_EQ(2U, frame->num);
292 EXPECT_EQ(0U, frame->rel_pc);
293 EXPECT_EQ(0x23000U, frame->pc);
294 EXPECT_EQ(0x10070U, frame->sp);
295 EXPECT_EQ("Frame2", frame->function_name);
296 EXPECT_EQ(2U, frame->function_offset);
297 EXPECT_EQ("/fake/libanother.so", frame->map_name);
298 EXPECT_EQ(0U, frame->map_offset);
299 EXPECT_EQ(0x23000U, frame->map_start);
300 EXPECT_EQ(0x24000U, frame->map_end);
301 EXPECT_EQ(0U, frame->map_load_bias);
302 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
303}
304
305// Verify SP in a non-existant map is okay.
306TEST_F(UnwinderTest, sp_not_in_map) {
307 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
308 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
309
310 regs_.FakeSetPc(0x1000);
311 regs_.FakeSetSp(0x53000);
312 ElfInterfaceFake::FakePushStepData(StepData(0x21002, 0x50020, false));
313 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
314
315 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
316 unwinder.Unwind();
317
318 ASSERT_EQ(2U, unwinder.NumFrames());
319
320 auto* frame = &unwinder.frames()[0];
321 EXPECT_EQ(0U, frame->num);
322 EXPECT_EQ(0U, frame->rel_pc);
323 EXPECT_EQ(0x1000U, frame->pc);
324 EXPECT_EQ(0x53000U, frame->sp);
325 EXPECT_EQ("Frame0", frame->function_name);
326 EXPECT_EQ(0U, frame->function_offset);
327 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
328 EXPECT_EQ(0U, frame->map_offset);
329 EXPECT_EQ(0x1000U, frame->map_start);
330 EXPECT_EQ(0x8000U, frame->map_end);
331 EXPECT_EQ(0U, frame->map_load_bias);
332 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
333
334 frame = &unwinder.frames()[1];
335 EXPECT_EQ(1U, frame->num);
336 EXPECT_EQ(0x1000U, frame->rel_pc);
337 EXPECT_EQ(0x21000U, frame->pc);
338 EXPECT_EQ(0x50020U, frame->sp);
339 EXPECT_EQ("Frame1", frame->function_name);
340 EXPECT_EQ(1U, frame->function_offset);
341 EXPECT_EQ("/system/fake/libunwind.so", frame->map_name);
342 EXPECT_EQ(0U, frame->map_offset);
343 EXPECT_EQ(0x20000U, frame->map_start);
344 EXPECT_EQ(0x22000U, frame->map_end);
345 EXPECT_EQ(0U, frame->map_load_bias);
346 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
347}
348
349// Verify PC in a device stops the unwind.
350TEST_F(UnwinderTest, pc_in_device_stops_unwind) {
351 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
352 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
353 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
354
355 regs_.FakeSetPc(0x13000);
356 regs_.FakeSetSp(0x10000);
357 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10010, false));
358 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
359 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
360
361 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
362 unwinder.Unwind();
363
364 ASSERT_EQ(1U, unwinder.NumFrames());
365}
366
367// Verify SP in a device stops the unwind.
368TEST_F(UnwinderTest, sp_in_device_stops_unwind) {
369 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
370 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
371 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
372
373 regs_.FakeSetPc(0x1000);
374 regs_.FakeSetSp(0x13000);
375 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10010, false));
376 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
377 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
378
379 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
380 unwinder.Unwind();
381
382 ASSERT_EQ(1U, unwinder.NumFrames());
383}
384
385// Verify a no map info frame gets a frame.
386TEST_F(UnwinderTest, pc_without_map) {
387 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
388
389 regs_.FakeSetPc(0x41000);
390 regs_.FakeSetSp(0x13000);
391
392 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
393 unwinder.Unwind();
394
395 ASSERT_EQ(1U, unwinder.NumFrames());
396
397 auto* frame = &unwinder.frames()[0];
398 EXPECT_EQ(0U, frame->num);
399 EXPECT_EQ(0x41000U, frame->rel_pc);
400 EXPECT_EQ(0x41000U, frame->pc);
401 EXPECT_EQ(0x13000U, frame->sp);
402 EXPECT_EQ("", frame->function_name);
403 EXPECT_EQ(0U, frame->function_offset);
404 EXPECT_EQ("", frame->map_name);
405 EXPECT_EQ(0U, frame->map_offset);
406 EXPECT_EQ(0U, frame->map_start);
407 EXPECT_EQ(0U, frame->map_end);
408 EXPECT_EQ(0U, frame->map_load_bias);
409 EXPECT_EQ(0, frame->map_flags);
410}
411
412// Verify that a speculative frame is added.
413TEST_F(UnwinderTest, speculative_frame) {
414 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
415 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
416
417 // Fake as if code called a nullptr function.
418 regs_.FakeSetPc(0);
419 regs_.FakeSetSp(0x10000);
420 regs_.FakeSetReturnAddress(0x1202);
421 regs_.FakeSetReturnAddressValid(true);
422
423 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
424 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
425
426 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
427 unwinder.Unwind();
428
429 ASSERT_EQ(3U, unwinder.NumFrames());
430
431 auto* frame = &unwinder.frames()[0];
432 EXPECT_EQ(0U, frame->num);
433 EXPECT_EQ(0U, frame->rel_pc);
434 EXPECT_EQ(0U, frame->pc);
435 EXPECT_EQ(0x10000U, frame->sp);
436 EXPECT_EQ("", frame->function_name);
437 EXPECT_EQ(0U, frame->function_offset);
438 EXPECT_EQ("", frame->map_name);
439 EXPECT_EQ(0U, frame->map_offset);
440 EXPECT_EQ(0U, frame->map_start);
441 EXPECT_EQ(0U, frame->map_end);
442 EXPECT_EQ(0U, frame->map_load_bias);
443 EXPECT_EQ(0, frame->map_flags);
444
445 frame = &unwinder.frames()[1];
446 EXPECT_EQ(1U, frame->num);
447 EXPECT_EQ(0x200U, frame->rel_pc);
448 EXPECT_EQ(0x1200U, frame->pc);
449 EXPECT_EQ(0x10000U, frame->sp);
450 EXPECT_EQ("Frame0", frame->function_name);
451 EXPECT_EQ(0U, frame->function_offset);
452 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
453 EXPECT_EQ(0U, frame->map_offset);
454 EXPECT_EQ(0x1000U, frame->map_start);
455 EXPECT_EQ(0x8000U, frame->map_end);
456 EXPECT_EQ(0U, frame->map_load_bias);
457 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
458
459 frame = &unwinder.frames()[2];
460 EXPECT_EQ(2U, frame->num);
461 EXPECT_EQ(0x100U, frame->rel_pc);
462 EXPECT_EQ(0x23100U, frame->pc);
463 EXPECT_EQ(0x10020U, frame->sp);
464 EXPECT_EQ("Frame1", frame->function_name);
465 EXPECT_EQ(1U, frame->function_offset);
466 EXPECT_EQ("/fake/libanother.so", frame->map_name);
467 EXPECT_EQ(0U, frame->map_offset);
468 EXPECT_EQ(0x23000U, frame->map_start);
469 EXPECT_EQ(0x24000U, frame->map_end);
470 EXPECT_EQ(0U, frame->map_load_bias);
471 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
472}
473
474// Verify that a speculative frame is added then removed because no other
475// frames are added.
476TEST_F(UnwinderTest, speculative_frame_removed) {
477 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
478 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
479
480 // Fake as if code called a nullptr function.
481 regs_.FakeSetPc(0);
482 regs_.FakeSetSp(0x10000);
483 regs_.FakeSetReturnAddress(0x1202);
484 regs_.FakeSetReturnAddressValid(true);
485
486 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
487 unwinder.Unwind();
488
489 ASSERT_EQ(1U, unwinder.NumFrames());
490
491 auto* frame = &unwinder.frames()[0];
492 EXPECT_EQ(0U, frame->num);
493 EXPECT_EQ(0U, frame->rel_pc);
494 EXPECT_EQ(0U, frame->pc);
495 EXPECT_EQ(0x10000U, frame->sp);
496 EXPECT_EQ("", frame->function_name);
497 EXPECT_EQ(0U, frame->function_offset);
498 EXPECT_EQ("", frame->map_name);
499 EXPECT_EQ(0U, frame->map_offset);
500 EXPECT_EQ(0U, frame->map_start);
501 EXPECT_EQ(0U, frame->map_end);
502 EXPECT_EQ(0U, frame->map_load_bias);
503 EXPECT_EQ(0, frame->map_flags);
504}
505
506// Verify format frame code.
507TEST_F(UnwinderTest, format_frame_static) {
508 FrameData frame;
509 frame.num = 1;
510 frame.rel_pc = 0x1000;
511 frame.pc = 0x4000;
512 frame.sp = 0x1000;
513 frame.function_name = "function";
514 frame.function_offset = 100;
515 frame.map_name = "/fake/libfake.so";
516 frame.map_offset = 0x2000;
517 frame.map_start = 0x3000;
518 frame.map_end = 0x6000;
519 frame.map_flags = PROT_READ;
520
521 EXPECT_EQ(" #01 pc 0000000000001000 (offset 0x2000) /fake/libfake.so (function+100)",
522 Unwinder::FormatFrame(frame, false));
523 EXPECT_EQ(" #01 pc 00001000 (offset 0x2000) /fake/libfake.so (function+100)",
524 Unwinder::FormatFrame(frame, true));
525
526 frame.map_offset = 0;
527 EXPECT_EQ(" #01 pc 0000000000001000 /fake/libfake.so (function+100)",
528 Unwinder::FormatFrame(frame, false));
529 EXPECT_EQ(" #01 pc 00001000 /fake/libfake.so (function+100)",
530 Unwinder::FormatFrame(frame, true));
531
532 frame.function_offset = 0;
533 EXPECT_EQ(" #01 pc 0000000000001000 /fake/libfake.so (function)",
534 Unwinder::FormatFrame(frame, false));
535 EXPECT_EQ(" #01 pc 00001000 /fake/libfake.so (function)", Unwinder::FormatFrame(frame, true));
536
537 frame.function_name = "";
538 EXPECT_EQ(" #01 pc 0000000000001000 /fake/libfake.so", Unwinder::FormatFrame(frame, false));
539 EXPECT_EQ(" #01 pc 00001000 /fake/libfake.so", Unwinder::FormatFrame(frame, true));
540
541 frame.map_name = "";
542 EXPECT_EQ(" #01 pc 0000000000001000 <anonymous:3000>", Unwinder::FormatFrame(frame, false));
543 EXPECT_EQ(" #01 pc 00001000 <anonymous:3000>", Unwinder::FormatFrame(frame, true));
544
545 frame.map_start = 0;
546 frame.map_end = 0;
547 EXPECT_EQ(" #01 pc 0000000000001000 <unknown>", Unwinder::FormatFrame(frame, false));
548 EXPECT_EQ(" #01 pc 00001000 <unknown>", Unwinder::FormatFrame(frame, true));
549}
550
551// Verify format frame code.
552TEST_F(UnwinderTest, format_frame) {
553 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 10));
554
555 regs_.FakeSetPc(0x2300);
556 regs_.FakeSetSp(0x10000);
557
558 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
559 unwinder.Unwind();
560
561 ASSERT_EQ(1U, unwinder.NumFrames());
562
563 regs_.FakeSetMachineType(EM_ARM);
564 EXPECT_EQ(" #00 pc 00001300 /system/fake/libc.so (Frame0+10)", unwinder.FormatFrame(0));
565 regs_.FakeSetMachineType(EM_386);
566 EXPECT_EQ(" #00 pc 00001300 /system/fake/libc.so (Frame0+10)", unwinder.FormatFrame(0));
567
568 regs_.FakeSetMachineType(EM_AARCH64);
569 EXPECT_EQ(" #00 pc 0000000000001300 /system/fake/libc.so (Frame0+10)", unwinder.FormatFrame(0));
570 regs_.FakeSetMachineType(EM_X86_64);
571 EXPECT_EQ(" #00 pc 0000000000001300 /system/fake/libc.so (Frame0+10)", unwinder.FormatFrame(0));
572
573 EXPECT_EQ("", unwinder.FormatFrame(1));
574}
575
576} // namespace unwindstack