/* * 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 #include #include #include #include "Check.h" #include "DwarfEhFrameWithHdr.h" #include "DwarfEncoding.h" namespace unwindstack { static inline bool IsEncodingRelative(uint8_t encoding) { encoding >>= 4; return encoding > 0 && encoding <= DW_EH_PE_funcrel; } template bool DwarfEhFrameWithHdr::Init(uint64_t offset, uint64_t size, uint64_t load_bias) { load_bias_ = load_bias; memory_.clear_func_offset(); memory_.clear_text_offset(); memory_.set_data_offset(offset); memory_.set_cur_offset(offset); pc_offset_ = offset; // Read the first four bytes all at once. uint8_t data[4]; if (!memory_.ReadBytes(data, 4)) { last_error_.code = DWARF_ERROR_MEMORY_INVALID; last_error_.address = memory_.cur_offset(); return false; } version_ = data[0]; if (version_ != 1) { // Unknown version. last_error_.code = DWARF_ERROR_UNSUPPORTED_VERSION; return false; } ptr_encoding_ = data[1]; uint8_t fde_count_encoding = data[2]; table_encoding_ = data[3]; table_entry_size_ = memory_.template GetEncodedSize(table_encoding_); memory_.set_pc_offset(memory_.cur_offset()); if (!memory_.template ReadEncodedValue(ptr_encoding_, &ptr_offset_)) { last_error_.code = DWARF_ERROR_MEMORY_INVALID; last_error_.address = memory_.cur_offset(); return false; } memory_.set_pc_offset(memory_.cur_offset()); if (!memory_.template ReadEncodedValue(fde_count_encoding, &fde_count_)) { last_error_.code = DWARF_ERROR_MEMORY_INVALID; last_error_.address = memory_.cur_offset(); return false; } if (fde_count_ == 0) { last_error_.code = DWARF_ERROR_NO_FDES; return false; } entries_offset_ = memory_.cur_offset(); entries_end_ = offset + size; entries_data_offset_ = offset; cur_entries_offset_ = entries_offset_; return true; } template const DwarfFde* DwarfEhFrameWithHdr::GetFdeFromPc(uint64_t pc) { uint64_t fde_offset; if (!GetFdeOffsetFromPc(pc, &fde_offset)) { return nullptr; } const DwarfFde* fde = this->GetFdeFromOffset(fde_offset); if (fde == nullptr) { return nullptr; } // Guaranteed pc >= pc_start, need to check pc in the fde range. if (pc < fde->pc_end) { return fde; } last_error_.code = DWARF_ERROR_ILLEGAL_STATE; return nullptr; } template const typename DwarfEhFrameWithHdr::FdeInfo* DwarfEhFrameWithHdr::GetFdeInfoFromIndex(size_t index) { auto entry = fde_info_.find(index); if (entry != fde_info_.end()) { return &fde_info_[index]; } FdeInfo* info = &fde_info_[index]; memory_.set_data_offset(entries_data_offset_); memory_.set_cur_offset(entries_offset_ + 2 * index * table_entry_size_); memory_.set_pc_offset(0); uint64_t value; if (!memory_.template ReadEncodedValue(table_encoding_, &value) || !memory_.template ReadEncodedValue(table_encoding_, &info->offset)) { last_error_.code = DWARF_ERROR_MEMORY_INVALID; last_error_.address = memory_.cur_offset(); fde_info_.erase(index); return nullptr; } // Relative encodings require adding in the load bias. if (IsEncodingRelative(table_encoding_)) { value += load_bias_; } info->pc = value; return info; } template bool DwarfEhFrameWithHdr::GetFdeOffsetBinary(uint64_t pc, uint64_t* fde_offset, uint64_t total_entries) { CHECK(fde_count_ > 0); CHECK(total_entries <= fde_count_); size_t first = 0; size_t last = total_entries; while (first < last) { size_t current = (first + last) / 2; const FdeInfo* info = GetFdeInfoFromIndex(current); if (info == nullptr) { return false; } if (pc == info->pc) { *fde_offset = info->offset; return true; } if (pc < info->pc) { last = current; } else { first = current + 1; } } if (last != 0) { const FdeInfo* info = GetFdeInfoFromIndex(last - 1); if (info == nullptr) { return false; } *fde_offset = info->offset; return true; } return false; } template bool DwarfEhFrameWithHdr::GetFdeOffsetSequential(uint64_t pc, uint64_t* fde_offset) { CHECK(fde_count_ != 0); last_error_.code = DWARF_ERROR_NONE; last_error_.address = 0; // We can do a binary search if the pc is in the range of the elements // that have already been cached. if (!fde_info_.empty()) { const FdeInfo* info = &fde_info_[fde_info_.size() - 1]; if (pc >= info->pc) { *fde_offset = info->offset; return true; } if (pc < info->pc) { return GetFdeOffsetBinary(pc, fde_offset, fde_info_.size()); } } if (cur_entries_offset_ == 0) { // All entries read, or error encountered. return false; } memory_.set_data_offset(entries_data_offset_); memory_.set_cur_offset(cur_entries_offset_); memory_.set_pc_offset(0); cur_entries_offset_ = 0; FdeInfo* prev_info = nullptr; for (size_t current = fde_info_.size(); current < fde_count_ && memory_.cur_offset() < entries_end_; current++) { FdeInfo* info = &fde_info_[current]; uint64_t value; if (!memory_.template ReadEncodedValue(table_encoding_, &value) || !memory_.template ReadEncodedValue(table_encoding_, &info->offset)) { fde_info_.erase(current); last_error_.code = DWARF_ERROR_MEMORY_INVALID; last_error_.address = memory_.cur_offset(); return false; } // Relative encodings require adding in the load bias. if (IsEncodingRelative(table_encoding_)) { value += load_bias_; } info->pc = value; if (pc < info->pc) { if (prev_info == nullptr) { return false; } cur_entries_offset_ = memory_.cur_offset(); *fde_offset = prev_info->offset; return true; } prev_info = info; } if (fde_count_ == fde_info_.size() && pc >= prev_info->pc) { *fde_offset = prev_info->offset; return true; } return false; } template bool DwarfEhFrameWithHdr::GetFdeOffsetFromPc(uint64_t pc, uint64_t* fde_offset) { if (fde_count_ == 0) { return false; } if (table_entry_size_ > 0) { // Do a binary search since the size of each table entry is fixed. return GetFdeOffsetBinary(pc, fde_offset, fde_count_); } else { // Do a sequential search since each table entry size is variable. return GetFdeOffsetSequential(pc, fde_offset); } } template void DwarfEhFrameWithHdr::GetFdes(std::vector* fdes) { for (size_t i = 0; i < fde_count_; i++) { const FdeInfo* info = GetFdeInfoFromIndex(i); if (info == nullptr) { break; } const DwarfFde* fde = this->GetFdeFromOffset(info->offset); if (fde == nullptr) { break; } fdes->push_back(fde); } } // Explicitly instantiate DwarfEhFrameWithHdr template class DwarfEhFrameWithHdr; template class DwarfEhFrameWithHdr; } // namespace unwindstack