Blame backends/riscv_symbol.c

Packit 032894
/* RISC-V specific symbolic name handling.
Packit 032894
   This file is part of elfutils.
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 <assert.h>
Packit 032894
#include <elf.h>
Packit 032894
#include <stddef.h>
Packit 032894
#include <string.h>
Packit 032894
Packit 032894
#define BACKEND riscv_
Packit 032894
#include "libebl_CPU.h"
Packit 032894
Packit 032894
Packit 032894
/* Check for the simple reloc types.  */
Packit 032894
Elf_Type
Packit 032894
riscv_reloc_simple_type (Ebl *ebl __attribute__ ((unused)), int type,
Packit 032894
			 int *addsub)
Packit 032894
{
Packit 032894
  switch (type)
Packit 032894
    {
Packit 032894
    case R_RISCV_SET8:
Packit 032894
      return ELF_T_BYTE;
Packit 032894
    case R_RISCV_SET16:
Packit 032894
      return ELF_T_HALF;
Packit 032894
    case R_RISCV_32:
Packit 032894
    case R_RISCV_SET32:
Packit 032894
      return ELF_T_WORD;
Packit 032894
    case R_RISCV_64:
Packit 032894
      return ELF_T_XWORD;
Packit 032894
    case R_RISCV_ADD16:
Packit 032894
      *addsub = 1;
Packit 032894
      return ELF_T_HALF;
Packit 032894
    case R_RISCV_SUB16:
Packit 032894
      *addsub = -1;
Packit 032894
      return ELF_T_HALF;
Packit 032894
    case R_RISCV_ADD32:
Packit 032894
      *addsub = 1;
Packit 032894
      return ELF_T_WORD;
Packit 032894
    case R_RISCV_SUB32:
Packit 032894
      *addsub = -1;
Packit 032894
      return ELF_T_WORD;
Packit 032894
    case R_RISCV_ADD64:
Packit 032894
      *addsub = 1;
Packit 032894
      return ELF_T_XWORD;
Packit 032894
    case R_RISCV_SUB64:
Packit 032894
      *addsub = -1;
Packit 032894
      return ELF_T_XWORD;
Packit 032894
    default:
Packit 032894
      return ELF_T_NUM;
Packit 032894
    }
Packit 032894
}
Packit 032894
Packit 032894
/* Check whether machine flags are valid.  */
Packit 032894
bool
Packit 032894
riscv_machine_flag_check (GElf_Word flags)
Packit 032894
{
Packit 032894
  return ((flags &~ (EF_RISCV_RVC
Packit 032894
		     | EF_RISCV_FLOAT_ABI)) == 0);
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
riscv_check_special_symbol (Elf *elf, const GElf_Sym *sym,
Packit 032894
			    const char *name, const GElf_Shdr *destshdr)
Packit 032894
{
Packit 032894
  if (name == NULL)
Packit 032894
    return false;
Packit 032894
Packit 032894
  size_t shstrndx;
Packit 032894
  if (elf_getshdrstrndx (elf, &shstrndx) != 0)
Packit 032894
    return false;
Packit 032894
  const char *sname = elf_strptr (elf, shstrndx, destshdr->sh_name);
Packit 032894
  if (sname == NULL)
Packit 032894
    return false;
Packit 032894
Packit 032894
  /* _GLOBAL_OFFSET_TABLE_ points to the start of the .got section, but it
Packit 032894
     is preceded by the .got.plt section in the output .got section.  */
Packit 032894
  if (strcmp (name, "_GLOBAL_OFFSET_TABLE_") == 0)
Packit 032894
    return (strcmp (sname, ".got") == 0
Packit 032894
	    && sym->st_value >= destshdr->sh_addr
Packit 032894
	    && sym->st_value < destshdr->sh_addr + destshdr->sh_size);
Packit 032894
Packit 032894
  /* __global_pointer$ points to the .sdata section with an offset of
Packit 032894
     0x800.  It might however fall in the .got section, in which case we
Packit 032894
     cannot check the offset.  The size always should be zero.  */
Packit 032894
  if (strcmp (name, "__global_pointer$") == 0)
Packit 032894
    return (((strcmp (sname, ".sdata") == 0
Packit 032894
	      && sym->st_value == destshdr->sh_addr + 0x800)
Packit 032894
	     || strcmp (sname, ".got") == 0)
Packit 032894
	    && sym->st_size == 0);
Packit 032894
Packit 032894
  return false;
Packit 032894
}