mirror of
https://github.com/klzgrad/naiveproxy.git
synced 2024-12-11 06:36:11 +03:00
340 lines
13 KiB
C++
340 lines
13 KiB
C++
|
// Copyright 2018 The Chromium Authors. All rights reserved.
|
||
|
// Use of this source code is governed by a BSD-style license that can be
|
||
|
// found in the LICENSE file.
|
||
|
|
||
|
#include "base/trace_event/cfi_backtrace_android.h"
|
||
|
|
||
|
#include <sys/mman.h>
|
||
|
#include <sys/types.h>
|
||
|
|
||
|
#include "base/android/apk_assets.h"
|
||
|
|
||
|
#if !defined(ARCH_CPU_ARMEL)
|
||
|
#error This file should not be built for this architecture.
|
||
|
#endif
|
||
|
|
||
|
/*
|
||
|
Basics of unwinding:
|
||
|
For each instruction in a function we need to know what is the offset of SP
|
||
|
(Stack Pointer) to reach the previous function's stack frame. To know which
|
||
|
function is being invoked, we need the return address of the next function. The
|
||
|
CFI information for an instruction is made up of 2 offsets, CFA (Call Frame
|
||
|
Address) offset and RA (Return Address) offset. The CFA offset is the change in
|
||
|
SP made by the function till the current instruction. This depends on amount of
|
||
|
memory allocated on stack by the function plus some registers that the function
|
||
|
stores that needs to be restored at the end of function. So, at each instruction
|
||
|
the CFA offset tells the offset from original SP before the function call. The
|
||
|
RA offset tells us the offset from the previous SP into the current function
|
||
|
where the return address is stored.
|
||
|
|
||
|
The unwind table file has 2 tables UNW_INDEX and UNW_DATA, inspired from ARM
|
||
|
EHABI format. The first table contains function addresses and an index into the
|
||
|
UNW_DATA table. The second table contains one or more rows for the function
|
||
|
unwind information.
|
||
|
|
||
|
UNW_INDEX contains two columns of N rows each, where N is the number of
|
||
|
functions.
|
||
|
1. First column 4 byte rows of all the function start address as offset from
|
||
|
start of the binary, in sorted order.
|
||
|
2. For each function addr, the second column contains 2 byte indices in order.
|
||
|
The indices are offsets (in count of 2 bytes) of the CFI data from start of
|
||
|
UNW_DATA.
|
||
|
The last entry in the table always contains CANT_UNWIND index to specify the
|
||
|
end address of the last function.
|
||
|
|
||
|
UNW_DATA contains data of all the functions. Each function data contains N rows.
|
||
|
The data found at the address pointed from UNW_INDEX will be:
|
||
|
2 bytes: N - number of rows that belong to current function.
|
||
|
N * 4 bytes: N rows of data. 16 bits : Address offset from function start.
|
||
|
14 bits : CFA offset / 4.
|
||
|
2 bits : RA offset / 4.
|
||
|
If the RA offset of a row is 0, then use the offset of the previous rows in the
|
||
|
same function.
|
||
|
TODO(ssid): Make sure RA offset is always present.
|
||
|
|
||
|
See extract_unwind_tables.py for details about how this data is extracted from
|
||
|
breakpad symbol files.
|
||
|
*/
|
||
|
|
||
|
extern "C" {
|
||
|
|
||
|
// The address of |__executable_start| gives the start address of the
|
||
|
// executable or shared library. This value is used to find the offset address
|
||
|
// of the instruction in binary from PC.
|
||
|
extern char __executable_start;
|
||
|
|
||
|
// The address |_etext| gives the end of the .text section in the binary. This
|
||
|
// value is more accurate than parsing the memory map since the mapped
|
||
|
// regions are usualy larger than the .text section.
|
||
|
extern char _etext;
|
||
|
}
|
||
|
|
||
|
namespace base {
|
||
|
namespace trace_event {
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
// The value of index when the function does not have unwind information.
|
||
|
constexpr uint32_t kCantUnwind = 0xFFFF;
|
||
|
|
||
|
// The mask on the CFI row data that is used to get the high 14 bits and
|
||
|
// multiply it by 4 to get CFA offset. Since the last 2 bits are masked out, a
|
||
|
// shift is not necessary.
|
||
|
constexpr uint16_t kCFAMask = 0xfffc;
|
||
|
|
||
|
// The mask on the CFI row data that is used to get the low 2 bits and multiply
|
||
|
// it by 4 to get the RA offset.
|
||
|
constexpr uint16_t kRAMask = 0x3;
|
||
|
constexpr uint16_t kRAShift = 2;
|
||
|
|
||
|
// The code in this file assumes we are running in 32-bit builds since all the
|
||
|
// addresses in the unwind table are specified in 32 bits.
|
||
|
static_assert(sizeof(uintptr_t) == 4,
|
||
|
"The unwind table format is only valid for 32 bit builds.");
|
||
|
|
||
|
// The CFI data in UNW_DATA table starts with number of rows (N) and then
|
||
|
// followed by N rows of 4 bytes long. The CFIUnwindDataRow represents a single
|
||
|
// row of CFI data of a function in the table. Since we cast the memory at the
|
||
|
// address after the address of number of rows, into an array of
|
||
|
// CFIUnwindDataRow, the size of the struct should be 4 bytes and the order of
|
||
|
// the members is fixed according to the given format. The first 2 bytes tell
|
||
|
// the address of function and last 2 bytes give the CFI data for the offset.
|
||
|
struct CFIUnwindDataRow {
|
||
|
// The address of the instruction in terms of offset from the start of the
|
||
|
// function.
|
||
|
uint16_t addr_offset;
|
||
|
// Represents the CFA and RA offsets to get information about next stack
|
||
|
// frame. This is the CFI data at the point before executing the instruction
|
||
|
// at |addr_offset| from the start of the function.
|
||
|
uint16_t cfi_data;
|
||
|
|
||
|
// Return the RA offset for the current unwind row.
|
||
|
size_t ra_offset() const { return (cfi_data & kRAMask) << kRAShift; }
|
||
|
|
||
|
// Returns the CFA offset for the current unwind row.
|
||
|
size_t cfa_offset() const { return cfi_data & kCFAMask; }
|
||
|
};
|
||
|
|
||
|
static_assert(
|
||
|
sizeof(CFIUnwindDataRow) == 4,
|
||
|
"The CFIUnwindDataRow struct must be exactly 4 bytes for searching.");
|
||
|
|
||
|
} // namespace
|
||
|
|
||
|
// static
|
||
|
CFIBacktraceAndroid* CFIBacktraceAndroid::GetInitializedInstance() {
|
||
|
static CFIBacktraceAndroid* instance = new CFIBacktraceAndroid();
|
||
|
return instance;
|
||
|
}
|
||
|
|
||
|
// static
|
||
|
uintptr_t CFIBacktraceAndroid::executable_start_addr() {
|
||
|
return reinterpret_cast<uintptr_t>(&__executable_start);
|
||
|
}
|
||
|
|
||
|
// static
|
||
|
uintptr_t CFIBacktraceAndroid::executable_end_addr() {
|
||
|
return reinterpret_cast<uintptr_t>(&_etext);
|
||
|
}
|
||
|
|
||
|
CFIBacktraceAndroid::CFIBacktraceAndroid()
|
||
|
: thread_local_cfi_cache_(
|
||
|
[](void* ptr) { delete static_cast<CFICache*>(ptr); }) {
|
||
|
Initialize();
|
||
|
}
|
||
|
|
||
|
CFIBacktraceAndroid::~CFIBacktraceAndroid() {}
|
||
|
|
||
|
void CFIBacktraceAndroid::Initialize() {
|
||
|
// This file name is defined by extract_unwind_tables.gni.
|
||
|
static constexpr char kCfiFileName[] = "assets/unwind_cfi_32";
|
||
|
MemoryMappedFile::Region cfi_region;
|
||
|
int fd = base::android::OpenApkAsset(kCfiFileName, &cfi_region);
|
||
|
if (fd < 0)
|
||
|
return;
|
||
|
cfi_mmap_ = std::make_unique<MemoryMappedFile>();
|
||
|
// The CFI region starts at |cfi_region.offset|.
|
||
|
if (!cfi_mmap_->Initialize(base::File(fd), cfi_region))
|
||
|
return;
|
||
|
|
||
|
ParseCFITables();
|
||
|
can_unwind_stack_frames_ = true;
|
||
|
}
|
||
|
|
||
|
void CFIBacktraceAndroid::ParseCFITables() {
|
||
|
// The first 4 bytes in the file is the size of UNW_INDEX table.
|
||
|
static constexpr size_t kUnwIndexRowSize =
|
||
|
sizeof(*unw_index_function_col_) + sizeof(*unw_index_indices_col_);
|
||
|
size_t unw_index_size = 0;
|
||
|
memcpy(&unw_index_size, cfi_mmap_->data(), sizeof(unw_index_size));
|
||
|
DCHECK_EQ(0u, unw_index_size % kUnwIndexRowSize);
|
||
|
// UNW_INDEX table starts after 4 bytes.
|
||
|
unw_index_function_col_ =
|
||
|
reinterpret_cast<const uintptr_t*>(cfi_mmap_->data()) + 1;
|
||
|
unw_index_row_count_ = unw_index_size / kUnwIndexRowSize;
|
||
|
unw_index_indices_col_ = reinterpret_cast<const uint16_t*>(
|
||
|
unw_index_function_col_ + unw_index_row_count_);
|
||
|
|
||
|
// The UNW_DATA table data is right after the end of UNW_INDEX table.
|
||
|
// Interpret the UNW_DATA table as an array of 2 byte numbers since the
|
||
|
// indexes we have from the UNW_INDEX table are in terms of 2 bytes.
|
||
|
unw_data_start_addr_ = unw_index_indices_col_ + unw_index_row_count_;
|
||
|
}
|
||
|
|
||
|
size_t CFIBacktraceAndroid::Unwind(const void** out_trace, size_t max_depth) {
|
||
|
// This function walks the stack using the call frame information to find the
|
||
|
// return addresses of all the functions that belong to current binary in call
|
||
|
// stack. For each function the CFI table defines the offset of the previous
|
||
|
// call frame and offset where the return address is stored.
|
||
|
if (!can_unwind_stack_frames())
|
||
|
return 0;
|
||
|
|
||
|
// Get the current register state. This register state can be taken at any
|
||
|
// point in the function and the unwind information would be for this point.
|
||
|
// Define local variables before trying to get the current PC and SP to make
|
||
|
// sure the register state obtained is consistent with each other.
|
||
|
uintptr_t pc = 0, sp = 0;
|
||
|
asm volatile("mov %0, pc" : "=r"(pc));
|
||
|
asm volatile("mov %0, sp" : "=r"(sp));
|
||
|
|
||
|
return Unwind(pc, sp, out_trace, max_depth);
|
||
|
}
|
||
|
|
||
|
size_t CFIBacktraceAndroid::Unwind(uintptr_t pc,
|
||
|
uintptr_t sp,
|
||
|
const void** out_trace,
|
||
|
size_t max_depth) {
|
||
|
if (!can_unwind_stack_frames())
|
||
|
return 0;
|
||
|
|
||
|
// We can only unwind as long as the pc is within the chrome.so.
|
||
|
size_t depth = 0;
|
||
|
while (is_chrome_address(pc) && depth < max_depth) {
|
||
|
out_trace[depth++] = reinterpret_cast<void*>(pc);
|
||
|
// The offset of function from the start of the chrome.so binary:
|
||
|
uintptr_t func_addr = pc - executable_start_addr();
|
||
|
CFIRow cfi{};
|
||
|
if (!FindCFIRowForPC(func_addr, &cfi))
|
||
|
break;
|
||
|
|
||
|
// The rules for unwinding using the CFI information are:
|
||
|
// SP_prev = SP_cur + cfa_offset and
|
||
|
// PC_prev = * (SP_prev - ra_offset).
|
||
|
sp = sp + cfi.cfa_offset;
|
||
|
memcpy(&pc, reinterpret_cast<uintptr_t*>(sp - cfi.ra_offset),
|
||
|
sizeof(uintptr_t));
|
||
|
}
|
||
|
return depth;
|
||
|
}
|
||
|
|
||
|
void CFIBacktraceAndroid::AllocateCacheForCurrentThread() {
|
||
|
GetThreadLocalCFICache();
|
||
|
}
|
||
|
|
||
|
bool CFIBacktraceAndroid::FindCFIRowForPC(uintptr_t func_addr,
|
||
|
CFIBacktraceAndroid::CFIRow* cfi) {
|
||
|
if (!can_unwind_stack_frames())
|
||
|
return false;
|
||
|
|
||
|
auto* cache = GetThreadLocalCFICache();
|
||
|
*cfi = {0};
|
||
|
if (cache->Find(func_addr, cfi))
|
||
|
return true;
|
||
|
|
||
|
// Consider each column of UNW_INDEX table as arrays of uintptr_t (function
|
||
|
// addresses) and uint16_t (indices). Define start and end iterator on the
|
||
|
// first column array (addresses) and use std::lower_bound() to binary search
|
||
|
// on this array to find the required function address.
|
||
|
static const uintptr_t* const unw_index_fn_end =
|
||
|
unw_index_function_col_ + unw_index_row_count_;
|
||
|
const uintptr_t* found =
|
||
|
std::lower_bound(unw_index_function_col_, unw_index_fn_end, func_addr);
|
||
|
|
||
|
// If found is start, then the given function is not in the table. If the
|
||
|
// given pc is start of a function then we cannot unwind.
|
||
|
if (found == unw_index_function_col_ || *found == func_addr)
|
||
|
return false;
|
||
|
|
||
|
// std::lower_bound() returns the iter that corresponds to the first address
|
||
|
// that is greater than the given address. So, the required iter is always one
|
||
|
// less than the value returned by std::lower_bound().
|
||
|
--found;
|
||
|
uintptr_t func_start_addr = *found;
|
||
|
size_t row_num = found - unw_index_function_col_;
|
||
|
uint16_t index = unw_index_indices_col_[row_num];
|
||
|
DCHECK_LE(func_start_addr, func_addr);
|
||
|
// If the index is CANT_UNWIND then we do not have unwind infomation for the
|
||
|
// function.
|
||
|
if (index == kCantUnwind)
|
||
|
return false;
|
||
|
|
||
|
// The unwind data for the current function is at an offsset of the index
|
||
|
// found in UNW_INDEX table.
|
||
|
const uint16_t* unwind_data = unw_data_start_addr_ + index;
|
||
|
// The value of first 2 bytes is the CFI data row count for the function.
|
||
|
uint16_t row_count = 0;
|
||
|
memcpy(&row_count, unwind_data, sizeof(row_count));
|
||
|
// And the actual CFI rows start after 2 bytes from the |unwind_data|. Cast
|
||
|
// the data into an array of CFIUnwindDataRow since the struct is designed to
|
||
|
// represent each row. We should be careful to read only |row_count| number of
|
||
|
// elements in the array.
|
||
|
const CFIUnwindDataRow* function_data =
|
||
|
reinterpret_cast<const CFIUnwindDataRow*>(unwind_data + 1);
|
||
|
|
||
|
// Iterate through the CFI rows of the function to find the row that gives
|
||
|
// offset for the given instruction address.
|
||
|
CFIUnwindDataRow cfi_row = {0, 0};
|
||
|
uint16_t ra_offset = 0;
|
||
|
for (uint16_t i = 0; i < row_count; ++i) {
|
||
|
CFIUnwindDataRow row;
|
||
|
memcpy(&row, function_data + i, sizeof(CFIUnwindDataRow));
|
||
|
// The return address of the function is the instruction that is not yet
|
||
|
// been executed. The cfi row specifies the unwind info before executing the
|
||
|
// given instruction. If the given address is equal to the instruction
|
||
|
// offset, then use the current row. Or use the row with highest address
|
||
|
// less than the given address.
|
||
|
if (row.addr_offset + func_start_addr > func_addr)
|
||
|
break;
|
||
|
|
||
|
cfi_row = row;
|
||
|
// The ra offset of the last specified row should be used, if unspecified.
|
||
|
// So, keep updating the RA offset till we reach the correct CFI row.
|
||
|
// TODO(ssid): This should be fixed in the format and we should always
|
||
|
// output ra offset.
|
||
|
if (cfi_row.ra_offset())
|
||
|
ra_offset = cfi_row.ra_offset();
|
||
|
}
|
||
|
DCHECK_NE(0u, cfi_row.addr_offset);
|
||
|
*cfi = {cfi_row.cfa_offset(), ra_offset};
|
||
|
DCHECK(cfi->cfa_offset);
|
||
|
DCHECK(cfi->ra_offset);
|
||
|
|
||
|
// safe to update since the cache is thread local.
|
||
|
cache->Add(func_addr, *cfi);
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
CFIBacktraceAndroid::CFICache* CFIBacktraceAndroid::GetThreadLocalCFICache() {
|
||
|
auto* cache = static_cast<CFICache*>(thread_local_cfi_cache_.Get());
|
||
|
if (!cache) {
|
||
|
cache = new CFICache();
|
||
|
thread_local_cfi_cache_.Set(cache);
|
||
|
}
|
||
|
return cache;
|
||
|
}
|
||
|
|
||
|
void CFIBacktraceAndroid::CFICache::Add(uintptr_t address, CFIRow cfi) {
|
||
|
cache_[address % kLimit] = {address, cfi};
|
||
|
}
|
||
|
|
||
|
bool CFIBacktraceAndroid::CFICache::Find(uintptr_t address, CFIRow* cfi) {
|
||
|
if (cache_[address % kLimit].address == address) {
|
||
|
*cfi = cache_[address % kLimit].cfi;
|
||
|
return true;
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
} // namespace trace_event
|
||
|
} // namespace base
|