|
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 |
}
|