| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117 | /* Copyright (C) 2005 Free Software Foundation, Inc.   This file is part of the GNU C Library.   The GNU C Library is free software; you can redistribute it and/or   modify it under the terms of the GNU Lesser General Public   License as published by the Free Software Foundation; either   version 2.1 of the License, or (at your option) any later version.   The GNU C Library is distributed in the hope that it will be useful,   but WITHOUT ANY WARRANTY; without even the implied warranty of   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU   Lesser General Public License for more details.   You should have received a copy of the GNU Lesser General Public   License along with the GNU C Library; if not, see   <http://www.gnu.org/licenses/>.  */#include <link.h>#include <unwind.h>#if __FDPIC__#include <bits/elf-fdpic.h>static __always_inline int__dl_addr_in_loadaddr(void *p, struct elf32_fdpic_loadaddr loadaddr){ struct elf32_fdpic_loadmap *map = loadaddr.map; int c; for (c = 0; c < map->nsegs; c++)   if ((void *)map->segs[c].addr <= p &&       (char *)p < (char *)map->segs[c].addr + map->segs[c].p_memsz)     return 1; return 0;}#endifstruct unw_eh_callback_data{  _Unwind_Ptr pc;  _Unwind_Ptr exidx_start;  int exidx_len;};/* Callback to determins if the PC lies within an object, and remember the   location of the exception index table if it does.  */static intfind_exidx_callback (struct dl_phdr_info * info, size_t size, void * ptr){#if __FDPIC__  struct unw_eh_callback_data * data;  const ElfW(Phdr) *phdr;  int i;  int match = 0;  data = (struct unw_eh_callback_data *) ptr;  if (__dl_addr_in_loadaddr((void *) data->pc, info->dlpi_addr)) {    match = 1;    phdr = info->dlpi_phdr;    for (i = info->dlpi_phnum; i > 0; i--, phdr++) {      if (phdr->p_type == PT_ARM_EXIDX) {        data->exidx_start = (_Unwind_Ptr) __RELOC_POINTER(phdr->p_vaddr, info->dlpi_addr);        data->exidx_len = phdr->p_memsz;      }    }  }  return match;#else  struct unw_eh_callback_data * data;  const ElfW(Phdr) *phdr;  int i;  int match;  _Unwind_Ptr load_base;  data = (struct unw_eh_callback_data *) ptr;  load_base = info->dlpi_addr;  phdr = info->dlpi_phdr;  match = 0;  for (i = info->dlpi_phnum; i > 0; i--, phdr++)    {      if (phdr->p_type == PT_LOAD)        {          _Unwind_Ptr vaddr = phdr->p_vaddr + load_base;          if (data->pc >= vaddr && data->pc < vaddr + phdr->p_memsz)            match = 1;        }      else if (phdr->p_type == PT_ARM_EXIDX)	{	  data->exidx_start = (_Unwind_Ptr) (phdr->p_vaddr + load_base);	  data->exidx_len = phdr->p_memsz;	}    }  return match;#endif}/* Find the exception index table containing PC.  */_Unwind_Ptr __gnu_Unwind_Find_exidx (_Unwind_Ptr pc, int * pcount);_Unwind_Ptr __gnu_Unwind_Find_exidx (_Unwind_Ptr pc, int * pcount){  struct unw_eh_callback_data data;  data.pc = pc;  data.exidx_start = 0;  if (dl_iterate_phdr (find_exidx_callback, &data) <= 0)    return 0;  *pcount = data.exidx_len / 8;  return data.exidx_start;}
 |