Blame backends/ppc_initreg.c

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