blob: 01d9ed596c193b4cedfc201862fd555d5337c7ca [file] [log] [blame]
Jeff Brown10484a02011-10-21 12:07:49 -07001/*
2 * Copyright (C) 2011 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17/*
18 * Backtracing functions for x86.
19 */
20
21#define LOG_TAG "Corkscrew"
22//#define LOG_NDEBUG 0
23
24#include "../backtrace-arch.h"
25#include "../backtrace-helper.h"
26#include <corkscrew/ptrace.h>
27
28#include <stdlib.h>
29#include <signal.h>
30#include <stdbool.h>
31#include <limits.h>
32#include <errno.h>
33#include <sys/ptrace.h>
Jeff Brown10484a02011-10-21 12:07:49 -070034#include <cutils/log.h>
35
Elliott Hughes6b3bab32012-05-21 13:59:52 -070036#if defined(__BIONIC__)
37
38// Bionic offers the Linux kernel headers.
39#include <asm/sigcontext.h>
40#include <asm/ucontext.h>
41typedef struct ucontext ucontext_t;
42
43#else
44
45// glibc has its own renaming of the Linux kernel's structures.
Elliott Hughes71363a82012-05-18 11:56:17 -070046#define __USE_GNU // For REG_EBP, REG_ESP, and REG_EIP.
47#include <ucontext.h>
Jeff Brown10484a02011-10-21 12:07:49 -070048
Elliott Hughes6b3bab32012-05-21 13:59:52 -070049#endif
50
Jeff Brown10484a02011-10-21 12:07:49 -070051/* Unwind state. */
52typedef struct {
53 uint32_t ebp;
54 uint32_t eip;
55 uint32_t esp;
56} unwind_state_t;
57
Jeff Brownf0c58722011-11-03 17:58:44 -070058uintptr_t rewind_pc_arch(const memory_t* memory, uintptr_t pc) {
59 // TODO: Implement for x86.
60 return pc;
61}
62
63static ssize_t unwind_backtrace_common(const memory_t* memory,
64 const map_info_t* map_info_list,
Jeff Brown10484a02011-10-21 12:07:49 -070065 unwind_state_t* state, backtrace_frame_t* backtrace,
66 size_t ignore_depth, size_t max_depth) {
67 size_t ignored_frames = 0;
68 size_t returned_frames = 0;
69
Jeff Brownf0c58722011-11-03 17:58:44 -070070 for (size_t index = 0; state->ebp && returned_frames < max_depth; index++) {
71 backtrace_frame_t* frame = add_backtrace_entry(
72 index ? rewind_pc_arch(memory, state->eip) : state->eip,
Jeff Brown10484a02011-10-21 12:07:49 -070073 backtrace, ignore_depth, max_depth,
74 &ignored_frames, &returned_frames);
75 uint32_t next_esp = state->ebp + 8;
76 if (frame) {
77 frame->stack_top = state->esp;
78 if (state->esp < next_esp) {
79 frame->stack_size = next_esp - state->esp;
80 }
81 }
82 state->esp = next_esp;
Jeff Brownf0c58722011-11-03 17:58:44 -070083 if (!try_get_word(memory, state->ebp + 4, &state->eip)
84 || !try_get_word(memory, state->ebp, &state->ebp)
Jeff Brown10484a02011-10-21 12:07:49 -070085 || !state->eip) {
86 break;
87 }
88 }
89
90 return returned_frames;
91}
92
93ssize_t unwind_backtrace_signal_arch(siginfo_t* siginfo, void* sigcontext,
Jeff Brownf0c58722011-11-03 17:58:44 -070094 const map_info_t* map_info_list,
Jeff Brown10484a02011-10-21 12:07:49 -070095 backtrace_frame_t* backtrace, size_t ignore_depth, size_t max_depth) {
96 const ucontext_t* uc = (const ucontext_t*)sigcontext;
97
98 unwind_state_t state;
Elliott Hughes6b3bab32012-05-21 13:59:52 -070099#if defined(__BIONIC__)
100 state.ebp = uc->uc_mcontext.ebp;
101 state.esp = uc->uc_mcontext.esp;
102 state.eip = uc->uc_mcontext.eip;
103#else
Elliott Hughes71363a82012-05-18 11:56:17 -0700104 state.ebp = uc->uc_mcontext.gregs[REG_EBP];
105 state.esp = uc->uc_mcontext.gregs[REG_ESP];
106 state.eip = uc->uc_mcontext.gregs[REG_EIP];
Elliott Hughes6b3bab32012-05-21 13:59:52 -0700107#endif
Jeff Brown10484a02011-10-21 12:07:49 -0700108
Jeff Brownf0c58722011-11-03 17:58:44 -0700109 memory_t memory;
110 init_memory(&memory, map_info_list);
111 return unwind_backtrace_common(&memory, map_info_list,
112 &state, backtrace, ignore_depth, max_depth);
Jeff Brown10484a02011-10-21 12:07:49 -0700113}
114
115ssize_t unwind_backtrace_ptrace_arch(pid_t tid, const ptrace_context_t* context,
116 backtrace_frame_t* backtrace, size_t ignore_depth, size_t max_depth) {
117 pt_regs_x86_t regs;
118 if (ptrace(PTRACE_GETREGS, tid, 0, &regs)) {
119 return -1;
120 }
121
122 unwind_state_t state;
123 state.ebp = regs.ebp;
124 state.eip = regs.eip;
125 state.esp = regs.esp;
126
Jeff Brownf0c58722011-11-03 17:58:44 -0700127 memory_t memory;
128 init_memory_ptrace(&memory, tid);
129 return unwind_backtrace_common(&memory, context->map_info_list,
130 &state, backtrace, ignore_depth, max_depth);
Jeff Brown10484a02011-10-21 12:07:49 -0700131}