| /* |
| * Copyright (C) 2011 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. |
| */ |
| |
| #define LOG_TAG "Corkscrew" |
| //#define LOG_NDEBUG 0 |
| |
| #include "../ptrace-arch.h" |
| |
| #include <sys/exec_elf.h> |
| #include <cutils/log.h> |
| |
| #ifndef PT_ARM_EXIDX |
| #define PT_ARM_EXIDX 0x70000001 |
| #endif |
| |
| static void load_exidx_header(pid_t pid, map_info_t* mi, |
| uintptr_t* out_exidx_start, size_t* out_exidx_size) { |
| uint32_t elf_phoff; |
| uint32_t elf_phentsize_ehsize; |
| uint32_t elf_shentsize_phnum; |
| if (try_get_word_ptrace(pid, mi->start + offsetof(Elf32_Ehdr, e_phoff), &elf_phoff) |
| && try_get_word_ptrace(pid, mi->start + offsetof(Elf32_Ehdr, e_ehsize), |
| &elf_phentsize_ehsize) |
| && try_get_word_ptrace(pid, mi->start + offsetof(Elf32_Ehdr, e_phnum), |
| &elf_shentsize_phnum)) { |
| uint32_t elf_phentsize = elf_phentsize_ehsize >> 16; |
| uint32_t elf_phnum = elf_shentsize_phnum & 0xffff; |
| for (uint32_t i = 0; i < elf_phnum; i++) { |
| uintptr_t elf_phdr = mi->start + elf_phoff + i * elf_phentsize; |
| uint32_t elf_phdr_type; |
| if (!try_get_word_ptrace(pid, elf_phdr + offsetof(Elf32_Phdr, p_type), &elf_phdr_type)) { |
| break; |
| } |
| if (elf_phdr_type == PT_ARM_EXIDX) { |
| uint32_t elf_phdr_offset; |
| uint32_t elf_phdr_filesz; |
| if (!try_get_word_ptrace(pid, elf_phdr + offsetof(Elf32_Phdr, p_offset), |
| &elf_phdr_offset) |
| || !try_get_word_ptrace(pid, elf_phdr + offsetof(Elf32_Phdr, p_filesz), |
| &elf_phdr_filesz)) { |
| break; |
| } |
| *out_exidx_start = mi->start + elf_phdr_offset; |
| *out_exidx_size = elf_phdr_filesz / 8; |
| ALOGV("Parsed EXIDX header info for %s: start=0x%08x, size=%d", mi->name, |
| *out_exidx_start, *out_exidx_size); |
| return; |
| } |
| } |
| } |
| *out_exidx_start = 0; |
| *out_exidx_size = 0; |
| } |
| |
| void load_ptrace_map_info_data_arch(pid_t pid, map_info_t* mi, map_info_data_t* data) { |
| load_exidx_header(pid, mi, &data->exidx_start, &data->exidx_size); |
| } |
| |
| void free_ptrace_map_info_data_arch(map_info_t* mi, map_info_data_t* data) { |
| } |