blob: 7fbae4cf8abd5bbb640bf84fd659a688b778021d [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>
Christopher Ferris02fdb562017-12-08 15:04:49 -080031#include <unwindstack/RegsArm.h>
32#include <unwindstack/RegsArm64.h>
33#include <unwindstack/RegsX86.h>
34#include <unwindstack/RegsX86_64.h>
Douglas Leung61b1a1a2017-11-08 10:53:53 +010035#include <unwindstack/RegsMips.h>
36#include <unwindstack/RegsMips64.h>
Christopher Ferrisf6f691b2017-09-25 19:23:07 -070037#include <unwindstack/Unwinder.h>
38
39#include "ElfFake.h"
40#include "MemoryFake.h"
41#include "RegsFake.h"
42
43namespace unwindstack {
44
45class MapsFake : public Maps {
46 public:
47 MapsFake() = default;
48 virtual ~MapsFake() = default;
49
50 bool Parse() { return true; }
51
52 void FakeClear() { maps_.clear(); }
53
Christopher Ferrisbe788d82017-11-27 14:50:38 -080054 void FakeAddMapInfo(MapInfo* map_info) { maps_.push_back(map_info); }
Christopher Ferrisf6f691b2017-09-25 19:23:07 -070055};
56
57class UnwinderTest : public ::testing::Test {
58 protected:
59 static void SetUpTestCase() {
60 maps_.FakeClear();
Christopher Ferrisbe788d82017-11-27 14:50:38 -080061 MapInfo* info = new MapInfo(0x1000, 0x8000, 0, PROT_READ | PROT_WRITE, "/system/fake/libc.so");
Christopher Ferris02fdb562017-12-08 15:04:49 -080062 ElfFake* elf = new ElfFake(new MemoryFake);
Christopher Ferris0b79ae12018-01-25 12:15:56 -080063 info->elf.reset(elf);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -070064 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
Christopher Ferrisf6f691b2017-09-25 19:23:07 -070065 maps_.FakeAddMapInfo(info);
66
Christopher Ferrisbe788d82017-11-27 14:50:38 -080067 info = new MapInfo(0x10000, 0x12000, 0, PROT_READ | PROT_WRITE, "[stack]");
Christopher Ferrisf6f691b2017-09-25 19:23:07 -070068 maps_.FakeAddMapInfo(info);
69
Christopher Ferrisbe788d82017-11-27 14:50:38 -080070 info = new MapInfo(0x13000, 0x15000, 0, PROT_READ | PROT_WRITE | MAPS_FLAGS_DEVICE_MAP,
71 "/dev/fake_device");
Christopher Ferrisf6f691b2017-09-25 19:23:07 -070072 maps_.FakeAddMapInfo(info);
73
Christopher Ferrisbe788d82017-11-27 14:50:38 -080074 info = new MapInfo(0x20000, 0x22000, 0, PROT_READ | PROT_WRITE, "/system/fake/libunwind.so");
Christopher Ferris02fdb562017-12-08 15:04:49 -080075 elf = new ElfFake(new MemoryFake);
Christopher Ferris0b79ae12018-01-25 12:15:56 -080076 info->elf.reset(elf);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -070077 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
78 maps_.FakeAddMapInfo(info);
79
Christopher Ferrisbe788d82017-11-27 14:50:38 -080080 info = new MapInfo(0x23000, 0x24000, 0, PROT_READ | PROT_WRITE, "/fake/libanother.so");
Christopher Ferris02fdb562017-12-08 15:04:49 -080081 elf = new ElfFake(new MemoryFake);
Christopher Ferris0b79ae12018-01-25 12:15:56 -080082 info->elf.reset(elf);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -070083 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
84 maps_.FakeAddMapInfo(info);
Christopher Ferrisc5a3baa2017-10-17 18:42:03 -070085
Christopher Ferrisbe788d82017-11-27 14:50:38 -080086 info = new MapInfo(0x33000, 0x34000, 0, PROT_READ | PROT_WRITE, "/fake/compressed.so");
Christopher Ferris02fdb562017-12-08 15:04:49 -080087 elf = new ElfFake(new MemoryFake);
Christopher Ferris0b79ae12018-01-25 12:15:56 -080088 info->elf.reset(elf);
Christopher Ferrise69f4702017-10-19 16:08:58 -070089 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
90 maps_.FakeAddMapInfo(info);
91
Christopher Ferrisbe788d82017-11-27 14:50:38 -080092 info = new MapInfo(0x43000, 0x44000, 0x1d000, PROT_READ | PROT_WRITE, "/fake/fake.apk");
Christopher Ferris02fdb562017-12-08 15:04:49 -080093 elf = new ElfFake(new MemoryFake);
Christopher Ferris0b79ae12018-01-25 12:15:56 -080094 info->elf.reset(elf);
Christopher Ferrisc5a3baa2017-10-17 18:42:03 -070095 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
96 maps_.FakeAddMapInfo(info);
Christopher Ferrise69f4702017-10-19 16:08:58 -070097
Christopher Ferrisbe788d82017-11-27 14:50:38 -080098 info = new MapInfo(0x53000, 0x54000, 0, PROT_READ | PROT_WRITE, "/fake/fake.oat");
Christopher Ferrise69f4702017-10-19 16:08:58 -070099 maps_.FakeAddMapInfo(info);
Christopher Ferris02fdb562017-12-08 15:04:49 -0800100
Christopher Ferrise762f1f2018-02-06 14:51:48 -0800101 info = new MapInfo(0xa3000, 0xa4000, 0, PROT_READ | PROT_WRITE | PROT_EXEC, "/fake/fake.vdex");
102 info->load_bias = 0;
103 maps_.FakeAddMapInfo(info);
104
Christopher Ferrise4df5f52018-02-12 13:24:18 -0800105 info = new MapInfo(0xa5000, 0xa6000, 0, PROT_READ | PROT_WRITE | PROT_EXEC,
106 "/fake/fake_load_bias.so");
107 elf = new ElfFake(new MemoryFake);
108 info->elf.reset(elf);
109 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
110 elf->FakeSetLoadBias(0x5000);
111 maps_.FakeAddMapInfo(info);
112
113 info = new MapInfo(0xa7000, 0xa8000, 0, PROT_READ | PROT_WRITE | PROT_EXEC,
114 "/fake/fake_offset.oat");
115 elf = new ElfFake(new MemoryFake);
116 info->elf.reset(elf);
117 elf->FakeSetInterface(new ElfInterfaceFake(nullptr));
118 info->elf_offset = 0x8000;
119 maps_.FakeAddMapInfo(info);
120
Christopher Ferris02fdb562017-12-08 15:04:49 -0800121 process_memory_.reset(new MemoryFake);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700122 }
123
124 void SetUp() override {
125 ElfInterfaceFake::FakeClear();
Christopher Ferrisd06001d2017-11-30 18:56:01 -0800126 regs_.FakeSetArch(ARCH_ARM);
Christopher Ferrisfda7edd2017-10-31 16:10:42 -0700127 regs_.FakeSetReturnAddressValid(false);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700128 }
129
130 static MapsFake maps_;
131 static RegsFake regs_;
132 static std::shared_ptr<Memory> process_memory_;
133};
134
135MapsFake UnwinderTest::maps_;
136RegsFake UnwinderTest::regs_(5, 0);
137std::shared_ptr<Memory> UnwinderTest::process_memory_(nullptr);
138
139TEST_F(UnwinderTest, multiple_frames) {
140 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
141 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
142 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
143
144 regs_.FakeSetPc(0x1000);
145 regs_.FakeSetSp(0x10000);
146 ElfInterfaceFake::FakePushStepData(StepData(0x1102, 0x10010, false));
147 ElfInterfaceFake::FakePushStepData(StepData(0x1202, 0x10020, false));
148 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
149
150 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
151 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800152 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700153
154 ASSERT_EQ(3U, unwinder.NumFrames());
155
156 auto* frame = &unwinder.frames()[0];
157 EXPECT_EQ(0U, frame->num);
158 EXPECT_EQ(0U, frame->rel_pc);
159 EXPECT_EQ(0x1000U, frame->pc);
160 EXPECT_EQ(0x10000U, frame->sp);
161 EXPECT_EQ("Frame0", frame->function_name);
162 EXPECT_EQ(0U, frame->function_offset);
163 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
164 EXPECT_EQ(0U, frame->map_offset);
165 EXPECT_EQ(0x1000U, frame->map_start);
166 EXPECT_EQ(0x8000U, frame->map_end);
167 EXPECT_EQ(0U, frame->map_load_bias);
168 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
169
170 frame = &unwinder.frames()[1];
171 EXPECT_EQ(1U, frame->num);
172 EXPECT_EQ(0x100U, frame->rel_pc);
173 EXPECT_EQ(0x1100U, frame->pc);
174 EXPECT_EQ(0x10010U, frame->sp);
175 EXPECT_EQ("Frame1", frame->function_name);
176 EXPECT_EQ(1U, frame->function_offset);
177 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
178 EXPECT_EQ(0U, frame->map_offset);
179 EXPECT_EQ(0x1000U, frame->map_start);
180 EXPECT_EQ(0x8000U, frame->map_end);
181 EXPECT_EQ(0U, frame->map_load_bias);
182 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
183
184 frame = &unwinder.frames()[2];
185 EXPECT_EQ(2U, frame->num);
186 EXPECT_EQ(0x200U, frame->rel_pc);
187 EXPECT_EQ(0x1200U, frame->pc);
188 EXPECT_EQ(0x10020U, frame->sp);
189 EXPECT_EQ("Frame2", frame->function_name);
190 EXPECT_EQ(2U, frame->function_offset);
191 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
192 EXPECT_EQ(0U, frame->map_offset);
193 EXPECT_EQ(0x1000U, frame->map_start);
194 EXPECT_EQ(0x8000U, frame->map_end);
195 EXPECT_EQ(0U, frame->map_load_bias);
196 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
197}
198
Christopher Ferrise4df5f52018-02-12 13:24:18 -0800199TEST_F(UnwinderTest, non_zero_load_bias) {
200 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
201
202 regs_.FakeSetPc(0xa5500);
203 regs_.FakeSetSp(0x10000);
204 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
205
206 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
207 unwinder.Unwind();
208 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
209
210 ASSERT_EQ(1U, unwinder.NumFrames());
211
212 auto* frame = &unwinder.frames()[0];
213 EXPECT_EQ(0U, frame->num);
214 EXPECT_EQ(0x5500U, frame->rel_pc);
215 EXPECT_EQ(0xa5500U, frame->pc);
216 EXPECT_EQ(0x10000U, frame->sp);
217 EXPECT_EQ("Frame0", frame->function_name);
218 EXPECT_EQ(0U, frame->function_offset);
219 EXPECT_EQ("/fake/fake_load_bias.so", frame->map_name);
220 EXPECT_EQ(0U, frame->map_offset);
221 EXPECT_EQ(0xa5000U, frame->map_start);
222 EXPECT_EQ(0xa6000U, frame->map_end);
223 EXPECT_EQ(0x5000U, frame->map_load_bias);
224 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, frame->map_flags);
225}
226
227TEST_F(UnwinderTest, non_zero_elf_offset) {
228 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
229
230 regs_.FakeSetPc(0xa7500);
231 regs_.FakeSetSp(0x10000);
232 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
233
234 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
235 unwinder.Unwind();
236 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
237
238 ASSERT_EQ(1U, unwinder.NumFrames());
239
240 auto* frame = &unwinder.frames()[0];
241 EXPECT_EQ(0U, frame->num);
242 EXPECT_EQ(0x8500U, frame->rel_pc);
243 EXPECT_EQ(0xa7500U, frame->pc);
244 EXPECT_EQ(0x10000U, frame->sp);
245 EXPECT_EQ("Frame0", frame->function_name);
246 EXPECT_EQ(0U, frame->function_offset);
247 EXPECT_EQ("/fake/fake_offset.oat", frame->map_name);
248 EXPECT_EQ(0U, frame->map_offset);
249 EXPECT_EQ(0xa7000U, frame->map_start);
250 EXPECT_EQ(0xa8000U, frame->map_end);
251 EXPECT_EQ(0U, frame->map_load_bias);
252 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, frame->map_flags);
253}
254
Christopher Ferrisc5a3baa2017-10-17 18:42:03 -0700255TEST_F(UnwinderTest, non_zero_map_offset) {
256 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
257
258 regs_.FakeSetPc(0x43000);
259 regs_.FakeSetSp(0x10000);
260 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
261
262 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
263 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800264 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrisc5a3baa2017-10-17 18:42:03 -0700265
266 ASSERT_EQ(1U, unwinder.NumFrames());
267
268 auto* frame = &unwinder.frames()[0];
269 EXPECT_EQ(0U, frame->num);
270 EXPECT_EQ(0U, frame->rel_pc);
271 EXPECT_EQ(0x43000U, frame->pc);
272 EXPECT_EQ(0x10000U, frame->sp);
273 EXPECT_EQ("Frame0", frame->function_name);
274 EXPECT_EQ(0U, frame->function_offset);
275 EXPECT_EQ("/fake/fake.apk", frame->map_name);
276 EXPECT_EQ(0x1d000U, frame->map_offset);
277 EXPECT_EQ(0x43000U, frame->map_start);
278 EXPECT_EQ(0x44000U, frame->map_end);
279 EXPECT_EQ(0U, frame->map_load_bias);
280 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
281}
282
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700283// Verify that no attempt to continue after the step indicates it is done.
284TEST_F(UnwinderTest, no_frames_after_finished) {
285 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
286 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
287 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
288 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame3", 3));
289 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame4", 4));
290
291 regs_.FakeSetPc(0x1000);
292 regs_.FakeSetSp(0x10000);
293 ElfInterfaceFake::FakePushStepData(StepData(0x1000, 0x10000, true));
294 ElfInterfaceFake::FakePushStepData(StepData(0x1102, 0x10010, false));
295 ElfInterfaceFake::FakePushStepData(StepData(0x1202, 0x10020, false));
296
297 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
298 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800299 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700300
301 ASSERT_EQ(1U, unwinder.NumFrames());
302
303 auto* frame = &unwinder.frames()[0];
304 EXPECT_EQ(0U, frame->num);
305 EXPECT_EQ(0U, frame->rel_pc);
306 EXPECT_EQ(0x1000U, frame->pc);
307 EXPECT_EQ(0x10000U, frame->sp);
308 EXPECT_EQ("Frame0", frame->function_name);
309 EXPECT_EQ(0U, frame->function_offset);
310 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
311 EXPECT_EQ(0U, frame->map_offset);
312 EXPECT_EQ(0x1000U, frame->map_start);
313 EXPECT_EQ(0x8000U, frame->map_end);
314 EXPECT_EQ(0U, frame->map_load_bias);
315 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
316}
317
318// Verify the maximum frames to save.
319TEST_F(UnwinderTest, max_frames) {
320 for (size_t i = 0; i < 30; i++) {
321 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame" + std::to_string(i), i));
322 ElfInterfaceFake::FakePushStepData(StepData(0x1102 + i * 0x100, 0x10010 + i * 0x10, false));
323 }
324
325 regs_.FakeSetPc(0x1000);
326 regs_.FakeSetSp(0x10000);
327
328 Unwinder unwinder(20, &maps_, &regs_, process_memory_);
329 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800330 EXPECT_EQ(ERROR_MAX_FRAMES_EXCEEDED, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700331
332 ASSERT_EQ(20U, unwinder.NumFrames());
333
334 for (size_t i = 0; i < 20; i++) {
335 auto* frame = &unwinder.frames()[i];
336 EXPECT_EQ(i, frame->num);
337 EXPECT_EQ(i * 0x100, frame->rel_pc) << "Failed at frame " << i;
338 EXPECT_EQ(0x1000 + i * 0x100, frame->pc) << "Failed at frame " << i;
339 EXPECT_EQ(0x10000 + 0x10 * i, frame->sp) << "Failed at frame " << i;
340 EXPECT_EQ("Frame" + std::to_string(i), frame->function_name) << "Failed at frame " << i;
341 EXPECT_EQ(i, frame->function_offset) << "Failed at frame " << i;
342 EXPECT_EQ("/system/fake/libc.so", frame->map_name) << "Failed at frame " << i;
343 EXPECT_EQ(0U, frame->map_offset) << "Failed at frame " << i;
344 EXPECT_EQ(0x1000U, frame->map_start) << "Failed at frame " << i;
345 EXPECT_EQ(0x8000U, frame->map_end) << "Failed at frame " << i;
346 EXPECT_EQ(0U, frame->map_load_bias) << "Failed at frame " << i;
347 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags) << "Failed at frame " << i;
348 }
349}
350
351// Verify that initial map names frames are removed.
352TEST_F(UnwinderTest, verify_frames_skipped) {
353 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
354 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
355 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
356
357 regs_.FakeSetPc(0x20000);
358 regs_.FakeSetSp(0x10000);
359 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10010, false));
360 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
361 ElfInterfaceFake::FakePushStepData(StepData(0x20002, 0x10030, false));
362 ElfInterfaceFake::FakePushStepData(StepData(0x21002, 0x10040, false));
363 ElfInterfaceFake::FakePushStepData(StepData(0x1002, 0x10050, false));
364 ElfInterfaceFake::FakePushStepData(StepData(0x21002, 0x10060, false));
365 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10070, false));
366 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
367
368 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
Christopher Ferris1cb84ce2017-10-24 15:36:00 -0700369 std::vector<std::string> skip_libs{"libunwind.so", "libanother.so"};
370 unwinder.Unwind(&skip_libs);
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800371 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700372
373 ASSERT_EQ(3U, unwinder.NumFrames());
374
375 auto* frame = &unwinder.frames()[0];
376 EXPECT_EQ(0U, frame->num);
377 EXPECT_EQ(0U, frame->rel_pc);
378 EXPECT_EQ(0x1000U, frame->pc);
379 EXPECT_EQ(0x10050U, frame->sp);
380 EXPECT_EQ("Frame0", frame->function_name);
381 EXPECT_EQ(0U, frame->function_offset);
382 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
383 EXPECT_EQ(0U, frame->map_offset);
384 EXPECT_EQ(0x1000U, frame->map_start);
385 EXPECT_EQ(0x8000U, frame->map_end);
386 EXPECT_EQ(0U, frame->map_load_bias);
387 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
388
389 frame = &unwinder.frames()[1];
390 EXPECT_EQ(1U, frame->num);
391 EXPECT_EQ(0x1000U, frame->rel_pc);
392 EXPECT_EQ(0x21000U, frame->pc);
393 EXPECT_EQ(0x10060U, frame->sp);
394 EXPECT_EQ("Frame1", frame->function_name);
395 EXPECT_EQ(1U, frame->function_offset);
396 EXPECT_EQ("/system/fake/libunwind.so", frame->map_name);
397 EXPECT_EQ(0U, frame->map_offset);
398 EXPECT_EQ(0x20000U, frame->map_start);
399 EXPECT_EQ(0x22000U, frame->map_end);
400 EXPECT_EQ(0U, frame->map_load_bias);
401 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
402
403 frame = &unwinder.frames()[2];
404 EXPECT_EQ(2U, frame->num);
405 EXPECT_EQ(0U, frame->rel_pc);
406 EXPECT_EQ(0x23000U, frame->pc);
407 EXPECT_EQ(0x10070U, frame->sp);
408 EXPECT_EQ("Frame2", frame->function_name);
409 EXPECT_EQ(2U, frame->function_offset);
410 EXPECT_EQ("/fake/libanother.so", frame->map_name);
411 EXPECT_EQ(0U, frame->map_offset);
412 EXPECT_EQ(0x23000U, frame->map_start);
413 EXPECT_EQ(0x24000U, frame->map_end);
414 EXPECT_EQ(0U, frame->map_load_bias);
415 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
416}
417
418// Verify SP in a non-existant map is okay.
419TEST_F(UnwinderTest, sp_not_in_map) {
420 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
421 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
422
423 regs_.FakeSetPc(0x1000);
Christopher Ferrise69f4702017-10-19 16:08:58 -0700424 regs_.FakeSetSp(0x63000);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700425 ElfInterfaceFake::FakePushStepData(StepData(0x21002, 0x50020, false));
426 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
427
428 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
429 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800430 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700431
432 ASSERT_EQ(2U, unwinder.NumFrames());
433
434 auto* frame = &unwinder.frames()[0];
435 EXPECT_EQ(0U, frame->num);
436 EXPECT_EQ(0U, frame->rel_pc);
437 EXPECT_EQ(0x1000U, frame->pc);
Christopher Ferrise69f4702017-10-19 16:08:58 -0700438 EXPECT_EQ(0x63000U, frame->sp);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700439 EXPECT_EQ("Frame0", frame->function_name);
440 EXPECT_EQ(0U, frame->function_offset);
441 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
442 EXPECT_EQ(0U, frame->map_offset);
443 EXPECT_EQ(0x1000U, frame->map_start);
444 EXPECT_EQ(0x8000U, frame->map_end);
445 EXPECT_EQ(0U, frame->map_load_bias);
446 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
447
448 frame = &unwinder.frames()[1];
449 EXPECT_EQ(1U, frame->num);
450 EXPECT_EQ(0x1000U, frame->rel_pc);
451 EXPECT_EQ(0x21000U, frame->pc);
452 EXPECT_EQ(0x50020U, frame->sp);
453 EXPECT_EQ("Frame1", frame->function_name);
454 EXPECT_EQ(1U, frame->function_offset);
455 EXPECT_EQ("/system/fake/libunwind.so", frame->map_name);
456 EXPECT_EQ(0U, frame->map_offset);
457 EXPECT_EQ(0x20000U, frame->map_start);
458 EXPECT_EQ(0x22000U, frame->map_end);
459 EXPECT_EQ(0U, frame->map_load_bias);
460 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
461}
462
463// Verify PC in a device stops the unwind.
464TEST_F(UnwinderTest, pc_in_device_stops_unwind) {
465 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
466 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
467 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
468
469 regs_.FakeSetPc(0x13000);
470 regs_.FakeSetSp(0x10000);
471 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10010, false));
472 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
473 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
474
475 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
476 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800477 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700478
479 ASSERT_EQ(1U, unwinder.NumFrames());
480}
481
482// Verify SP in a device stops the unwind.
483TEST_F(UnwinderTest, sp_in_device_stops_unwind) {
484 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
485 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
486 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
487
488 regs_.FakeSetPc(0x1000);
489 regs_.FakeSetSp(0x13000);
490 ElfInterfaceFake::FakePushStepData(StepData(0x23002, 0x10010, false));
491 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
492 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
493
494 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
495 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800496 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700497
498 ASSERT_EQ(1U, unwinder.NumFrames());
499}
500
501// Verify a no map info frame gets a frame.
502TEST_F(UnwinderTest, pc_without_map) {
503 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
504
505 regs_.FakeSetPc(0x41000);
506 regs_.FakeSetSp(0x13000);
507
508 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
509 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800510 EXPECT_EQ(ERROR_INVALID_MAP, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700511
512 ASSERT_EQ(1U, unwinder.NumFrames());
513
514 auto* frame = &unwinder.frames()[0];
515 EXPECT_EQ(0U, frame->num);
516 EXPECT_EQ(0x41000U, frame->rel_pc);
517 EXPECT_EQ(0x41000U, frame->pc);
518 EXPECT_EQ(0x13000U, frame->sp);
519 EXPECT_EQ("", frame->function_name);
520 EXPECT_EQ(0U, frame->function_offset);
521 EXPECT_EQ("", frame->map_name);
522 EXPECT_EQ(0U, frame->map_offset);
523 EXPECT_EQ(0U, frame->map_start);
524 EXPECT_EQ(0U, frame->map_end);
525 EXPECT_EQ(0U, frame->map_load_bias);
526 EXPECT_EQ(0, frame->map_flags);
527}
528
529// Verify that a speculative frame is added.
530TEST_F(UnwinderTest, speculative_frame) {
531 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
532 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
533
534 // Fake as if code called a nullptr function.
535 regs_.FakeSetPc(0);
536 regs_.FakeSetSp(0x10000);
537 regs_.FakeSetReturnAddress(0x1202);
538 regs_.FakeSetReturnAddressValid(true);
539
540 ElfInterfaceFake::FakePushStepData(StepData(0x23102, 0x10020, false));
541 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
542
543 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
544 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800545 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700546
547 ASSERT_EQ(3U, unwinder.NumFrames());
548
549 auto* frame = &unwinder.frames()[0];
550 EXPECT_EQ(0U, frame->num);
551 EXPECT_EQ(0U, frame->rel_pc);
552 EXPECT_EQ(0U, frame->pc);
553 EXPECT_EQ(0x10000U, frame->sp);
554 EXPECT_EQ("", frame->function_name);
555 EXPECT_EQ(0U, frame->function_offset);
556 EXPECT_EQ("", frame->map_name);
557 EXPECT_EQ(0U, frame->map_offset);
558 EXPECT_EQ(0U, frame->map_start);
559 EXPECT_EQ(0U, frame->map_end);
560 EXPECT_EQ(0U, frame->map_load_bias);
561 EXPECT_EQ(0, frame->map_flags);
562
563 frame = &unwinder.frames()[1];
564 EXPECT_EQ(1U, frame->num);
565 EXPECT_EQ(0x200U, frame->rel_pc);
566 EXPECT_EQ(0x1200U, frame->pc);
567 EXPECT_EQ(0x10000U, frame->sp);
568 EXPECT_EQ("Frame0", frame->function_name);
569 EXPECT_EQ(0U, frame->function_offset);
570 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
571 EXPECT_EQ(0U, frame->map_offset);
572 EXPECT_EQ(0x1000U, frame->map_start);
573 EXPECT_EQ(0x8000U, frame->map_end);
574 EXPECT_EQ(0U, frame->map_load_bias);
575 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
576
577 frame = &unwinder.frames()[2];
578 EXPECT_EQ(2U, frame->num);
579 EXPECT_EQ(0x100U, frame->rel_pc);
580 EXPECT_EQ(0x23100U, frame->pc);
581 EXPECT_EQ(0x10020U, frame->sp);
582 EXPECT_EQ("Frame1", frame->function_name);
583 EXPECT_EQ(1U, frame->function_offset);
584 EXPECT_EQ("/fake/libanother.so", frame->map_name);
585 EXPECT_EQ(0U, frame->map_offset);
586 EXPECT_EQ(0x23000U, frame->map_start);
587 EXPECT_EQ(0x24000U, frame->map_end);
588 EXPECT_EQ(0U, frame->map_load_bias);
589 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
590}
591
592// Verify that a speculative frame is added then removed because no other
593// frames are added.
594TEST_F(UnwinderTest, speculative_frame_removed) {
595 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
596 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
597
598 // Fake as if code called a nullptr function.
599 regs_.FakeSetPc(0);
600 regs_.FakeSetSp(0x10000);
601 regs_.FakeSetReturnAddress(0x1202);
602 regs_.FakeSetReturnAddressValid(true);
603
604 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
605 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800606 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700607
608 ASSERT_EQ(1U, unwinder.NumFrames());
609
610 auto* frame = &unwinder.frames()[0];
611 EXPECT_EQ(0U, frame->num);
612 EXPECT_EQ(0U, frame->rel_pc);
613 EXPECT_EQ(0U, frame->pc);
614 EXPECT_EQ(0x10000U, frame->sp);
615 EXPECT_EQ("", frame->function_name);
616 EXPECT_EQ(0U, frame->function_offset);
617 EXPECT_EQ("", frame->map_name);
618 EXPECT_EQ(0U, frame->map_offset);
619 EXPECT_EQ(0U, frame->map_start);
620 EXPECT_EQ(0U, frame->map_end);
621 EXPECT_EQ(0U, frame->map_load_bias);
622 EXPECT_EQ(0, frame->map_flags);
623}
624
Christopher Ferrise69f4702017-10-19 16:08:58 -0700625// Verify that an unwind stops when a frame is in given suffix.
626TEST_F(UnwinderTest, map_ignore_suffixes) {
627 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
628 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
629 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
630 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame3", 3));
631
632 // Fake as if code called a nullptr function.
633 regs_.FakeSetPc(0x1000);
634 regs_.FakeSetSp(0x10000);
635 ElfInterfaceFake::FakePushStepData(StepData(0x43402, 0x10010, false));
636 ElfInterfaceFake::FakePushStepData(StepData(0x53502, 0x10020, false));
637 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
638
639 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
Christopher Ferris1cb84ce2017-10-24 15:36:00 -0700640 std::vector<std::string> suffixes{"oat"};
Christopher Ferrise69f4702017-10-19 16:08:58 -0700641 unwinder.Unwind(nullptr, &suffixes);
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800642 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
Christopher Ferrise69f4702017-10-19 16:08:58 -0700643
644 ASSERT_EQ(2U, unwinder.NumFrames());
645 // Make sure the elf was not initialized.
646 MapInfo* map_info = maps_.Find(0x53000);
647 ASSERT_TRUE(map_info != nullptr);
648 EXPECT_TRUE(map_info->elf == nullptr);
649
650 auto* frame = &unwinder.frames()[0];
651 EXPECT_EQ(0U, frame->num);
652 EXPECT_EQ(0U, frame->rel_pc);
653 EXPECT_EQ(0x1000U, frame->pc);
654 EXPECT_EQ(0x10000U, frame->sp);
655 EXPECT_EQ("Frame0", frame->function_name);
656 EXPECT_EQ(0U, frame->function_offset);
657 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
658 EXPECT_EQ(0U, frame->map_offset);
659 EXPECT_EQ(0x1000U, frame->map_start);
660 EXPECT_EQ(0x8000U, frame->map_end);
661 EXPECT_EQ(0U, frame->map_load_bias);
662 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
663
664 frame = &unwinder.frames()[1];
665 EXPECT_EQ(1U, frame->num);
666 EXPECT_EQ(0x400U, frame->rel_pc);
667 EXPECT_EQ(0x43400U, frame->pc);
668 EXPECT_EQ(0x10010U, frame->sp);
669 EXPECT_EQ("Frame1", frame->function_name);
670 EXPECT_EQ(1U, frame->function_offset);
671 EXPECT_EQ("/fake/fake.apk", frame->map_name);
672 EXPECT_EQ(0x1d000U, frame->map_offset);
673 EXPECT_EQ(0x43000U, frame->map_start);
674 EXPECT_EQ(0x44000U, frame->map_end);
675 EXPECT_EQ(0U, frame->map_load_bias);
676 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
677}
678
Christopher Ferrisfda7edd2017-10-31 16:10:42 -0700679// Verify that an unwind stops when the sp and pc don't change.
680TEST_F(UnwinderTest, sp_pc_do_not_change) {
681 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
682 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
683 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame2", 2));
684 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame3", 3));
685 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame4", 4));
686
687 regs_.FakeSetPc(0x1000);
688 regs_.FakeSetSp(0x10000);
689 ElfInterfaceFake::FakePushStepData(StepData(0x33402, 0x10010, false));
690 ElfInterfaceFake::FakePushStepData(StepData(0x33502, 0x10020, false));
691 ElfInterfaceFake::FakePushStepData(StepData(0x33502, 0x10020, false));
692 ElfInterfaceFake::FakePushStepData(StepData(0x33502, 0x10020, false));
693 ElfInterfaceFake::FakePushStepData(StepData(0x33502, 0x10020, false));
694 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
695
696 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
697 unwinder.Unwind();
Christopher Ferris2fcf4cf2018-01-23 17:52:23 -0800698 EXPECT_EQ(ERROR_REPEATED_FRAME, unwinder.LastErrorCode());
Christopher Ferrisfda7edd2017-10-31 16:10:42 -0700699
700 ASSERT_EQ(3U, unwinder.NumFrames());
701
702 auto* frame = &unwinder.frames()[0];
703 EXPECT_EQ(0U, frame->num);
704 EXPECT_EQ(0U, frame->rel_pc);
705 EXPECT_EQ(0x1000U, frame->pc);
706 EXPECT_EQ(0x10000U, frame->sp);
707 EXPECT_EQ("Frame0", frame->function_name);
708 EXPECT_EQ(0U, frame->function_offset);
709 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
710 EXPECT_EQ(0U, frame->map_offset);
711 EXPECT_EQ(0x1000U, frame->map_start);
712 EXPECT_EQ(0x8000U, frame->map_end);
713 EXPECT_EQ(0U, frame->map_load_bias);
714 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
715
716 frame = &unwinder.frames()[1];
717 EXPECT_EQ(1U, frame->num);
718 EXPECT_EQ(0x400U, frame->rel_pc);
719 EXPECT_EQ(0x33400U, frame->pc);
720 EXPECT_EQ(0x10010U, frame->sp);
721 EXPECT_EQ("Frame1", frame->function_name);
722 EXPECT_EQ(1U, frame->function_offset);
723 EXPECT_EQ("/fake/compressed.so", frame->map_name);
724 EXPECT_EQ(0U, frame->map_offset);
725 EXPECT_EQ(0x33000U, frame->map_start);
726 EXPECT_EQ(0x34000U, frame->map_end);
727 EXPECT_EQ(0U, frame->map_load_bias);
728 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
729
730 frame = &unwinder.frames()[2];
731 EXPECT_EQ(2U, frame->num);
732 EXPECT_EQ(0x500U, frame->rel_pc);
733 EXPECT_EQ(0x33500U, frame->pc);
734 EXPECT_EQ(0x10020U, frame->sp);
735 EXPECT_EQ("Frame2", frame->function_name);
736 EXPECT_EQ(2U, frame->function_offset);
737 EXPECT_EQ("/fake/compressed.so", frame->map_name);
738 EXPECT_EQ(0U, frame->map_offset);
739 EXPECT_EQ(0x33000U, frame->map_start);
740 EXPECT_EQ(0x34000U, frame->map_end);
741 EXPECT_EQ(0U, frame->map_load_bias);
742 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
743}
744
Christopher Ferrise762f1f2018-02-06 14:51:48 -0800745TEST_F(UnwinderTest, dex_pc_in_map) {
746 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
747 regs_.FakeSetPc(0x1000);
748 regs_.FakeSetSp(0x10000);
749 regs_.FakeSetDexPc(0xa3400);
750
751 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
752 unwinder.Unwind();
753 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
754
755 ASSERT_EQ(2U, unwinder.NumFrames());
756
757 auto* frame = &unwinder.frames()[0];
758 EXPECT_EQ(0U, frame->num);
759 EXPECT_EQ(0x400U, frame->rel_pc);
760 EXPECT_EQ(0xa3400U, frame->pc);
761 EXPECT_EQ(0x10000U, frame->sp);
762 EXPECT_EQ("", frame->function_name);
763 EXPECT_EQ(0U, frame->function_offset);
764 EXPECT_EQ("/fake/fake.vdex", frame->map_name);
765 EXPECT_EQ(0U, frame->map_offset);
766 EXPECT_EQ(0xa3000U, frame->map_start);
767 EXPECT_EQ(0xa4000U, frame->map_end);
768 EXPECT_EQ(0U, frame->map_load_bias);
769 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, frame->map_flags);
770
771 frame = &unwinder.frames()[1];
772 EXPECT_EQ(1U, frame->num);
773 EXPECT_EQ(0U, frame->rel_pc);
774 EXPECT_EQ(0x1000U, frame->pc);
775 EXPECT_EQ(0x10000U, frame->sp);
776 EXPECT_EQ("Frame0", frame->function_name);
777 EXPECT_EQ(0U, frame->function_offset);
778 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
779 EXPECT_EQ(0U, frame->map_offset);
780 EXPECT_EQ(0x1000U, frame->map_start);
781 EXPECT_EQ(0x8000U, frame->map_end);
782 EXPECT_EQ(0U, frame->map_load_bias);
783 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
784}
785
786TEST_F(UnwinderTest, dex_pc_not_in_map) {
787 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
788 regs_.FakeSetPc(0x1000);
789 regs_.FakeSetSp(0x10000);
790 regs_.FakeSetDexPc(0x50000);
791
792 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
793 unwinder.Unwind();
794 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
795
796 ASSERT_EQ(2U, unwinder.NumFrames());
797
798 auto* frame = &unwinder.frames()[0];
799 EXPECT_EQ(0U, frame->num);
800 EXPECT_EQ(0x50000U, frame->rel_pc);
801 EXPECT_EQ(0x50000U, frame->pc);
802 EXPECT_EQ(0x10000U, frame->sp);
803 EXPECT_EQ("", frame->function_name);
804 EXPECT_EQ(0U, frame->function_offset);
805 EXPECT_EQ("", frame->map_name);
806 EXPECT_EQ(0U, frame->map_offset);
807 EXPECT_EQ(0U, frame->map_start);
808 EXPECT_EQ(0U, frame->map_end);
809 EXPECT_EQ(0U, frame->map_load_bias);
810 EXPECT_EQ(0, frame->map_flags);
811
812 frame = &unwinder.frames()[1];
813 EXPECT_EQ(1U, frame->num);
814 EXPECT_EQ(0U, frame->rel_pc);
815 EXPECT_EQ(0x1000U, frame->pc);
816 EXPECT_EQ(0x10000U, frame->sp);
817 EXPECT_EQ("Frame0", frame->function_name);
818 EXPECT_EQ(0U, frame->function_offset);
819 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
820 EXPECT_EQ(0U, frame->map_offset);
821 EXPECT_EQ(0x1000U, frame->map_start);
822 EXPECT_EQ(0x8000U, frame->map_end);
823 EXPECT_EQ(0U, frame->map_load_bias);
824 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
825}
826
Christopher Ferrisa8c39732018-02-12 08:46:19 -0800827TEST_F(UnwinderTest, dex_pc_multiple_frames) {
828 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 0));
829 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame1", 1));
830 regs_.FakeSetPc(0x1000);
831 regs_.FakeSetSp(0x10000);
832 regs_.FakeSetDexPc(0xa3400);
833 ElfInterfaceFake::FakePushStepData(StepData(0x33402, 0x10010, false));
834 ElfInterfaceFake::FakePushStepData(StepData(0, 0, true));
835
836 Unwinder unwinder(64, &maps_, &regs_, process_memory_);
837 unwinder.Unwind();
838 EXPECT_EQ(ERROR_NONE, unwinder.LastErrorCode());
839
840 ASSERT_EQ(3U, unwinder.NumFrames());
841
842 auto* frame = &unwinder.frames()[0];
843 EXPECT_EQ(0U, frame->num);
844 EXPECT_EQ(0x400U, frame->rel_pc);
845 EXPECT_EQ(0xa3400U, frame->pc);
846 EXPECT_EQ(0x10000U, frame->sp);
847 EXPECT_EQ("", frame->function_name);
848 EXPECT_EQ(0U, frame->function_offset);
849 EXPECT_EQ("/fake/fake.vdex", frame->map_name);
850 EXPECT_EQ(0U, frame->map_offset);
851 EXPECT_EQ(0xa3000U, frame->map_start);
852 EXPECT_EQ(0xa4000U, frame->map_end);
853 EXPECT_EQ(0U, frame->map_load_bias);
854 EXPECT_EQ(PROT_READ | PROT_WRITE | PROT_EXEC, frame->map_flags);
855
856 frame = &unwinder.frames()[1];
857 EXPECT_EQ(1U, frame->num);
858 EXPECT_EQ(0U, frame->rel_pc);
859 EXPECT_EQ(0x1000U, frame->pc);
860 EXPECT_EQ(0x10000U, frame->sp);
861 EXPECT_EQ("Frame0", frame->function_name);
862 EXPECT_EQ(0U, frame->function_offset);
863 EXPECT_EQ("/system/fake/libc.so", frame->map_name);
864 EXPECT_EQ(0U, frame->map_offset);
865 EXPECT_EQ(0x1000U, frame->map_start);
866 EXPECT_EQ(0x8000U, frame->map_end);
867 EXPECT_EQ(0U, frame->map_load_bias);
868 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
869
870 frame = &unwinder.frames()[2];
871 EXPECT_EQ(2U, frame->num);
872 EXPECT_EQ(0x400U, frame->rel_pc);
873 EXPECT_EQ(0x33400U, frame->pc);
874 EXPECT_EQ(0x10010U, frame->sp);
875 EXPECT_EQ("Frame1", frame->function_name);
876 EXPECT_EQ(1U, frame->function_offset);
877 EXPECT_EQ("/fake/compressed.so", frame->map_name);
878 EXPECT_EQ(0U, frame->map_offset);
879 EXPECT_EQ(0x33000U, frame->map_start);
880 EXPECT_EQ(0x34000U, frame->map_end);
881 EXPECT_EQ(0U, frame->map_load_bias);
882 EXPECT_EQ(PROT_READ | PROT_WRITE, frame->map_flags);
883}
884
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700885// Verify format frame code.
886TEST_F(UnwinderTest, format_frame_static) {
887 FrameData frame;
888 frame.num = 1;
889 frame.rel_pc = 0x1000;
890 frame.pc = 0x4000;
891 frame.sp = 0x1000;
892 frame.function_name = "function";
893 frame.function_offset = 100;
894 frame.map_name = "/fake/libfake.so";
895 frame.map_offset = 0x2000;
896 frame.map_start = 0x3000;
897 frame.map_end = 0x6000;
898 frame.map_flags = PROT_READ;
899
900 EXPECT_EQ(" #01 pc 0000000000001000 (offset 0x2000) /fake/libfake.so (function+100)",
901 Unwinder::FormatFrame(frame, false));
902 EXPECT_EQ(" #01 pc 00001000 (offset 0x2000) /fake/libfake.so (function+100)",
903 Unwinder::FormatFrame(frame, true));
904
905 frame.map_offset = 0;
906 EXPECT_EQ(" #01 pc 0000000000001000 /fake/libfake.so (function+100)",
907 Unwinder::FormatFrame(frame, false));
908 EXPECT_EQ(" #01 pc 00001000 /fake/libfake.so (function+100)",
909 Unwinder::FormatFrame(frame, true));
910
911 frame.function_offset = 0;
912 EXPECT_EQ(" #01 pc 0000000000001000 /fake/libfake.so (function)",
913 Unwinder::FormatFrame(frame, false));
914 EXPECT_EQ(" #01 pc 00001000 /fake/libfake.so (function)", Unwinder::FormatFrame(frame, true));
915
916 frame.function_name = "";
917 EXPECT_EQ(" #01 pc 0000000000001000 /fake/libfake.so", Unwinder::FormatFrame(frame, false));
918 EXPECT_EQ(" #01 pc 00001000 /fake/libfake.so", Unwinder::FormatFrame(frame, true));
919
920 frame.map_name = "";
921 EXPECT_EQ(" #01 pc 0000000000001000 <anonymous:3000>", Unwinder::FormatFrame(frame, false));
922 EXPECT_EQ(" #01 pc 00001000 <anonymous:3000>", Unwinder::FormatFrame(frame, true));
923
924 frame.map_start = 0;
925 frame.map_end = 0;
926 EXPECT_EQ(" #01 pc 0000000000001000 <unknown>", Unwinder::FormatFrame(frame, false));
927 EXPECT_EQ(" #01 pc 00001000 <unknown>", Unwinder::FormatFrame(frame, true));
928}
929
Christopher Ferris02fdb562017-12-08 15:04:49 -0800930static std::string ArchToString(ArchEnum arch) {
931 if (arch == ARCH_ARM) {
932 return "Arm";
933 } else if (arch == ARCH_ARM64) {
934 return "Arm64";
935 } else if (arch == ARCH_X86) {
936 return "X86";
937 } else if (arch == ARCH_X86_64) {
938 return "X86_64";
939 } else {
940 return "Unknown";
941 }
942}
943
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700944// Verify format frame code.
945TEST_F(UnwinderTest, format_frame) {
Christopher Ferris02fdb562017-12-08 15:04:49 -0800946 std::vector<Regs*> reg_list;
947 RegsArm* arm = new RegsArm;
948 arm->set_pc(0x2300);
949 arm->set_sp(0x10000);
950 reg_list.push_back(arm);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700951
Christopher Ferris02fdb562017-12-08 15:04:49 -0800952 RegsArm64* arm64 = new RegsArm64;
953 arm64->set_pc(0x2300);
954 arm64->set_sp(0x10000);
955 reg_list.push_back(arm64);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700956
Christopher Ferris02fdb562017-12-08 15:04:49 -0800957 RegsX86* x86 = new RegsX86;
958 x86->set_pc(0x2300);
959 x86->set_sp(0x10000);
960 reg_list.push_back(x86);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700961
Christopher Ferris02fdb562017-12-08 15:04:49 -0800962 RegsX86_64* x86_64 = new RegsX86_64;
963 x86_64->set_pc(0x2300);
964 x86_64->set_sp(0x10000);
965 reg_list.push_back(x86_64);
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700966
Douglas Leung61b1a1a2017-11-08 10:53:53 +0100967 RegsMips* mips = new RegsMips;
968 mips->set_pc(0x2300);
969 mips->set_sp(0x10000);
970 reg_list.push_back(mips);
971
972 RegsMips64* mips64 = new RegsMips64;
973 mips64->set_pc(0x2300);
974 mips64->set_sp(0x10000);
975 reg_list.push_back(mips64);
976
Christopher Ferris02fdb562017-12-08 15:04:49 -0800977 for (auto regs : reg_list) {
978 ElfInterfaceFake::FakePushFunctionData(FunctionData("Frame0", 10));
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700979
Christopher Ferris02fdb562017-12-08 15:04:49 -0800980 Unwinder unwinder(64, &maps_, regs, process_memory_);
981 unwinder.Unwind();
Christopher Ferrisf6f691b2017-09-25 19:23:07 -0700982
Christopher Ferris02fdb562017-12-08 15:04:49 -0800983 ASSERT_EQ(1U, unwinder.NumFrames());
984 std::string expected;
985 switch (regs->Arch()) {
986 case ARCH_ARM:
987 case ARCH_X86:
Douglas Leung61b1a1a2017-11-08 10:53:53 +0100988 case ARCH_MIPS:
Christopher Ferris02fdb562017-12-08 15:04:49 -0800989 expected = " #00 pc 00001300 /system/fake/libc.so (Frame0+10)";
990 break;
991 case ARCH_ARM64:
992 case ARCH_X86_64:
Douglas Leung61b1a1a2017-11-08 10:53:53 +0100993 case ARCH_MIPS64:
Christopher Ferris02fdb562017-12-08 15:04:49 -0800994 expected = " #00 pc 0000000000001300 /system/fake/libc.so (Frame0+10)";
995 break;
996 default:
997 expected = "";
998 }
999 EXPECT_EQ(expected, unwinder.FormatFrame(0))
1000 << "Mismatch of frame format for regs arch " << ArchToString(regs->Arch());
1001 delete regs;
1002 }
Christopher Ferrisf6f691b2017-09-25 19:23:07 -07001003}
1004
1005} // namespace unwindstack