Blame backends/ppc_initreg.c

Packit Service 97d2fb
/* Fetch live process registers from TID.
Packit Service 97d2fb
   Copyright (C) 2013 Red Hat, Inc.
Packit Service 97d2fb
   This file is part of elfutils.
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 <stdlib.h>
Packit Service 97d2fb
#if defined(__powerpc__) && defined(__linux__)
Packit Service 97d2fb
# include <sys/ptrace.h>
Packit Service 97d2fb
# include <sys/user.h>
Packit Service 97d2fb
#endif
Packit Service 97d2fb
Packit Service 97d2fb
#include "system.h"
Packit Service 97d2fb
Packit Service 97d2fb
#define BACKEND ppc_
Packit Service 97d2fb
#include "libebl_CPU.h"
Packit Service 97d2fb
Packit Service 97d2fb
bool
Packit Service 97d2fb
ppc_dwarf_to_regno (Ebl *ebl __attribute__ ((unused)), unsigned *regno)
Packit Service 97d2fb
{
Packit Service 97d2fb
  switch (*regno)
Packit Service 97d2fb
  {
Packit Service 97d2fb
    case 108:
Packit Service 97d2fb
      // LR uses both 65 and 108 numbers, there is no consistency for it.
Packit Service 97d2fb
      *regno = 65;
Packit Service 97d2fb
      return true;
Packit Service 97d2fb
    case 0 ... 107:
Packit Service 97d2fb
    case 109 ... (114 - 1) -1:
Packit Service 97d2fb
      return true;
Packit Service 97d2fb
    case 1200 ... 1231:
Packit Service 97d2fb
      *regno = *regno - 1200 + (114 - 1);
Packit Service 97d2fb
      return true;
Packit Service 97d2fb
    default:
Packit Service 97d2fb
      return false;
Packit Service 97d2fb
  }
Packit Service 97d2fb
  abort ();
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
__typeof (ppc_dwarf_to_regno)
Packit Service 97d2fb
     ppc64_dwarf_to_regno
Packit Service 97d2fb
     __attribute__ ((alias ("ppc_dwarf_to_regno")));
Packit Service 97d2fb
Packit Service 97d2fb
bool
Packit Service 97d2fb
ppc_set_initial_registers_tid (pid_t tid __attribute__ ((unused)),
Packit Service 97d2fb
			  ebl_tid_registers_t *setfunc __attribute__ ((unused)),
Packit Service 97d2fb
			       void *arg __attribute__ ((unused)))
Packit Service 97d2fb
{
Packit Service 97d2fb
#if !defined(__powerpc__) || !defined(__linux__)
Packit Service 97d2fb
  return false;
Packit Service 97d2fb
#else /* __powerpc__ */
Packit Service 97d2fb
  union
Packit Service 97d2fb
    {
Packit Service 97d2fb
      struct pt_regs r;
Packit Service 97d2fb
      long l[sizeof (struct pt_regs) / sizeof (long)];
Packit Service 97d2fb
    }
Packit Service 97d2fb
  user_regs;
Packit Service 97d2fb
  eu_static_assert (sizeof (struct pt_regs) % sizeof (long) == 0);
Packit Service 97d2fb
  /* PTRACE_GETREGS is EIO on kernel-2.6.18-308.el5.ppc64.  */
Packit Service 97d2fb
  errno = 0;
Packit Service 97d2fb
  for (unsigned regno = 0; regno < sizeof (user_regs) / sizeof (long);
Packit Service 97d2fb
       regno++)
Packit Service 97d2fb
    {
Packit Service 97d2fb
      user_regs.l[regno] = ptrace (PTRACE_PEEKUSER, tid,
Packit Service 97d2fb
				   (void *) (uintptr_t) (regno
Packit Service 97d2fb
							 * sizeof (long)),
Packit Service 97d2fb
				   NULL);
Packit Service 97d2fb
      if (errno != 0)
Packit Service 97d2fb
	return false;
Packit Service 97d2fb
    }
Packit Service 97d2fb
#define GPRS (sizeof (user_regs.r.gpr) / sizeof (*user_regs.r.gpr))
Packit Service 97d2fb
  Dwarf_Word dwarf_regs[GPRS];
Packit Service 97d2fb
  for (unsigned gpr = 0; gpr < GPRS; gpr++)
Packit Service 97d2fb
    dwarf_regs[gpr] = user_regs.r.gpr[gpr];
Packit Service 97d2fb
  if (! setfunc (0, GPRS, dwarf_regs, arg))
Packit Service 97d2fb
    return false;
Packit Service 97d2fb
  dwarf_regs[0] = user_regs.r.link;
Packit Service 97d2fb
  // LR uses both 65 and 108 numbers, there is no consistency for it.
Packit Service 97d2fb
  if (! setfunc (65, 1, dwarf_regs, arg))
Packit Service 97d2fb
    return false;
Packit Service 97d2fb
  /* Registers like msr, ctr, xer, dar, dsisr etc. are probably irrelevant
Packit Service 97d2fb
     for CFI.  */
Packit Service 97d2fb
  dwarf_regs[0] = user_regs.r.nip;
Packit Service 97d2fb
  return setfunc (-1, 1, dwarf_regs, arg);
Packit Service 97d2fb
#endif /* __powerpc__ */
Packit Service 97d2fb
}
Packit Service 97d2fb
Packit Service 97d2fb
__typeof (ppc_set_initial_registers_tid)
Packit Service 97d2fb
     ppc64_set_initial_registers_tid
Packit Service 97d2fb
     __attribute__ ((alias ("ppc_set_initial_registers_tid")));