Use hex_string instead of phex_nz for error.
[deliverable/binutils-gdb.git] / binutils / readelf.c
index 6994289522e77c99b5fb152bea1626bfa9180324..ba4fd9b9262358f042915e9de9613a9af6d09138 100644 (file)
 #include "elf/sh.h"
 #include "elf/sparc.h"
 #include "elf/spu.h"
+#include "elf/tic6x.h"
 #include "elf/v850.h"
 #include "elf/vax.h"
 #include "elf/x86-64.h"
@@ -285,6 +286,11 @@ static void (* byte_put) (unsigned char *, bfd_vma, int);
 #define streq(a,b)       (strcmp ((a), (b)) == 0)
 #define strneq(a,b,n)    (strncmp ((a), (b), (n)) == 0)
 #define const_strneq(a,b) (strncmp ((a), (b), sizeof (b) - 1) == 0)
+
+#define REMOVE_ARCH_BITS(ADDR) do {            \
+    if (elf_header.e_machine == EM_ARM)                \
+      (ADDR) &= ~1;                            \
+  } while (0)
 \f
 static void *
 get_data (void * var, FILE * file, long offset, size_t size, size_t nmemb,
@@ -544,6 +550,33 @@ find_section (const char * name)
   return NULL;
 }
 
+/* Return a pointer to a section containing ADDR, or NULL if no such
+   section exists.  */
+
+static Elf_Internal_Shdr *
+find_section_by_address (bfd_vma addr)
+{
+  unsigned int i;
+
+  for (i = 0; i < elf_header.e_shnum; i++)
+    {
+      Elf_Internal_Shdr *sec = section_headers + i;
+      if (addr >= sec->sh_addr && addr < sec->sh_addr + sec->sh_size)
+       return sec;
+    }
+
+  return NULL;
+}
+
+/* Read an unsigned LEB128 encoded value from p.  Set *PLEN to the number of
+   bytes read.  */
+
+static unsigned long
+read_uleb128 (unsigned char *data, unsigned int *length_return)
+{
+  return read_leb128 (data, length_return, 0);
+}
+
 /* Guess the relocation size commonly used by the specific machines.  */
 
 static int
@@ -616,6 +649,7 @@ guess_is_rela (unsigned int e_machine)
     case EM_SPARC32PLUS:
     case EM_SPARCV9:
     case EM_SPU:
+    case EM_TI_C6000:
     case EM_V850:
     case EM_CYGNUS_V850:
     case EM_VAX:
@@ -1229,6 +1263,10 @@ dump_relocations (FILE * file,
        case EM_C166:
          rtype = elf_xc16x_reloc_type (type);
          break;
+
+       case EM_TI_C6000:
+         rtype = elf_tic6x_reloc_type (type);
+         break;
        }
 
       if (rtype == NULL)
@@ -1606,6 +1644,21 @@ get_score_dynamic_type (unsigned long type)
     }
 }
 
+static const char *
+get_tic6x_dynamic_type (unsigned long type)
+{
+  switch (type)
+    {
+    case DT_C6000_GSYM_OFFSET: return "C6000_GSYM_OFFSET";
+    case DT_C6000_GSTR_OFFSET: return "C6000_GSTR_OFFSET";
+    case DT_C6000_DSBT_BASE:   return "C6000_DSBT_BASE";
+    case DT_C6000_DSBT_SIZE:   return "C6000_DSBT_SIZE";
+    case DT_C6000_PREEMPTMAP:  return "C6000_PREEMPTMAP";
+    case DT_C6000_DSBT_INDEX:  return "C6000_DSBT_INDEX";
+    default:
+      return NULL;
+    }
+}
 
 static const char *
 get_dynamic_type (unsigned long type)
@@ -1718,6 +1771,9 @@ get_dynamic_type (unsigned long type)
            case EM_SCORE:
              result = get_score_dynamic_type (type);
              break;
+           case EM_TI_C6000:
+             result = get_tic6x_dynamic_type (type);
+             break;
            default:
              result = NULL;
              break;
@@ -2512,6 +2568,10 @@ get_machine_flags (unsigned e_flags, unsigned e_machine)
        case EM_S390:
          if (e_flags & EF_S390_HIGH_GPRS)
            strcat (buf, ", highgprs");
+
+       case EM_TI_C6000:
+         if ((e_flags & EF_C6000_REL))
+           strcat (buf, ", relocatable module");
        }
     }
 
@@ -2541,9 +2601,42 @@ get_osabi_name (unsigned int osabi)
     case ELFOSABI_NSK:         return "HP - Non-Stop Kernel";
     case ELFOSABI_AROS:                return "AROS";
     case ELFOSABI_FENIXOS:     return "FenixOS";
-    case ELFOSABI_STANDALONE:  return _("Standalone App");
-    case ELFOSABI_ARM:         return "ARM";
     default:
+      if (osabi >= 64)
+       switch (elf_header.e_machine)
+         {
+         case EM_ARM:
+           switch (osabi)
+             {
+             case ELFOSABI_ARM:        return "ARM";
+             default:
+               break;
+             }
+           break;
+
+         case EM_MSP430:
+         case EM_MSP430_OLD:
+           switch (osabi)
+             {
+             case ELFOSABI_STANDALONE: return _("Standalone App");
+             default:
+               break;
+             }
+           break;
+
+         case EM_TI_C6000:
+           switch (osabi)
+             {
+             case ELFOSABI_C6000_ELFABI:       return _("Bare-metal C6000");
+             case ELFOSABI_C6000_LINUX:        return "Linux C6000";
+             default:
+               break;
+             }
+           break;
+
+         default:
+           break;
+         }
       snprintf (buff, sizeof (buff), _("<unknown: %x>"), osabi);
       return buff;
     }
@@ -2630,6 +2723,19 @@ get_ia64_segment_type (unsigned long type)
   return NULL;
 }
 
+static const char *
+get_tic6x_segment_type (unsigned long type)
+{
+  switch (type)
+    {
+    case PT_C6000_PHATTR:      return "C6000_PHATTR";
+    default:
+      break;
+    }
+
+  return NULL;
+}
+
 static const char *
 get_segment_type (unsigned long p_type)
 {
@@ -2671,6 +2777,9 @@ get_segment_type (unsigned long p_type)
            case EM_IA_64:
              result = get_ia64_segment_type (p_type);
              break;
+           case EM_TI_C6000:
+             result = get_tic6x_segment_type (p_type);
+             break;
            default:
              result = NULL;
              break;
@@ -2831,6 +2940,33 @@ get_arm_section_type_name (unsigned int sh_type)
   return NULL;
 }
 
+static const char *
+get_tic6x_section_type_name (unsigned int sh_type)
+{
+  switch (sh_type)
+    {
+    case SHT_C6000_UNWIND:
+      return "C6000_UNWIND";
+    case SHT_C6000_PREEMPTMAP:
+      return "C6000_PREEMPTMAP";
+    case SHT_C6000_ATTRIBUTES:
+      return "C6000_ATTRIBUTES";
+    case SHT_TI_ICODE:
+      return "TI_ICODE";
+    case SHT_TI_XREF:
+      return "TI_XREF";
+    case SHT_TI_HANDLER:
+      return "TI_HANDLER";
+    case SHT_TI_INITINFO:
+      return "TI_INITINFO";
+    case SHT_TI_PHATTRS:
+      return "TI_PHATTRS";
+    default:
+      break;
+    }
+  return NULL;
+}
+
 static const char *
 get_section_type_name (unsigned int sh_type)
 {
@@ -2889,6 +3025,9 @@ get_section_type_name (unsigned int sh_type)
            case EM_ARM:
              result = get_arm_section_type_name (sh_type);
              break;
+           case EM_TI_C6000:
+             result = get_tic6x_section_type_name (sh_type);
+             break;
            default:
              result = NULL;
              break;
@@ -4633,11 +4772,14 @@ get_group_flags (unsigned int flags)
   static char buff[32];
   switch (flags)
     {
+    case 0:
+      return "";
+
     case GRP_COMDAT:
-      return "COMDAT";
+      return "COMDAT ";
 
    default:
-      snprintf (buff, sizeof (buff), _("[<unknown>: 0x%x]"), flags);
+      snprintf (buff, sizeof (buff), _("[<unknown>: 0x%x] "), flags);
       break;
     }
   return buff;
@@ -4797,7 +4939,7 @@ process_section_groups (FILE * file)
 
          if (do_section_groups)
            {
-             printf ("\n%s group section [%5u] `%s' [%s] contains %u sections:\n",
+             printf ("\n%sgroup section [%5u] `%s' [%s] contains %u sections:\n",
                      get_group_flags (entry), i, name, group_name, size);
 
              printf (_("   [Index]    Name\n"));
@@ -5086,16 +5228,22 @@ find_symbol_for_address (Elf_Internal_Sym * symtab,
   Elf_Internal_Sym * best = NULL;
   unsigned long i;
 
+  REMOVE_ARCH_BITS (addr.offset);
+
   for (i = 0, sym = symtab; i < nsyms; ++i, ++sym)
     {
+      bfd_vma value = sym->st_value;
+
+      REMOVE_ARCH_BITS (value);
+
       if (ELF_ST_TYPE (sym->st_info) == STT_FUNC
          && sym->st_name != 0
          && (addr.section == SHN_UNDEF || addr.section == sym->st_shndx)
-         && addr.offset >= sym->st_value
-         && addr.offset - sym->st_value < dist)
+         && addr.offset >= value
+         && addr.offset - value < dist)
        {
          best = sym;
-         dist = addr.offset - sym->st_value;
+         dist = addr.offset - value;
          if (!dist)
            break;
        }
@@ -5758,6 +5906,579 @@ hppa_process_unwind (FILE * file)
   return 1;
 }
 
+struct arm_section
+{
+  unsigned char *data;
+
+  Elf_Internal_Shdr *sec;
+  Elf_Internal_Rela *rela;
+  unsigned long nrelas;
+  unsigned int rel_type;
+
+  Elf_Internal_Rela *next_rela;
+};
+
+struct arm_unw_aux_info
+{
+  FILE *file;
+
+  Elf_Internal_Sym *symtab;    /* The symbol table.  */
+  unsigned long nsyms;         /* Number of symbols.  */
+  char *strtab;                        /* The string table.  */
+  unsigned long strtab_size;   /* Size of string table.  */
+};
+
+static const char *
+arm_print_vma_and_name (struct arm_unw_aux_info *aux,
+                       bfd_vma fn, struct absaddr addr)
+{
+  const char *procname;
+  bfd_vma sym_offset;
+
+  if (addr.section == SHN_UNDEF)
+    addr.offset = fn;
+
+  find_symbol_for_address (aux->symtab, aux->nsyms, aux->strtab,
+                          aux->strtab_size, addr, &procname,
+                          &sym_offset);
+
+  print_vma (fn, PREFIX_HEX);
+
+  if (procname)
+    {
+      fputs (" <", stdout);
+      fputs (procname, stdout);
+
+      if (sym_offset)
+       printf ("+0x%lx", (unsigned long) sym_offset);
+      fputc ('>', stdout);
+    }
+
+  return procname;
+}
+
+static void
+arm_free_section (struct arm_section *arm_sec)
+{
+  if (arm_sec->data != NULL)
+    free (arm_sec->data);
+
+  if (arm_sec->rela != NULL)
+    free (arm_sec->rela);
+}
+
+static int
+arm_section_get_word (struct arm_unw_aux_info *aux,
+                     struct arm_section *arm_sec,
+                     Elf_Internal_Shdr *sec, bfd_vma word_offset,
+                     unsigned int *wordp, struct absaddr *addr)
+{
+  Elf_Internal_Rela *rp;
+  Elf_Internal_Sym *sym;
+  const char * relname;
+  unsigned int word;
+  bfd_boolean wrapped;
+
+  addr->section = SHN_UNDEF;
+  addr->offset = 0;
+
+  if (sec != arm_sec->sec)
+    {
+      Elf_Internal_Shdr *relsec;
+
+      arm_free_section (arm_sec);
+
+      arm_sec->sec = sec;
+      arm_sec->data = get_data (NULL, aux->file, sec->sh_offset, 1,
+                               sec->sh_size, _("unwind data"));
+
+      arm_sec->rela = NULL;
+      arm_sec->nrelas = 0;
+
+      for (relsec = section_headers;
+          relsec < section_headers + elf_header.e_shnum;
+          ++relsec)
+       {
+         if (relsec->sh_info >= elf_header.e_shnum
+             || section_headers + relsec->sh_info != sec)
+           continue;
+
+         if (relsec->sh_type == SHT_REL)
+           {
+             if (!slurp_rel_relocs (aux->file, relsec->sh_offset,
+                                    relsec->sh_size,
+                                    & arm_sec->rela, & arm_sec->nrelas))
+               return 0;
+             break;
+           }
+         else if (relsec->sh_type == SHT_RELA)
+           {
+             if (!slurp_rela_relocs (aux->file, relsec->sh_offset,
+                                     relsec->sh_size,
+                                     & arm_sec->rela, & arm_sec->nrelas))
+               return 0;
+             break;
+           }
+       }
+
+      arm_sec->next_rela = arm_sec->rela;
+    }
+
+  if (arm_sec->data == NULL)
+    return 0;
+
+  word = byte_get (arm_sec->data + word_offset, 4);
+
+  wrapped = FALSE;
+  for (rp = arm_sec->next_rela; rp != arm_sec->rela + arm_sec->nrelas; rp++)
+    {
+      bfd_vma prelval, offset;
+
+      if (rp->r_offset > word_offset && !wrapped)
+       {
+         rp = arm_sec->rela;
+         wrapped = TRUE;
+       }
+      if (rp->r_offset > word_offset)
+       break;
+
+      if (rp->r_offset & 3)
+       {
+         warn (_("Skipping unexpected relocation at offset 0x%lx\n"),
+               (unsigned long) rp->r_offset);
+         continue;
+       }
+
+      if (rp->r_offset < word_offset)
+       continue;
+
+      relname = elf_arm_reloc_type (ELF32_R_TYPE (rp->r_info));
+
+      if (streq (relname, "R_ARM_NONE"))
+       continue;
+
+      if (! streq (relname, "R_ARM_PREL31"))
+       {
+         warn (_("Skipping unexpected relocation type %s\n"), relname);
+         continue;
+       }
+
+      sym = aux->symtab + ELF32_R_SYM (rp->r_info);
+
+      if (arm_sec->rel_type == SHT_REL)
+       {
+         offset = word & 0x7fffffff;
+         if (offset & 0x40000000)
+           offset |= ~ (bfd_vma) 0x7fffffff;
+       }
+      else
+       offset = rp->r_addend;
+
+      offset += sym->st_value;
+      prelval = offset - (arm_sec->sec->sh_addr + rp->r_offset);
+
+      word = (word & ~ (bfd_vma) 0x7fffffff) | (prelval & 0x7fffffff);
+      addr->section = sym->st_shndx;
+      addr->offset = offset;
+      break;
+    }
+
+  *wordp = word;
+  arm_sec->next_rela = rp;
+
+  return 1;
+}
+
+static void
+decode_arm_unwind (struct arm_unw_aux_info *aux,
+                  unsigned int word, unsigned int remaining,
+                  bfd_vma data_offset, Elf_Internal_Shdr *data_sec,
+                  struct arm_section *data_arm_sec)
+{
+  int per_index;
+  unsigned int more_words;
+  struct absaddr addr;
+
+#define ADVANCE                                                        \
+  if (remaining == 0 && more_words)                            \
+    {                                                          \
+      data_offset += 4;                                                \
+      if (!arm_section_get_word (aux, data_arm_sec, data_sec,  \
+                                data_offset, &word, &addr))    \
+       return;                                                 \
+      remaining = 4;                                           \
+      more_words--;                                            \
+    }                                                          \
+
+#define GET_OP(OP)                     \
+  ADVANCE;                             \
+  if (remaining)                       \
+    {                                  \
+      remaining--;                     \
+      (OP) = word >> 24;               \
+      word <<= 8;                      \
+    }                                  \
+  else                                 \
+    {                                  \
+      printf ("[Truncated opcode]\n"); \
+      return;                          \
+    }                                  \
+  printf (_("0x%02x "), OP)
+
+  if (remaining == 0)
+    {
+      /* Fetch the first word.  */
+      if (!arm_section_get_word (aux, data_arm_sec, data_sec, data_offset,
+                                &word, &addr))
+       return;
+      remaining = 4;
+    }
+
+  if ((word & 0x80000000) == 0)
+    {
+      /* Expand prel31 for personality routine.  */
+      bfd_vma fn;
+      const char *procname;
+
+      fn = word;
+      if (fn & 0x40000000)
+       fn |= ~ (bfd_vma) 0x7fffffff;
+      fn = fn + data_sec->sh_addr + data_offset;
+
+      printf (_("  Personality routine: "));
+      procname = arm_print_vma_and_name (aux, fn, addr);
+      fputc ('\n', stdout);
+
+      /* The GCC personality routines use the standard compact
+        encoding, starting with one byte giving the number of
+        words.  */
+      if (procname != NULL
+         && (const_strneq (procname, "__gcc_personality_v0")
+             || const_strneq (procname, "__gxx_personality_v0")
+             || const_strneq (procname, "__gcj_personality_v0")
+             || const_strneq (procname, "__gnu_objc_personality_v0")))
+       {
+         remaining = 0;
+         more_words = 1;
+         ADVANCE;
+         if (!remaining)
+           {
+             printf (_("  [Truncated data]\n"));
+             return;
+           }
+         more_words = word >> 24;
+         word <<= 8;
+         remaining--;
+       }
+      else
+       return;
+    }
+  else
+    {
+      
+      per_index = (word >> 24) & 0x7f;
+      if (per_index != 0 && per_index != 1 && per_index != 2)
+       {
+         printf (_("  [reserved compact index %d]\n"), per_index);
+         return;
+       }
+
+      printf (_("  Compact model %d\n"), per_index);
+      if (per_index == 0)
+       {
+         more_words = 0;
+         word <<= 8;
+         remaining--;
+       }
+      else
+       {
+         more_words = (word >> 16) & 0xff;
+         word <<= 16;
+         remaining -= 2;
+       }
+    }
+
+  /* Decode the unwinding instructions.  */
+  while (1)
+    {
+      unsigned int op, op2;
+
+      ADVANCE;
+      if (remaining == 0)
+       break;
+      remaining--;
+      op = word >> 24;
+      word <<= 8;
+
+      printf (_("  0x%02x "), op);
+
+      if ((op & 0xc0) == 0x00)
+       {
+         int offset = ((op & 0x3f) << 2) + 4;
+         printf (_("     vsp = vsp + %d"), offset);
+       }
+      else if ((op & 0xc0) == 0x40)
+       {
+         int offset = ((op & 0x3f) << 2) + 4;
+         printf (_("     vsp = vsp - %d"), offset);
+       }
+      else if ((op & 0xf0) == 0x80)
+       {
+         GET_OP (op2);
+         if (op == 0x80 && op2 == 0)
+           printf (_("Refuse to unwind"));
+         else
+           {
+             unsigned int mask = ((op & 0x0f) << 8) | op2;
+             int first = 1;
+             int i;
+             printf ("pop {");
+             for (i = 0; i < 12; i++)
+               if (mask & (1 << i))
+                 {
+                   if (first)
+                     first = 0;
+                   else
+                     printf (", ");
+                   printf ("r%d", 4 + i);
+                 }
+             printf ("}");
+           }
+       }
+      else if ((op & 0xf0) == 0x90)
+       {
+         if (op == 0x9d || op == 0x9f)
+           printf (_("     [Reserved]"));
+         else
+           printf (_("     vsp = r%d"), op & 0x0f);
+       }
+      else if ((op & 0xf0) == 0xa0)
+       {
+         int end = 4 + (op & 0x07);
+         int first = 1;
+         int i;
+         printf ("     pop {");
+         for (i = 4; i <= end; i++)
+           {
+             if (first)
+               first = 0;
+             else
+               printf (", ");
+             printf ("r%d", i);
+           }
+         if (op & 0x08)
+           {
+             if (first)
+               printf (", ");
+             printf ("r14");
+           }
+         printf ("}");
+       }
+      else if (op == 0xb0)
+       printf (_("     finish"));
+      else if (op == 0xb1)
+       {
+         GET_OP (op2);
+         if (op2 == 0 || (op2 & 0xf0) != 0)
+           printf (_("[Spare]"));
+         else
+           {
+             unsigned int mask = op2 & 0x0f;
+             int first = 1;
+             int i;
+             printf ("pop {");
+             for (i = 0; i < 12; i++)
+               if (mask & (1 << i))
+                 {
+                   if (first)
+                     first = 0;
+                   else
+                     printf (", ");
+                   printf ("r%d", i);
+                 }
+             printf ("}");
+           }
+       }
+      else if (op == 0xb2)
+       {
+         unsigned char buf[9];
+         unsigned int i, len;
+         unsigned long offset;
+         for (i = 0; i < sizeof (buf); i++)
+           {
+             GET_OP (buf[i]);
+             if ((buf[i] & 0x80) == 0)
+               break;
+           }
+         assert (i < sizeof (buf));
+         offset = read_uleb128 (buf, &len);
+         assert (len == i + 1);
+         offset = offset * 4 + 0x204;
+         printf (_("vsp = vsp + %ld"), offset);
+       }
+      else
+       {
+         if (op == 0xb3 || op == 0xc6 || op == 0xc7 || op == 0xc8 || op == 0xc9)
+           {
+             GET_OP (op2);
+             printf (_("[unsupported two-byte opcode]"));
+           }
+         else
+           {
+             printf (_("     [unsupported opcode]"));
+           }
+       }
+      printf ("\n");
+    }
+
+  /* Decode the descriptors.  Not implemented.  */
+}
+
+static void
+dump_arm_unwind (struct arm_unw_aux_info *aux, Elf_Internal_Shdr *exidx_sec)
+{
+  struct arm_section exidx_arm_sec, extab_arm_sec;
+  unsigned int i, exidx_len;
+
+  memset (&exidx_arm_sec, 0, sizeof (exidx_arm_sec));
+  memset (&extab_arm_sec, 0, sizeof (extab_arm_sec));
+  exidx_len = exidx_sec->sh_size / 8;
+
+  for (i = 0; i < exidx_len; i++)
+    {
+      unsigned int exidx_fn, exidx_entry;
+      struct absaddr fn_addr, entry_addr;
+      bfd_vma fn;
+
+      fputc ('\n', stdout);
+
+      if (!arm_section_get_word (aux, &exidx_arm_sec, exidx_sec,
+                                8 * i, &exidx_fn, &fn_addr)
+         || !arm_section_get_word (aux, &exidx_arm_sec, exidx_sec,
+                                   8 * i + 4, &exidx_entry, &entry_addr))
+       {
+         arm_free_section (&exidx_arm_sec);
+         arm_free_section (&extab_arm_sec);
+         return;
+       }
+
+      fn = exidx_fn & 0x7fffffff;
+      if (fn & 0x40000000)
+       fn |= ~ (bfd_vma) 0x7fffffff;
+      fn = fn + exidx_sec->sh_addr + 8 * i;
+
+      arm_print_vma_and_name (aux, fn, entry_addr);
+      fputs (": ", stdout);
+
+      if (exidx_entry == 1)
+       {
+         print_vma (exidx_entry, PREFIX_HEX);
+         fputs (" [cantunwind]\n", stdout);
+       }
+      else if (exidx_entry & 0x80000000)
+       {
+         print_vma (exidx_entry, PREFIX_HEX);
+         fputc ('\n', stdout);
+         decode_arm_unwind (aux, exidx_entry, 4, 0, NULL, NULL);
+       }
+      else
+       {
+         bfd_vma table, table_offset = 0;
+         Elf_Internal_Shdr *table_sec;
+
+         fputs ("@", stdout);
+         table = exidx_entry;
+         if (table & 0x40000000)
+           table |= ~ (bfd_vma) 0x7fffffff;
+         table = table + exidx_sec->sh_addr + 8 * i + 4;
+         print_vma (table, PREFIX_HEX);
+         printf ("\n");
+
+         /* Locate the matching .ARM.extab.  */
+         if (entry_addr.section != SHN_UNDEF
+             && entry_addr.section < elf_header.e_shnum)
+           {
+             table_sec = section_headers + entry_addr.section;
+             table_offset = entry_addr.offset;
+           }
+         else
+           {
+             table_sec = find_section_by_address (table);
+             if (table_sec != NULL)
+               table_offset = table - table_sec->sh_addr;
+           }
+         if (table_sec == NULL)
+           {
+             warn (_("Could not locate .ARM.extab section containing 0x%lx.\n"),
+                   (unsigned long) table);
+             continue;
+           }
+         decode_arm_unwind (aux, 0, 0, table_offset, table_sec,
+                            &extab_arm_sec);
+       }
+    }
+
+  printf ("\n");
+
+  arm_free_section (&exidx_arm_sec);
+  arm_free_section (&extab_arm_sec);
+}
+
+static int
+arm_process_unwind (FILE *file)
+{
+  struct arm_unw_aux_info aux;
+  Elf_Internal_Shdr *unwsec = NULL;
+  Elf_Internal_Shdr *strsec;
+  Elf_Internal_Shdr *sec;
+  unsigned long i;
+
+  memset (& aux, 0, sizeof (aux));
+  aux.file = file;
+
+  if (string_table == NULL)
+    return 1;
+
+  for (i = 0, sec = section_headers; i < elf_header.e_shnum; ++i, ++sec)
+    {
+      if (sec->sh_type == SHT_SYMTAB && sec->sh_link < elf_header.e_shnum)
+       {
+         aux.nsyms = sec->sh_size / sec->sh_entsize;
+         aux.symtab = GET_ELF_SYMBOLS (file, sec);
+
+         strsec = section_headers + sec->sh_link;
+         aux.strtab = get_data (NULL, file, strsec->sh_offset,
+                                1, strsec->sh_size, _("string table"));
+         aux.strtab_size = aux.strtab != NULL ? strsec->sh_size : 0;
+       }
+      else if (sec->sh_type == SHT_ARM_EXIDX)
+       unwsec = sec;
+    }
+
+  if (!unwsec)
+    printf (_("\nThere are no unwind sections in this file.\n"));
+
+  for (i = 0, sec = section_headers; i < elf_header.e_shnum; ++i, ++sec)
+    {
+      if (sec->sh_type == SHT_ARM_EXIDX)
+       {
+         printf (_("\nUnwind table index '%s' at offset 0x%lx contains %lu entries:\n"),
+                 SECTION_NAME (sec),
+                 (unsigned long) sec->sh_offset,
+                 (unsigned long) (sec->sh_size / (2 * eh_addr_size)));
+
+         dump_arm_unwind (&aux, sec);
+       }
+    }
+
+  if (aux.symtab)
+    free (aux.symtab);
+  if (aux.strtab)
+    free ((char *) aux.strtab);
+
+  return 1;
+}
+
 static int
 process_unwind (FILE * file)
 {
@@ -5767,6 +6488,7 @@ process_unwind (FILE * file)
     int (* handler)(FILE *);
   } handlers[] =
   {
+    { EM_ARM, arm_process_unwind },
     { EM_IA_64, ia64_process_unwind },
     { EM_PARISC, hppa_process_unwind },
     { 0, 0 }
@@ -6652,7 +7374,15 @@ get_ver_flags (unsigned int flags)
       strcat (buff, "WEAK ");
     }
 
-  if (flags & ~(VER_FLG_BASE | VER_FLG_WEAK))
+  if (flags & VER_FLG_INFO)
+    {
+      if (flags & (VER_FLG_BASE|VER_FLG_WEAK))
+       strcat (buff, "| ");
+
+      strcat (buff, "INFO ");
+    }
+
+  if (flags & ~(VER_FLG_BASE | VER_FLG_WEAK | VER_FLG_INFO))
     strcat (buff, "| <unknown>");
 
   return buff;
@@ -8178,6 +8908,8 @@ is_32bit_abs_reloc (unsigned int reloc_type)
        || reloc_type == 23; /* R_SPARC_UA32.  */
     case EM_SPU:
       return reloc_type == 6; /* R_SPU_ADDR32 */
+    case EM_TI_C6000:
+      return reloc_type == 1; /* R_C6000_ABS32.  */
     case EM_CYGNUS_V850:
     case EM_V850:
       return reloc_type == 6; /* R_V850_ABS32.  */
@@ -8361,6 +9093,8 @@ is_16bit_abs_reloc (unsigned int reloc_type)
     case EM_ALTERA_NIOS2:
     case EM_NIOS32:
       return reloc_type == 9; /* R_NIOS_16.  */
+    case EM_TI_C6000:
+      return reloc_type == 2; /* R_C6000_ABS16.  */
     case EM_XC16X:
     case EM_C166:
       return reloc_type == 2; /* R_XC16C_ABS_16.  */
@@ -8397,6 +9131,7 @@ is_none_reloc (unsigned int reloc_type)
     case EM_L1OM:    /* R_X86_64_NONE.  */
     case EM_MN10300: /* R_MN10300_NONE.  */
     case EM_M32R:    /* R_M32R_NONE.  */
+    case EM_TI_C6000:/* R_C6000_NONE.  */
     case EM_XC16X:
     case EM_C166:    /* R_XC16X_NONE.  */
       return reloc_type == 0;
@@ -9071,7 +9806,8 @@ static const char * arm_attr_tag_THUMB_ISA_use[] =
 static const char * arm_attr_tag_VFP_arch[] =
   {"No", "VFPv1", "VFPv2", "VFPv3", "VFPv3-D16", "VFPv4", "VFPv4-D16"};
 static const char * arm_attr_tag_WMMX_arch[] = {"No", "WMMXv1", "WMMXv2"};
-static const char * arm_attr_tag_Advanced_SIMD_arch[] = {"No", "NEONv1"};
+static const char * arm_attr_tag_Advanced_SIMD_arch[] = 
+  {"No", "NEONv1", "NEONv1 with Fused-MAC"};
 static const char * arm_attr_tag_PCS_config[] =
   {"None", "Bare platform", "Linux application", "Linux DSO", "PalmOS 2004",
    "PalmOS (reserved)", "SymbianOS 2004", "SymbianOS (reserved)"};
@@ -9114,10 +9850,17 @@ static const char * arm_attr_tag_VFP_HP_extension[] =
   {"Not Allowed", "Allowed"};
 static const char * arm_attr_tag_ABI_FP_16bit_format[] =
   {"None", "IEEE 754", "Alternative Format"};
+static const char * arm_attr_tag_MPextension_use[] = 
+  {"Not Allowed", "Allowed"};
+static const char * arm_attr_tag_DIV_use[] =
+  {"Allowed in Thumb-ISA, v7-R or v7-M", "Not allowed", 
+    "Allowed in v7-A with integer division extension"};
 static const char * arm_attr_tag_T2EE_use[] = {"Not Allowed", "Allowed"};
 static const char * arm_attr_tag_Virtualization_use[] =
+  {"Not Allowed", "TrustZone", "Virtualization Extensions", 
+    "TrustZone and Virtualization Extensions"};
+static const char * arm_attr_tag_MPextension_use_legacy[] = 
   {"Not Allowed", "Allowed"};
-static const char * arm_attr_tag_MPextension_use[] = {"Not Allowed", "Allowed"};
 
 #define LOOKUP(id, name) \
   {id, #name, 0x80 | ARRAY_SIZE(arm_attr_tag_##name), arm_attr_tag_##name}
@@ -9155,42 +9898,17 @@ static arm_attr_public_tag arm_attr_public_tags[] =
   LOOKUP(34, CPU_unaligned_access),
   LOOKUP(36, VFP_HP_extension),
   LOOKUP(38, ABI_FP_16bit_format),
+  LOOKUP(42, MPextension_use),
+  LOOKUP(44, DIV_use),
   {64, "nodefaults", 0, NULL},
   {65, "also_compatible_with", 0, NULL},
   LOOKUP(66, T2EE_use),
   {67, "conformance", 1, NULL},
   LOOKUP(68, Virtualization_use),
-  LOOKUP(70, MPextension_use)
+  LOOKUP(70, MPextension_use_legacy)
 };
 #undef LOOKUP
 
-/* Read an unsigned LEB128 encoded value from p.  Set *PLEN to the number of
-   bytes read.  */
-
-static unsigned int
-read_uleb128 (unsigned char * p, unsigned int * plen)
-{
-  unsigned char c;
-  unsigned int val;
-  int shift;
-  int len;
-
-  val = 0;
-  shift = 0;
-  len = 0;
-  do
-    {
-      c = *(p++);
-      len++;
-      val |= ((unsigned int)c & 0x7f) << shift;
-      shift += 7;
-    }
-  while (c & 0x80);
-
-  *plen = len;
-  return val;
-}
-
 static unsigned char *
 display_arm_attribute (unsigned char * p)
 {
This page took 0.035912 seconds and 4 git commands to generate.