183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm/* libunwind - a platform-independent unwind library
283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm   Copyright (C) 2003-2005 Hewlett-Packard Co
383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmThis file is part of libunwind.
683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmPermission is hereby granted, free of charge, to any person obtaining
883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidma copy of this software and associated documentation files (the
983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm"Software"), to deal in the Software without restriction, including
1083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmwithout limitation the rights to use, copy, modify, merge, publish,
1183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmdistribute, sublicense, and/or sell copies of the Software, and to
1283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmpermit persons to whom the Software is furnished to do so, subject to
1383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmthe following conditions:
1483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
1583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmThe above copyright notice and this permission notice shall be
1683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmincluded in all copies or substantial portions of the Software.
1783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
1883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmTHE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
1983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmEXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
2083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmMERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
2183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmNONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
2283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmLIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
2383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmOF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
2483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmWITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.  */
2583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
2683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm#if UNW_TARGET_IA64
2783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
2883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm#include "libunwind_i.h"
2983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm#include "tdep-ia64/rse.h"
3083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
3183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmstatic inline int
3283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmbsp_match (unw_cursor_t *c, unw_word_t *wp)
3383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm{
3483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  unw_word_t bsp, pfs, sol;
3583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
3683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  if (unw_get_reg (c, UNW_IA64_BSP, &bsp) < 0
3783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm      || unw_get_reg (c, UNW_IA64_AR_PFS, &pfs) < 0)
3883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm    abort ();
3983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
4083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  /* simulate the effect of "br.call sigsetjmp" on ar.bsp: */
4183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  sol = (pfs >> 7) & 0x7f;
4283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  bsp = rse_skip_regs (bsp, sol);
4383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
4483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  if (bsp != wp[JB_BSP])
4583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm    return 0;
4683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
4783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  if (unlikely (sol == 0))
4883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm    {
4983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm      unw_word_t sp, prev_sp;
5083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm      unw_cursor_t tmp = *c;
5183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
5283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm      /* The caller of {sig,}setjmp() cannot have a NULL-frame.  If we
5383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	 see a NULL-frame, we haven't reached the right target yet.
5483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	 To have a NULL-frame, the number of locals must be zero and
5583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	 the stack-frame must also be empty.  */
5683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
5783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm      if (unw_step (&tmp) < 0)
5883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	abort ();
5983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
6083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm      if (unw_get_reg (&tmp, UNW_REG_SP, &sp) < 0
6183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	  || unw_get_reg (&tmp, UNW_REG_SP, &prev_sp) < 0)
6283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	abort ();
6383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
6483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm      if (sp == prev_sp)
6583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	/* got a NULL-frame; keep looking... */
6683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	return 0;
6783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm    }
6883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  return 1;
6983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm}
7083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
7183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm/* On ia64 we cannot always call sigprocmask() at
7283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm   _UI_siglongjmp_cont() because the signal may have switched stacks
7383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm   and the old stack's register-backing store may have overflown,
7483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm   leaving us no space to allocate the stacked registers needed to
7583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm   call sigprocmask().  Fortunately, we can just let unw_resume() (via
7683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm   sigreturn) take care of restoring the signal-mask.  That's faster
7783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm   anyhow.  */
7883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmstatic inline int
7983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmresume_restores_sigmask (unw_cursor_t *c, unw_word_t *wp)
8083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm{
8183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  unw_word_t sc_addr = ((struct cursor *) c)->sigcontext_addr;
8283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  struct sigcontext *sc = (struct sigcontext *) sc_addr;
8383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  sigset_t current_mask;
8483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  void *mp;
8583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
8683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  if (!sc_addr)
8783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm    return 0;
8883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
8983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  /* let unw_resume() install the desired signal mask */
9083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
9183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  if (wp[JB_MASK_SAVED])
9283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm    mp = &wp[JB_MASK];
9383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  else
9483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm    {
9583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm      if (sigprocmask (SIG_BLOCK, NULL, &current_mask) < 0)
9683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm	abort ();
9783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm      mp = &current_mask;
9883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm    }
9983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  memcpy (&sc->sc_mask, mp, sizeof (sc->sc_mask));
10083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  return 1;
10183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm}
10283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
10383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm#else /* !UNW_TARGET_IA64 */
10483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
10583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmstatic inline int
10683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmbsp_match (unw_cursor_t *c, unw_word_t *wp)
10783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm{
10883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  return 1;
10983820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm}
11083820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
11183820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmstatic inline int
11283820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidmresume_restores_sigmask (unw_cursor_t *c, unw_word_t *wp)
11383820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm{
11483820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  /* We may want to do this analogously as for ia64... */
11583820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm  return 0;
11683820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm}
11783820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm
11883820a78d1ac4daf3a1808f34bae8a8e195db5f9hp.com!davidm#endif /* !UNW_TARGET_IA64 */
119