Blame backends/alpha_symbol.c

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