Blame backends/aarch64_initreg.c

Packit 032894
/* Fetch live process registers from TID.
Packit 032894
   Copyright (C) 2013, 2014 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 "system.h"
Packit 032894
#include <assert.h>
Packit 032894
#if defined(__aarch64__) && defined(__linux__)
Packit 032894
# include <linux/uio.h>
Packit 032894
# include <sys/user.h>
Packit 032894
# include <sys/ptrace.h>
Packit 032894
/* Deal with old glibc defining user_pt_regs instead of user_regs_struct.  */
Packit 032894
# ifndef HAVE_SYS_USER_REGS
Packit 032894
#  define user_regs_struct user_pt_regs
Packit 032894
#  define user_fpsimd_struct user_fpsimd_state
Packit 032894
# endif
Packit 032894
#endif
Packit 032894
Packit 032894
#define BACKEND aarch64_
Packit 032894
#include "libebl_CPU.h"
Packit 032894
Packit 032894
bool
Packit 032894
aarch64_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(__aarch64__) || !defined(__linux__)
Packit 032894
  return false;
Packit 032894
#else /* __aarch64__ */
Packit 032894
Packit 032894
  /* General registers.  */
Packit 032894
  struct user_regs_struct gregs;
Packit 032894
  struct iovec iovec;
Packit 032894
  iovec.iov_base = &gregs;
Packit 032894
  iovec.iov_len = sizeof (gregs);
Packit 032894
  if (ptrace (PTRACE_GETREGSET, tid, NT_PRSTATUS, &iovec) != 0)
Packit 032894
    return false;
Packit 032894
Packit 032894
  /* X0..X30 plus SP.  */
Packit 032894
  if (! setfunc (0, 32, (Dwarf_Word *) &gregs.regs[0], arg))
Packit 032894
    return false;
Packit 032894
Packit 032894
  /* PC.  */
Packit 032894
  if (! setfunc (-1, 1, (Dwarf_Word *) &gregs.pc, arg))
Packit 032894
    return false;
Packit 032894
Packit 032894
  /* ELR cannot be found.  */
Packit 032894
Packit 032894
  /* FP registers (only 64bits are used).  */
Packit 032894
  struct user_fpsimd_struct fregs;
Packit 032894
  iovec.iov_base = &fregs;
Packit 032894
  iovec.iov_len = sizeof (fregs);
Packit 032894
  if (ptrace (PTRACE_GETREGSET, tid, NT_FPREGSET, &iovec) != 0)
Packit 032894
    return false;
Packit 032894
Packit 032894
  Dwarf_Word dwarf_fregs[32];
Packit 032894
  for (int r = 0; r < 32; r++)
Packit 032894
    dwarf_fregs[r] = fregs.vregs[r] & 0xFFFFFFFF;
Packit 032894
Packit 032894
  if (! setfunc (64, 32, dwarf_fregs, arg))
Packit 032894
    return false;
Packit 032894
Packit 032894
  return true;
Packit 032894
#endif /* __aarch64__ */
Packit 032894
}