123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074 |
- /* unwind-ia64.c -- utility routines to dump IA-64 unwind info for readelf.
- Copyright (C) 2000-2015 Free Software Foundation, Inc.
- Contributed by David Mosberger-Tang <davidm@hpl.hp.com>
- This file is part of GNU Binutils.
- This program is free software; you can redistribute it and/or modify
- it under the terms of the GNU General Public License as published by
- the Free Software Foundation; either version 3, or (at your option)
- any later version.
- This program is distributed in the hope that it will be useful,
- but WITHOUT ANY WARRANTY; without even the implied warranty of
- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
- GNU General Public License for more details.
- You should have received a copy of the GNU General Public License
- along with this program; if not, write to the Free Software
- Foundation, 51 Franklin Street - Fifth Floor, Boston,
- MA 02110-1301, USA. */
- #include "config.h"
- #include "unwind-ia64.h"
- #include <stdio.h>
- #include <string.h>
- #if __GNUC__ >= 2
- /* Define BFD64 here, even if our default architecture is 32 bit ELF
- as this will allow us to read in and parse 64bit and 32bit ELF files.
- Only do this if we believe that the compiler can support a 64 bit
- data type. For now we only rely on GCC being able to do this. */
- #define BFD64
- #endif
- #include "bfd.h"
- static bfd_vma unw_rlen = 0;
- static void unw_print_brmask (char *, unsigned int);
- static void unw_print_grmask (char *, unsigned int);
- static void unw_print_frmask (char *, unsigned int);
- static void unw_print_abreg (char *, unsigned int);
- static void unw_print_xyreg (char *, unsigned int, unsigned int);
- static void
- unw_print_brmask (char *cp, unsigned int mask)
- {
- int sep = 0;
- int i;
- for (i = 0; mask && (i < 5); ++i)
- {
- if (mask & 1)
- {
- if (sep)
- *cp++ = ',';
- *cp++ = 'b';
- *cp++ = i + 1 + '0';
- sep = 1;
- }
- mask >>= 1;
- }
- *cp = '\0';
- }
- static void
- unw_print_grmask (char *cp, unsigned int mask)
- {
- int sep = 0;
- int i;
- for (i = 0; i < 4; ++i)
- {
- if (mask & 1)
- {
- if (sep)
- *cp++ = ',';
- *cp++ = 'r';
- *cp++ = i + 4 + '0';
- sep = 1;
- }
- mask >>= 1;
- }
- *cp = '\0';
- }
- static void
- unw_print_frmask (char *cp, unsigned int mask)
- {
- int sep = 0;
- int i;
- for (i = 0; i < 20; ++i)
- {
- if (mask & 1)
- {
- if (sep)
- *cp++ = ',';
- *cp++ = 'f';
- if (i < 4)
- *cp++ = i + 2 + '0';
- else
- {
- *cp++ = (i + 2) / 10 + 1 + '0';
- *cp++ = (i + 2) % 10 + '0';
- }
- sep = 1;
- }
- mask >>= 1;
- }
- *cp = '\0';
- }
- static void
- unw_print_abreg (char *cp, unsigned int abreg)
- {
- static const char * const special_reg[16] =
- {
- "pr", "psp", "@priunat", "rp", "ar.bsp", "ar.bspstore", "ar.rnat",
- "ar.unat", "ar.fpsr", "ar.pfs", "ar.lc",
- "Unknown11", "Unknown12", "Unknown13", "Unknown14", "Unknown15"
- };
- switch ((abreg >> 5) & 0x3)
- {
- case 0: /* gr */
- sprintf (cp, "r%u", (abreg & 0x1f));
- break;
- case 1: /* fr */
- sprintf (cp, "f%u", (abreg & 0x1f));
- break;
- case 2: /* br */
- sprintf (cp, "b%u", (abreg & 0x1f));
- break;
- case 3: /* special */
- strcpy (cp, special_reg[abreg & 0xf]);
- break;
- }
- }
- static void
- unw_print_xyreg (char *cp, unsigned int x, unsigned int ytreg)
- {
- switch ((x << 1) | ((ytreg >> 7) & 1))
- {
- case 0: /* gr */
- sprintf (cp, "r%u", (ytreg & 0x1f));
- break;
- case 1: /* fr */
- sprintf (cp, "f%u", (ytreg & 0x1f));
- break;
- case 2: /* br */
- sprintf (cp, "b%u", (ytreg & 0x1f));
- break;
- }
- }
- #define UNW_REG_BSP "bsp"
- #define UNW_REG_BSPSTORE "bspstore"
- #define UNW_REG_FPSR "fpsr"
- #define UNW_REG_LC "lc"
- #define UNW_REG_PFS "pfs"
- #define UNW_REG_PR "pr"
- #define UNW_REG_PSP "psp"
- #define UNW_REG_RNAT "rnat"
- #define UNW_REG_RP "rp"
- #define UNW_REG_UNAT "unat"
- typedef bfd_vma unw_word;
- #define UNW_DEC_BAD_CODE(code) \
- printf ("Unknown code 0x%02x\n", code)
- #define UNW_DEC_PROLOGUE(fmt, body, rlen, arg) \
- do \
- { \
- unw_rlen = rlen; \
- *(int *)arg = body; \
- printf (" %s:%s(rlen=%lu)\n", \
- fmt, body ? "body" : "prologue", (unsigned long) rlen); \
- } \
- while (0)
- #define UNW_DEC_PROLOGUE_GR(fmt, rlen, mask, grsave, arg) \
- do \
- { \
- char regname[16], maskstr[64], *sep; \
- \
- unw_rlen = rlen; \
- *(int *)arg = 0; \
- \
- maskstr[0] = '\0'; \
- sep = ""; \
- if (mask & 0x8) \
- { \
- strcat (maskstr, "rp"); \
- sep = ","; \
- } \
- if (mask & 0x4) \
- { \
- strcat (maskstr, sep); \
- strcat (maskstr, "ar.pfs"); \
- sep = ","; \
- } \
- if (mask & 0x2) \
- { \
- strcat (maskstr, sep); \
- strcat (maskstr, "psp"); \
- sep = ","; \
- } \
- if (mask & 0x1) \
- { \
- strcat (maskstr, sep); \
- strcat (maskstr, "pr"); \
- } \
- sprintf (regname, "r%u", grsave); \
- printf (" %s:prologue_gr(mask=[%s],grsave=%s,rlen=%lu)\n", \
- fmt, maskstr, regname, (unsigned long) rlen); \
- } \
- while (0)
- #define UNW_DEC_FR_MEM(fmt, frmask, arg) \
- do \
- { \
- char frstr[200]; \
- \
- unw_print_frmask (frstr, frmask); \
- printf ("\t%s:fr_mem(frmask=[%s])\n", fmt, frstr); \
- } \
- while (0)
- #define UNW_DEC_GR_MEM(fmt, grmask, arg) \
- do \
- { \
- char grstr[200]; \
- \
- unw_print_grmask (grstr, grmask); \
- printf ("\t%s:gr_mem(grmask=[%s])\n", fmt, grstr); \
- } \
- while (0)
- #define UNW_DEC_FRGR_MEM(fmt, grmask, frmask, arg) \
- do \
- { \
- char frstr[200], grstr[20]; \
- \
- unw_print_grmask (grstr, grmask); \
- unw_print_frmask (frstr, frmask); \
- printf ("\t%s:frgr_mem(grmask=[%s],frmask=[%s])\n", fmt, grstr, frstr); \
- } \
- while (0)
- #define UNW_DEC_BR_MEM(fmt, brmask, arg) \
- do \
- { \
- char brstr[20]; \
- \
- unw_print_brmask (brstr, brmask); \
- printf ("\t%s:br_mem(brmask=[%s])\n", fmt, brstr); \
- } \
- while (0)
- #define UNW_DEC_BR_GR(fmt, brmask, gr, arg) \
- do \
- { \
- char brstr[20]; \
- \
- unw_print_brmask (brstr, brmask); \
- printf ("\t%s:br_gr(brmask=[%s],gr=r%u)\n", fmt, brstr, gr); \
- } \
- while (0)
- #define UNW_DEC_REG_GR(fmt, src, dst, arg) \
- printf ("\t%s:%s_gr(reg=r%u)\n", fmt, src, dst)
- #define UNW_DEC_RP_BR(fmt, dst, arg) \
- printf ("\t%s:rp_br(reg=b%u)\n", fmt, dst)
- #define UNW_DEC_REG_WHEN(fmt, reg, t, arg) \
- printf ("\t%s:%s_when(t=%lu)\n", fmt, reg, (unsigned long) t)
- #define UNW_DEC_REG_SPREL(fmt, reg, spoff, arg) \
- printf ("\t%s:%s_sprel(spoff=0x%lx)\n", \
- fmt, reg, 4*(unsigned long)spoff)
- #define UNW_DEC_REG_PSPREL(fmt, reg, pspoff, arg) \
- printf ("\t%s:%s_psprel(pspoff=0x10-0x%lx)\n", \
- fmt, reg, 4*(unsigned long)pspoff)
- #define UNW_DEC_GR_GR(fmt, grmask, gr, arg) \
- do \
- { \
- char grstr[20]; \
- \
- unw_print_grmask (grstr, grmask); \
- printf ("\t%s:gr_gr(grmask=[%s],r%u)\n", fmt, grstr, gr); \
- } \
- while (0)
- #define UNW_DEC_ABI(fmt, abi, context, arg) \
- do \
- { \
- static const char * const abiname[] = \
- { \
- "@svr4", "@hpux", "@nt" \
- }; \
- char buf[20]; \
- const char *abistr = buf; \
- \
- if (abi < 3) \
- abistr = abiname[abi]; \
- else \
- sprintf (buf, "0x%x", abi); \
- printf ("\t%s:unwabi(abi=%s,context=0x%02x)\n", \
- fmt, abistr, context); \
- } \
- while (0)
- #define UNW_DEC_PRIUNAT_GR(fmt, r, arg) \
- printf ("\t%s:priunat_gr(reg=r%u)\n", fmt, r)
- #define UNW_DEC_PRIUNAT_WHEN_GR(fmt, t, arg) \
- printf ("\t%s:priunat_when_gr(t=%lu)\n", fmt, (unsigned long) t)
- #define UNW_DEC_PRIUNAT_WHEN_MEM(fmt, t, arg) \
- printf ("\t%s:priunat_when_mem(t=%lu)\n", fmt, (unsigned long) t)
- #define UNW_DEC_PRIUNAT_PSPREL(fmt, pspoff, arg) \
- printf ("\t%s:priunat_psprel(pspoff=0x10-0x%lx)\n", \
- fmt, 4*(unsigned long)pspoff)
- #define UNW_DEC_PRIUNAT_SPREL(fmt, spoff, arg) \
- printf ("\t%s:priunat_sprel(spoff=0x%lx)\n", \
- fmt, 4*(unsigned long)spoff)
- #define UNW_DEC_MEM_STACK_F(fmt, t, size, arg) \
- printf ("\t%s:mem_stack_f(t=%lu,size=%lu)\n", \
- fmt, (unsigned long) t, 16*(unsigned long)size)
- #define UNW_DEC_MEM_STACK_V(fmt, t, arg) \
- printf ("\t%s:mem_stack_v(t=%lu)\n", fmt, (unsigned long) t)
- #define UNW_DEC_SPILL_BASE(fmt, pspoff, arg) \
- printf ("\t%s:spill_base(pspoff=0x10-0x%lx)\n", \
- fmt, 4*(unsigned long)pspoff)
- #define UNW_DEC_SPILL_MASK(fmt, dp, arg, end) \
- do \
- { \
- static const char *spill_type = "-frb"; \
- unsigned const char *imaskp = dp; \
- unsigned char mask = 0; \
- bfd_vma insn = 0; \
- \
- /* PR 18420. */ \
- if ((dp + (unw_rlen / 4)) > end) \
- { \
- printf ("\nERROR: unwind length too long (0x%lx > 0x%lx)\n\n",\
- (long) (unw_rlen / 4), (long)(end - dp)); \
- /* FIXME: Should we reset unw_rlen ? */ \
- break; \
- } \
- printf ("\t%s:spill_mask(imask=[", fmt); \
- for (insn = 0; insn < unw_rlen; ++insn) \
- { \
- if ((insn % 4) == 0) \
- mask = *imaskp++; \
- if (insn > 0 && (insn % 3) == 0) \
- putchar (','); \
- putchar (spill_type[(mask >> (2 * (3 - (insn & 0x3)))) & 0x3]); \
- } \
- printf ("])\n"); \
- dp = imaskp; \
- } \
- while (0)
- #define UNW_DEC_SPILL_SPREL(fmt, t, abreg, spoff, arg) \
- do \
- { \
- char regname[20]; \
- \
- unw_print_abreg (regname, abreg); \
- printf ("\t%s:spill_sprel(reg=%s,t=%lu,spoff=0x%lx)\n", \
- fmt, regname, (unsigned long) t, 4*(unsigned long)off); \
- } \
- while (0)
- #define UNW_DEC_SPILL_PSPREL(fmt, t, abreg, pspoff, arg) \
- do \
- { \
- char regname[20]; \
- \
- unw_print_abreg (regname, abreg); \
- printf ("\t%s:spill_psprel(reg=%s,t=%lu,pspoff=0x10-0x%lx)\n", \
- fmt, regname, (unsigned long) t, 4*(unsigned long)pspoff); \
- } \
- while (0)
- #define UNW_DEC_RESTORE(fmt, t, abreg, arg) \
- do \
- { \
- char regname[20]; \
- \
- unw_print_abreg (regname, abreg); \
- printf ("\t%s:restore(t=%lu,reg=%s)\n", \
- fmt, (unsigned long) t, regname); \
- } \
- while (0)
- #define UNW_DEC_SPILL_REG(fmt, t, abreg, x, ytreg, arg) \
- do \
- { \
- char abregname[20], tregname[20]; \
- \
- unw_print_abreg (abregname, abreg); \
- unw_print_xyreg (tregname, x, ytreg); \
- printf ("\t%s:spill_reg(t=%lu,reg=%s,treg=%s)\n", \
- fmt, (unsigned long) t, abregname, tregname); \
- } \
- while (0)
- #define UNW_DEC_SPILL_SPREL_P(fmt, qp, t, abreg, spoff, arg) \
- do \
- { \
- char regname[20]; \
- \
- unw_print_abreg (regname, abreg); \
- printf ("\t%s:spill_sprel_p(qp=p%u,t=%lu,reg=%s,spoff=0x%lx)\n", \
- fmt, qp, (unsigned long) t, regname, 4 * (unsigned long)spoff); \
- } \
- while (0)
- #define UNW_DEC_SPILL_PSPREL_P(fmt, qp, t, abreg, pspoff, arg) \
- do \
- { \
- char regname[20]; \
- \
- unw_print_abreg (regname, abreg); \
- printf ("\t%s:spill_psprel_p(qp=p%u,t=%lu,reg=%s,pspoff=0x10-0x%lx)\n",\
- fmt, qp, (unsigned long) t, regname, 4*(unsigned long)pspoff);\
- } \
- while (0)
- #define UNW_DEC_RESTORE_P(fmt, qp, t, abreg, arg) \
- do \
- { \
- char regname[20]; \
- \
- unw_print_abreg (regname, abreg); \
- printf ("\t%s:restore_p(qp=p%u,t=%lu,reg=%s)\n", \
- fmt, qp, (unsigned long) t, regname); \
- } \
- while (0)
- #define UNW_DEC_SPILL_REG_P(fmt, qp, t, abreg, x, ytreg, arg) \
- do \
- { \
- char regname[20], tregname[20]; \
- \
- unw_print_abreg (regname, abreg); \
- unw_print_xyreg (tregname, x, ytreg); \
- printf ("\t%s:spill_reg_p(qp=p%u,t=%lu,reg=%s,treg=%s)\n", \
- fmt, qp, (unsigned long) t, regname, tregname); \
- } \
- while (0)
- #define UNW_DEC_LABEL_STATE(fmt, label, arg) \
- printf ("\t%s:label_state(label=%lu)\n", fmt, (unsigned long) label)
- #define UNW_DEC_COPY_STATE(fmt, label, arg) \
- printf ("\t%s:copy_state(label=%lu)\n", fmt, (unsigned long) label)
- #define UNW_DEC_EPILOGUE(fmt, t, ecount, arg) \
- printf ("\t%s:epilogue(t=%lu,ecount=%lu)\n", \
- fmt, (unsigned long) t, (unsigned long) ecount)
- /*
- * Generic IA-64 unwind info decoder.
- *
- * This file is used both by the Linux kernel and objdump. Please
- * keep the two copies of this file in sync (modulo differences in the
- * prototypes...).
- *
- * You need to customize the decoder by defining the following
- * macros/constants before including this file:
- *
- * Types:
- * unw_word Unsigned integer type with at least 64 bits
- *
- * Register names:
- * UNW_REG_BSP
- * UNW_REG_BSPSTORE
- * UNW_REG_FPSR
- * UNW_REG_LC
- * UNW_REG_PFS
- * UNW_REG_PR
- * UNW_REG_RNAT
- * UNW_REG_PSP
- * UNW_REG_RP
- * UNW_REG_UNAT
- *
- * Decoder action macros:
- * UNW_DEC_BAD_CODE(code)
- * UNW_DEC_ABI(fmt,abi,context,arg)
- * UNW_DEC_BR_GR(fmt,brmask,gr,arg)
- * UNW_DEC_BR_MEM(fmt,brmask,arg)
- * UNW_DEC_COPY_STATE(fmt,label,arg)
- * UNW_DEC_EPILOGUE(fmt,t,ecount,arg)
- * UNW_DEC_FRGR_MEM(fmt,grmask,frmask,arg)
- * UNW_DEC_FR_MEM(fmt,frmask,arg)
- * UNW_DEC_GR_GR(fmt,grmask,gr,arg)
- * UNW_DEC_GR_MEM(fmt,grmask,arg)
- * UNW_DEC_LABEL_STATE(fmt,label,arg)
- * UNW_DEC_MEM_STACK_F(fmt,t,size,arg)
- * UNW_DEC_MEM_STACK_V(fmt,t,arg)
- * UNW_DEC_PRIUNAT_GR(fmt,r,arg)
- * UNW_DEC_PRIUNAT_WHEN_GR(fmt,t,arg)
- * UNW_DEC_PRIUNAT_WHEN_MEM(fmt,t,arg)
- * UNW_DEC_PRIUNAT_WHEN_PSPREL(fmt,pspoff,arg)
- * UNW_DEC_PRIUNAT_WHEN_SPREL(fmt,spoff,arg)
- * UNW_DEC_PROLOGUE(fmt,body,rlen,arg)
- * UNW_DEC_PROLOGUE_GR(fmt,rlen,mask,grsave,arg)
- * UNW_DEC_REG_PSPREL(fmt,reg,pspoff,arg)
- * UNW_DEC_REG_REG(fmt,src,dst,arg)
- * UNW_DEC_REG_SPREL(fmt,reg,spoff,arg)
- * UNW_DEC_REG_WHEN(fmt,reg,t,arg)
- * UNW_DEC_RESTORE(fmt,t,abreg,arg)
- * UNW_DEC_RESTORE_P(fmt,qp,t,abreg,arg)
- * UNW_DEC_SPILL_BASE(fmt,pspoff,arg)
- * UNW_DEC_SPILL_MASK(fmt,imaskp,arg)
- * UNW_DEC_SPILL_PSPREL(fmt,t,abreg,pspoff,arg)
- * UNW_DEC_SPILL_PSPREL_P(fmt,qp,t,abreg,pspoff,arg)
- * UNW_DEC_SPILL_REG(fmt,t,abreg,x,ytreg,arg)
- * UNW_DEC_SPILL_REG_P(fmt,qp,t,abreg,x,ytreg,arg)
- * UNW_DEC_SPILL_SPREL(fmt,t,abreg,spoff,arg)
- * UNW_DEC_SPILL_SPREL_P(fmt,qp,t,abreg,pspoff,arg)
- */
- static unw_word
- unw_decode_uleb128 (const unsigned char **dpp)
- {
- unsigned shift = 0;
- unw_word byte, result = 0;
- const unsigned char *bp = *dpp;
- while (1)
- {
- byte = *bp++;
- result |= (byte & 0x7f) << shift;
- if ((byte & 0x80) == 0)
- break;
- shift += 7;
- }
- *dpp = bp;
- return result;
- }
- static const unsigned char *
- unw_decode_x1 (const unsigned char *dp, unsigned int code ATTRIBUTE_UNUSED,
- void *arg ATTRIBUTE_UNUSED)
- {
- unsigned char byte1, abreg;
- unw_word t, off;
- byte1 = *dp++;
- t = unw_decode_uleb128 (&dp);
- off = unw_decode_uleb128 (&dp);
- abreg = (byte1 & 0x7f);
- if (byte1 & 0x80)
- UNW_DEC_SPILL_SPREL ("X1", t, abreg, off, arg);
- else
- UNW_DEC_SPILL_PSPREL ("X1", t, abreg, off, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_x2 (const unsigned char *dp, unsigned int code ATTRIBUTE_UNUSED,
- void *arg ATTRIBUTE_UNUSED)
- {
- unsigned char byte1, byte2, abreg, x, ytreg;
- unw_word t;
- byte1 = *dp++;
- byte2 = *dp++;
- t = unw_decode_uleb128 (&dp);
- abreg = (byte1 & 0x7f);
- ytreg = byte2;
- x = (byte1 >> 7) & 1;
- if ((byte1 & 0x80) == 0 && ytreg == 0)
- UNW_DEC_RESTORE ("X2", t, abreg, arg);
- else
- UNW_DEC_SPILL_REG ("X2", t, abreg, x, ytreg, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_x3 (const unsigned char *dp, unsigned int code ATTRIBUTE_UNUSED,
- void *arg ATTRIBUTE_UNUSED)
- {
- unsigned char byte1, byte2, abreg, qp;
- unw_word t, off;
- byte1 = *dp++;
- byte2 = *dp++;
- t = unw_decode_uleb128 (&dp);
- off = unw_decode_uleb128 (&dp);
- qp = (byte1 & 0x3f);
- abreg = (byte2 & 0x7f);
- if (byte1 & 0x80)
- UNW_DEC_SPILL_SPREL_P ("X3", qp, t, abreg, off, arg);
- else
- UNW_DEC_SPILL_PSPREL_P ("X3", qp, t, abreg, off, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_x4 (const unsigned char *dp, unsigned int code ATTRIBUTE_UNUSED,
- void *arg ATTRIBUTE_UNUSED)
- {
- unsigned char byte1, byte2, byte3, qp, abreg, x, ytreg;
- unw_word t;
- byte1 = *dp++;
- byte2 = *dp++;
- byte3 = *dp++;
- t = unw_decode_uleb128 (&dp);
- qp = (byte1 & 0x3f);
- abreg = (byte2 & 0x7f);
- x = (byte2 >> 7) & 1;
- ytreg = byte3;
- if ((byte2 & 0x80) == 0 && byte3 == 0)
- UNW_DEC_RESTORE_P ("X4", qp, t, abreg, arg);
- else
- UNW_DEC_SPILL_REG_P ("X4", qp, t, abreg, x, ytreg, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_r1 (const unsigned char *dp, unsigned int code, void *arg,
- const unsigned char * end ATTRIBUTE_UNUSED)
- {
- int body = (code & 0x20) != 0;
- unw_word rlen;
- rlen = (code & 0x1f);
- UNW_DEC_PROLOGUE ("R1", body, rlen, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_r2 (const unsigned char *dp, unsigned int code, void *arg,
- const unsigned char * end ATTRIBUTE_UNUSED)
- {
- unsigned char byte1, mask, grsave;
- unw_word rlen;
- byte1 = *dp++;
- mask = ((code & 0x7) << 1) | ((byte1 >> 7) & 1);
- grsave = (byte1 & 0x7f);
- rlen = unw_decode_uleb128 (& dp);
- UNW_DEC_PROLOGUE_GR ("R2", rlen, mask, grsave, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_r3 (const unsigned char *dp, unsigned int code, void *arg,
- const unsigned char * end ATTRIBUTE_UNUSED)
- {
- unw_word rlen;
- rlen = unw_decode_uleb128 (& dp);
- UNW_DEC_PROLOGUE ("R3", ((code & 0x3) == 1), rlen, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_p1 (const unsigned char *dp, unsigned int code,
- void *arg ATTRIBUTE_UNUSED,
- const unsigned char * end ATTRIBUTE_UNUSED)
- {
- unsigned char brmask = (code & 0x1f);
- UNW_DEC_BR_MEM ("P1", brmask, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_p2_p5 (const unsigned char *dp, unsigned int code,
- void *arg ATTRIBUTE_UNUSED,
- const unsigned char * end)
- {
- if ((code & 0x10) == 0)
- {
- unsigned char byte1 = *dp++;
- UNW_DEC_BR_GR ("P2", ((code & 0xf) << 1) | ((byte1 >> 7) & 1),
- (byte1 & 0x7f), arg);
- }
- else if ((code & 0x08) == 0)
- {
- unsigned char byte1 = *dp++, r, dst;
- r = ((code & 0x7) << 1) | ((byte1 >> 7) & 1);
- dst = (byte1 & 0x7f);
- switch (r)
- {
- case 0:
- UNW_DEC_REG_GR ("P3", UNW_REG_PSP, dst, arg);
- break;
- case 1:
- UNW_DEC_REG_GR ("P3", UNW_REG_RP, dst, arg);
- break;
- case 2:
- UNW_DEC_REG_GR ("P3", UNW_REG_PFS, dst, arg);
- break;
- case 3:
- UNW_DEC_REG_GR ("P3", UNW_REG_PR, dst, arg);
- break;
- case 4:
- UNW_DEC_REG_GR ("P3", UNW_REG_UNAT, dst, arg);
- break;
- case 5:
- UNW_DEC_REG_GR ("P3", UNW_REG_LC, dst, arg);
- break;
- case 6:
- UNW_DEC_RP_BR ("P3", dst, arg);
- break;
- case 7:
- UNW_DEC_REG_GR ("P3", UNW_REG_RNAT, dst, arg);
- break;
- case 8:
- UNW_DEC_REG_GR ("P3", UNW_REG_BSP, dst, arg);
- break;
- case 9:
- UNW_DEC_REG_GR ("P3", UNW_REG_BSPSTORE, dst, arg);
- break;
- case 10:
- UNW_DEC_REG_GR ("P3", UNW_REG_FPSR, dst, arg);
- break;
- case 11:
- UNW_DEC_PRIUNAT_GR ("P3", dst, arg);
- break;
- default:
- UNW_DEC_BAD_CODE (r);
- break;
- }
- }
- else if ((code & 0x7) == 0)
- UNW_DEC_SPILL_MASK ("P4", dp, arg, end);
- else if ((code & 0x7) == 1)
- {
- unw_word grmask, frmask, byte1, byte2, byte3;
- byte1 = *dp++;
- byte2 = *dp++;
- byte3 = *dp++;
- grmask = ((byte1 >> 4) & 0xf);
- frmask = ((byte1 & 0xf) << 16) | (byte2 << 8) | byte3;
- UNW_DEC_FRGR_MEM ("P5", grmask, frmask, arg);
- }
- else
- UNW_DEC_BAD_CODE (code);
- return dp;
- }
- static const unsigned char *
- unw_decode_p6 (const unsigned char *dp, unsigned int code,
- void *arg ATTRIBUTE_UNUSED,
- const unsigned char * end ATTRIBUTE_UNUSED)
- {
- int gregs = (code & 0x10) != 0;
- unsigned char mask = (code & 0x0f);
- if (gregs)
- UNW_DEC_GR_MEM ("P6", mask, arg);
- else
- UNW_DEC_FR_MEM ("P6", mask, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_p7_p10 (const unsigned char *dp, unsigned int code, void *arg,
- const unsigned char * end ATTRIBUTE_UNUSED)
- {
- unsigned char r, byte1, byte2;
- unw_word t, size;
- if ((code & 0x10) == 0)
- {
- r = (code & 0xf);
- t = unw_decode_uleb128 (&dp);
- switch (r)
- {
- case 0:
- size = unw_decode_uleb128 (&dp);
- UNW_DEC_MEM_STACK_F ("P7", t, size, arg);
- break;
- case 1:
- UNW_DEC_MEM_STACK_V ("P7", t, arg);
- break;
- case 2:
- UNW_DEC_SPILL_BASE ("P7", t, arg);
- break;
- case 3:
- UNW_DEC_REG_SPREL ("P7", UNW_REG_PSP, t, arg);
- break;
- case 4:
- UNW_DEC_REG_WHEN ("P7", UNW_REG_RP, t, arg);
- break;
- case 5:
- UNW_DEC_REG_PSPREL ("P7", UNW_REG_RP, t, arg);
- break;
- case 6:
- UNW_DEC_REG_WHEN ("P7", UNW_REG_PFS, t, arg);
- break;
- case 7:
- UNW_DEC_REG_PSPREL ("P7", UNW_REG_PFS, t, arg);
- break;
- case 8:
- UNW_DEC_REG_WHEN ("P7", UNW_REG_PR, t, arg);
- break;
- case 9:
- UNW_DEC_REG_PSPREL ("P7", UNW_REG_PR, t, arg);
- break;
- case 10:
- UNW_DEC_REG_WHEN ("P7", UNW_REG_LC, t, arg);
- break;
- case 11:
- UNW_DEC_REG_PSPREL ("P7", UNW_REG_LC, t, arg);
- break;
- case 12:
- UNW_DEC_REG_WHEN ("P7", UNW_REG_UNAT, t, arg);
- break;
- case 13:
- UNW_DEC_REG_PSPREL ("P7", UNW_REG_UNAT, t, arg);
- break;
- case 14:
- UNW_DEC_REG_WHEN ("P7", UNW_REG_FPSR, t, arg);
- break;
- case 15:
- UNW_DEC_REG_PSPREL ("P7", UNW_REG_FPSR, t, arg);
- break;
- default:
- UNW_DEC_BAD_CODE (r);
- break;
- }
- }
- else
- {
- switch (code & 0xf)
- {
- case 0x0: /* p8 */
- {
- r = *dp++;
- t = unw_decode_uleb128 (&dp);
- switch (r)
- {
- case 1:
- UNW_DEC_REG_SPREL ("P8", UNW_REG_RP, t, arg);
- break;
- case 2:
- UNW_DEC_REG_SPREL ("P8", UNW_REG_PFS, t, arg);
- break;
- case 3:
- UNW_DEC_REG_SPREL ("P8", UNW_REG_PR, t, arg);
- break;
- case 4:
- UNW_DEC_REG_SPREL ("P8", UNW_REG_LC, t, arg);
- break;
- case 5:
- UNW_DEC_REG_SPREL ("P8", UNW_REG_UNAT, t, arg);
- break;
- case 6:
- UNW_DEC_REG_SPREL ("P8", UNW_REG_FPSR, t, arg);
- break;
- case 7:
- UNW_DEC_REG_WHEN ("P8", UNW_REG_BSP, t, arg);
- break;
- case 8:
- UNW_DEC_REG_PSPREL ("P8", UNW_REG_BSP, t, arg);
- break;
- case 9:
- UNW_DEC_REG_SPREL ("P8", UNW_REG_BSP, t, arg);
- break;
- case 10:
- UNW_DEC_REG_WHEN ("P8", UNW_REG_BSPSTORE, t, arg);
- break;
- case 11:
- UNW_DEC_REG_PSPREL ("P8", UNW_REG_BSPSTORE, t, arg);
- break;
- case 12:
- UNW_DEC_REG_SPREL ("P8", UNW_REG_BSPSTORE, t, arg);
- break;
- case 13:
- UNW_DEC_REG_WHEN ("P8", UNW_REG_RNAT, t, arg);
- break;
- case 14:
- UNW_DEC_REG_PSPREL ("P8", UNW_REG_RNAT, t, arg);
- break;
- case 15:
- UNW_DEC_REG_SPREL ("P8", UNW_REG_RNAT, t, arg);
- break;
- case 16:
- UNW_DEC_PRIUNAT_WHEN_GR ("P8", t, arg);
- break;
- case 17:
- UNW_DEC_PRIUNAT_PSPREL ("P8", t, arg);
- break;
- case 18:
- UNW_DEC_PRIUNAT_SPREL ("P8", t, arg);
- break;
- case 19:
- UNW_DEC_PRIUNAT_WHEN_MEM ("P8", t, arg);
- break;
- default:
- UNW_DEC_BAD_CODE (r);
- break;
- }
- }
- break;
- case 0x1:
- byte1 = *dp++;
- byte2 = *dp++;
- UNW_DEC_GR_GR ("P9", (byte1 & 0xf), (byte2 & 0x7f), arg);
- break;
- case 0xf: /* p10 */
- byte1 = *dp++;
- byte2 = *dp++;
- UNW_DEC_ABI ("P10", byte1, byte2, arg);
- break;
- case 0x9:
- return unw_decode_x1 (dp, code, arg);
- case 0xa:
- return unw_decode_x2 (dp, code, arg);
- case 0xb:
- return unw_decode_x3 (dp, code, arg);
- case 0xc:
- return unw_decode_x4 (dp, code, arg);
- default:
- UNW_DEC_BAD_CODE (code);
- break;
- }
- }
- return dp;
- }
- static const unsigned char *
- unw_decode_b1 (const unsigned char *dp, unsigned int code,
- void *arg ATTRIBUTE_UNUSED,
- const unsigned char * end ATTRIBUTE_UNUSED)
- {
- unw_word label = (code & 0x1f);
- if ((code & 0x20) != 0)
- UNW_DEC_COPY_STATE ("B1", label, arg);
- else
- UNW_DEC_LABEL_STATE ("B1", label, arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_b2 (const unsigned char *dp, unsigned int code,
- void *arg ATTRIBUTE_UNUSED,
- const unsigned char * end ATTRIBUTE_UNUSED)
- {
- unw_word t;
- t = unw_decode_uleb128 (& dp);
- UNW_DEC_EPILOGUE ("B2", t, (code & 0x1f), arg);
- return dp;
- }
- static const unsigned char *
- unw_decode_b3_x4 (const unsigned char *dp, unsigned int code, void *arg,
- const unsigned char * end ATTRIBUTE_UNUSED)
- {
- unw_word t, ecount, label;
- if ((code & 0x10) == 0)
- {
- t = unw_decode_uleb128 (&dp);
- ecount = unw_decode_uleb128 (&dp);
- UNW_DEC_EPILOGUE ("B3", t, ecount, arg);
- }
- else if ((code & 0x07) == 0)
- {
- label = unw_decode_uleb128 (&dp);
- if ((code & 0x08) != 0)
- UNW_DEC_COPY_STATE ("B4", label, arg);
- else
- UNW_DEC_LABEL_STATE ("B4", label, arg);
- }
- else
- switch (code & 0x7)
- {
- case 1:
- return unw_decode_x1 (dp, code, arg);
- case 2:
- return unw_decode_x2 (dp, code, arg);
- case 3:
- return unw_decode_x3 (dp, code, arg);
- case 4:
- return unw_decode_x4 (dp, code, arg);
- default:
- UNW_DEC_BAD_CODE (code);
- break;
- }
- return dp;
- }
- typedef const unsigned char *(*unw_decoder)
- (const unsigned char *, unsigned int, void *, const unsigned char *);
- static const unw_decoder unw_decode_table[2][8] =
- {
- /* prologue table: */
- {
- unw_decode_r1, /* 0 */
- unw_decode_r1,
- unw_decode_r2,
- unw_decode_r3,
- unw_decode_p1, /* 4 */
- unw_decode_p2_p5,
- unw_decode_p6,
- unw_decode_p7_p10
- },
- {
- unw_decode_r1, /* 0 */
- unw_decode_r1,
- unw_decode_r2,
- unw_decode_r3,
- unw_decode_b1, /* 4 */
- unw_decode_b1,
- unw_decode_b2,
- unw_decode_b3_x4
- }
- };
- /* Decode one descriptor and return address of next descriptor. */
- const unsigned char *
- unw_decode (const unsigned char *dp, int inside_body,
- void *ptr_inside_body, const unsigned char * end)
- {
- unw_decoder decoder;
- unsigned char code;
- code = *dp++;
- decoder = unw_decode_table[inside_body][code >> 5];
- return (*decoder) (dp, code, ptr_inside_body, end);
- }
|