C++程序  |  292行  |  8.19 KB


/* Testing framework, for developing code to copy vex's x87 simulation
   state to and from a real x87 state image (the 108-byte thing). 

   Includes code from fp_80_64.c.
*/

#include "../pub/libvex_basictypes.h"
#include "../pub/libvex_ir.h"
#include "../priv/guest-x86/gdefs.h"
#include <stdio.h>
#include <assert.h>
#include <stdlib.h>

/* Get definitions of convert_f64le_to_f80le and
   convert_f80le_to_f64le. */
#define USED_AS_INCLUDE
#include "fp_80_64.c"
#undef  USED_AS_INCLUDE


////////////////////////////////////////////////////////////////

/* Layout of the real x87 state. */

typedef
   struct {
      UShort env[14];
      UChar  reg[80];
   }
   Fpu_State;

/* Offsets, in 16-bit ints, into the FPU environment (env) area. */
#define FP_ENV_CTRL   0
#define FP_ENV_STAT   2
#define FP_ENV_TAG    4
#define FP_ENV_IP     6 /* and 7 */
#define FP_ENV_CS     8
#define FP_ENV_OPOFF  10 /* and 11 */
#define FP_ENV_OPSEL  12
#define FP_REG(ii)    (10*(7-(ii)))


/* Layout of vex's FP state is defined in ../priv/guest-x86/gdefs.h */

static void x87_to_vex ( /*IN*/UChar* x87_state, /*OUT*/UChar* vex_state )
{
   Int        r;
   UInt       tag;
   Double*    vexRegs = (Double*)(vex_state + OFFB_F0);
   UChar*     vexTags = (UChar*)(vex_state + OFFB_FTAG0);
   Fpu_State* x87     = (Fpu_State*)x87_state;
   UInt       ftop    = (x87->env[FP_ENV_STAT] >> 11) & 7;
   UInt       tagw    = x87->env[FP_ENV_TAG];

   /* Copy registers and tags */
   for (r = 0; r < 8; r++) {
      tag = (tagw >> (2*r)) & 3;
      if (tag == 3) {
         /* register is empty */
         vexRegs[r] = 0.0;
         vexTags[r] = 0;
      } else {
         /* register is non-empty */
         convert_f80le_to_f64le( &x87->reg[FP_REG(r)], (UChar*)&vexRegs[r] );
         vexTags[r] = 1;
      }
   }

   /* stack pointer */
   *(UInt*)(vex_state + OFFB_FTOP) = ftop;

   /* TODO: Check the CW is 037F.  Or at least, bottom 6 bits are 1
      (all exceptions masked), and 11:10, which is rounding control,
      is set to ..?
   */
}


static void vex_to_x87 ( /*IN*/UChar* vex_state, /*OUT*/UChar* x87_state )
{
   Int        i, r;
   UInt       tagw;
   Double*    vexRegs = (Double*)(vex_state + OFFB_F0);
   UChar*     vexTags = (UChar*)(vex_state + OFFB_FTAG0);
   Fpu_State* x87     = (Fpu_State*)x87_state;
   UInt       ftop    = *(UInt*)(vex_state + OFFB_FTOP);

   for (i = 0; i < 14; i++)
      x87->env[i] = 0;

   x87->env[1] = x87->env[3] = x87->env[5] = x87->env[13] = 0xFFFF;
   x87->env[FP_ENV_CTRL] = 0x037F;
   x87->env[FP_ENV_STAT] = (ftop & 7) << 11;

   tagw = 0;
   for (r = 0; r < 8; r++) {
      if (vexTags[r] == 0) {
         /* register is empty */
         tagw |= (3 << (2*r));
         convert_f64le_to_f80le( (UChar*)&vexRegs[r], &x87->reg[FP_REG(r)] );
      } else {
         /* register is full. */
         tagw |= (0 << (2*r));
         convert_f64le_to_f80le( (UChar*)&vexRegs[r],  &x87->reg[FP_REG(r)] );
      }
   }
   x87->env[FP_ENV_TAG] = tagw;
}

////////////////////////////////////////////////////////////////

// fwds ...
static void printFpuState ( UChar* fpu_state );
static void printVexState ( UChar* vex_state );


/* Capture the FPU state.  Convert it to vex.  Convert it back
   to x87.  Print it at all stages.
*/
void capture_convert_show ( /* preallocated storage */
                            UChar* x87_state0,
                            UChar* x87_state1,
                            UChar* vex_state )
{
   asm volatile ("fsave (%0)"
                 :
                 : "r" (x87_state0)
                 : "memory" );
   x87_to_vex(x87_state0, vex_state);
   vex_to_x87(vex_state, x87_state1);
   printf("\n\n=================================================\n\n");
   printFpuState(x87_state0);
   printf("\n\n");
   printVexState(vex_state);
   printf("\n\n");
#if 0
   asm volatile("frstor (%0) ; fsave (%0)"
                 :
                 : "r" (x87_state1)
                 : "memory" );
#endif
   printFpuState(x87_state1);
   printf("\n\n");
   x87_to_vex(x87_state1, vex_state);
   printVexState(vex_state);
   printf("\n\n");
}

int main ( void )
{
  UChar*  x87_state0 = malloc(sizeof(Fpu_State));
  UChar* x87_state1 = malloc(sizeof(Fpu_State));
  UChar* vex_state = malloc(1000);
  asm volatile ("finit");
  capture_convert_show(x87_state0, x87_state1, vex_state);
  asm volatile ("fldpi");
  capture_convert_show(x87_state0, x87_state1, vex_state);
  asm volatile ("fldz ; fld1 ; fdiv %st(1)");
  asm volatile ("fldln2 ; fldlg2 ; fchs ; fsqrt");
  capture_convert_show(x87_state0, x87_state1, vex_state);
  return 1;
}

////////////////////////////////////////////////////////////////

/* Bitfield offsets for exceptions in the FPU status and control words. */
#define FP_E_INVAL    0
#define FP_E_DENOR    1
#define FP_E_DIVZ     2
#define FP_E_OVERF    3
#define FP_E_UNDER    4
#define FP_E_LOS      5

/* More bitfield offsets, but for the status word only. */
#define FP_E_STACKF   6
#define FP_E_SUMMARY  7
#define FP_F_C0       8
#define FP_F_C1       9
#define FP_F_C2      10
#define FP_F_C3      14
/* top-of-stack ptr is bits 13,12,11 of the word */
#define FP_F_TOS_LO  11
#define FP_F_TOS_HI  13

/* Register tags. */
#define FP_TAG_VALID 0
#define FP_TAG_ZERO  1
#define FP_TAG_SPEC  2
#define FP_TAG_EMPTY 3

char* fp_tag_names[4]
   = { "Valid", "Zero", "Spec", "Empty" };

char* fp_exception_names[6]
   = { "INVAL", "DENOR", "DIVZ", "OVERF", "UNDERF", "LOS" };


UInt fp_get_tos ( Fpu_State* x87 )
{
   return (x87->env[FP_ENV_STAT] >> FP_F_TOS_LO) & 7;
}

UInt fp_get_tag ( Fpu_State* x87, UInt regno )
{
   assert(!(regno < 0 || regno > 7));
   return (x87->env[FP_ENV_TAG] >> (2*regno)) & 3;
}

UInt fp_get_statusword_flag ( Fpu_State* x87, UInt flagno )
{
   assert(!(flagno < 0 || flagno > 15));
   return (x87->env[FP_ENV_STAT] >> flagno) & 0x1;
}

UInt fp_get_controlword_flag ( Fpu_State* x87, UInt flagno )
{
   assert(!(flagno < 0 || flagno > 15));
   return (x87->env[FP_ENV_CTRL] >> flagno) & 0x1;
}


static void printFpuState ( UChar* fpu_state )
{
   Fpu_State* x87 = (Fpu_State*)fpu_state;

   Int i, j, k;
   assert(sizeof(Fpu_State)==108);
   for (i = 7; i >= 0; i--) {
      printf ( " %s fpreg%d: 0x", 
               (UInt)i == fp_get_tos(x87) ? "**" : "  ", i );
      for (k = 0, j = FP_REG(i)+9; k < 10; k++,j--)
         printf ( "%02x", (UInt)x87->reg[j]);
      printf ( "  %5s  ", fp_tag_names[fp_get_tag(x87,i)] );
      printf("\n");
      //printf ( "%20.16e\n", fp_get_reg(i) );
   }
   printf("     fctrl:     0x%04x  masked: ", 
          (UInt)x87->env[FP_ENV_CTRL] );
   for (i = FP_E_INVAL; i <= FP_E_LOS; i++)
      if (fp_get_controlword_flag(x87,i))
         printf ( "%s ", fp_exception_names[i] );
   printf ( "\n" );

   printf("     fstat:     0x%04x  except:", 
          (UInt)x87->env[FP_ENV_STAT] );
   for (i = FP_E_INVAL; i <= FP_E_LOS; i++)
      if (fp_get_statusword_flag(x87,i))
         printf ( "%s ", fp_exception_names[i] );
   printf ( "  top: %d  ", fp_get_tos(x87) );
   printf ( "c3210: %d%d%d%d",
            fp_get_statusword_flag(x87,FP_F_C3),
            fp_get_statusword_flag(x87,FP_F_C2),
            fp_get_statusword_flag(x87,FP_F_C1),
            fp_get_statusword_flag(x87,FP_F_C0) );
   printf ( "  STACKF: %d\n", fp_get_statusword_flag(x87,FP_E_STACKF) );

   printf("      ftag:     0x%04x  ", (UInt)x87->env[FP_ENV_TAG] );
   for (i = 7; i >= 0; i--)
      printf ( "%d:%s ", i, fp_tag_names[fp_get_tag(x87,i)] );
   printf("\n");

   printf("       fip: 0x%08x\n", 
           (((UInt)x87->env[FP_ENV_IP+1]) << 16) |
            ((UInt)x87->env[FP_ENV_IP]) );
   printf("       fcs:     0x%04x\n", 
           ((UInt)x87->env[FP_ENV_CS]) );
   printf("    fopoff: 0x%08x\n", 
           (((UInt)x87->env[FP_ENV_OPOFF+1]) << 16) |
            ((UInt)x87->env[FP_ENV_OPOFF]) );
   printf("    fopsel:     0x%04x\n", 
           ((UInt)x87->env[FP_ENV_OPSEL]) );
}


static void printVexState ( UChar* vex_state )
{
   Int r;
   ULong*     vexRegs = (ULong*)(vex_state + OFFB_F0);
   UChar*     vexTags = (UChar*)(vex_state + OFFB_FTAG0);
   UInt       ftop    = *(UInt*)(vex_state + OFFB_FTOP);

   for (r = 7; r >= 0; r--) {
      printf("%s %%f%d:  0x%llx  %s\n", 
              r == ftop ? "##" : "  ",
              r,
              vexRegs[r], 
	      vexTags[r] == 0 ? "Empty" : "Full" );
   }

}