Blame sysdeps/microblaze/backtrace_linux.c

Packit 6c4009
/* Copyright (C) 2005-2018 Free Software Foundation, Inc.
Packit 6c4009
Packit 6c4009
   This file is part of the GNU C Library.
Packit 6c4009
Packit 6c4009
   The GNU C Library is free software; you can redistribute it and/or
Packit 6c4009
   modify it under the terms of the GNU Lesser General Public License as
Packit 6c4009
   published by the Free Software Foundation; either version 2.1 of the
Packit 6c4009
   License, or (at your option) any later version.
Packit 6c4009
Packit 6c4009
   The GNU C Library is distributed in the hope that it will be useful,
Packit 6c4009
   but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit 6c4009
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit 6c4009
   Lesser General Public License for more details.
Packit 6c4009
Packit 6c4009
   You should have received a copy of the GNU Lesser General Public
Packit 6c4009
   License along with the GNU C Library; if not, see
Packit 6c4009
   <http://www.gnu.org/licenses/>.  */
Packit 6c4009
Packit 6c4009
#include <stddef.h>
Packit 6c4009
#include <asm/sigcontext.h>
Packit 6c4009
#include <linux/signal.h>
Packit 6c4009
#include <asm-generic/ucontext.h>
Packit 6c4009
#include <asm/unistd.h>
Packit 6c4009
Packit 6c4009
int
Packit 6c4009
_identify_sighandler (unsigned long fp, unsigned long pc,
Packit 6c4009
                      unsigned long *pprev_fp, unsigned long *pprev_pc,
Packit 6c4009
                      unsigned long *retaddr)
Packit 6c4009
{
Packit 6c4009
  unsigned long *tramp = 0;
Packit 6c4009
  struct ucontext *uc;
Packit 6c4009
Packit 6c4009
  if (*retaddr == 0)
Packit 6c4009
    {
Packit 6c4009
      /* Kernel inserts the tramp between the signal handler frame and the
Packit 6c4009
         caller frame in signal handling.  */
Packit 6c4009
      tramp = (unsigned long *) pc;
Packit 6c4009
      tramp += 2;
Packit 6c4009
      if ((*tramp == (0x31800000 | __NR_rt_sigreturn))
Packit 6c4009
          && (*(tramp+1) == 0xb9cc0008))
Packit 6c4009
        {
Packit 6c4009
          /* Signal handler function argument are:
Packit 6c4009
             int sig_num, siginfo_t * info, void * ucontext
Packit 6c4009
             therefore ucontext is the 3rd argument.  */
Packit 6c4009
          unsigned long ucptr = ((unsigned long) tramp
Packit 6c4009
                                 - sizeof (struct ucontext));
Packit 6c4009
          uc = (struct ucontext *) ucptr;
Packit 6c4009
          *pprev_pc = uc->uc_mcontext.regs.pc;
Packit 6c4009
          /* Need to record the return address since the return address of the
Packit 6c4009
             function which causes this signal may not be recorded in the
Packit 6c4009
             stack.  */
Packit 6c4009
          *pprev_fp = uc->uc_mcontext.regs.r1;
Packit 6c4009
          *retaddr = uc->uc_mcontext.regs.r15;
Packit 6c4009
          /* It is a signal handler.  */
Packit 6c4009
          return 1;
Packit 6c4009
        }
Packit 6c4009
    }
Packit 6c4009
  return 0;
Packit 6c4009
}