Blame backends/ppc_symbol.c

Packit Service 97d2fb
/* PPC specific symbolic name handling.
Packit Service 97d2fb
   Copyright (C) 2004, 2005, 2007, 2014, 2015 Red Hat, Inc.
Packit Service 97d2fb
   This file is part of elfutils.
Packit Service 97d2fb
   Written by Ulrich Drepper <drepper@redhat.com>, 2004.
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 <assert.h>
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		ppc_
Packit Service 97d2fb
#include "libebl_CPU.h"
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
/* Check for the simple reloc types.  */
Packit Service 97d2fb
Elf_Type
Packit Service 97d2fb
ppc_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_PPC_ADDR32:
Packit Service 97d2fb
    case R_PPC_UADDR32:
Packit Service 97d2fb
      return ELF_T_WORD;
Packit Service 97d2fb
    case R_PPC_UADDR16:
Packit Service 97d2fb
      return ELF_T_HALF;
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 machine flags are valid.  */
Packit Service 97d2fb
bool
Packit Service 97d2fb
ppc_machine_flag_check (GElf_Word flags)
Packit Service 97d2fb
{
Packit Service 97d2fb
  return ((flags &~ (EF_PPC_EMB
Packit Service 97d2fb
		     | EF_PPC_RELOCATABLE
Packit Service 97d2fb
		     | EF_PPC_RELOCATABLE_LIB)) == 0);
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
const char *
Packit Service 97d2fb
ppc_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_PPC_GOT:
Packit Service 97d2fb
      return "PPC_GOT";
Packit Service 97d2fb
    case DT_PPC_OPT:
Packit Service 97d2fb
      return "PPC_OPT";
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
Packit Service 97d2fb
bool
Packit Service 97d2fb
ppc_dynamic_tag_check (int64_t tag)
Packit Service 97d2fb
{
Packit Service 97d2fb
  return (tag == DT_PPC_GOT
Packit Service 97d2fb
	  || tag == DT_PPC_OPT);
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
/* Look for DT_PPC_GOT.  */
Packit Service 97d2fb
static bool
Packit Service 97d2fb
find_dyn_got (Elf *elf, GElf_Addr *addr)
Packit Service 97d2fb
{
Packit Service 97d2fb
  size_t phnum;
Packit Service 97d2fb
  if (elf_getphdrnum (elf, &phnum) != 0)
Packit Service 97d2fb
    return false;
Packit Service 97d2fb
Packit Service 97d2fb
  for (size_t i = 0; i < phnum; ++i)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      GElf_Phdr phdr_mem;
Packit Service 97d2fb
      GElf_Phdr *phdr = gelf_getphdr (elf, i, &phdr_mem);
Packit Service 97d2fb
      if (phdr == NULL || phdr->p_type != PT_DYNAMIC)
Packit Service 97d2fb
	continue;
Packit Service 97d2fb
Packit Service 97d2fb
      Elf_Scn *scn = gelf_offscn (elf, phdr->p_offset);
Packit Service 97d2fb
      GElf_Shdr shdr_mem;
Packit Service 97d2fb
      GElf_Shdr *shdr = gelf_getshdr (scn, &shdr_mem);
Packit Service 97d2fb
      Elf_Data *data = elf_getdata (scn, NULL);
Packit Service 97d2fb
      if (shdr != NULL && shdr->sh_type == SHT_DYNAMIC && data != NULL
Packit Service 97d2fb
	  && shdr->sh_entsize != 0)
Packit Service 97d2fb
	for (unsigned int j = 0; j < shdr->sh_size / shdr->sh_entsize; ++j)
Packit Service 97d2fb
	  {
Packit Service 97d2fb
	    GElf_Dyn dyn_mem;
Packit Service 97d2fb
	    GElf_Dyn *dyn = gelf_getdyn (data, j, &dyn_mem);
Packit Service 97d2fb
	    if (dyn != NULL && dyn->d_tag == DT_PPC_GOT)
Packit Service 97d2fb
	      {
Packit Service 97d2fb
		*addr = dyn->d_un.d_ptr;
Packit Service 97d2fb
		return true;
Packit Service 97d2fb
	      }
Packit Service 97d2fb
	  }
Packit Service 97d2fb
Packit Service 97d2fb
      /* There is only one PT_DYNAMIC entry.  */
Packit Service 97d2fb
      break;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  return false;
Packit Service 97d2fb
}
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
ppc_check_special_symbol (Elf *elf, const GElf_Sym *sym,
Packit Service 97d2fb
			  const char *name, const GElf_Shdr *destshdr)
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
    {
Packit Service 97d2fb
      /* In -msecure-plt mode, DT_PPC_GOT is present and must match.  */
Packit Service 97d2fb
      GElf_Addr gotaddr;
Packit Service 97d2fb
      if (find_dyn_got (elf, &gotaddr))
Packit Service 97d2fb
	return sym->st_value == gotaddr;
Packit Service 97d2fb
Packit Service 97d2fb
      /* In -mbss-plt mode, any place in the section is valid.  */
Packit Service 97d2fb
      return true;
Packit Service 97d2fb
    }
Packit Service 97d2fb
Packit Service 97d2fb
  size_t shstrndx;
Packit Service 97d2fb
  if (elf_getshdrstrndx (elf, &shstrndx) != 0)
Packit Service 97d2fb
    return false;
Packit Service 97d2fb
  const char *sname = elf_strptr (elf, shstrndx, destshdr->sh_name);
Packit Service 97d2fb
  if (sname == NULL)
Packit Service 97d2fb
    return false;
Packit Service 97d2fb
Packit Service 97d2fb
  /* Small data area.  Normally points to .sdata, in which case we
Packit Service 97d2fb
     check it is at an offset of 0x8000.  It might however fall in the
Packit Service 97d2fb
     .data section, in which case we cannot check the offset.  The
Packit Service 97d2fb
     size always should be zero.  */
Packit Service 97d2fb
  if (strcmp (name, "_SDA_BASE_") == 0)
Packit Service 97d2fb
    return (((strcmp (sname, ".sdata") == 0
Packit Service 97d2fb
	      && sym->st_value == destshdr->sh_addr + 0x8000)
Packit Service 97d2fb
	     || strcmp (sname, ".data") == 0)
Packit Service 97d2fb
	    && sym->st_size == 0);
Packit Service 97d2fb
Packit Service 97d2fb
  if (strcmp (name, "_SDA2_BASE_") == 0)
Packit Service 97d2fb
    return (strcmp (sname, ".sdata2") == 0
Packit Service 97d2fb
	    && sym->st_value == destshdr->sh_addr + 0x8000
Packit Service 97d2fb
	    && sym->st_size == 0);
Packit Service 97d2fb
Packit Service 97d2fb
  return false;
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
Packit Service 97d2fb
/* Check if backend uses a bss PLT in this file.  */
Packit Service 97d2fb
bool
Packit Service 97d2fb
ppc_bss_plt_p (Elf *elf)
Packit Service 97d2fb
{
Packit Service 97d2fb
  GElf_Addr addr;
Packit Service 97d2fb
  return ! find_dyn_got (elf, &addr);
Packit Service 97d2fb
}