| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 1 | /* | 
 | 2 |  * Copyright (C) 2013 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 |  | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 17 | #include <stdint.h> | 
| Christopher Ferris | c58287d | 2014-05-09 16:59:06 -0700 | [diff] [blame] | 18 | #include <ucontext.h> | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 19 |  | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 20 | #include <memory> | 
 | 21 | #include <string> | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 22 |  | 
 | 23 | #define UNW_LOCAL_ONLY | 
 | 24 | #include <libunwind.h> | 
 | 25 |  | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 26 | #include <backtrace/Backtrace.h> | 
 | 27 |  | 
| Christopher Ferris | e296091 | 2014-03-07 19:42:19 -0800 | [diff] [blame] | 28 | #include "BacktraceLog.h" | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 29 | #include "UnwindCurrent.h" | 
 | 30 |  | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 31 | std::string UnwindCurrent::GetFunctionNameRaw(uintptr_t pc, uintptr_t* offset) { | 
 | 32 |   *offset = 0; | 
 | 33 |   char buf[512]; | 
 | 34 |   unw_word_t value; | 
 | 35 |   if (unw_get_proc_name_by_ip(unw_local_addr_space, pc, buf, sizeof(buf), | 
 | 36 |                               &value, &context_) >= 0 && buf[0] != '\0') { | 
 | 37 |     *offset = static_cast<uintptr_t>(value); | 
 | 38 |     return buf; | 
| Christopher Ferris | a2efd3a | 2014-05-06 15:23:59 -0700 | [diff] [blame] | 39 |   } | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 40 |   return ""; | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 41 | } | 
 | 42 |  | 
| Christopher Ferris | a2efd3a | 2014-05-06 15:23:59 -0700 | [diff] [blame] | 43 | void UnwindCurrent::GetUnwContextFromUcontext(const ucontext_t* ucontext) { | 
 | 44 |   unw_tdep_context_t* unw_context = reinterpret_cast<unw_tdep_context_t*>(&context_); | 
 | 45 |  | 
 | 46 | #if defined(__arm__) | 
 | 47 |   unw_context->regs[0] = ucontext->uc_mcontext.arm_r0; | 
 | 48 |   unw_context->regs[1] = ucontext->uc_mcontext.arm_r1; | 
 | 49 |   unw_context->regs[2] = ucontext->uc_mcontext.arm_r2; | 
 | 50 |   unw_context->regs[3] = ucontext->uc_mcontext.arm_r3; | 
 | 51 |   unw_context->regs[4] = ucontext->uc_mcontext.arm_r4; | 
 | 52 |   unw_context->regs[5] = ucontext->uc_mcontext.arm_r5; | 
 | 53 |   unw_context->regs[6] = ucontext->uc_mcontext.arm_r6; | 
 | 54 |   unw_context->regs[7] = ucontext->uc_mcontext.arm_r7; | 
 | 55 |   unw_context->regs[8] = ucontext->uc_mcontext.arm_r8; | 
 | 56 |   unw_context->regs[9] = ucontext->uc_mcontext.arm_r9; | 
 | 57 |   unw_context->regs[10] = ucontext->uc_mcontext.arm_r10; | 
 | 58 |   unw_context->regs[11] = ucontext->uc_mcontext.arm_fp; | 
 | 59 |   unw_context->regs[12] = ucontext->uc_mcontext.arm_ip; | 
 | 60 |   unw_context->regs[13] = ucontext->uc_mcontext.arm_sp; | 
 | 61 |   unw_context->regs[14] = ucontext->uc_mcontext.arm_lr; | 
 | 62 |   unw_context->regs[15] = ucontext->uc_mcontext.arm_pc; | 
 | 63 | #else | 
 | 64 |   unw_context->uc_mcontext = ucontext->uc_mcontext; | 
 | 65 | #endif | 
 | 66 | } | 
 | 67 |  | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 68 | bool UnwindCurrent::UnwindFromContext(size_t num_ignore_frames, ucontext_t* ucontext) { | 
 | 69 |   if (ucontext == nullptr) { | 
 | 70 |     int ret = unw_getcontext(&context_); | 
 | 71 |     if (ret < 0) { | 
 | 72 |       BACK_LOGW("unw_getcontext failed %d", ret); | 
 | 73 |       return false; | 
| Christopher Ferris | 17224d5 | 2014-04-11 13:05:07 -0700 | [diff] [blame] | 74 |     } | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 75 |   } else { | 
 | 76 |     GetUnwContextFromUcontext(ucontext); | 
 | 77 |   } | 
 | 78 |  | 
 | 79 |   // The cursor structure is pretty large, do not put it on the stack. | 
 | 80 |   std::unique_ptr<unw_cursor_t> cursor(new unw_cursor_t); | 
 | 81 |   int ret = unw_init_local(cursor.get(), &context_); | 
 | 82 |   if (ret < 0) { | 
 | 83 |     BACK_LOGW("unw_init_local failed %d", ret); | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 84 |     return false; | 
 | 85 |   } | 
 | 86 |  | 
| Christopher Ferris | 4675682 | 2014-01-14 20:16:30 -0800 | [diff] [blame] | 87 |   size_t num_frames = 0; | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 88 |   do { | 
 | 89 |     unw_word_t pc; | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 90 |     ret = unw_get_reg(cursor.get(), UNW_REG_IP, &pc); | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 91 |     if (ret < 0) { | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 92 |       BACK_LOGW("Failed to read IP %d", ret); | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 93 |       break; | 
 | 94 |     } | 
 | 95 |     unw_word_t sp; | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 96 |     ret = unw_get_reg(cursor.get(), UNW_REG_SP, &sp); | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 97 |     if (ret < 0) { | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 98 |       BACK_LOGW("Failed to read SP %d", ret); | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 99 |       break; | 
 | 100 |     } | 
 | 101 |  | 
| Christopher Ferris | ca09ce9 | 2015-03-31 17:28:22 -0700 | [diff] [blame] | 102 |     frames_.resize(num_frames+1); | 
 | 103 |     backtrace_frame_data_t* frame = &frames_.at(num_frames); | 
 | 104 |     frame->num = num_frames; | 
 | 105 |     frame->pc = static_cast<uintptr_t>(pc); | 
 | 106 |     frame->sp = static_cast<uintptr_t>(sp); | 
 | 107 |     frame->stack_size = 0; | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 108 |  | 
| Christopher Ferris | ca09ce9 | 2015-03-31 17:28:22 -0700 | [diff] [blame] | 109 |     FillInMap(frame->pc, &frame->map); | 
 | 110 |     // Check to see if we should skip this frame because it's coming | 
 | 111 |     // from within the library, and we are doing a local unwind. | 
 | 112 |     if (ucontext != nullptr || num_frames != 0 || !DiscardFrame(*frame)) { | 
 | 113 |       if (num_ignore_frames == 0) { | 
 | 114 |         // GetFunctionName is an expensive call, only do it if we are | 
 | 115 |         // keeping the frame. | 
 | 116 |         frame->func_name = GetFunctionName(frame->pc, &frame->func_offset); | 
 | 117 |         if (num_frames > 0) { | 
 | 118 |           // Set the stack size for the previous frame. | 
 | 119 |           backtrace_frame_data_t* prev = &frames_.at(num_frames-1); | 
 | 120 |           prev->stack_size = frame->sp - prev->sp; | 
 | 121 |         } | 
 | 122 |         num_frames++; | 
 | 123 |       } else { | 
 | 124 |         num_ignore_frames--; | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 125 |       } | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 126 |     } | 
| Christopher Ferris | 2c43cff | 2015-03-26 19:18:36 -0700 | [diff] [blame] | 127 |     ret = unw_step (cursor.get()); | 
| Christopher Ferris | 4675682 | 2014-01-14 20:16:30 -0800 | [diff] [blame] | 128 |   } while (ret > 0 && num_frames < MAX_BACKTRACE_FRAMES); | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 129 |  | 
| Christopher Ferris | 17e91d4 | 2013-10-21 13:30:52 -0700 | [diff] [blame] | 130 |   return true; | 
 | 131 | } |