Merge "Add a new unwind method on error."
am: 9b91324cb0
Change-Id: I99aca55a9ae9ffd2c7220885ec033486996b31a6
This commit is contained in:
commit
fb8ea2626b
24 changed files with 735 additions and 238 deletions
|
|
@ -114,6 +114,11 @@ noinline int do_action_on_thread(const char* arg) {
|
||||||
return reinterpret_cast<uintptr_t>(result);
|
return reinterpret_cast<uintptr_t>(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
noinline int crash_null() {
|
||||||
|
int (*null_func)() = nullptr;
|
||||||
|
return null_func();
|
||||||
|
}
|
||||||
|
|
||||||
noinline int crash3(int a) {
|
noinline int crash3(int a) {
|
||||||
*reinterpret_cast<int*>(0xdead) = a;
|
*reinterpret_cast<int*>(0xdead) = a;
|
||||||
return a*4;
|
return a*4;
|
||||||
|
|
@ -169,6 +174,7 @@ static int usage() {
|
||||||
fprintf(stderr, " nostack crash with a NULL stack pointer\n");
|
fprintf(stderr, " nostack crash with a NULL stack pointer\n");
|
||||||
fprintf(stderr, "\n");
|
fprintf(stderr, "\n");
|
||||||
fprintf(stderr, " heap-usage cause a libc abort by abusing a heap function\n");
|
fprintf(stderr, " heap-usage cause a libc abort by abusing a heap function\n");
|
||||||
|
fprintf(stderr, " call-null cause a crash by calling through a nullptr\n");
|
||||||
fprintf(stderr, " leak leak memory until we get OOM-killed\n");
|
fprintf(stderr, " leak leak memory until we get OOM-killed\n");
|
||||||
fprintf(stderr, "\n");
|
fprintf(stderr, "\n");
|
||||||
fprintf(stderr, " abort call abort()\n");
|
fprintf(stderr, " abort call abort()\n");
|
||||||
|
|
@ -239,6 +245,8 @@ noinline int do_action(const char* arg) {
|
||||||
crashnostack();
|
crashnostack();
|
||||||
} else if (!strcasecmp(arg, "exit")) {
|
} else if (!strcasecmp(arg, "exit")) {
|
||||||
exit(1);
|
exit(1);
|
||||||
|
} else if (!strcasecmp(arg, "call-null")) {
|
||||||
|
return crash_null();
|
||||||
} else if (!strcasecmp(arg, "crash") || !strcmp(arg, "SIGSEGV")) {
|
} else if (!strcasecmp(arg, "crash") || !strcmp(arg, "SIGSEGV")) {
|
||||||
return crash(42);
|
return crash(42);
|
||||||
} else if (!strcasecmp(arg, "abort")) {
|
} else if (!strcasecmp(arg, "abort")) {
|
||||||
|
|
|
||||||
|
|
@ -68,6 +68,32 @@ static bool IsUnwindLibrary(const std::string& map_name) {
|
||||||
return library == "libunwindstack.so" || library == "libbacktrace.so";
|
return library == "libunwindstack.so" || library == "libbacktrace.so";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void SetFrameInfo(unwindstack::Regs* regs, unwindstack::MapInfo* map_info,
|
||||||
|
uint64_t adjusted_rel_pc, backtrace_frame_data_t* frame) {
|
||||||
|
// This will point to the adjusted absolute pc. regs->pc() is
|
||||||
|
// unaltered.
|
||||||
|
frame->pc = map_info->start + adjusted_rel_pc;
|
||||||
|
frame->sp = regs->sp();
|
||||||
|
frame->rel_pc = adjusted_rel_pc;
|
||||||
|
frame->stack_size = 0;
|
||||||
|
|
||||||
|
frame->map.start = map_info->start;
|
||||||
|
frame->map.end = map_info->end;
|
||||||
|
frame->map.offset = map_info->offset;
|
||||||
|
frame->map.flags = map_info->flags;
|
||||||
|
frame->map.name = map_info->name;
|
||||||
|
|
||||||
|
unwindstack::Elf* elf = map_info->elf;
|
||||||
|
frame->map.load_bias = elf->GetLoadBias();
|
||||||
|
uint64_t func_offset = 0;
|
||||||
|
if (elf->GetFunctionName(adjusted_rel_pc, &frame->func_name, &func_offset)) {
|
||||||
|
frame->func_name = demangle(frame->func_name.c_str());
|
||||||
|
} else {
|
||||||
|
frame->func_name = "";
|
||||||
|
}
|
||||||
|
frame->func_offset = func_offset;
|
||||||
|
}
|
||||||
|
|
||||||
static bool Unwind(unwindstack::Regs* regs, BacktraceMap* back_map,
|
static bool Unwind(unwindstack::Regs* regs, BacktraceMap* back_map,
|
||||||
std::vector<backtrace_frame_data_t>* frames, size_t num_ignore_frames) {
|
std::vector<backtrace_frame_data_t>* frames, size_t num_ignore_frames) {
|
||||||
UnwindStackMap* stack_map = reinterpret_cast<UnwindStackMap*>(back_map);
|
UnwindStackMap* stack_map = reinterpret_cast<UnwindStackMap*>(back_map);
|
||||||
|
|
@ -75,70 +101,96 @@ static bool Unwind(unwindstack::Regs* regs, BacktraceMap* back_map,
|
||||||
bool adjust_rel_pc = false;
|
bool adjust_rel_pc = false;
|
||||||
size_t num_frames = 0;
|
size_t num_frames = 0;
|
||||||
frames->clear();
|
frames->clear();
|
||||||
|
bool return_address_attempted = false;
|
||||||
|
auto process_memory = stack_map->process_memory();
|
||||||
while (num_frames < MAX_BACKTRACE_FRAMES) {
|
while (num_frames < MAX_BACKTRACE_FRAMES) {
|
||||||
if (regs->pc() == 0) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
unwindstack::MapInfo* map_info = maps->Find(regs->pc());
|
unwindstack::MapInfo* map_info = maps->Find(regs->pc());
|
||||||
|
bool stepped;
|
||||||
|
bool in_device_map = false;
|
||||||
if (map_info == nullptr) {
|
if (map_info == nullptr) {
|
||||||
break;
|
stepped = false;
|
||||||
}
|
if (num_ignore_frames == 0) {
|
||||||
|
frames->resize(num_frames + 1);
|
||||||
unwindstack::Elf* elf = map_info->GetElf(stack_map->process_memory(), true);
|
backtrace_frame_data_t* frame = &frames->at(num_frames);
|
||||||
uint64_t rel_pc = elf->GetRelPc(regs->pc(), map_info);
|
frame->pc = regs->pc();
|
||||||
|
frame->sp = regs->sp();
|
||||||
bool skip_frame = num_frames == 0 && IsUnwindLibrary(map_info->name);
|
frame->rel_pc = frame->pc;
|
||||||
if (num_ignore_frames == 0 && !skip_frame) {
|
num_frames++;
|
||||||
uint64_t adjusted_rel_pc = rel_pc;
|
|
||||||
if (adjust_rel_pc) {
|
|
||||||
adjusted_rel_pc = regs->GetAdjustedPc(rel_pc, elf);
|
|
||||||
}
|
|
||||||
frames->resize(num_frames + 1);
|
|
||||||
backtrace_frame_data_t* frame = &frames->at(num_frames);
|
|
||||||
frame->num = num_frames;
|
|
||||||
// This will point to the adjusted absolute pc. regs->pc() is
|
|
||||||
// unaltered.
|
|
||||||
frame->pc = map_info->start + adjusted_rel_pc;
|
|
||||||
frame->sp = regs->sp();
|
|
||||||
frame->rel_pc = adjusted_rel_pc;
|
|
||||||
frame->stack_size = 0;
|
|
||||||
|
|
||||||
frame->map.start = map_info->start;
|
|
||||||
frame->map.end = map_info->end;
|
|
||||||
frame->map.offset = map_info->offset;
|
|
||||||
frame->map.load_bias = elf->GetLoadBias();
|
|
||||||
frame->map.flags = map_info->flags;
|
|
||||||
frame->map.name = map_info->name;
|
|
||||||
|
|
||||||
uint64_t func_offset = 0;
|
|
||||||
if (elf->GetFunctionName(adjusted_rel_pc, &frame->func_name, &func_offset)) {
|
|
||||||
frame->func_name = demangle(frame->func_name.c_str());
|
|
||||||
} else {
|
} else {
|
||||||
frame->func_name = "";
|
num_ignore_frames--;
|
||||||
}
|
}
|
||||||
frame->func_offset = func_offset;
|
} else {
|
||||||
if (num_frames > 0) {
|
unwindstack::Elf* elf = map_info->GetElf(process_memory, true);
|
||||||
// Set the stack size for the previous frame.
|
uint64_t rel_pc = elf->GetRelPc(regs->pc(), map_info);
|
||||||
backtrace_frame_data_t* prev = &frames->at(num_frames - 1);
|
|
||||||
prev->stack_size = frame->sp - prev->sp;
|
if (frames->size() != 0 || !IsUnwindLibrary(map_info->name)) {
|
||||||
|
if (num_ignore_frames == 0) {
|
||||||
|
uint64_t adjusted_rel_pc = rel_pc;
|
||||||
|
if (adjust_rel_pc) {
|
||||||
|
adjusted_rel_pc = regs->GetAdjustedPc(rel_pc, elf);
|
||||||
|
}
|
||||||
|
|
||||||
|
frames->resize(num_frames + 1);
|
||||||
|
backtrace_frame_data_t* frame = &frames->at(num_frames);
|
||||||
|
frame->num = num_frames;
|
||||||
|
SetFrameInfo(regs, map_info, adjusted_rel_pc, frame);
|
||||||
|
|
||||||
|
if (num_frames > 0) {
|
||||||
|
// Set the stack size for the previous frame.
|
||||||
|
backtrace_frame_data_t* prev = &frames->at(num_frames - 1);
|
||||||
|
prev->stack_size = frame->sp - prev->sp;
|
||||||
|
}
|
||||||
|
num_frames++;
|
||||||
|
} else {
|
||||||
|
num_ignore_frames--;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (map_info->flags & PROT_DEVICE_MAP) {
|
||||||
|
// Do not stop here, fall through in case we are
|
||||||
|
// in the speculative unwind path and need to remove
|
||||||
|
// some of the speculative frames.
|
||||||
|
stepped = false;
|
||||||
|
in_device_map = true;
|
||||||
|
} else {
|
||||||
|
unwindstack::MapInfo* sp_info = maps->Find(regs->sp());
|
||||||
|
if (sp_info->flags & PROT_DEVICE_MAP) {
|
||||||
|
// Do not stop here, fall through in case we are
|
||||||
|
// in the speculative unwind path and need to remove
|
||||||
|
// some of the speculative frames.
|
||||||
|
stepped = false;
|
||||||
|
in_device_map = true;
|
||||||
|
} else {
|
||||||
|
bool finished;
|
||||||
|
stepped = elf->Step(rel_pc + map_info->elf_offset, regs, process_memory.get(), &finished);
|
||||||
|
if (stepped && finished) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
num_frames++;
|
|
||||||
} else if (!skip_frame && num_ignore_frames > 0) {
|
|
||||||
num_ignore_frames--;
|
|
||||||
}
|
}
|
||||||
adjust_rel_pc = true;
|
adjust_rel_pc = true;
|
||||||
|
|
||||||
// Do not unwind through a device map.
|
if (!stepped) {
|
||||||
if (map_info->flags & PROT_DEVICE_MAP) {
|
if (return_address_attempted) {
|
||||||
break;
|
// Remove the speculative frame.
|
||||||
}
|
if (frames->size() > 0) {
|
||||||
unwindstack::MapInfo* sp_info = maps->Find(regs->sp());
|
frames->pop_back();
|
||||||
if (sp_info->flags & PROT_DEVICE_MAP) {
|
}
|
||||||
break;
|
break;
|
||||||
}
|
} else if (in_device_map) {
|
||||||
|
// Do not attempt any other unwinding, pc or sp is in a device
|
||||||
if (!elf->Step(rel_pc + map_info->elf_offset, regs, stack_map->process_memory().get())) {
|
// map.
|
||||||
break;
|
break;
|
||||||
|
} else {
|
||||||
|
// Stepping didn't work, try this secondary method.
|
||||||
|
if (!regs->SetPcFromReturnAddress(process_memory.get())) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return_address_attempted = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return_address_attempted = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -82,6 +82,14 @@ struct dump_thread_t {
|
||||||
int32_t done;
|
int32_t done;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
typedef Backtrace* (*create_func_t)(pid_t, pid_t, BacktraceMap*);
|
||||||
|
typedef BacktraceMap* (*map_create_func_t)(pid_t, bool);
|
||||||
|
|
||||||
|
static void VerifyLevelDump(Backtrace* backtrace, create_func_t create_func = nullptr,
|
||||||
|
map_create_func_t map_func = nullptr);
|
||||||
|
static void VerifyMaxDump(Backtrace* backtrace, create_func_t create_func = nullptr,
|
||||||
|
map_create_func_t map_func = nullptr);
|
||||||
|
|
||||||
static uint64_t NanoTime() {
|
static uint64_t NanoTime() {
|
||||||
struct timespec t = { 0, 0 };
|
struct timespec t = { 0, 0 };
|
||||||
clock_gettime(CLOCK_MONOTONIC, &t);
|
clock_gettime(CLOCK_MONOTONIC, &t);
|
||||||
|
|
@ -147,7 +155,7 @@ static bool ReadyLevelBacktrace(Backtrace* backtrace) {
|
||||||
return found;
|
return found;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void VerifyLevelDump(Backtrace* backtrace) {
|
static void VerifyLevelDump(Backtrace* backtrace, create_func_t, map_create_func_t) {
|
||||||
ASSERT_GT(backtrace->NumFrames(), static_cast<size_t>(0))
|
ASSERT_GT(backtrace->NumFrames(), static_cast<size_t>(0))
|
||||||
<< DumpFrames(backtrace);
|
<< DumpFrames(backtrace);
|
||||||
ASSERT_LT(backtrace->NumFrames(), static_cast<size_t>(MAX_BACKTRACE_FRAMES))
|
ASSERT_LT(backtrace->NumFrames(), static_cast<size_t>(MAX_BACKTRACE_FRAMES))
|
||||||
|
|
@ -189,7 +197,7 @@ static bool ReadyMaxBacktrace(Backtrace* backtrace) {
|
||||||
return (backtrace->NumFrames() == MAX_BACKTRACE_FRAMES);
|
return (backtrace->NumFrames() == MAX_BACKTRACE_FRAMES);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void VerifyMaxDump(Backtrace* backtrace) {
|
static void VerifyMaxDump(Backtrace* backtrace, create_func_t, map_create_func_t) {
|
||||||
ASSERT_EQ(backtrace->NumFrames(), static_cast<size_t>(MAX_BACKTRACE_FRAMES))
|
ASSERT_EQ(backtrace->NumFrames(), static_cast<size_t>(MAX_BACKTRACE_FRAMES))
|
||||||
<< DumpFrames(backtrace);
|
<< DumpFrames(backtrace);
|
||||||
// Verify that the last frame is our recursive call.
|
// Verify that the last frame is our recursive call.
|
||||||
|
|
@ -251,10 +259,14 @@ TEST(libbacktrace, local_trace) {
|
||||||
|
|
||||||
static void VerifyIgnoreFrames(Backtrace* bt_all, Backtrace* bt_ign1, Backtrace* bt_ign2,
|
static void VerifyIgnoreFrames(Backtrace* bt_all, Backtrace* bt_ign1, Backtrace* bt_ign2,
|
||||||
const char* cur_proc) {
|
const char* cur_proc) {
|
||||||
EXPECT_EQ(bt_all->NumFrames(), bt_ign1->NumFrames() + 1)
|
ASSERT_EQ(bt_all->NumFrames(), bt_ign1->NumFrames() + 1) << "All backtrace:\n"
|
||||||
<< "All backtrace:\n" << DumpFrames(bt_all) << "Ignore 1 backtrace:\n" << DumpFrames(bt_ign1);
|
<< DumpFrames(bt_all)
|
||||||
EXPECT_EQ(bt_all->NumFrames(), bt_ign2->NumFrames() + 2)
|
<< "Ignore 1 backtrace:\n"
|
||||||
<< "All backtrace:\n" << DumpFrames(bt_all) << "Ignore 2 backtrace:\n" << DumpFrames(bt_ign2);
|
<< DumpFrames(bt_ign1);
|
||||||
|
ASSERT_EQ(bt_all->NumFrames(), bt_ign2->NumFrames() + 2) << "All backtrace:\n"
|
||||||
|
<< DumpFrames(bt_all)
|
||||||
|
<< "Ignore 2 backtrace:\n"
|
||||||
|
<< DumpFrames(bt_ign2);
|
||||||
|
|
||||||
// Check all of the frames are the same > the current frame.
|
// Check all of the frames are the same > the current frame.
|
||||||
bool check = (cur_proc == nullptr);
|
bool check = (cur_proc == nullptr);
|
||||||
|
|
@ -305,9 +317,8 @@ TEST(libbacktrace, local_max_trace) {
|
||||||
}
|
}
|
||||||
|
|
||||||
static void VerifyProcTest(pid_t pid, pid_t tid, bool (*ReadyFunc)(Backtrace*),
|
static void VerifyProcTest(pid_t pid, pid_t tid, bool (*ReadyFunc)(Backtrace*),
|
||||||
void (*VerifyFunc)(Backtrace*),
|
void (*VerifyFunc)(Backtrace*, create_func_t, map_create_func_t),
|
||||||
Backtrace* (*back_func)(pid_t, pid_t, BacktraceMap*),
|
create_func_t create_func, map_create_func_t map_create_func) {
|
||||||
BacktraceMap* (*map_func)(pid_t, bool)) {
|
|
||||||
pid_t ptrace_tid;
|
pid_t ptrace_tid;
|
||||||
if (tid < 0) {
|
if (tid < 0) {
|
||||||
ptrace_tid = pid;
|
ptrace_tid = pid;
|
||||||
|
|
@ -324,13 +335,13 @@ static void VerifyProcTest(pid_t pid, pid_t tid, bool (*ReadyFunc)(Backtrace*),
|
||||||
WaitForStop(ptrace_tid);
|
WaitForStop(ptrace_tid);
|
||||||
|
|
||||||
std::unique_ptr<BacktraceMap> map;
|
std::unique_ptr<BacktraceMap> map;
|
||||||
map.reset(map_func(pid, false));
|
map.reset(map_create_func(pid, false));
|
||||||
std::unique_ptr<Backtrace> backtrace(back_func(pid, tid, map.get()));
|
std::unique_ptr<Backtrace> backtrace(create_func(pid, tid, map.get()));
|
||||||
ASSERT_TRUE(backtrace.get() != nullptr);
|
ASSERT_TRUE(backtrace.get() != nullptr);
|
||||||
ASSERT_TRUE(backtrace->Unwind(0));
|
ASSERT_TRUE(backtrace->Unwind(0));
|
||||||
ASSERT_EQ(BACKTRACE_UNWIND_NO_ERROR, backtrace->GetError());
|
ASSERT_EQ(BACKTRACE_UNWIND_NO_ERROR, backtrace->GetError());
|
||||||
if (ReadyFunc(backtrace.get())) {
|
if (ReadyFunc(backtrace.get())) {
|
||||||
VerifyFunc(backtrace.get());
|
VerifyFunc(backtrace.get(), create_func, map_create_func);
|
||||||
verified = true;
|
verified = true;
|
||||||
} else {
|
} else {
|
||||||
last_dump = DumpFrames(backtrace.get());
|
last_dump = DumpFrames(backtrace.get());
|
||||||
|
|
@ -399,13 +410,15 @@ TEST(libbacktrace, ptrace_max_trace_new) {
|
||||||
ASSERT_EQ(waitpid(pid, &status, 0), pid);
|
ASSERT_EQ(waitpid(pid, &status, 0), pid);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void VerifyProcessIgnoreFrames(Backtrace* bt_all) {
|
static void VerifyProcessIgnoreFrames(Backtrace* bt_all, create_func_t create_func,
|
||||||
std::unique_ptr<Backtrace> ign1(Backtrace::Create(bt_all->Pid(), BACKTRACE_CURRENT_THREAD));
|
map_create_func_t map_create_func) {
|
||||||
|
std::unique_ptr<BacktraceMap> map(map_create_func(bt_all->Pid(), false));
|
||||||
|
std::unique_ptr<Backtrace> ign1(create_func(bt_all->Pid(), BACKTRACE_CURRENT_THREAD, map.get()));
|
||||||
ASSERT_TRUE(ign1.get() != nullptr);
|
ASSERT_TRUE(ign1.get() != nullptr);
|
||||||
ASSERT_TRUE(ign1->Unwind(1));
|
ASSERT_TRUE(ign1->Unwind(1));
|
||||||
ASSERT_EQ(BACKTRACE_UNWIND_NO_ERROR, ign1->GetError());
|
ASSERT_EQ(BACKTRACE_UNWIND_NO_ERROR, ign1->GetError());
|
||||||
|
|
||||||
std::unique_ptr<Backtrace> ign2(Backtrace::Create(bt_all->Pid(), BACKTRACE_CURRENT_THREAD));
|
std::unique_ptr<Backtrace> ign2(create_func(bt_all->Pid(), BACKTRACE_CURRENT_THREAD, map.get()));
|
||||||
ASSERT_TRUE(ign2.get() != nullptr);
|
ASSERT_TRUE(ign2.get() != nullptr);
|
||||||
ASSERT_TRUE(ign2->Unwind(2));
|
ASSERT_TRUE(ign2->Unwind(2));
|
||||||
ASSERT_EQ(BACKTRACE_UNWIND_NO_ERROR, ign2->GetError());
|
ASSERT_EQ(BACKTRACE_UNWIND_NO_ERROR, ign2->GetError());
|
||||||
|
|
@ -1702,9 +1715,8 @@ static void SetValueAndLoop(void* data) {
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void UnwindThroughSignal(bool use_action,
|
static void UnwindThroughSignal(bool use_action, create_func_t create_func,
|
||||||
Backtrace* (*back_func)(pid_t, pid_t, BacktraceMap*),
|
map_create_func_t map_create_func) {
|
||||||
BacktraceMap* (*map_func)(pid_t, bool)) {
|
|
||||||
volatile int value = 0;
|
volatile int value = 0;
|
||||||
pid_t pid;
|
pid_t pid;
|
||||||
if ((pid = fork()) == 0) {
|
if ((pid = fork()) == 0) {
|
||||||
|
|
@ -1730,8 +1742,8 @@ static void UnwindThroughSignal(bool use_action,
|
||||||
|
|
||||||
WaitForStop(pid);
|
WaitForStop(pid);
|
||||||
|
|
||||||
std::unique_ptr<BacktraceMap> map(map_func(pid, false));
|
std::unique_ptr<BacktraceMap> map(map_create_func(pid, false));
|
||||||
std::unique_ptr<Backtrace> backtrace(back_func(pid, pid, map.get()));
|
std::unique_ptr<Backtrace> backtrace(create_func(pid, pid, map.get()));
|
||||||
|
|
||||||
size_t bytes_read = backtrace->Read(reinterpret_cast<uintptr_t>(const_cast<int*>(&value)),
|
size_t bytes_read = backtrace->Read(reinterpret_cast<uintptr_t>(const_cast<int*>(&value)),
|
||||||
reinterpret_cast<uint8_t*>(&read_value), sizeof(read_value));
|
reinterpret_cast<uint8_t*>(&read_value), sizeof(read_value));
|
||||||
|
|
@ -1758,9 +1770,9 @@ static void UnwindThroughSignal(bool use_action,
|
||||||
|
|
||||||
WaitForStop(pid);
|
WaitForStop(pid);
|
||||||
|
|
||||||
map.reset(map_func(pid, false));
|
map.reset(map_create_func(pid, false));
|
||||||
ASSERT_TRUE(map.get() != nullptr);
|
ASSERT_TRUE(map.get() != nullptr);
|
||||||
backtrace.reset(back_func(pid, pid, map.get()));
|
backtrace.reset(create_func(pid, pid, map.get()));
|
||||||
ASSERT_TRUE(backtrace->Unwind(0));
|
ASSERT_TRUE(backtrace->Unwind(0));
|
||||||
bool found = false;
|
bool found = false;
|
||||||
for (frame_iter = backtrace->begin(); frame_iter != backtrace->end(); ++frame_iter) {
|
for (frame_iter = backtrace->begin(); frame_iter != backtrace->end(); ++frame_iter) {
|
||||||
|
|
|
||||||
|
|
@ -61,6 +61,7 @@ cc_library {
|
||||||
"Maps.cpp",
|
"Maps.cpp",
|
||||||
"Memory.cpp",
|
"Memory.cpp",
|
||||||
"Regs.cpp",
|
"Regs.cpp",
|
||||||
|
"Unwinder.cpp",
|
||||||
"Symbols.cpp",
|
"Symbols.cpp",
|
||||||
],
|
],
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -47,7 +47,7 @@ const DwarfFde* DwarfSection::GetFdeFromPc(uint64_t pc) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DwarfSection::Step(uint64_t pc, Regs* regs, Memory* process_memory) {
|
bool DwarfSection::Step(uint64_t pc, Regs* regs, Memory* process_memory, bool* finished) {
|
||||||
last_error_ = DWARF_ERROR_NONE;
|
last_error_ = DWARF_ERROR_NONE;
|
||||||
const DwarfFde* fde = GetFdeFromPc(pc);
|
const DwarfFde* fde = GetFdeFromPc(pc);
|
||||||
if (fde == nullptr || fde->cie == nullptr) {
|
if (fde == nullptr || fde->cie == nullptr) {
|
||||||
|
|
@ -62,7 +62,7 @@ bool DwarfSection::Step(uint64_t pc, Regs* regs, Memory* process_memory) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Now eval the actual registers.
|
// Now eval the actual registers.
|
||||||
return Eval(fde->cie, process_memory, loc_regs, regs);
|
return Eval(fde->cie, process_memory, loc_regs, regs, finished);
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename AddressType>
|
template <typename AddressType>
|
||||||
|
|
@ -92,7 +92,8 @@ bool DwarfSectionImpl<AddressType>::EvalExpression(const DwarfLocation& loc, uin
|
||||||
|
|
||||||
template <typename AddressType>
|
template <typename AddressType>
|
||||||
bool DwarfSectionImpl<AddressType>::Eval(const DwarfCie* cie, Memory* regular_memory,
|
bool DwarfSectionImpl<AddressType>::Eval(const DwarfCie* cie, Memory* regular_memory,
|
||||||
const dwarf_loc_regs_t& loc_regs, Regs* regs) {
|
const dwarf_loc_regs_t& loc_regs, Regs* regs,
|
||||||
|
bool* finished) {
|
||||||
RegsImpl<AddressType>* cur_regs = reinterpret_cast<RegsImpl<AddressType>*>(regs);
|
RegsImpl<AddressType>* cur_regs = reinterpret_cast<RegsImpl<AddressType>*>(regs);
|
||||||
if (cie->return_address_register >= cur_regs->total_regs()) {
|
if (cie->return_address_register >= cur_regs->total_regs()) {
|
||||||
last_error_ = DWARF_ERROR_ILLEGAL_VALUE;
|
last_error_ = DWARF_ERROR_ILLEGAL_VALUE;
|
||||||
|
|
@ -224,12 +225,14 @@ bool DwarfSectionImpl<AddressType>::Eval(const DwarfCie* cie, Memory* regular_me
|
||||||
// Find the return address location.
|
// Find the return address location.
|
||||||
if (return_address_undefined) {
|
if (return_address_undefined) {
|
||||||
cur_regs->set_pc(0);
|
cur_regs->set_pc(0);
|
||||||
|
*finished = true;
|
||||||
} else {
|
} else {
|
||||||
cur_regs->set_pc((*cur_regs)[cie->return_address_register]);
|
cur_regs->set_pc((*cur_regs)[cie->return_address_register]);
|
||||||
|
*finished = false;
|
||||||
}
|
}
|
||||||
cur_regs->set_sp(cfa);
|
cur_regs->set_sp(cfa);
|
||||||
// Stop if the cfa and pc are the same.
|
// Return false if the unwind is not finished or the cfa and pc didn't change.
|
||||||
return prev_cfa != cfa || prev_pc != cur_regs->pc();
|
return *finished || prev_cfa != cfa || prev_pc != cur_regs->pc();
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename AddressType>
|
template <typename AddressType>
|
||||||
|
|
|
||||||
|
|
@ -95,11 +95,17 @@ bool Elf::GetFunctionName(uint64_t addr, std::string* name, uint64_t* func_offse
|
||||||
gnu_debugdata_interface_->GetFunctionName(addr, name, func_offset)));
|
gnu_debugdata_interface_->GetFunctionName(addr, name, func_offset)));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Elf::Step(uint64_t rel_pc, Regs* regs, Memory* process_memory) {
|
bool Elf::Step(uint64_t rel_pc, Regs* regs, Memory* process_memory, bool* finished) {
|
||||||
return valid_ && (regs->StepIfSignalHandler(rel_pc, this, process_memory) ||
|
if (!valid_) {
|
||||||
interface_->Step(rel_pc, regs, process_memory) ||
|
return false;
|
||||||
(gnu_debugdata_interface_ &&
|
}
|
||||||
gnu_debugdata_interface_->Step(rel_pc, regs, process_memory)));
|
if (regs->StepIfSignalHandler(rel_pc, this, process_memory)) {
|
||||||
|
*finished = false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return interface_->Step(rel_pc, regs, process_memory, finished) ||
|
||||||
|
(gnu_debugdata_interface_ &&
|
||||||
|
gnu_debugdata_interface_->Step(rel_pc, regs, process_memory, finished));
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64_t Elf::GetLoadBias() {
|
uint64_t Elf::GetLoadBias() {
|
||||||
|
|
|
||||||
|
|
@ -348,7 +348,7 @@ bool ElfInterface::GetFunctionNameWithTemplate(uint64_t addr, std::string* name,
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ElfInterface::Step(uint64_t pc, Regs* regs, Memory* process_memory) {
|
bool ElfInterface::Step(uint64_t pc, Regs* regs, Memory* process_memory, bool* finished) {
|
||||||
// Need to subtract off the load_bias to get the correct pc.
|
// Need to subtract off the load_bias to get the correct pc.
|
||||||
if (pc < load_bias_) {
|
if (pc < load_bias_) {
|
||||||
return false;
|
return false;
|
||||||
|
|
@ -357,16 +357,15 @@ bool ElfInterface::Step(uint64_t pc, Regs* regs, Memory* process_memory) {
|
||||||
|
|
||||||
// Try the eh_frame first.
|
// Try the eh_frame first.
|
||||||
DwarfSection* eh_frame = eh_frame_.get();
|
DwarfSection* eh_frame = eh_frame_.get();
|
||||||
if (eh_frame != nullptr && eh_frame->Step(pc, regs, process_memory)) {
|
if (eh_frame != nullptr && eh_frame->Step(pc, regs, process_memory, finished)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Try the debug_frame next.
|
// Try the debug_frame next.
|
||||||
DwarfSection* debug_frame = debug_frame_.get();
|
DwarfSection* debug_frame = debug_frame_.get();
|
||||||
if (debug_frame != nullptr && debug_frame->Step(pc, regs, process_memory)) {
|
if (debug_frame != nullptr && debug_frame->Step(pc, regs, process_memory, finished)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -99,22 +99,25 @@ bool ElfInterfaceArm::HandleType(uint64_t offset, uint32_t type) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ElfInterfaceArm::Step(uint64_t pc, Regs* regs, Memory* process_memory) {
|
bool ElfInterfaceArm::Step(uint64_t pc, Regs* regs, Memory* process_memory, bool* finished) {
|
||||||
// Dwarf unwind information is precise about whether a pc is covered or not,
|
// Dwarf unwind information is precise about whether a pc is covered or not,
|
||||||
// but arm unwind information only has ranges of pc. In order to avoid
|
// but arm unwind information only has ranges of pc. In order to avoid
|
||||||
// incorrectly doing a bad unwind using arm unwind information for a
|
// incorrectly doing a bad unwind using arm unwind information for a
|
||||||
// different function, always try and unwind with the dwarf information first.
|
// different function, always try and unwind with the dwarf information first.
|
||||||
return ElfInterface32::Step(pc, regs, process_memory) || StepExidx(pc, regs, process_memory);
|
return ElfInterface32::Step(pc, regs, process_memory, finished) ||
|
||||||
|
StepExidx(pc, regs, process_memory, finished);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool ElfInterfaceArm::StepExidx(uint64_t pc, Regs* regs, Memory* process_memory) {
|
bool ElfInterfaceArm::StepExidx(uint64_t pc, Regs* regs, Memory* process_memory, bool* finished) {
|
||||||
RegsArm* regs_arm = reinterpret_cast<RegsArm*>(regs);
|
RegsArm* regs_arm = reinterpret_cast<RegsArm*>(regs);
|
||||||
uint64_t entry_offset;
|
uint64_t entry_offset;
|
||||||
if (!FindEntry(pc, &entry_offset)) {
|
if (!FindEntry(pc, &entry_offset)) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
ArmExidx arm(regs_arm, memory_, process_memory);
|
ArmExidx arm(regs_arm, memory_, process_memory);
|
||||||
arm.set_cfa(regs_arm->sp());
|
arm.set_cfa(regs_arm->sp());
|
||||||
|
bool return_value = false;
|
||||||
if (arm.ExtractEntryData(entry_offset) && arm.Eval()) {
|
if (arm.ExtractEntryData(entry_offset) && arm.Eval()) {
|
||||||
// If the pc was not set, then use the LR registers for the PC.
|
// If the pc was not set, then use the LR registers for the PC.
|
||||||
if (!arm.pc_set()) {
|
if (!arm.pc_set()) {
|
||||||
|
|
@ -125,9 +128,15 @@ bool ElfInterfaceArm::StepExidx(uint64_t pc, Regs* regs, Memory* process_memory)
|
||||||
}
|
}
|
||||||
regs_arm->set_sp(arm.cfa());
|
regs_arm->set_sp(arm.cfa());
|
||||||
(*regs_arm)[ARM_REG_SP] = regs_arm->sp();
|
(*regs_arm)[ARM_REG_SP] = regs_arm->sp();
|
||||||
|
*finished = false;
|
||||||
|
return_value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (arm.status() == ARM_STATUS_NO_UNWIND) {
|
||||||
|
*finished = true;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return return_value;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace unwindstack
|
} // namespace unwindstack
|
||||||
|
|
|
||||||
|
|
@ -70,9 +70,9 @@ class ElfInterfaceArm : public ElfInterface32 {
|
||||||
|
|
||||||
bool HandleType(uint64_t offset, uint32_t type) override;
|
bool HandleType(uint64_t offset, uint32_t type) override;
|
||||||
|
|
||||||
bool Step(uint64_t pc, Regs* regs, Memory* process_memory) override;
|
bool Step(uint64_t pc, Regs* regs, Memory* process_memory, bool* finished) override;
|
||||||
|
|
||||||
bool StepExidx(uint64_t pc, Regs* regs, Memory* process_memory);
|
bool StepExidx(uint64_t pc, Regs* regs, Memory* process_memory, bool* finished);
|
||||||
|
|
||||||
uint64_t start_offset() { return start_offset_; }
|
uint64_t start_offset() { return start_offset_; }
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -91,6 +91,15 @@ void RegsArm::SetFromRaw() {
|
||||||
set_sp(regs_[ARM_REG_SP]);
|
set_sp(regs_[ARM_REG_SP]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool RegsArm::SetPcFromReturnAddress(Memory*) {
|
||||||
|
if (pc() == regs_[ARM_REG_LR]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
set_pc(regs_[ARM_REG_LR]);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
RegsArm64::RegsArm64()
|
RegsArm64::RegsArm64()
|
||||||
: RegsImpl<uint64_t>(ARM64_REG_LAST, ARM64_REG_SP, Location(LOCATION_REGISTER, ARM64_REG_LR)) {}
|
: RegsImpl<uint64_t>(ARM64_REG_LAST, ARM64_REG_SP, Location(LOCATION_REGISTER, ARM64_REG_LR)) {}
|
||||||
|
|
||||||
|
|
@ -114,6 +123,15 @@ void RegsArm64::SetFromRaw() {
|
||||||
set_sp(regs_[ARM64_REG_SP]);
|
set_sp(regs_[ARM64_REG_SP]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool RegsArm64::SetPcFromReturnAddress(Memory*) {
|
||||||
|
if (pc() == regs_[ARM64_REG_LR]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
set_pc(regs_[ARM64_REG_LR]);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
RegsX86::RegsX86()
|
RegsX86::RegsX86()
|
||||||
: RegsImpl<uint32_t>(X86_REG_LAST, X86_REG_SP, Location(LOCATION_SP_OFFSET, -4)) {}
|
: RegsImpl<uint32_t>(X86_REG_LAST, X86_REG_SP, Location(LOCATION_SP_OFFSET, -4)) {}
|
||||||
|
|
||||||
|
|
@ -137,6 +155,17 @@ void RegsX86::SetFromRaw() {
|
||||||
set_sp(regs_[X86_REG_SP]);
|
set_sp(regs_[X86_REG_SP]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool RegsX86::SetPcFromReturnAddress(Memory* process_memory) {
|
||||||
|
// Attempt to get the return address from the top of the stack.
|
||||||
|
uint32_t new_pc;
|
||||||
|
if (!process_memory->Read(sp_, &new_pc, sizeof(new_pc)) || new_pc == pc()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
set_pc(new_pc);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
RegsX86_64::RegsX86_64()
|
RegsX86_64::RegsX86_64()
|
||||||
: RegsImpl<uint64_t>(X86_64_REG_LAST, X86_64_REG_SP, Location(LOCATION_SP_OFFSET, -8)) {}
|
: RegsImpl<uint64_t>(X86_64_REG_LAST, X86_64_REG_SP, Location(LOCATION_SP_OFFSET, -8)) {}
|
||||||
|
|
||||||
|
|
@ -161,6 +190,17 @@ void RegsX86_64::SetFromRaw() {
|
||||||
set_sp(regs_[X86_64_REG_SP]);
|
set_sp(regs_[X86_64_REG_SP]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool RegsX86_64::SetPcFromReturnAddress(Memory* process_memory) {
|
||||||
|
// Attempt to get the return address from the top of the stack.
|
||||||
|
uint64_t new_pc;
|
||||||
|
if (!process_memory->Read(sp_, &new_pc, sizeof(new_pc)) || new_pc == pc()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
set_pc(new_pc);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static Regs* ReadArm(void* remote_data) {
|
static Regs* ReadArm(void* remote_data) {
|
||||||
arm_user_regs* user = reinterpret_cast<arm_user_regs*>(remote_data);
|
arm_user_regs* user = reinterpret_cast<arm_user_regs*>(remote_data);
|
||||||
|
|
||||||
|
|
|
||||||
145
libunwindstack/Unwinder.cpp
Normal file
145
libunwindstack/Unwinder.cpp
Normal file
|
|
@ -0,0 +1,145 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2017 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <elf.h>
|
||||||
|
#include <inttypes.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <android-base/stringprintf.h>
|
||||||
|
|
||||||
|
#include <unwindstack/Elf.h>
|
||||||
|
#include <unwindstack/MapInfo.h>
|
||||||
|
#include <unwindstack/Unwinder.h>
|
||||||
|
|
||||||
|
namespace unwindstack {
|
||||||
|
|
||||||
|
void Unwinder::FillInFrame(MapInfo* map_info, uint64_t* rel_pc) {
|
||||||
|
size_t frame_num = frames_.size();
|
||||||
|
frames_.resize(frame_num + 1);
|
||||||
|
FrameData* frame = &frames_.at(frame_num);
|
||||||
|
frame->num = frame_num;
|
||||||
|
frame->pc = regs_->pc();
|
||||||
|
frame->sp = regs_->sp();
|
||||||
|
frame->rel_pc = frame->pc;
|
||||||
|
|
||||||
|
if (map_info == nullptr) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Elf* elf = map_info->GetElf(process_memory_, true);
|
||||||
|
*rel_pc = elf->GetRelPc(regs_->pc(), map_info);
|
||||||
|
if (frame_num != 0) {
|
||||||
|
// Don't adjust the first frame pc.
|
||||||
|
frame->rel_pc = regs_->GetAdjustedPc(*rel_pc, elf);
|
||||||
|
|
||||||
|
// Adjust the original pc.
|
||||||
|
frame->pc -= *rel_pc - frame->rel_pc;
|
||||||
|
} else {
|
||||||
|
frame->rel_pc = *rel_pc;
|
||||||
|
}
|
||||||
|
|
||||||
|
frame->map_name = map_info->name;
|
||||||
|
frame->map_offset = map_info->elf_offset;
|
||||||
|
frame->map_start = map_info->start;
|
||||||
|
frame->map_end = map_info->end;
|
||||||
|
|
||||||
|
if (!elf->GetFunctionName(frame->rel_pc, &frame->function_name, &frame->function_offset)) {
|
||||||
|
frame->function_name = "";
|
||||||
|
frame->function_offset = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Unwinder::Unwind() {
|
||||||
|
frames_.clear();
|
||||||
|
|
||||||
|
bool return_address_attempt = false;
|
||||||
|
for (; frames_.size() < max_frames_;) {
|
||||||
|
MapInfo* map_info = maps_->Find(regs_->pc());
|
||||||
|
|
||||||
|
uint64_t rel_pc;
|
||||||
|
FillInFrame(map_info, &rel_pc);
|
||||||
|
|
||||||
|
bool stepped;
|
||||||
|
if (map_info == nullptr) {
|
||||||
|
stepped = false;
|
||||||
|
} else {
|
||||||
|
bool finished;
|
||||||
|
stepped = map_info->elf->Step(rel_pc + map_info->elf_offset, regs_, process_memory_.get(),
|
||||||
|
&finished);
|
||||||
|
if (stepped && finished) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!stepped) {
|
||||||
|
if (return_address_attempt) {
|
||||||
|
// Remove the speculative frame.
|
||||||
|
frames_.pop_back();
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
// Steping didn't work, try this secondary method.
|
||||||
|
if (!regs_->SetPcFromReturnAddress(process_memory_.get())) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return_address_attempt = true;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return_address_attempt = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string Unwinder::FormatFrame(size_t frame_num) {
|
||||||
|
if (frame_num >= frames_.size()) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
return FormatFrame(frames_[frame_num],
|
||||||
|
regs_->MachineType() == EM_ARM || regs_->MachineType() == EM_386);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string Unwinder::FormatFrame(const FrameData& frame, bool bits32) {
|
||||||
|
std::string data;
|
||||||
|
|
||||||
|
if (bits32) {
|
||||||
|
data += android::base::StringPrintf(" #%02zu pc %08" PRIx64, frame.num, frame.rel_pc);
|
||||||
|
} else {
|
||||||
|
data += android::base::StringPrintf(" #%02zu pc %016" PRIx64, frame.num, frame.rel_pc);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (frame.map_offset != 0) {
|
||||||
|
data += android::base::StringPrintf(" (offset 0x%" PRIx64 ")", frame.map_offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (frame.map_start == frame.map_end) {
|
||||||
|
// No valid map associated with this frame.
|
||||||
|
data += " <unknown>";
|
||||||
|
} else if (!frame.map_name.empty()) {
|
||||||
|
data += " " + frame.map_name;
|
||||||
|
} else {
|
||||||
|
data += android::base::StringPrintf(" <anonymous:%" PRIx64 ">", frame.map_start);
|
||||||
|
}
|
||||||
|
if (!frame.function_name.empty()) {
|
||||||
|
data += " (" + frame.function_name;
|
||||||
|
if (frame.function_offset != 0) {
|
||||||
|
data += android::base::StringPrintf("+%" PRId64, frame.function_offset);
|
||||||
|
}
|
||||||
|
data += ')';
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace unwindstack
|
||||||
|
|
@ -76,7 +76,7 @@ class DwarfSection {
|
||||||
|
|
||||||
virtual bool Init(uint64_t offset, uint64_t size) = 0;
|
virtual bool Init(uint64_t offset, uint64_t size) = 0;
|
||||||
|
|
||||||
virtual bool Eval(const DwarfCie*, Memory*, const dwarf_loc_regs_t&, Regs*) = 0;
|
virtual bool Eval(const DwarfCie*, Memory*, const dwarf_loc_regs_t&, Regs*, bool*) = 0;
|
||||||
|
|
||||||
virtual bool GetFdeOffsetFromPc(uint64_t pc, uint64_t* fde_offset) = 0;
|
virtual bool GetFdeOffsetFromPc(uint64_t pc, uint64_t* fde_offset) = 0;
|
||||||
|
|
||||||
|
|
@ -100,7 +100,7 @@ class DwarfSection {
|
||||||
|
|
||||||
virtual uint64_t AdjustPcFromFde(uint64_t pc) = 0;
|
virtual uint64_t AdjustPcFromFde(uint64_t pc) = 0;
|
||||||
|
|
||||||
bool Step(uint64_t pc, Regs* regs, Memory* process_memory);
|
bool Step(uint64_t pc, Regs* regs, Memory* process_memory, bool* finished);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
DwarfMemory memory_;
|
DwarfMemory memory_;
|
||||||
|
|
@ -119,7 +119,7 @@ class DwarfSectionImpl : public DwarfSection {
|
||||||
virtual ~DwarfSectionImpl() = default;
|
virtual ~DwarfSectionImpl() = default;
|
||||||
|
|
||||||
bool Eval(const DwarfCie* cie, Memory* regular_memory, const dwarf_loc_regs_t& loc_regs,
|
bool Eval(const DwarfCie* cie, Memory* regular_memory, const dwarf_loc_regs_t& loc_regs,
|
||||||
Regs* regs) override;
|
Regs* regs, bool* finished) override;
|
||||||
|
|
||||||
const DwarfCie* GetCie(uint64_t offset);
|
const DwarfCie* GetCie(uint64_t offset);
|
||||||
bool FillInCie(DwarfCie* cie);
|
bool FillInCie(DwarfCie* cie);
|
||||||
|
|
|
||||||
|
|
@ -50,7 +50,7 @@ class Elf {
|
||||||
|
|
||||||
uint64_t GetRelPc(uint64_t pc, const MapInfo* map_info);
|
uint64_t GetRelPc(uint64_t pc, const MapInfo* map_info);
|
||||||
|
|
||||||
bool Step(uint64_t rel_pc, Regs* regs, Memory* process_memory);
|
bool Step(uint64_t rel_pc, Regs* regs, Memory* process_memory, bool* finished);
|
||||||
|
|
||||||
ElfInterface* CreateInterfaceFromMemory(Memory* memory);
|
ElfInterface* CreateInterfaceFromMemory(Memory* memory);
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -59,7 +59,7 @@ class ElfInterface {
|
||||||
|
|
||||||
virtual bool GetFunctionName(uint64_t addr, std::string* name, uint64_t* offset) = 0;
|
virtual bool GetFunctionName(uint64_t addr, std::string* name, uint64_t* offset) = 0;
|
||||||
|
|
||||||
virtual bool Step(uint64_t rel_pc, Regs* regs, Memory* process_memory);
|
virtual bool Step(uint64_t rel_pc, Regs* regs, Memory* process_memory, bool* finished);
|
||||||
|
|
||||||
Memory* CreateGnuDebugdataMemory();
|
Memory* CreateGnuDebugdataMemory();
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -63,6 +63,8 @@ class Regs {
|
||||||
|
|
||||||
virtual void SetFromRaw() = 0;
|
virtual void SetFromRaw() = 0;
|
||||||
|
|
||||||
|
virtual bool SetPcFromReturnAddress(Memory* process_memory) = 0;
|
||||||
|
|
||||||
uint16_t sp_reg() { return sp_reg_; }
|
uint16_t sp_reg() { return sp_reg_; }
|
||||||
uint16_t total_regs() { return total_regs_; }
|
uint16_t total_regs() { return total_regs_; }
|
||||||
|
|
||||||
|
|
@ -113,6 +115,8 @@ class RegsArm : public RegsImpl<uint32_t> {
|
||||||
|
|
||||||
void SetFromRaw() override;
|
void SetFromRaw() override;
|
||||||
|
|
||||||
|
bool SetPcFromReturnAddress(Memory* process_memory) override;
|
||||||
|
|
||||||
bool StepIfSignalHandler(uint64_t rel_pc, Elf* elf, Memory* process_memory) override;
|
bool StepIfSignalHandler(uint64_t rel_pc, Elf* elf, Memory* process_memory) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -127,6 +131,8 @@ class RegsArm64 : public RegsImpl<uint64_t> {
|
||||||
|
|
||||||
void SetFromRaw() override;
|
void SetFromRaw() override;
|
||||||
|
|
||||||
|
bool SetPcFromReturnAddress(Memory* process_memory) override;
|
||||||
|
|
||||||
bool StepIfSignalHandler(uint64_t rel_pc, Elf* elf, Memory* process_memory) override;
|
bool StepIfSignalHandler(uint64_t rel_pc, Elf* elf, Memory* process_memory) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -141,6 +147,8 @@ class RegsX86 : public RegsImpl<uint32_t> {
|
||||||
|
|
||||||
void SetFromRaw() override;
|
void SetFromRaw() override;
|
||||||
|
|
||||||
|
bool SetPcFromReturnAddress(Memory* process_memory) override;
|
||||||
|
|
||||||
bool StepIfSignalHandler(uint64_t rel_pc, Elf* elf, Memory* process_memory) override;
|
bool StepIfSignalHandler(uint64_t rel_pc, Elf* elf, Memory* process_memory) override;
|
||||||
|
|
||||||
void SetFromUcontext(x86_ucontext_t* ucontext);
|
void SetFromUcontext(x86_ucontext_t* ucontext);
|
||||||
|
|
@ -157,6 +165,8 @@ class RegsX86_64 : public RegsImpl<uint64_t> {
|
||||||
|
|
||||||
void SetFromRaw() override;
|
void SetFromRaw() override;
|
||||||
|
|
||||||
|
bool SetPcFromReturnAddress(Memory* process_memory) override;
|
||||||
|
|
||||||
bool StepIfSignalHandler(uint64_t rel_pc, Elf* elf, Memory* process_memory) override;
|
bool StepIfSignalHandler(uint64_t rel_pc, Elf* elf, Memory* process_memory) override;
|
||||||
|
|
||||||
void SetFromUcontext(x86_64_ucontext_t* ucontext);
|
void SetFromUcontext(x86_64_ucontext_t* ucontext);
|
||||||
|
|
|
||||||
78
libunwindstack/include/unwindstack/Unwinder.h
Normal file
78
libunwindstack/include/unwindstack/Unwinder.h
Normal file
|
|
@ -0,0 +1,78 @@
|
||||||
|
/*
|
||||||
|
* Copyright (C) 2017 The Android Open Source Project
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
* you may not use this file except in compliance with the License.
|
||||||
|
* You may obtain a copy of the License at
|
||||||
|
*
|
||||||
|
* http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* Unless required by applicable law or agreed to in writing, software
|
||||||
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
* See the License for the specific language governing permissions and
|
||||||
|
* limitations under the License.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _LIBUNWINDSTACK_UNWINDER_H
|
||||||
|
#define _LIBUNWINDSTACK_UNWINDER_H
|
||||||
|
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <sys/types.h>
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include <unwindstack/Maps.h>
|
||||||
|
#include <unwindstack/Memory.h>
|
||||||
|
#include <unwindstack/Regs.h>
|
||||||
|
|
||||||
|
namespace unwindstack {
|
||||||
|
|
||||||
|
struct FrameData {
|
||||||
|
size_t num;
|
||||||
|
|
||||||
|
uint64_t rel_pc;
|
||||||
|
uint64_t pc;
|
||||||
|
uint64_t sp;
|
||||||
|
|
||||||
|
std::string function_name;
|
||||||
|
uint64_t function_offset;
|
||||||
|
|
||||||
|
std::string map_name;
|
||||||
|
uint64_t map_offset;
|
||||||
|
uint64_t map_start;
|
||||||
|
uint64_t map_end;
|
||||||
|
};
|
||||||
|
|
||||||
|
class Unwinder {
|
||||||
|
public:
|
||||||
|
Unwinder(size_t max_frames, Maps* maps, Regs* regs, std::shared_ptr<Memory> process_memory)
|
||||||
|
: max_frames_(max_frames), maps_(maps), regs_(regs), process_memory_(process_memory) {
|
||||||
|
frames_.reserve(max_frames);
|
||||||
|
}
|
||||||
|
~Unwinder() = default;
|
||||||
|
|
||||||
|
void Unwind();
|
||||||
|
|
||||||
|
size_t NumFrames() { return frames_.size(); }
|
||||||
|
|
||||||
|
const std::vector<FrameData>& frames() { return frames_; }
|
||||||
|
|
||||||
|
std::string FormatFrame(size_t frame_num);
|
||||||
|
static std::string FormatFrame(const FrameData& frame, bool bits32);
|
||||||
|
|
||||||
|
private:
|
||||||
|
void FillInFrame(MapInfo* map_info, uint64_t* rel_pc);
|
||||||
|
|
||||||
|
size_t max_frames_;
|
||||||
|
Maps* maps_;
|
||||||
|
Regs* regs_;
|
||||||
|
std::vector<FrameData> frames_;
|
||||||
|
std::shared_ptr<Memory> process_memory_;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace unwindstack
|
||||||
|
|
||||||
|
#endif // _LIBUNWINDSTACK_UNWINDER_H
|
||||||
|
|
@ -98,7 +98,8 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_cfa_expr_eval_fail) {
|
||||||
regs[5] = 0x20;
|
regs[5] = 0x20;
|
||||||
regs[9] = 0x3000;
|
regs[9] = 0x3000;
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x2, 0x5000}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x2, 0x5000}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_MEMORY_INVALID, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_MEMORY_INVALID, this->section_->last_error());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -113,7 +114,8 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_cfa_expr_no_stack) {
|
||||||
regs[9] = 0x3000;
|
regs[9] = 0x3000;
|
||||||
this->memory_.SetMemory(0x5000, std::vector<uint8_t>{0x96, 0x96, 0x96});
|
this->memory_.SetMemory(0x5000, std::vector<uint8_t>{0x96, 0x96, 0x96});
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x2, 0x5000}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x2, 0x5000}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_ILLEGAL_STATE, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_ILLEGAL_STATE, this->section_->last_error());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -130,7 +132,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_cfa_expr) {
|
||||||
TypeParam cfa_value = 0x12345;
|
TypeParam cfa_value = 0x12345;
|
||||||
this->memory_.SetMemory(0x80000000, &cfa_value, sizeof(cfa_value));
|
this->memory_.SetMemory(0x80000000, &cfa_value, sizeof(cfa_value));
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x4, 0x5000}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x4, 0x5000}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
EXPECT_FALSE(finished);
|
||||||
EXPECT_EQ(0x12345U, regs.sp());
|
EXPECT_EQ(0x12345U, regs.sp());
|
||||||
EXPECT_EQ(0x20U, regs.pc());
|
EXPECT_EQ(0x20U, regs.pc());
|
||||||
}
|
}
|
||||||
|
|
@ -146,7 +150,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_cfa_val_expr) {
|
||||||
regs[9] = 0x3000;
|
regs[9] = 0x3000;
|
||||||
this->memory_.SetMemory(0x5000, std::vector<uint8_t>{0x0c, 0x00, 0x00, 0x00, 0x80});
|
this->memory_.SetMemory(0x5000, std::vector<uint8_t>{0x0c, 0x00, 0x00, 0x00, 0x80});
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_VAL_EXPRESSION, {0x4, 0x5000}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_VAL_EXPRESSION, {0x4, 0x5000}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
ASSERT_FALSE(finished);
|
||||||
EXPECT_EQ(0x80000000U, regs.sp());
|
EXPECT_EQ(0x80000000U, regs.sp());
|
||||||
EXPECT_EQ(0x20U, regs.pc());
|
EXPECT_EQ(0x20U, regs.pc());
|
||||||
}
|
}
|
||||||
|
|
@ -162,7 +168,8 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_cfa_expr_is_register) {
|
||||||
regs[9] = 0x3000;
|
regs[9] = 0x3000;
|
||||||
this->memory_.SetMemory(0x5000, std::vector<uint8_t>{0x50, 0x96, 0x96});
|
this->memory_.SetMemory(0x5000, std::vector<uint8_t>{0x50, 0x96, 0x96});
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x2, 0x5000}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x2, 0x5000}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_NOT_IMPLEMENTED, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_NOT_IMPLEMENTED, this->section_->last_error());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -171,7 +178,8 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_bad_regs) {
|
||||||
RegsFake<TypeParam> regs(10, 9);
|
RegsFake<TypeParam> regs(10, 9);
|
||||||
dwarf_loc_regs_t loc_regs;
|
dwarf_loc_regs_t loc_regs;
|
||||||
|
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -180,7 +188,8 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_no_cfa) {
|
||||||
RegsFake<TypeParam> regs(10, 9);
|
RegsFake<TypeParam> regs(10, 9);
|
||||||
dwarf_loc_regs_t loc_regs;
|
dwarf_loc_regs_t loc_regs;
|
||||||
|
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_CFA_NOT_DEFINED, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_CFA_NOT_DEFINED, this->section_->last_error());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -190,25 +199,26 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_cfa_bad) {
|
||||||
dwarf_loc_regs_t loc_regs;
|
dwarf_loc_regs_t loc_regs;
|
||||||
|
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {20, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {20, 0}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
||||||
|
|
||||||
this->section_->TestClearError();
|
this->section_->TestClearError();
|
||||||
loc_regs.erase(CFA_REG);
|
loc_regs.erase(CFA_REG);
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_INVALID, {0, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_INVALID, {0, 0}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
||||||
|
|
||||||
this->section_->TestClearError();
|
this->section_->TestClearError();
|
||||||
loc_regs.erase(CFA_REG);
|
loc_regs.erase(CFA_REG);
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_OFFSET, {0, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_OFFSET, {0, 0}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
||||||
|
|
||||||
this->section_->TestClearError();
|
this->section_->TestClearError();
|
||||||
loc_regs.erase(CFA_REG);
|
loc_regs.erase(CFA_REG);
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_VAL_OFFSET, {0, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_VAL_OFFSET, {0, 0}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -222,7 +232,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_cfa_register_prev) {
|
||||||
regs[5] = 0x20;
|
regs[5] = 0x20;
|
||||||
regs[9] = 0x3000;
|
regs[9] = 0x3000;
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {9, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {9, 0}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
EXPECT_FALSE(finished);
|
||||||
EXPECT_EQ(0x20U, regs.pc());
|
EXPECT_EQ(0x20U, regs.pc());
|
||||||
EXPECT_EQ(0x2000U, regs.sp());
|
EXPECT_EQ(0x2000U, regs.sp());
|
||||||
}
|
}
|
||||||
|
|
@ -238,7 +250,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_cfa_register_from_value) {
|
||||||
regs[6] = 0x4000;
|
regs[6] = 0x4000;
|
||||||
regs[9] = 0x3000;
|
regs[9] = 0x3000;
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {6, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {6, 0}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
EXPECT_FALSE(finished);
|
||||||
EXPECT_EQ(0x20U, regs.pc());
|
EXPECT_EQ(0x20U, regs.pc());
|
||||||
EXPECT_EQ(0x4000U, regs.sp());
|
EXPECT_EQ(0x4000U, regs.sp());
|
||||||
}
|
}
|
||||||
|
|
@ -254,7 +268,8 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_double_indirection) {
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
||||||
loc_regs[1] = DwarfLocation{DWARF_LOCATION_REGISTER, {3, 0}};
|
loc_regs[1] = DwarfLocation{DWARF_LOCATION_REGISTER, {3, 0}};
|
||||||
loc_regs[9] = DwarfLocation{DWARF_LOCATION_REGISTER, {1, 0}};
|
loc_regs[9] = DwarfLocation{DWARF_LOCATION_REGISTER, {1, 0}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_ILLEGAL_STATE, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_ILLEGAL_STATE, this->section_->last_error());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -268,7 +283,8 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_invalid_register) {
|
||||||
regs[8] = 0x10;
|
regs[8] = 0x10;
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
||||||
loc_regs[1] = DwarfLocation{DWARF_LOCATION_REGISTER, {10, 0}};
|
loc_regs[1] = DwarfLocation{DWARF_LOCATION_REGISTER, {10, 0}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
EXPECT_EQ(DWARF_ERROR_ILLEGAL_VALUE, this->section_->last_error());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -292,7 +308,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_different_reg_locations) {
|
||||||
loc_regs[1] = DwarfLocation{DWARF_LOCATION_VAL_OFFSET, {0x100, 0}};
|
loc_regs[1] = DwarfLocation{DWARF_LOCATION_VAL_OFFSET, {0x100, 0}};
|
||||||
loc_regs[2] = DwarfLocation{DWARF_LOCATION_OFFSET, {0x50, 0}};
|
loc_regs[2] = DwarfLocation{DWARF_LOCATION_OFFSET, {0x50, 0}};
|
||||||
loc_regs[3] = DwarfLocation{DWARF_LOCATION_UNDEFINED, {0, 0}};
|
loc_regs[3] = DwarfLocation{DWARF_LOCATION_UNDEFINED, {0, 0}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
EXPECT_FALSE(finished);
|
||||||
EXPECT_EQ(0x10U, regs.pc());
|
EXPECT_EQ(0x10U, regs.pc());
|
||||||
EXPECT_EQ(0x2100U, regs.sp());
|
EXPECT_EQ(0x2100U, regs.sp());
|
||||||
EXPECT_EQ(0x2200U, regs[1]);
|
EXPECT_EQ(0x2200U, regs[1]);
|
||||||
|
|
@ -315,7 +333,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_return_address_undefined) {
|
||||||
regs[8] = 0x10;
|
regs[8] = 0x10;
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
||||||
loc_regs[5] = DwarfLocation{DWARF_LOCATION_UNDEFINED, {0, 0}};
|
loc_regs[5] = DwarfLocation{DWARF_LOCATION_UNDEFINED, {0, 0}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
EXPECT_TRUE(finished);
|
||||||
EXPECT_EQ(0U, regs.pc());
|
EXPECT_EQ(0U, regs.pc());
|
||||||
EXPECT_EQ(0x10U, regs.sp());
|
EXPECT_EQ(0x10U, regs.sp());
|
||||||
}
|
}
|
||||||
|
|
@ -330,7 +350,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_return_address) {
|
||||||
regs[5] = 0x20;
|
regs[5] = 0x20;
|
||||||
regs[8] = 0x10;
|
regs[8] = 0x10;
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
EXPECT_FALSE(finished);
|
||||||
EXPECT_EQ(0x20U, regs.pc());
|
EXPECT_EQ(0x20U, regs.pc());
|
||||||
EXPECT_EQ(0x10U, regs.sp());
|
EXPECT_EQ(0x10U, regs.sp());
|
||||||
}
|
}
|
||||||
|
|
@ -347,7 +369,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_ignore_large_reg_loc) {
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
||||||
// This should not result in any errors.
|
// This should not result in any errors.
|
||||||
loc_regs[20] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
loc_regs[20] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
EXPECT_FALSE(finished);
|
||||||
EXPECT_EQ(0x20U, regs.pc());
|
EXPECT_EQ(0x20U, regs.pc());
|
||||||
EXPECT_EQ(0x10U, regs.sp());
|
EXPECT_EQ(0x10U, regs.sp());
|
||||||
}
|
}
|
||||||
|
|
@ -365,7 +389,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_reg_expr) {
|
||||||
this->memory_.SetMemory(0x80000000, &cfa_value, sizeof(cfa_value));
|
this->memory_.SetMemory(0x80000000, &cfa_value, sizeof(cfa_value));
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
||||||
loc_regs[5] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x4, 0x5000}};
|
loc_regs[5] = DwarfLocation{DWARF_LOCATION_EXPRESSION, {0x4, 0x5000}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
EXPECT_FALSE(finished);
|
||||||
EXPECT_EQ(0x3000U, regs.sp());
|
EXPECT_EQ(0x3000U, regs.sp());
|
||||||
EXPECT_EQ(0x12345U, regs.pc());
|
EXPECT_EQ(0x12345U, regs.pc());
|
||||||
}
|
}
|
||||||
|
|
@ -381,7 +407,9 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_reg_val_expr) {
|
||||||
this->memory_.SetMemory(0x5000, std::vector<uint8_t>{0x0c, 0x00, 0x00, 0x00, 0x80});
|
this->memory_.SetMemory(0x5000, std::vector<uint8_t>{0x0c, 0x00, 0x00, 0x00, 0x80});
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
||||||
loc_regs[5] = DwarfLocation{DWARF_LOCATION_VAL_EXPRESSION, {0x4, 0x5000}};
|
loc_regs[5] = DwarfLocation{DWARF_LOCATION_VAL_EXPRESSION, {0x4, 0x5000}};
|
||||||
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_TRUE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
|
EXPECT_FALSE(finished);
|
||||||
EXPECT_EQ(0x3000U, regs.sp());
|
EXPECT_EQ(0x3000U, regs.sp());
|
||||||
EXPECT_EQ(0x80000000U, regs.pc());
|
EXPECT_EQ(0x80000000U, regs.pc());
|
||||||
}
|
}
|
||||||
|
|
@ -396,7 +424,8 @@ TYPED_TEST_P(DwarfSectionImplTest, Eval_same_cfa_same_pc) {
|
||||||
regs[5] = 0x100;
|
regs[5] = 0x100;
|
||||||
regs[8] = 0x2000;
|
regs[8] = 0x2000;
|
||||||
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
loc_regs[CFA_REG] = DwarfLocation{DWARF_LOCATION_REGISTER, {8, 0}};
|
||||||
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s));
|
bool finished;
|
||||||
|
ASSERT_FALSE(this->section_->Eval(&cie, &this->memory_, loc_regs, ®s, &finished));
|
||||||
EXPECT_EQ(0x2000U, regs.sp());
|
EXPECT_EQ(0x2000U, regs.sp());
|
||||||
EXPECT_EQ(0x100U, regs.pc());
|
EXPECT_EQ(0x100U, regs.pc());
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -32,7 +32,7 @@ class MockDwarfSection : public DwarfSection {
|
||||||
|
|
||||||
MOCK_METHOD4(Log, bool(uint8_t, uint64_t, uint64_t, const DwarfFde*));
|
MOCK_METHOD4(Log, bool(uint8_t, uint64_t, uint64_t, const DwarfFde*));
|
||||||
|
|
||||||
MOCK_METHOD4(Eval, bool(const DwarfCie*, Memory*, const dwarf_loc_regs_t&, Regs*));
|
MOCK_METHOD5(Eval, bool(const DwarfCie*, Memory*, const dwarf_loc_regs_t&, Regs*, bool*));
|
||||||
|
|
||||||
MOCK_METHOD3(GetCfaLocationInfo, bool(uint64_t, const DwarfFde*, dwarf_loc_regs_t*));
|
MOCK_METHOD3(GetCfaLocationInfo, bool(uint64_t, const DwarfFde*, dwarf_loc_regs_t*));
|
||||||
|
|
||||||
|
|
@ -104,7 +104,8 @@ TEST_F(DwarfSectionTest, Step_fail_fde) {
|
||||||
EXPECT_CALL(mock_section, GetFdeOffsetFromPc(0x1000, ::testing::_))
|
EXPECT_CALL(mock_section, GetFdeOffsetFromPc(0x1000, ::testing::_))
|
||||||
.WillOnce(::testing::Return(false));
|
.WillOnce(::testing::Return(false));
|
||||||
|
|
||||||
ASSERT_FALSE(mock_section.Step(0x1000, nullptr, nullptr));
|
bool finished;
|
||||||
|
ASSERT_FALSE(mock_section.Step(0x1000, nullptr, nullptr, &finished));
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(DwarfSectionTest, Step_fail_cie_null) {
|
TEST_F(DwarfSectionTest, Step_fail_cie_null) {
|
||||||
|
|
@ -118,7 +119,8 @@ TEST_F(DwarfSectionTest, Step_fail_cie_null) {
|
||||||
.WillOnce(::testing::Return(true));
|
.WillOnce(::testing::Return(true));
|
||||||
EXPECT_CALL(mock_section, GetFdeFromOffset(::testing::_)).WillOnce(::testing::Return(&fde));
|
EXPECT_CALL(mock_section, GetFdeFromOffset(::testing::_)).WillOnce(::testing::Return(&fde));
|
||||||
|
|
||||||
ASSERT_FALSE(mock_section.Step(0x1000, nullptr, nullptr));
|
bool finished;
|
||||||
|
ASSERT_FALSE(mock_section.Step(0x1000, nullptr, nullptr, &finished));
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(DwarfSectionTest, Step_fail_cfa_location) {
|
TEST_F(DwarfSectionTest, Step_fail_cfa_location) {
|
||||||
|
|
@ -136,7 +138,8 @@ TEST_F(DwarfSectionTest, Step_fail_cfa_location) {
|
||||||
EXPECT_CALL(mock_section, GetCfaLocationInfo(0x1000, &fde, ::testing::_))
|
EXPECT_CALL(mock_section, GetCfaLocationInfo(0x1000, &fde, ::testing::_))
|
||||||
.WillOnce(::testing::Return(false));
|
.WillOnce(::testing::Return(false));
|
||||||
|
|
||||||
ASSERT_FALSE(mock_section.Step(0x1000, nullptr, nullptr));
|
bool finished;
|
||||||
|
ASSERT_FALSE(mock_section.Step(0x1000, nullptr, nullptr, &finished));
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(DwarfSectionTest, Step_pass) {
|
TEST_F(DwarfSectionTest, Step_pass) {
|
||||||
|
|
@ -155,10 +158,11 @@ TEST_F(DwarfSectionTest, Step_pass) {
|
||||||
.WillOnce(::testing::Return(true));
|
.WillOnce(::testing::Return(true));
|
||||||
|
|
||||||
MemoryFake process;
|
MemoryFake process;
|
||||||
EXPECT_CALL(mock_section, Eval(&cie, &process, ::testing::_, nullptr))
|
EXPECT_CALL(mock_section, Eval(&cie, &process, ::testing::_, nullptr, ::testing::_))
|
||||||
.WillOnce(::testing::Return(true));
|
.WillOnce(::testing::Return(true));
|
||||||
|
|
||||||
ASSERT_TRUE(mock_section.Step(0x1000, nullptr, &process));
|
bool finished;
|
||||||
|
ASSERT_TRUE(mock_section.Step(0x1000, nullptr, &process, &finished));
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace unwindstack
|
} // namespace unwindstack
|
||||||
|
|
|
||||||
|
|
@ -322,7 +322,8 @@ TEST_F(ElfInterfaceArmTest, StepExidx) {
|
||||||
ElfInterfaceArm interface(&memory_);
|
ElfInterfaceArm interface(&memory_);
|
||||||
|
|
||||||
// FindEntry fails.
|
// FindEntry fails.
|
||||||
ASSERT_FALSE(interface.StepExidx(0x7000, nullptr, nullptr));
|
bool finished;
|
||||||
|
ASSERT_FALSE(interface.StepExidx(0x7000, nullptr, nullptr, &finished));
|
||||||
|
|
||||||
// ExtractEntry should fail.
|
// ExtractEntry should fail.
|
||||||
interface.set_start_offset(0x1000);
|
interface.set_start_offset(0x1000);
|
||||||
|
|
@ -335,15 +336,16 @@ TEST_F(ElfInterfaceArmTest, StepExidx) {
|
||||||
regs[ARM_REG_LR] = 0x20000;
|
regs[ARM_REG_LR] = 0x20000;
|
||||||
regs.set_sp(regs[ARM_REG_SP]);
|
regs.set_sp(regs[ARM_REG_SP]);
|
||||||
regs.set_pc(0x1234);
|
regs.set_pc(0x1234);
|
||||||
ASSERT_FALSE(interface.StepExidx(0x7000, ®s, &process_memory_));
|
ASSERT_FALSE(interface.StepExidx(0x7000, ®s, &process_memory_, &finished));
|
||||||
|
|
||||||
// Eval should fail.
|
// Eval should fail.
|
||||||
memory_.SetData32(0x1004, 0x81000000);
|
memory_.SetData32(0x1004, 0x81000000);
|
||||||
ASSERT_FALSE(interface.StepExidx(0x7000, ®s, &process_memory_));
|
ASSERT_FALSE(interface.StepExidx(0x7000, ®s, &process_memory_, &finished));
|
||||||
|
|
||||||
// Everything should pass.
|
// Everything should pass.
|
||||||
memory_.SetData32(0x1004, 0x80b0b0b0);
|
memory_.SetData32(0x1004, 0x80b0b0b0);
|
||||||
ASSERT_TRUE(interface.StepExidx(0x7000, ®s, &process_memory_));
|
ASSERT_TRUE(interface.StepExidx(0x7000, ®s, &process_memory_, &finished));
|
||||||
|
ASSERT_FALSE(finished);
|
||||||
ASSERT_EQ(0x1000U, regs.sp());
|
ASSERT_EQ(0x1000U, regs.sp());
|
||||||
ASSERT_EQ(0x1000U, regs[ARM_REG_SP]);
|
ASSERT_EQ(0x1000U, regs[ARM_REG_SP]);
|
||||||
ASSERT_EQ(0x20000U, regs.pc());
|
ASSERT_EQ(0x20000U, regs.pc());
|
||||||
|
|
@ -367,11 +369,57 @@ TEST_F(ElfInterfaceArmTest, StepExidx_pc_set) {
|
||||||
regs.set_pc(0x1234);
|
regs.set_pc(0x1234);
|
||||||
|
|
||||||
// Everything should pass.
|
// Everything should pass.
|
||||||
ASSERT_TRUE(interface.StepExidx(0x7000, ®s, &process_memory_));
|
bool finished;
|
||||||
|
ASSERT_TRUE(interface.StepExidx(0x7000, ®s, &process_memory_, &finished));
|
||||||
|
ASSERT_FALSE(finished);
|
||||||
ASSERT_EQ(0x10004U, regs.sp());
|
ASSERT_EQ(0x10004U, regs.sp());
|
||||||
ASSERT_EQ(0x10004U, regs[ARM_REG_SP]);
|
ASSERT_EQ(0x10004U, regs[ARM_REG_SP]);
|
||||||
ASSERT_EQ(0x10U, regs.pc());
|
ASSERT_EQ(0x10U, regs.pc());
|
||||||
ASSERT_EQ(0x10U, regs[ARM_REG_PC]);
|
ASSERT_EQ(0x10U, regs[ARM_REG_PC]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
TEST_F(ElfInterfaceArmTest, StepExidx_cant_unwind) {
|
||||||
|
ElfInterfaceArm interface(&memory_);
|
||||||
|
|
||||||
|
interface.set_start_offset(0x1000);
|
||||||
|
interface.set_total_entries(1);
|
||||||
|
memory_.SetData32(0x1000, 0x6000);
|
||||||
|
memory_.SetData32(0x1004, 1);
|
||||||
|
|
||||||
|
RegsArm regs;
|
||||||
|
regs[ARM_REG_SP] = 0x10000;
|
||||||
|
regs[ARM_REG_LR] = 0x20000;
|
||||||
|
regs.set_sp(regs[ARM_REG_SP]);
|
||||||
|
regs.set_pc(0x1234);
|
||||||
|
|
||||||
|
bool finished;
|
||||||
|
ASSERT_TRUE(interface.StepExidx(0x7000, ®s, &process_memory_, &finished));
|
||||||
|
ASSERT_TRUE(finished);
|
||||||
|
ASSERT_EQ(0x10000U, regs.sp());
|
||||||
|
ASSERT_EQ(0x10000U, regs[ARM_REG_SP]);
|
||||||
|
ASSERT_EQ(0x1234U, regs.pc());
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(ElfInterfaceArmTest, StepExidx_refuse_unwind) {
|
||||||
|
ElfInterfaceArm interface(&memory_);
|
||||||
|
|
||||||
|
interface.set_start_offset(0x1000);
|
||||||
|
interface.set_total_entries(1);
|
||||||
|
memory_.SetData32(0x1000, 0x6000);
|
||||||
|
memory_.SetData32(0x1004, 0x808000b0);
|
||||||
|
|
||||||
|
RegsArm regs;
|
||||||
|
regs[ARM_REG_SP] = 0x10000;
|
||||||
|
regs[ARM_REG_LR] = 0x20000;
|
||||||
|
regs.set_sp(regs[ARM_REG_SP]);
|
||||||
|
regs.set_pc(0x1234);
|
||||||
|
|
||||||
|
bool finished;
|
||||||
|
ASSERT_TRUE(interface.StepExidx(0x7000, ®s, &process_memory_, &finished));
|
||||||
|
ASSERT_TRUE(finished);
|
||||||
|
ASSERT_EQ(0x10000U, regs.sp());
|
||||||
|
ASSERT_EQ(0x10000U, regs[ARM_REG_SP]);
|
||||||
|
ASSERT_EQ(0x1234U, regs.pc());
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace unwindstack
|
} // namespace unwindstack
|
||||||
|
|
|
||||||
|
|
@ -129,7 +129,8 @@ TEST_F(ElfTest, elf_invalid) {
|
||||||
uint64_t func_offset;
|
uint64_t func_offset;
|
||||||
ASSERT_FALSE(elf.GetFunctionName(0, &name, &func_offset));
|
ASSERT_FALSE(elf.GetFunctionName(0, &name, &func_offset));
|
||||||
|
|
||||||
ASSERT_FALSE(elf.Step(0, nullptr, nullptr));
|
bool finished;
|
||||||
|
ASSERT_FALSE(elf.Step(0, nullptr, nullptr, &finished));
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(ElfTest, elf32_invalid_machine) {
|
TEST_F(ElfTest, elf32_invalid_machine) {
|
||||||
|
|
|
||||||
|
|
@ -35,6 +35,7 @@ class RegsFake : public RegsImpl<TypeParam> {
|
||||||
|
|
||||||
uint64_t GetAdjustedPc(uint64_t, Elf*) override { return 0; }
|
uint64_t GetAdjustedPc(uint64_t, Elf*) override { return 0; }
|
||||||
void SetFromRaw() override {}
|
void SetFromRaw() override {}
|
||||||
|
bool SetPcFromReturnAddress(Memory*) override { return false; }
|
||||||
bool StepIfSignalHandler(uint64_t, Elf*, Memory*) override { return false; }
|
bool StepIfSignalHandler(uint64_t, Elf*, Memory*) override { return false; }
|
||||||
bool GetReturnAddressFromDefault(Memory*, uint64_t*) { return false; }
|
bool GetReturnAddressFromDefault(Memory*, uint64_t*) { return false; }
|
||||||
};
|
};
|
||||||
|
|
|
||||||
|
|
@ -46,7 +46,7 @@ class ElfInterfaceFake : public ElfInterface {
|
||||||
void InitHeaders() override {}
|
void InitHeaders() override {}
|
||||||
bool GetSoname(std::string*) override { return false; }
|
bool GetSoname(std::string*) override { return false; }
|
||||||
bool GetFunctionName(uint64_t, std::string*, uint64_t*) override { return false; }
|
bool GetFunctionName(uint64_t, std::string*, uint64_t*) override { return false; }
|
||||||
bool Step(uint64_t, Regs*, Memory*) override { return false; }
|
bool Step(uint64_t, Regs*, Memory*, bool*) override { return false; }
|
||||||
};
|
};
|
||||||
|
|
||||||
template <typename TypeParam>
|
template <typename TypeParam>
|
||||||
|
|
@ -62,6 +62,7 @@ class RegsTestImpl : public RegsImpl<TypeParam> {
|
||||||
|
|
||||||
uint64_t GetAdjustedPc(uint64_t, Elf*) override { return 0; }
|
uint64_t GetAdjustedPc(uint64_t, Elf*) override { return 0; }
|
||||||
void SetFromRaw() override {}
|
void SetFromRaw() override {}
|
||||||
|
bool SetPcFromReturnAddress(Memory*) override { return false; }
|
||||||
bool StepIfSignalHandler(uint64_t, Elf*, Memory*) override { return false; }
|
bool StepIfSignalHandler(uint64_t, Elf*, Memory*) override { return false; }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -31,12 +31,13 @@
|
||||||
#include <thread>
|
#include <thread>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include <unwindstack/Elf.h>
|
#include <android-base/stringprintf.h>
|
||||||
#include <unwindstack/MapInfo.h>
|
|
||||||
#include <unwindstack/Maps.h>
|
#include <unwindstack/Maps.h>
|
||||||
#include <unwindstack/Memory.h>
|
#include <unwindstack/Memory.h>
|
||||||
#include <unwindstack/Regs.h>
|
#include <unwindstack/Regs.h>
|
||||||
#include <unwindstack/RegsGetLocal.h>
|
#include <unwindstack/RegsGetLocal.h>
|
||||||
|
#include <unwindstack/Unwinder.h>
|
||||||
|
|
||||||
#include "TestUtils.h"
|
#include "TestUtils.h"
|
||||||
|
|
||||||
|
|
@ -56,11 +57,11 @@ static void ResetGlobals() {
|
||||||
g_ucontext = 0;
|
g_ucontext = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::vector<const char*> kFunctionOrder{"InnerFunction", "MiddleFunction", "OuterFunction"};
|
static std::vector<const char*> kFunctionOrder{"OuterFunction", "MiddleFunction", "InnerFunction"};
|
||||||
|
|
||||||
static std::vector<const char*> kFunctionSignalOrder{"SignalInnerFunction", "SignalMiddleFunction",
|
static std::vector<const char*> kFunctionSignalOrder{"OuterFunction", "MiddleFunction",
|
||||||
"SignalOuterFunction", "InnerFunction",
|
"InnerFunction", "SignalOuterFunction",
|
||||||
"MiddleFunction", "OuterFunction"};
|
"SignalMiddleFunction", "SignalInnerFunction"};
|
||||||
|
|
||||||
static void SignalHandler(int, siginfo_t*, void* sigcontext) {
|
static void SignalHandler(int, siginfo_t*, void* sigcontext) {
|
||||||
g_ucontext = reinterpret_cast<uintptr_t>(sigcontext);
|
g_ucontext = reinterpret_cast<uintptr_t>(sigcontext);
|
||||||
|
|
@ -86,62 +87,44 @@ static void SignalCallerHandler(int, siginfo_t*, void*) {
|
||||||
SignalOuterFunction();
|
SignalOuterFunction();
|
||||||
}
|
}
|
||||||
|
|
||||||
static std::string ErrorMsg(const std::vector<const char*>& function_names, size_t index,
|
static std::string ErrorMsg(const std::vector<const char*>& function_names, Unwinder& unwinder) {
|
||||||
std::stringstream& unwind_stream) {
|
std::string unwind;
|
||||||
|
for (size_t i = 0; i < unwinder.NumFrames(); i++) {
|
||||||
|
unwind += unwinder.FormatFrame(i) + '\n';
|
||||||
|
}
|
||||||
|
|
||||||
return std::string(
|
return std::string(
|
||||||
"Unwind completed without finding all frames\n"
|
"Unwind completed without finding all frames\n"
|
||||||
" Looking for function: ") +
|
" Looking for function: ") +
|
||||||
function_names[index] + "\n" + "Unwind data:\n" + unwind_stream.str();
|
function_names.front() + "\n" + "Unwind data:\n" + unwind;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void VerifyUnwind(pid_t pid, Maps* maps, Regs* regs,
|
static void VerifyUnwind(pid_t pid, Maps* maps, Regs* regs,
|
||||||
std::vector<const char*>& function_names) {
|
std::vector<const char*> expected_function_names) {
|
||||||
size_t function_name_index = 0;
|
auto process_memory(Memory::CreateProcessMemory(pid));
|
||||||
|
|
||||||
auto process_memory = Memory::CreateProcessMemory(pid);
|
Unwinder unwinder(512, maps, regs, process_memory);
|
||||||
std::stringstream unwind_stream;
|
unwinder.Unwind();
|
||||||
unwind_stream << std::hex;
|
|
||||||
for (size_t frame_num = 0; frame_num < 64; frame_num++) {
|
|
||||||
ASSERT_NE(0U, regs->pc()) << ErrorMsg(function_names, function_name_index, unwind_stream);
|
|
||||||
MapInfo* map_info = maps->Find(regs->pc());
|
|
||||||
ASSERT_TRUE(map_info != nullptr) << ErrorMsg(function_names, function_name_index, unwind_stream);
|
|
||||||
|
|
||||||
Elf* elf = map_info->GetElf(process_memory, true);
|
std::string expected_function = expected_function_names.back();
|
||||||
uint64_t rel_pc = elf->GetRelPc(regs->pc(), map_info);
|
expected_function_names.pop_back();
|
||||||
uint64_t adjusted_rel_pc = rel_pc;
|
for (auto& frame : unwinder.frames()) {
|
||||||
if (frame_num != 0) {
|
if (frame.function_name == expected_function) {
|
||||||
adjusted_rel_pc = regs->GetAdjustedPc(rel_pc, elf);
|
if (expected_function_names.empty()) {
|
||||||
}
|
break;
|
||||||
unwind_stream << " PC: 0x" << regs->pc() << " Rel: 0x" << adjusted_rel_pc;
|
|
||||||
unwind_stream << " Map: ";
|
|
||||||
if (!map_info->name.empty()) {
|
|
||||||
unwind_stream << map_info->name;
|
|
||||||
} else {
|
|
||||||
unwind_stream << " anonymous";
|
|
||||||
}
|
|
||||||
unwind_stream << "<" << map_info->start << "-" << map_info->end << ">";
|
|
||||||
|
|
||||||
std::string name;
|
|
||||||
uint64_t func_offset;
|
|
||||||
if (elf->GetFunctionName(adjusted_rel_pc, &name, &func_offset)) {
|
|
||||||
if (name == function_names[function_name_index]) {
|
|
||||||
if (++function_name_index == function_names.size()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
unwind_stream << " " << name;
|
expected_function = expected_function_names.back();
|
||||||
|
expected_function_names.pop_back();
|
||||||
}
|
}
|
||||||
unwind_stream << "\n";
|
|
||||||
ASSERT_TRUE(elf->Step(rel_pc + map_info->elf_offset, regs, process_memory.get()))
|
|
||||||
<< ErrorMsg(function_names, function_name_index, unwind_stream);
|
|
||||||
}
|
}
|
||||||
ASSERT_TRUE(false) << ErrorMsg(function_names, function_name_index, unwind_stream);
|
|
||||||
|
ASSERT_TRUE(expected_function_names.empty()) << ErrorMsg(expected_function_names, unwinder);
|
||||||
}
|
}
|
||||||
|
|
||||||
// This test assumes that this code is compiled with optimizations turned
|
// This test assumes that this code is compiled with optimizations turned
|
||||||
// off. If this doesn't happen, then all of the calls will be optimized
|
// off. If this doesn't happen, then all of the calls will be optimized
|
||||||
// away.
|
// away.
|
||||||
extern "C" void InnerFunction(bool local) {
|
extern "C" void InnerFunction(bool local, bool trigger_invalid_call) {
|
||||||
if (local) {
|
if (local) {
|
||||||
LocalMaps maps;
|
LocalMaps maps;
|
||||||
ASSERT_TRUE(maps.Parse());
|
ASSERT_TRUE(maps.Parse());
|
||||||
|
|
@ -152,17 +135,21 @@ extern "C" void InnerFunction(bool local) {
|
||||||
} else {
|
} else {
|
||||||
g_ready_for_remote = true;
|
g_ready_for_remote = true;
|
||||||
g_ready = true;
|
g_ready = true;
|
||||||
|
if (trigger_invalid_call) {
|
||||||
|
void (*crash_func)() = nullptr;
|
||||||
|
crash_func();
|
||||||
|
}
|
||||||
while (!g_finish.load()) {
|
while (!g_finish.load()) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void MiddleFunction(bool local) {
|
extern "C" void MiddleFunction(bool local, bool trigger_invalid_call) {
|
||||||
InnerFunction(local);
|
InnerFunction(local, trigger_invalid_call);
|
||||||
}
|
}
|
||||||
|
|
||||||
extern "C" void OuterFunction(bool local) {
|
extern "C" void OuterFunction(bool local, bool trigger_invalid_call) {
|
||||||
MiddleFunction(local);
|
MiddleFunction(local, trigger_invalid_call);
|
||||||
}
|
}
|
||||||
|
|
||||||
class UnwindTest : public ::testing::Test {
|
class UnwindTest : public ::testing::Test {
|
||||||
|
|
@ -171,7 +158,7 @@ class UnwindTest : public ::testing::Test {
|
||||||
};
|
};
|
||||||
|
|
||||||
TEST_F(UnwindTest, local) {
|
TEST_F(UnwindTest, local) {
|
||||||
OuterFunction(true);
|
OuterFunction(true, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
void WaitForRemote(pid_t pid, uint64_t addr, bool leave_attached, bool* completed) {
|
void WaitForRemote(pid_t pid, uint64_t addr, bool leave_attached, bool* completed) {
|
||||||
|
|
@ -206,7 +193,7 @@ void WaitForRemote(pid_t pid, uint64_t addr, bool leave_attached, bool* complete
|
||||||
TEST_F(UnwindTest, remote) {
|
TEST_F(UnwindTest, remote) {
|
||||||
pid_t pid;
|
pid_t pid;
|
||||||
if ((pid = fork()) == 0) {
|
if ((pid = fork()) == 0) {
|
||||||
OuterFunction(false);
|
OuterFunction(false, false);
|
||||||
exit(0);
|
exit(0);
|
||||||
}
|
}
|
||||||
ASSERT_NE(-1, pid);
|
ASSERT_NE(-1, pid);
|
||||||
|
|
@ -231,7 +218,7 @@ TEST_F(UnwindTest, from_context) {
|
||||||
std::atomic_int tid(0);
|
std::atomic_int tid(0);
|
||||||
std::thread thread([&]() {
|
std::thread thread([&]() {
|
||||||
tid = syscall(__NR_gettid);
|
tid = syscall(__NR_gettid);
|
||||||
OuterFunction(false);
|
OuterFunction(false, false);
|
||||||
});
|
});
|
||||||
|
|
||||||
struct sigaction act, oldact;
|
struct sigaction act, oldact;
|
||||||
|
|
@ -273,25 +260,27 @@ TEST_F(UnwindTest, from_context) {
|
||||||
thread.join();
|
thread.join();
|
||||||
}
|
}
|
||||||
|
|
||||||
static void RemoteThroughSignal(unsigned int sa_flags) {
|
static void RemoteThroughSignal(int signal, unsigned int sa_flags) {
|
||||||
pid_t pid;
|
pid_t pid;
|
||||||
if ((pid = fork()) == 0) {
|
if ((pid = fork()) == 0) {
|
||||||
struct sigaction act, oldact;
|
struct sigaction act, oldact;
|
||||||
memset(&act, 0, sizeof(act));
|
memset(&act, 0, sizeof(act));
|
||||||
act.sa_sigaction = SignalCallerHandler;
|
act.sa_sigaction = SignalCallerHandler;
|
||||||
act.sa_flags = SA_RESTART | SA_ONSTACK | sa_flags;
|
act.sa_flags = SA_RESTART | SA_ONSTACK | sa_flags;
|
||||||
ASSERT_EQ(0, sigaction(SIGUSR1, &act, &oldact));
|
ASSERT_EQ(0, sigaction(signal, &act, &oldact));
|
||||||
|
|
||||||
OuterFunction(false);
|
OuterFunction(false, signal == SIGSEGV);
|
||||||
exit(0);
|
exit(0);
|
||||||
}
|
}
|
||||||
ASSERT_NE(-1, pid);
|
ASSERT_NE(-1, pid);
|
||||||
TestScopedPidReaper reap(pid);
|
TestScopedPidReaper reap(pid);
|
||||||
|
|
||||||
bool completed;
|
bool completed;
|
||||||
WaitForRemote(pid, reinterpret_cast<uint64_t>(&g_ready_for_remote), false, &completed);
|
if (signal != SIGSEGV) {
|
||||||
ASSERT_TRUE(completed) << "Timed out waiting for remote process to be ready.";
|
WaitForRemote(pid, reinterpret_cast<uint64_t>(&g_ready_for_remote), false, &completed);
|
||||||
ASSERT_EQ(0, kill(pid, SIGUSR1));
|
ASSERT_TRUE(completed) << "Timed out waiting for remote process to be ready.";
|
||||||
|
ASSERT_EQ(0, kill(pid, SIGUSR1));
|
||||||
|
}
|
||||||
WaitForRemote(pid, reinterpret_cast<uint64_t>(&g_signal_ready_for_remote), true, &completed);
|
WaitForRemote(pid, reinterpret_cast<uint64_t>(&g_signal_ready_for_remote), true, &completed);
|
||||||
ASSERT_TRUE(completed) << "Timed out waiting for remote process to be in signal handler.";
|
ASSERT_TRUE(completed) << "Timed out waiting for remote process to be in signal handler.";
|
||||||
|
|
||||||
|
|
@ -307,11 +296,19 @@ static void RemoteThroughSignal(unsigned int sa_flags) {
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(UnwindTest, remote_through_signal) {
|
TEST_F(UnwindTest, remote_through_signal) {
|
||||||
RemoteThroughSignal(0);
|
RemoteThroughSignal(SIGUSR1, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
TEST_F(UnwindTest, remote_through_signal_sa_siginfo) {
|
TEST_F(UnwindTest, remote_through_signal_sa_siginfo) {
|
||||||
RemoteThroughSignal(SA_SIGINFO);
|
RemoteThroughSignal(SIGUSR1, SA_SIGINFO);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(UnwindTest, remote_through_signal_with_invalid_func) {
|
||||||
|
RemoteThroughSignal(SIGSEGV, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
TEST_F(UnwindTest, remote_through_signal_sa_siginfo_with_invalid_func) {
|
||||||
|
RemoteThroughSignal(SIGSEGV, SA_SIGINFO);
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace unwindstack
|
} // namespace unwindstack
|
||||||
|
|
|
||||||
|
|
@ -26,7 +26,11 @@
|
||||||
#include <sys/types.h>
|
#include <sys/types.h>
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <memory>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include <android-base/stringprintf.h>
|
||||||
|
|
||||||
#include <unwindstack/Elf.h>
|
#include <unwindstack/Elf.h>
|
||||||
#include <unwindstack/MapInfo.h>
|
#include <unwindstack/MapInfo.h>
|
||||||
|
|
@ -55,6 +59,62 @@ static bool Detach(pid_t pid) {
|
||||||
return ptrace(PTRACE_DETACH, pid, 0, 0) == 0;
|
return ptrace(PTRACE_DETACH, pid, 0, 0) == 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string GetFrameInfo(size_t frame_num, unwindstack::Regs* regs,
|
||||||
|
const std::shared_ptr<unwindstack::Memory>& process_memory,
|
||||||
|
unwindstack::MapInfo* map_info, uint64_t* rel_pc) {
|
||||||
|
bool bits32;
|
||||||
|
switch (regs->MachineType()) {
|
||||||
|
case EM_ARM:
|
||||||
|
case EM_386:
|
||||||
|
bits32 = true;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
bits32 = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (map_info == nullptr) {
|
||||||
|
if (bits32) {
|
||||||
|
return android::base::StringPrintf(" #%02zu pc %08" PRIx64, frame_num, regs->pc());
|
||||||
|
} else {
|
||||||
|
return android::base::StringPrintf(" #%02zu pc %016" PRIx64, frame_num, regs->pc());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
unwindstack::Elf* elf = map_info->GetElf(process_memory, true);
|
||||||
|
*rel_pc = elf->GetRelPc(regs->pc(), map_info);
|
||||||
|
uint64_t adjusted_rel_pc = *rel_pc;
|
||||||
|
// Don't need to adjust the first frame pc.
|
||||||
|
if (frame_num != 0) {
|
||||||
|
adjusted_rel_pc = regs->GetAdjustedPc(*rel_pc, elf);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string line;
|
||||||
|
if (bits32) {
|
||||||
|
line = android::base::StringPrintf(" #%02zu pc %08" PRIx64, frame_num, adjusted_rel_pc);
|
||||||
|
} else {
|
||||||
|
line = android::base::StringPrintf(" #%02zu pc %016" PRIx64, frame_num, adjusted_rel_pc);
|
||||||
|
}
|
||||||
|
if (!map_info->name.empty()) {
|
||||||
|
line += " " + map_info->name;
|
||||||
|
if (map_info->elf_offset != 0) {
|
||||||
|
line += android::base::StringPrintf(" (offset 0x%" PRIx64 ")", map_info->elf_offset);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
line += android::base::StringPrintf(" <anonymous:%" PRIx64 ">", map_info->offset);
|
||||||
|
}
|
||||||
|
uint64_t func_offset;
|
||||||
|
std::string func_name;
|
||||||
|
if (elf->GetFunctionName(adjusted_rel_pc, &func_name, &func_offset)) {
|
||||||
|
line += " (" + func_name;
|
||||||
|
if (func_offset != 0) {
|
||||||
|
line += android::base::StringPrintf("+%" PRId64, func_offset);
|
||||||
|
}
|
||||||
|
line += ')';
|
||||||
|
}
|
||||||
|
return line;
|
||||||
|
}
|
||||||
|
|
||||||
void DoUnwind(pid_t pid) {
|
void DoUnwind(pid_t pid) {
|
||||||
unwindstack::RemoteMaps remote_maps(pid);
|
unwindstack::RemoteMaps remote_maps(pid);
|
||||||
if (!remote_maps.Parse()) {
|
if (!remote_maps.Parse()) {
|
||||||
|
|
@ -68,7 +128,6 @@ void DoUnwind(pid_t pid) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool bits32 = true;
|
|
||||||
printf("ABI: ");
|
printf("ABI: ");
|
||||||
switch (regs->MachineType()) {
|
switch (regs->MachineType()) {
|
||||||
case EM_ARM:
|
case EM_ARM:
|
||||||
|
|
@ -79,11 +138,9 @@ void DoUnwind(pid_t pid) {
|
||||||
break;
|
break;
|
||||||
case EM_AARCH64:
|
case EM_AARCH64:
|
||||||
printf("arm64");
|
printf("arm64");
|
||||||
bits32 = false;
|
|
||||||
break;
|
break;
|
||||||
case EM_X86_64:
|
case EM_X86_64:
|
||||||
printf("x86_64");
|
printf("x86_64");
|
||||||
bits32 = false;
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
printf("unknown\n");
|
printf("unknown\n");
|
||||||
|
|
@ -92,52 +149,48 @@ void DoUnwind(pid_t pid) {
|
||||||
printf("\n");
|
printf("\n");
|
||||||
|
|
||||||
auto process_memory = unwindstack::Memory::CreateProcessMemory(pid);
|
auto process_memory = unwindstack::Memory::CreateProcessMemory(pid);
|
||||||
|
bool return_address_attempt = false;
|
||||||
|
std::vector<std::string> frames;
|
||||||
for (size_t frame_num = 0; frame_num < 64; frame_num++) {
|
for (size_t frame_num = 0; frame_num < 64; frame_num++) {
|
||||||
if (regs->pc() == 0) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
unwindstack::MapInfo* map_info = remote_maps.Find(regs->pc());
|
unwindstack::MapInfo* map_info = remote_maps.Find(regs->pc());
|
||||||
|
uint64_t rel_pc;
|
||||||
|
frames.push_back(GetFrameInfo(frame_num, regs, process_memory, map_info, &rel_pc));
|
||||||
|
bool stepped;
|
||||||
if (map_info == nullptr) {
|
if (map_info == nullptr) {
|
||||||
printf("Failed to find map data for the pc\n");
|
stepped = false;
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
unwindstack::Elf* elf = map_info->GetElf(process_memory, true);
|
|
||||||
|
|
||||||
uint64_t rel_pc = elf->GetRelPc(regs->pc(), map_info);
|
|
||||||
uint64_t adjusted_rel_pc = rel_pc;
|
|
||||||
// Don't need to adjust the first frame pc.
|
|
||||||
if (frame_num != 0) {
|
|
||||||
adjusted_rel_pc = regs->GetAdjustedPc(rel_pc, elf);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string name;
|
|
||||||
if (bits32) {
|
|
||||||
printf(" #%02zu pc %08" PRIx64, frame_num, adjusted_rel_pc);
|
|
||||||
} else {
|
} else {
|
||||||
printf(" #%02zu pc %016" PRIx64, frame_num, adjusted_rel_pc);
|
bool finished;
|
||||||
|
stepped =
|
||||||
|
map_info->elf->Step(rel_pc + map_info->elf_offset, regs, process_memory.get(), &finished);
|
||||||
|
if (stepped && finished) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (!map_info->name.empty()) {
|
if (!stepped) {
|
||||||
printf(" %s", map_info->name.c_str());
|
if (return_address_attempt) {
|
||||||
if (map_info->elf_offset != 0) {
|
// We tried the return address and it didn't work, remove the last
|
||||||
printf(" (offset 0x%" PRIx64 ")", map_info->elf_offset);
|
// two frames. If this bad frame is the only frame, only remove
|
||||||
|
// the last frame.
|
||||||
|
frames.pop_back();
|
||||||
|
if (frame_num != 1) {
|
||||||
|
frames.pop_back();
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
// Steping didn't work, try this secondary method.
|
||||||
|
if (!regs->SetPcFromReturnAddress(process_memory.get())) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return_address_attempt = true;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
printf(" <anonymous:%" PRIx64 ">", map_info->offset);
|
return_address_attempt = false;
|
||||||
}
|
}
|
||||||
uint64_t func_offset;
|
}
|
||||||
if (elf->GetFunctionName(adjusted_rel_pc, &name, &func_offset)) {
|
|
||||||
printf(" (%s", name.c_str());
|
|
||||||
if (func_offset != 0) {
|
|
||||||
printf("+%" PRId64, func_offset);
|
|
||||||
}
|
|
||||||
printf(")");
|
|
||||||
}
|
|
||||||
printf("\n");
|
|
||||||
|
|
||||||
if (!elf->Step(rel_pc + map_info->elf_offset, regs, process_memory.get())) {
|
// Print the frames.
|
||||||
break;
|
for (auto& frame : frames) {
|
||||||
}
|
printf("%s\n", frame.c_str());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Add table
Reference in a new issue