Blame backends/alpha_symbol.c

Packit 032894
/* Alpha specific symbolic name handling.
Packit 032894
   Copyright (C) 2002-2011 Red Hat, Inc.
Packit 032894
   This file is part of elfutils.
Packit 032894
   Written by Ulrich Drepper <drepper@redhat.com>, 2002.
Packit 032894
Packit 032894
   This file is free software; you can redistribute it and/or modify
Packit 032894
   it under the terms of either
Packit 032894
Packit 032894
     * the GNU Lesser General Public License as published by the Free
Packit 032894
       Software Foundation; either version 3 of the License, or (at
Packit 032894
       your option) any later version
Packit 032894
Packit 032894
   or
Packit 032894
Packit 032894
     * the GNU General Public License as published by the Free
Packit 032894
       Software Foundation; either version 2 of the License, or (at
Packit 032894
       your option) any later version
Packit 032894
Packit 032894
   or both in parallel, as here.
Packit 032894
Packit 032894
   elfutils is distributed in the hope that it will be useful, but
Packit 032894
   WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 032894
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 032894
   General Public License for more details.
Packit 032894
Packit 032894
   You should have received copies of the GNU General Public License and
Packit 032894
   the GNU Lesser General Public License along with this program.  If
Packit 032894
   not, see <http://www.gnu.org/licenses/>.  */
Packit 032894
Packit 032894
#ifdef HAVE_CONFIG_H
Packit 032894
# include <config.h>
Packit 032894
#endif
Packit 032894
Packit 032894
#include <elf.h>
Packit 032894
#include <stddef.h>
Packit 032894
#include <string.h>
Packit 032894
Packit 032894
#define BACKEND		alpha_
Packit 032894
#include "libebl_CPU.h"
Packit 032894
Packit 032894
Packit 032894
const char *
Packit 032894
alpha_dynamic_tag_name (int64_t tag, char *buf __attribute__ ((unused)),
Packit 032894
			size_t len __attribute__ ((unused)))
Packit 032894
{
Packit 032894
  switch (tag)
Packit 032894
    {
Packit 032894
    case DT_ALPHA_PLTRO:
Packit 032894
      return "ALPHA_PLTRO";
Packit 032894
    default:
Packit 032894
      break;
Packit 032894
    }
Packit 032894
  return NULL;
Packit 032894
}
Packit 032894
Packit 032894
bool
Packit 032894
alpha_dynamic_tag_check (int64_t tag)
Packit 032894
{
Packit 032894
  return tag == DT_ALPHA_PLTRO;
Packit 032894
}
Packit 032894
Packit 032894
/* Check for the simple reloc types.  */
Packit 032894
Elf_Type
Packit 032894
alpha_reloc_simple_type (Ebl *ebl __attribute__ ((unused)), int type,
Packit 032894
			 int *addsub __attribute__ ((unused)))
Packit 032894
{
Packit 032894
  switch (type)
Packit 032894
    {
Packit 032894
    case R_ALPHA_REFLONG:
Packit 032894
      return ELF_T_WORD;
Packit 032894
    case R_ALPHA_REFQUAD:
Packit 032894
      return ELF_T_XWORD;
Packit 032894
    default:
Packit 032894
      return ELF_T_NUM;
Packit 032894
    }
Packit 032894
}
Packit 032894
Packit 032894
Packit 032894
/* Check whether SHF_MASKPROC flags are valid.  */
Packit 032894
bool
Packit 032894
alpha_machine_section_flag_check (GElf_Xword sh_flags)
Packit 032894
{
Packit 032894
  return (sh_flags &~ (SHF_ALPHA_GPREL)) == 0;
Packit 032894
}
Packit 032894
Packit 032894
bool
Packit 032894
alpha_check_special_section (Ebl *ebl,
Packit 032894
			     int ndx __attribute__ ((unused)),
Packit 032894
			     const GElf_Shdr *shdr,
Packit 032894
			     const char *sname __attribute__ ((unused)))
Packit 032894
{
Packit 032894
  if ((shdr->sh_flags
Packit 032894
       & (SHF_WRITE | SHF_EXECINSTR)) == (SHF_WRITE | SHF_EXECINSTR)
Packit 032894
      && shdr->sh_addr != 0)
Packit 032894
    {
Packit 032894
      /* This is ordinarily flagged, but is valid for an old-style PLT.
Packit 032894
Packit 032894
	 Look for the SHT_DYNAMIC section and the DT_PLTGOT tag in it.
Packit 032894
	 Its d_ptr should match the .plt section's sh_addr.  */
Packit 032894
Packit 032894
      Elf_Scn *scn = NULL;
Packit 032894
      while ((scn = elf_nextscn (ebl->elf, scn)) != NULL)
Packit 032894
	{
Packit 032894
	  GElf_Shdr scn_shdr;
Packit 032894
	  if (likely (gelf_getshdr (scn, &scn_shdr) != NULL)
Packit 032894
	      && scn_shdr.sh_type == SHT_DYNAMIC
Packit 032894
	      && scn_shdr.sh_entsize != 0)
Packit 032894
	    {
Packit 032894
	      GElf_Addr pltgot = 0;
Packit 032894
	      Elf_Data *data = elf_getdata (scn, NULL);
Packit 032894
	      if (data != NULL)
Packit 032894
		for (size_t i = 0; i < data->d_size / scn_shdr.sh_entsize; ++i)
Packit 032894
		  {
Packit 032894
		    GElf_Dyn dyn;
Packit 032894
		    if (unlikely (gelf_getdyn (data, i, &dyn) == NULL))
Packit 032894
		      break;
Packit 032894
		    if (dyn.d_tag == DT_PLTGOT)
Packit 032894
		      pltgot = dyn.d_un.d_ptr;
Packit 032894
		    else if (dyn.d_tag == DT_ALPHA_PLTRO && dyn.d_un.d_val != 0)
Packit 032894
		      return false; /* This PLT should not be writable.  */
Packit 032894
		  }
Packit 032894
	      return pltgot == shdr->sh_addr;
Packit 032894
	    }
Packit 032894
	}
Packit 032894
    }
Packit 032894
Packit 032894
  return false;
Packit 032894
}
Packit 032894
Packit 032894
/* Check whether given symbol's st_value and st_size are OK despite failing
Packit 032894
   normal checks.  */
Packit 032894
bool
Packit 032894
alpha_check_special_symbol (Elf *elf __attribute__ ((unused)),
Packit 032894
			    const GElf_Sym *sym __attribute__ ((unused)),
Packit 032894
			    const char *name,
Packit 032894
			    const GElf_Shdr *destshdr __attribute__ ((unused)))
Packit 032894
{
Packit 032894
  if (name == NULL)
Packit 032894
    return false;
Packit 032894
Packit 032894
  if (strcmp (name, "_GLOBAL_OFFSET_TABLE_") == 0)
Packit 032894
    /* On Alpha any place in the section is valid.  */
Packit 032894
    return true;
Packit 032894
Packit 032894
  return false;
Packit 032894
}
Packit 032894
Packit 032894
/* Check whether only valid bits are set on the st_other symbol flag.
Packit 032894
   Standard ST_VISIBILITY have already been masked off.  */
Packit 032894
bool
Packit 032894
alpha_check_st_other_bits (unsigned char st_other)
Packit 032894
{
Packit 032894
  return ((((st_other & STO_ALPHA_STD_GPLOAD) == STO_ALPHA_NOPV)
Packit 032894
	   || ((st_other & STO_ALPHA_STD_GPLOAD) == STO_ALPHA_STD_GPLOAD))
Packit 032894
	  && (st_other &~ STO_ALPHA_STD_GPLOAD) == 0);
Packit 032894
}