198 lines
5.8 KiB
C
198 lines
5.8 KiB
C
/*
|
|
* Copyright (C) 2012 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.
|
|
*/
|
|
|
|
/*
|
|
* Backtracing functions for mips
|
|
*/
|
|
|
|
#define LOG_TAG "Corkscrew"
|
|
//#define LOG_NDEBUG 0
|
|
|
|
#include "../backtrace-arch.h"
|
|
#include "../backtrace-helper.h"
|
|
#include <corkscrew/ptrace.h>
|
|
|
|
#include <stdlib.h>
|
|
#include <signal.h>
|
|
#include <stdbool.h>
|
|
#include <limits.h>
|
|
#include <errno.h>
|
|
#include <sys/ptrace.h>
|
|
#include <sys/exec_elf.h>
|
|
#include <cutils/log.h>
|
|
|
|
/* For PTRACE_GETREGS */
|
|
typedef struct {
|
|
/* FIXME: check this definition */
|
|
uint64_t regs[32];
|
|
uint64_t lo;
|
|
uint64_t hi;
|
|
uint64_t epc;
|
|
uint64_t badvaddr;
|
|
uint64_t status;
|
|
uint64_t cause;
|
|
} user_regs_struct;
|
|
|
|
/* Machine context at the time a signal was raised. */
|
|
typedef struct ucontext {
|
|
/* FIXME: use correct definition */
|
|
uint32_t sp;
|
|
uint32_t ra;
|
|
uint32_t pc;
|
|
} ucontext_t;
|
|
|
|
/* Unwind state. */
|
|
typedef struct {
|
|
uint32_t sp;
|
|
uint32_t ra;
|
|
uint32_t pc;
|
|
} unwind_state_t;
|
|
|
|
uintptr_t rewind_pc_arch(const memory_t* memory, uintptr_t pc) {
|
|
if (pc == 0)
|
|
return pc;
|
|
if ((pc & 1) == 0)
|
|
return pc-8; /* jal/bal/jalr + branch delay slot */
|
|
return pc;
|
|
}
|
|
|
|
static ssize_t unwind_backtrace_common(const memory_t* memory,
|
|
const map_info_t* map_info_list,
|
|
unwind_state_t* state, backtrace_frame_t* backtrace,
|
|
size_t ignore_depth, size_t max_depth) {
|
|
size_t ignored_frames = 0;
|
|
size_t returned_frames = 0;
|
|
|
|
for (size_t index = 0; returned_frames < max_depth; index++) {
|
|
uintptr_t pc = index ? rewind_pc_arch(memory, state->pc) : state->pc;
|
|
backtrace_frame_t* frame;
|
|
uintptr_t addr;
|
|
int maxcheck = 1024;
|
|
int stack_size = 0, ra_offset = 0;
|
|
bool found_start = false;
|
|
|
|
frame = add_backtrace_entry(pc, backtrace, ignore_depth,
|
|
max_depth, &ignored_frames, &returned_frames);
|
|
|
|
if (frame)
|
|
frame->stack_top = state->sp;
|
|
|
|
ALOGV("#%d: frame=%p pc=%08x sp=%08x\n",
|
|
index, frame, frame->absolute_pc, frame->stack_top);
|
|
|
|
for (addr = state->pc; maxcheck-- > 0 && !found_start; addr -= 4) {
|
|
uint32_t op;
|
|
if (!try_get_word(memory, addr, &op))
|
|
break;
|
|
|
|
// ALOGV("@0x%08x: 0x%08x\n", addr, op);
|
|
switch (op & 0xffff0000) {
|
|
case 0x27bd0000: // addiu sp, imm
|
|
{
|
|
// looking for stack being decremented
|
|
int32_t immediate = ((((int)op) << 16) >> 16);
|
|
if (immediate < 0) {
|
|
stack_size = -immediate;
|
|
found_start = true;
|
|
ALOGV("@0x%08x: found stack adjustment=%d\n", addr, stack_size);
|
|
}
|
|
}
|
|
break;
|
|
case 0xafbf0000: // sw ra, imm(sp)
|
|
ra_offset = ((((int)op) << 16) >> 16);
|
|
ALOGV("@0x%08x: found ra offset=%d\n", addr, ra_offset);
|
|
break;
|
|
case 0x3c1c0000: // lui gp
|
|
ALOGV("@0x%08x: found function boundary\n", addr);
|
|
found_start = true;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (ra_offset) {
|
|
uint32_t next_ra;
|
|
if (!try_get_word(memory, state->sp + ra_offset, &next_ra))
|
|
break;
|
|
state->ra = next_ra;
|
|
ALOGV("New ra: 0x%08x\n", state->ra);
|
|
}
|
|
|
|
if (stack_size) {
|
|
if (frame)
|
|
frame->stack_size = stack_size;
|
|
state->sp += stack_size;
|
|
ALOGV("New sp: 0x%08x\n", state->sp);
|
|
}
|
|
|
|
if (state->pc == state->ra && stack_size == 0)
|
|
break;
|
|
|
|
if (state->ra == 0)
|
|
break;
|
|
|
|
state->pc = state->ra;
|
|
}
|
|
|
|
ALOGV("returning %d frames\n", returned_frames);
|
|
|
|
return returned_frames;
|
|
}
|
|
|
|
ssize_t unwind_backtrace_signal_arch(siginfo_t* siginfo, void* sigcontext,
|
|
const map_info_t* map_info_list,
|
|
backtrace_frame_t* backtrace, size_t ignore_depth, size_t max_depth) {
|
|
const ucontext_t* uc = (const ucontext_t*)sigcontext;
|
|
|
|
unwind_state_t state;
|
|
state.sp = uc->sp;
|
|
state.pc = uc->pc;
|
|
state.ra = uc->ra;
|
|
|
|
ALOGV("unwind_backtrace_signal_arch: "
|
|
"ignore_depth=%d max_depth=%d pc=0x%08x sp=0x%08x ra=0x%08x\n",
|
|
ignore_depth, max_depth, state.pc, state.sp, state.ra);
|
|
|
|
memory_t memory;
|
|
init_memory(&memory, map_info_list);
|
|
return unwind_backtrace_common(&memory, map_info_list,
|
|
&state, backtrace, ignore_depth, max_depth);
|
|
}
|
|
|
|
ssize_t unwind_backtrace_ptrace_arch(pid_t tid, const ptrace_context_t* context,
|
|
backtrace_frame_t* backtrace, size_t ignore_depth, size_t max_depth) {
|
|
|
|
user_regs_struct regs;
|
|
if (ptrace(PTRACE_GETREGS, tid, 0, ®s)) {
|
|
return -1;
|
|
}
|
|
|
|
unwind_state_t state;
|
|
state.sp = regs.regs[29];
|
|
state.ra = regs.regs[31];
|
|
state.pc = regs.epc;
|
|
|
|
ALOGV("unwind_backtrace_ptrace_arch: "
|
|
"ignore_depth=%d max_depth=%d pc=0x%08x sp=0x%08x ra=0x%08x\n",
|
|
ignore_depth, max_depth, state.pc, state.sp, state.ra);
|
|
|
|
memory_t memory;
|
|
init_memory_ptrace(&memory, tid);
|
|
return unwind_backtrace_common(&memory, context->map_info_list,
|
|
&state, backtrace, ignore_depth, max_depth);
|
|
}
|