LCOV - code coverage report
Current view: top level - backends - s390_unwind.c (source / functions) Hit Total Coverage
Test: elfutils-0.178 Lines: 7 48 14.6 %
Date: 2019-11-26 23:55:16 Functions: 1 1 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /* Get previous frame state for an existing frame state.
       2             :    Copyright (C) 2013 Red Hat, Inc.
       3             :    This file is part of elfutils.
       4             : 
       5             :    This file is free software; you can redistribute it and/or modify
       6             :    it under the terms of either
       7             : 
       8             :      * the GNU Lesser General Public License as published by the Free
       9             :        Software Foundation; either version 3 of the License, or (at
      10             :        your option) any later version
      11             : 
      12             :    or
      13             : 
      14             :      * the GNU General Public License as published by the Free
      15             :        Software Foundation; either version 2 of the License, or (at
      16             :        your option) any later version
      17             : 
      18             :    or both in parallel, as here.
      19             : 
      20             :    elfutils is distributed in the hope that it will be useful, but
      21             :    WITHOUT ANY WARRANTY; without even the implied warranty of
      22             :    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
      23             :    General Public License for more details.
      24             : 
      25             :    You should have received copies of the GNU General Public License and
      26             :    the GNU Lesser General Public License along with this program.  If
      27             :    not, see <http://www.gnu.org/licenses/>.  */
      28             : 
      29             : #ifdef HAVE_CONFIG_H
      30             : # include <config.h>
      31             : #endif
      32             : 
      33             : #include <stdlib.h>
      34             : #include <assert.h>
      35             : 
      36             : #define BACKEND s390_
      37             : #include "libebl_CPU.h"
      38             : 
      39             : /* s390/s390x do not annotate signal handler frame by CFI.  It would be also
      40             :    difficult as PC points into a stub built on stack.  Function below is called
      41             :    only if unwinder could not find CFI.  Function then verifies the register
      42             :    state for this frame really belongs to a signal frame.  In such case it
      43             :    fetches original registers saved by the signal frame.  */
      44             : 
      45             : bool
      46           8 : s390_unwind (Ebl *ebl, Dwarf_Addr pc, ebl_tid_registers_t *setfunc,
      47             :              ebl_tid_registers_get_t *getfunc, ebl_pid_memory_read_t *readfunc,
      48             :              void *arg, bool *signal_framep)
      49             : {
      50             :   /* Caller already assumed caller adjustment but S390 instructions are 4 bytes
      51             :      long.  Undo it.  */
      52           8 :   if ((pc & 0x3) != 0x3)
      53             :     return false;
      54           6 :   pc++;
      55             :   /* We can assume big-endian read here.  */
      56           6 :   Dwarf_Word instr;
      57           6 :   if (! readfunc (pc, &instr, arg))
      58             :     return false;
      59             :   /* Fetch only the very first two bytes.  */
      60           6 :   instr = (instr >> (ebl->class == ELFCLASS64 ? 48 : 16)) & 0xffff;
      61             :   /* See GDB s390_sigtramp_frame_sniffer.  */
      62             :   /* Check for 'svc' as the first instruction.  */
      63           6 :   if (((instr >> 8) & 0xff) != 0x0a)
      64             :     return false;
      65             :   /* Check for 'sigreturn' or 'rt_sigreturn' as the second instruction.  */
      66           0 :   if ((instr & 0xff) != 119 && (instr & 0xff) != 173)
      67             :     return false;
      68             :   /* See GDB s390_sigtramp_frame_unwind_cache.  */
      69           0 :   Dwarf_Word this_sp;
      70           0 :   if (! getfunc (0 + 15, 1, &this_sp, arg))
      71             :     return false;
      72           0 :   unsigned word_size = ebl->class == ELFCLASS64 ? 8 : 4;
      73           0 :   Dwarf_Addr next_cfa = this_sp + 16 * word_size + 32;
      74             :   /* "New-style RT frame" is not supported,
      75             :      assuming "Old-style RT frame and all non-RT frames".
      76             :      Pointer to the array of saved registers is at NEXT_CFA + 8.  */
      77           0 :   Dwarf_Word sigreg_ptr;
      78           0 :   if (! readfunc (next_cfa + 8, &sigreg_ptr, arg))
      79             :     return false;
      80             :   /* Skip PSW mask.  */
      81           0 :   sigreg_ptr += word_size;
      82             :   /* Read PSW address.  */
      83           0 :   Dwarf_Word val;
      84           0 :   if (! readfunc (sigreg_ptr, &val, arg))
      85             :     return false;
      86           0 :   if (! setfunc (-1, 1, &val, arg))
      87             :     return false;
      88           0 :   sigreg_ptr += word_size;
      89             :   /* Then the GPRs.  */
      90           0 :   Dwarf_Word gprs[16];
      91           0 :   for (int i = 0; i < 16; i++)
      92             :     {
      93           0 :       if (! readfunc (sigreg_ptr, &gprs[i], arg))
      94             :         return false;
      95           0 :       sigreg_ptr += word_size;
      96             :     }
      97             :   /* Then the ACRs.  Skip them, they are not used in CFI.  */
      98           0 :   for (int i = 0; i < 16; i++)
      99           0 :     sigreg_ptr += 4;
     100             :   /* The floating-point control word.  */
     101           0 :   sigreg_ptr += 8;
     102             :   /* And finally the FPRs.  */
     103           0 :   Dwarf_Word fprs[16];
     104           0 :   for (int i = 0; i < 16; i++)
     105             :     {
     106           0 :       if (! readfunc (sigreg_ptr, &val, arg))
     107             :         return false;
     108           0 :       if (ebl->class == ELFCLASS32)
     109             :         {
     110           0 :           Dwarf_Addr val_low;
     111           0 :           if (! readfunc (sigreg_ptr + 4, &val_low, arg))
     112           0 :             return false;
     113           0 :           val = (val << 32) | val_low;
     114             :         }
     115           0 :       fprs[i] = val;
     116           0 :       sigreg_ptr += 8;
     117             :     }
     118             :   /* If we have them, the GPR upper halves are appended at the end.  */
     119           0 :   if (ebl->class == ELFCLASS32)
     120             :     {
     121             :       /* Skip signal number.  */
     122           0 :       sigreg_ptr += 4;
     123           0 :       for (int i = 0; i < 16; i++)
     124             :         {
     125           0 :           if (! readfunc (sigreg_ptr, &val, arg))
     126             :             return false;
     127           0 :           Dwarf_Word val_low = gprs[i];
     128           0 :           val = (val << 32) | val_low;
     129           0 :           gprs[i] = val;
     130           0 :           sigreg_ptr += 4;
     131             :         }
     132             :     }
     133           0 :   if (! setfunc (0, 16, gprs, arg))
     134             :     return false;
     135           0 :   if (! setfunc (16, 16, fprs, arg))
     136             :     return false;
     137           0 :   *signal_framep = true;
     138           0 :   return true;
     139             : }

Generated by: LCOV version 1.13