/**************************************************************************** 
 | 
*            Realmode X86 Emulator Library 
 | 
* 
 | 
*  Copyright (C) 2007 Freescale Semiconductor, Inc. 
 | 
*  Jason Jin <Jason.jin@freescale.com> 
 | 
* 
 | 
*        Copyright (C) 1991-2004 SciTech Software, Inc. 
 | 
*                     Copyright (C) David Mosberger-Tang 
 | 
*                       Copyright (C) 1999 Egbert Eich 
 | 
* 
 | 
*  ======================================================================== 
 | 
* 
 | 
*  Permission to use, copy, modify, distribute, and sell this software and 
 | 
*  its documentation for any purpose is hereby granted without fee, 
 | 
*  provided that the above copyright notice appear in all copies and that 
 | 
*  both that copyright notice and this permission notice appear in 
 | 
*  supporting documentation, and that the name of the authors not be used 
 | 
*  in advertising or publicity pertaining to distribution of the software 
 | 
*  without specific, written prior permission.    The authors makes no 
 | 
*  representations about the suitability of this software for any purpose. 
 | 
*  It is provided "as is" without express or implied warranty. 
 | 
* 
 | 
*  THE AUTHORS DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, 
 | 
*  INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO 
 | 
*  EVENT SHALL THE AUTHORS BE LIABLE FOR ANY SPECIAL, INDIRECT OR 
 | 
*  CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF 
 | 
*  USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR 
 | 
*  OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR 
 | 
*  PERFORMANCE OF THIS SOFTWARE. 
 | 
* 
 | 
*  ======================================================================== 
 | 
* 
 | 
* Language:        ANSI C 
 | 
* Environment:    Any 
 | 
* Developer:    Kendall Bennett 
 | 
* 
 | 
* Description:    This file includes subroutines to implement the decoding 
 | 
*        and emulation of all the x86 processor instructions. 
 | 
* 
 | 
* There are approximately 250 subroutines in here, which correspond 
 | 
* to the 256 byte-"opcodes" found on the 8086.    The table which 
 | 
* dispatches this is found in the files optab.[ch]. 
 | 
* 
 | 
* Each opcode proc has a comment preceding it which gives it's table 
 | 
* address.  Several opcodes are missing (undefined) in the table. 
 | 
* 
 | 
* Each proc includes information for decoding (DECODE_PRINTF and 
 | 
* DECODE_PRINTF2), debugging (TRACE_REGS, SINGLE_STEP), and misc 
 | 
* functions (START_OF_INSTR, END_OF_INSTR). 
 | 
* 
 | 
* Many of the procedures are *VERY* similar in coding.    This has 
 | 
* allowed for a very large amount of code to be generated in a fairly 
 | 
* short amount of time (i.e. cut, paste, and modify).  The result is 
 | 
* that much of the code below could have been folded into subroutines 
 | 
* for a large reduction in size of this file.  The downside would be 
 | 
* that there would be a penalty in execution speed.  The file could 
 | 
* also have been *MUCH* larger by inlining certain functions which 
 | 
* were called.    This could have resulted even faster execution.     The 
 | 
* prime directive I used to decide whether to inline the code or to 
 | 
* modularize it, was basically: 1) no unnecessary subroutine calls, 
 | 
* 2) no routines more than about 200 lines in size, and 3) modularize 
 | 
* any code that I might not get right the first time.  The fetch_* 
 | 
* subroutines fall into the latter category.  The The decode_* fall 
 | 
* into the second category.  The coding of the "switch(mod){ .... }" 
 | 
* in many of the subroutines below falls into the first category. 
 | 
* Especially, the coding of {add,and,or,sub,...}_{byte,word} 
 | 
* subroutines are an especially glaring case of the third guideline. 
 | 
* Since so much of the code is cloned from other modules (compare 
 | 
* opcode #00 to opcode #01), making the basic operations subroutine 
 | 
* calls is especially important; otherwise mistakes in coding an 
 | 
* "add" would represent a nightmare in maintenance. 
 | 
* 
 | 
****************************************************************************/ 
 | 
  
 | 
#include <common.h> 
 | 
#include "x86emu/x86emui.h" 
 | 
  
 | 
/*----------------------------- Implementation ----------------------------*/ 
 | 
  
 | 
/* constant arrays to do several instructions in just one function */ 
 | 
  
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
static char *x86emu_GenOpName[8] = { 
 | 
    "ADD", "OR", "ADC", "SBB", "AND", "SUB", "XOR", "CMP"}; 
 | 
#endif 
 | 
  
 | 
/* used by several opcodes  */ 
 | 
static u8 (*genop_byte_operation[])(u8 d, u8 s) = 
 | 
{ 
 | 
    add_byte,        /* 00 */ 
 | 
    or_byte,        /* 01 */ 
 | 
    adc_byte,        /* 02 */ 
 | 
    sbb_byte,        /* 03 */ 
 | 
    and_byte,        /* 04 */ 
 | 
    sub_byte,        /* 05 */ 
 | 
    xor_byte,        /* 06 */ 
 | 
    cmp_byte,        /* 07 */ 
 | 
}; 
 | 
  
 | 
static u16 (*genop_word_operation[])(u16 d, u16 s) = 
 | 
{ 
 | 
    add_word,        /*00 */ 
 | 
    or_word,        /*01 */ 
 | 
    adc_word,        /*02 */ 
 | 
    sbb_word,        /*03 */ 
 | 
    and_word,        /*04 */ 
 | 
    sub_word,        /*05 */ 
 | 
    xor_word,        /*06 */ 
 | 
    cmp_word,        /*07 */ 
 | 
}; 
 | 
  
 | 
static u32 (*genop_long_operation[])(u32 d, u32 s) = 
 | 
{ 
 | 
    add_long,        /*00 */ 
 | 
    or_long,        /*01 */ 
 | 
    adc_long,        /*02 */ 
 | 
    sbb_long,        /*03 */ 
 | 
    and_long,        /*04 */ 
 | 
    sub_long,        /*05 */ 
 | 
    xor_long,        /*06 */ 
 | 
    cmp_long,        /*07 */ 
 | 
}; 
 | 
  
 | 
/* used by opcodes 80, c0, d0, and d2. */ 
 | 
static u8(*opcD0_byte_operation[])(u8 d, u8 s) = 
 | 
{ 
 | 
    rol_byte, 
 | 
    ror_byte, 
 | 
    rcl_byte, 
 | 
    rcr_byte, 
 | 
    shl_byte, 
 | 
    shr_byte, 
 | 
    shl_byte,        /* sal_byte === shl_byte  by definition */ 
 | 
    sar_byte, 
 | 
}; 
 | 
  
 | 
/* used by opcodes c1, d1, and d3. */ 
 | 
static u16(*opcD1_word_operation[])(u16 s, u8 d) = 
 | 
{ 
 | 
    rol_word, 
 | 
    ror_word, 
 | 
    rcl_word, 
 | 
    rcr_word, 
 | 
    shl_word, 
 | 
    shr_word, 
 | 
    shl_word,        /* sal_byte === shl_byte  by definition */ 
 | 
    sar_word, 
 | 
}; 
 | 
  
 | 
/* used by opcodes c1, d1, and d3. */ 
 | 
static u32 (*opcD1_long_operation[])(u32 s, u8 d) = 
 | 
{ 
 | 
    rol_long, 
 | 
    ror_long, 
 | 
    rcl_long, 
 | 
    rcr_long, 
 | 
    shl_long, 
 | 
    shr_long, 
 | 
    shl_long,        /* sal_byte === shl_byte  by definition */ 
 | 
    sar_long, 
 | 
}; 
 | 
  
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
  
 | 
static char *opF6_names[8] = 
 | 
  { "TEST\t", "", "NOT\t", "NEG\t", "MUL\t", "IMUL\t", "DIV\t", "IDIV\t" }; 
 | 
  
 | 
#endif 
 | 
  
 | 
/**************************************************************************** 
 | 
PARAMETERS: 
 | 
op1 - Instruction op code 
 | 
  
 | 
REMARKS: 
 | 
Handles illegal opcodes. 
 | 
****************************************************************************/ 
 | 
void x86emuOp_illegal_op( 
 | 
    u8 op1) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.R_SP != 0) { 
 | 
    DB(printf("ILLEGAL X86 OPCODE\n")); 
 | 
    TRACE_REGS(); 
 | 
    DB( printk("%04x:%04x: %02X ILLEGAL X86 OPCODE!\n", 
 | 
        M.x86.R_CS, M.x86.R_IP-1,op1)); 
 | 
    HALT_SYS(); 
 | 
    } 
 | 
    else { 
 | 
    /* If we get here, it means the stack pointer is back to zero 
 | 
     * so we are just returning from an emulator service call 
 | 
     * so therte is no need to display an error message. We trap 
 | 
     * the emulator with an 0xF1 opcode to finish the service 
 | 
     * call. 
 | 
     */ 
 | 
    X86EMU_halt_sys(); 
 | 
    } 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcodes 0x00, 0x08, 0x10, 0x18, 0x20, 0x28, 0x30, 0x38 
 | 
****************************************************************************/ 
 | 
void x86emuOp_genop_byte_RM_R(u8 op1) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
    u8 *destreg, *srcreg; 
 | 
    u8 destval; 
 | 
  
 | 
    op1 = (op1 >> 3) & 0x7; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF(x86emu_GenOpName[op1]); 
 | 
    DECODE_PRINTF("\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if(mod<3) 
 | 
    { destoffset = decode_rmXX_address(mod,rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = genop_byte_operation[op1](destval, *srcreg); 
 | 
    store_data_byte(destoffset, destval); 
 | 
    } 
 | 
    else 
 | 
    {            /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *destreg = genop_byte_operation[op1](*destreg, *srcreg); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcodes 0x01, 0x09, 0x11, 0x19, 0x21, 0x29, 0x31, 0x39 
 | 
****************************************************************************/ 
 | 
void x86emuOp_genop_word_RM_R(u8 op1) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    op1 = (op1 >> 3) & 0x7; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF(x86emu_GenOpName[op1]); 
 | 
    DECODE_PRINTF("\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
  
 | 
    if(mod<3) { 
 | 
    destoffset = decode_rmXX_address(mod,rl); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval; 
 | 
        u32 *srcreg; 
 | 
  
 | 
        DECODE_PRINTF(","); 
 | 
        destval = fetch_data_long(destoffset); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = genop_long_operation[op1](destval, *srcreg); 
 | 
        store_data_long(destoffset, destval); 
 | 
    } else { 
 | 
        u16 destval; 
 | 
        u16 *srcreg; 
 | 
  
 | 
        DECODE_PRINTF(","); 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = genop_word_operation[op1](destval, *srcreg); 
 | 
        store_data_word(destoffset, destval); 
 | 
    } 
 | 
    } else {            /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg,*srcreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = genop_long_operation[op1](*destreg, *srcreg); 
 | 
    } else { 
 | 
        u16 *destreg,*srcreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = genop_word_operation[op1](*destreg, *srcreg); 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcodes 0x02, 0x0a, 0x12, 0x1a, 0x22, 0x2a, 0x32, 0x3a 
 | 
****************************************************************************/ 
 | 
void x86emuOp_genop_byte_R_RM(u8 op1) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg, *srcreg; 
 | 
    uint srcoffset; 
 | 
    u8 srcval; 
 | 
  
 | 
    op1 = (op1 >> 3) & 0x7; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF(x86emu_GenOpName[op1]); 
 | 
    DECODE_PRINTF("\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcoffset = decode_rmXX_address(mod,rl); 
 | 
    srcval = fetch_data_byte(srcoffset); 
 | 
    } else {     /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    srcval = *srcreg; 
 | 
    } 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *destreg = genop_byte_operation[op1](*destreg, srcval); 
 | 
  
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcodes 0x03, 0x0b, 0x13, 0x1b, 0x23, 0x2b, 0x33, 0x3b 
 | 
****************************************************************************/ 
 | 
void x86emuOp_genop_word_R_RM(u8 op1) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint srcoffset; 
 | 
    u32 *destreg32, srcval; 
 | 
    u16 *destreg; 
 | 
  
 | 
    op1 = (op1 >> 3) & 0x7; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF(x86emu_GenOpName[op1]); 
 | 
    DECODE_PRINTF("\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    srcoffset = decode_rmXX_address(mod,rl); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        destreg32 = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_data_long(srcoffset); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg32 = genop_long_operation[op1](*destreg32, srcval); 
 | 
    } else { 
 | 
        destreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_data_word(srcoffset); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = genop_word_operation[op1](*destreg, srcval); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *srcreg; 
 | 
        destreg32 = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg32 = genop_long_operation[op1](*destreg32, *srcreg); 
 | 
    } else { 
 | 
        u16 *srcreg; 
 | 
        destreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = genop_word_operation[op1](*destreg, *srcreg); 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcodes 0x04, 0x0c, 0x14, 0x1c, 0x24, 0x2c, 0x34, 0x3c 
 | 
****************************************************************************/ 
 | 
void x86emuOp_genop_byte_AL_IMM(u8 op1) 
 | 
{ 
 | 
    u8 srcval; 
 | 
  
 | 
    op1 = (op1 >> 3) & 0x7; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF(x86emu_GenOpName[op1]); 
 | 
    DECODE_PRINTF("\tAL,"); 
 | 
    srcval = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("%x\n", srcval); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AL = genop_byte_operation[op1](M.x86.R_AL, srcval); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcodes 0x05, 0x0d, 0x15, 0x1d, 0x25, 0x2d, 0x35, 0x3d 
 | 
****************************************************************************/ 
 | 
void x86emuOp_genop_word_AX_IMM(u8 op1) 
 | 
{ 
 | 
    u32 srcval; 
 | 
  
 | 
    op1 = (op1 >> 3) & 0x7; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF(x86emu_GenOpName[op1]); 
 | 
    DECODE_PRINTF("\tEAX,"); 
 | 
    srcval = fetch_long_imm(); 
 | 
    } else { 
 | 
    DECODE_PRINTF(x86emu_GenOpName[op1]); 
 | 
    DECODE_PRINTF("\tAX,"); 
 | 
    srcval = fetch_word_imm(); 
 | 
    } 
 | 
    DECODE_PRINTF2("%x\n", srcval); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    M.x86.R_EAX = genop_long_operation[op1](M.x86.R_EAX, srcval); 
 | 
    } else { 
 | 
    M.x86.R_AX = genop_word_operation[op1](M.x86.R_AX, (u16)srcval); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x06 
 | 
****************************************************************************/ 
 | 
void x86emuOp_push_ES(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("PUSH\tES\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    push_word(M.x86.R_ES); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x07 
 | 
****************************************************************************/ 
 | 
void x86emuOp_pop_ES(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("POP\tES\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_ES = pop_word(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x0e 
 | 
****************************************************************************/ 
 | 
void x86emuOp_push_CS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("PUSH\tCS\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    push_word(M.x86.R_CS); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x0f. Escape for two-byte opcode (286 or better) 
 | 
****************************************************************************/ 
 | 
void x86emuOp_two_byte(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u8 op2 = (*sys_rdb)(((u32)M.x86.R_CS << 4) + (M.x86.R_IP++)); 
 | 
    INC_DECODED_INST_LEN(1); 
 | 
    (*x86emu_optab2[op2])(op2); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x16 
 | 
****************************************************************************/ 
 | 
void x86emuOp_push_SS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("PUSH\tSS\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    push_word(M.x86.R_SS); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x17 
 | 
****************************************************************************/ 
 | 
void x86emuOp_pop_SS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("POP\tSS\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_SS = pop_word(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x1e 
 | 
****************************************************************************/ 
 | 
void x86emuOp_push_DS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("PUSH\tDS\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    push_word(M.x86.R_DS); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x1f 
 | 
****************************************************************************/ 
 | 
void x86emuOp_pop_DS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("POP\tDS\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_DS = pop_word(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x26 
 | 
****************************************************************************/ 
 | 
void x86emuOp_segovr_ES(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("ES:\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_SEGOVR_ES; 
 | 
    /* 
 | 
     * note the lack of DECODE_CLEAR_SEGOVR(r) since, here is one of 4 
 | 
     * opcode subroutines we do not want to do this. 
 | 
     */ 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x27 
 | 
****************************************************************************/ 
 | 
void x86emuOp_daa(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("DAA\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AL = daa_byte(M.x86.R_AL); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x2e 
 | 
****************************************************************************/ 
 | 
void x86emuOp_segovr_CS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("CS:\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_SEGOVR_CS; 
 | 
    /* note no DECODE_CLEAR_SEGOVR here. */ 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x2f 
 | 
****************************************************************************/ 
 | 
void x86emuOp_das(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("DAS\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AL = das_byte(M.x86.R_AL); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x36 
 | 
****************************************************************************/ 
 | 
void x86emuOp_segovr_SS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("SS:\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_SEGOVR_SS; 
 | 
    /* no DECODE_CLEAR_SEGOVR ! */ 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x37 
 | 
****************************************************************************/ 
 | 
void x86emuOp_aaa(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("AAA\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AX = aaa_word(M.x86.R_AX); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x3e 
 | 
****************************************************************************/ 
 | 
void x86emuOp_segovr_DS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("DS:\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_SEGOVR_DS; 
 | 
    /* NO DECODE_CLEAR_SEGOVR! */ 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x3f 
 | 
****************************************************************************/ 
 | 
void x86emuOp_aas(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("AAS\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AX = aas_word(M.x86.R_AX); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x40 - 0x47 
 | 
****************************************************************************/ 
 | 
void x86emuOp_inc_register(u8 op1) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    op1 &= 0x7; 
 | 
    DECODE_PRINTF("INC\t"); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    u32 *reg; 
 | 
    reg = DECODE_RM_LONG_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *reg = inc_long(*reg); 
 | 
    } else { 
 | 
    u16 *reg; 
 | 
    reg = DECODE_RM_WORD_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *reg = inc_word(*reg); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x48 - 0x4F 
 | 
****************************************************************************/ 
 | 
void x86emuOp_dec_register(u8 op1) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    op1 &= 0x7; 
 | 
    DECODE_PRINTF("DEC\t"); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    u32 *reg; 
 | 
    reg = DECODE_RM_LONG_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *reg = dec_long(*reg); 
 | 
    } else { 
 | 
    u16 *reg; 
 | 
    reg = DECODE_RM_WORD_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *reg = dec_word(*reg); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x50 - 0x57 
 | 
****************************************************************************/ 
 | 
void x86emuOp_push_register(u8 op1) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    op1 &= 0x7; 
 | 
    DECODE_PRINTF("PUSH\t"); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    u32 *reg; 
 | 
    reg = DECODE_RM_LONG_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    push_long(*reg); 
 | 
    } else { 
 | 
    u16 *reg; 
 | 
    reg = DECODE_RM_WORD_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    push_word(*reg); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x58 - 0x5F 
 | 
****************************************************************************/ 
 | 
void x86emuOp_pop_register(u8 op1) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    op1 &= 0x7; 
 | 
    DECODE_PRINTF("POP\t"); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    u32 *reg; 
 | 
    reg = DECODE_RM_LONG_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *reg = pop_long(); 
 | 
    } else { 
 | 
    u16 *reg; 
 | 
    reg = DECODE_RM_WORD_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *reg = pop_word(); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x60 
 | 
****************************************************************************/ 
 | 
void x86emuOp_push_all(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("PUSHAD\n"); 
 | 
    } else { 
 | 
    DECODE_PRINTF("PUSHA\n"); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    u32 old_sp = M.x86.R_ESP; 
 | 
  
 | 
    push_long(M.x86.R_EAX); 
 | 
    push_long(M.x86.R_ECX); 
 | 
    push_long(M.x86.R_EDX); 
 | 
    push_long(M.x86.R_EBX); 
 | 
    push_long(old_sp); 
 | 
    push_long(M.x86.R_EBP); 
 | 
    push_long(M.x86.R_ESI); 
 | 
    push_long(M.x86.R_EDI); 
 | 
    } else { 
 | 
    u16 old_sp = M.x86.R_SP; 
 | 
  
 | 
    push_word(M.x86.R_AX); 
 | 
    push_word(M.x86.R_CX); 
 | 
    push_word(M.x86.R_DX); 
 | 
    push_word(M.x86.R_BX); 
 | 
    push_word(old_sp); 
 | 
    push_word(M.x86.R_BP); 
 | 
    push_word(M.x86.R_SI); 
 | 
    push_word(M.x86.R_DI); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x61 
 | 
****************************************************************************/ 
 | 
void x86emuOp_pop_all(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("POPAD\n"); 
 | 
    } else { 
 | 
    DECODE_PRINTF("POPA\n"); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    M.x86.R_EDI = pop_long(); 
 | 
    M.x86.R_ESI = pop_long(); 
 | 
    M.x86.R_EBP = pop_long(); 
 | 
    M.x86.R_ESP += 4;           /* skip ESP */ 
 | 
    M.x86.R_EBX = pop_long(); 
 | 
    M.x86.R_EDX = pop_long(); 
 | 
    M.x86.R_ECX = pop_long(); 
 | 
    M.x86.R_EAX = pop_long(); 
 | 
    } else { 
 | 
    M.x86.R_DI = pop_word(); 
 | 
    M.x86.R_SI = pop_word(); 
 | 
    M.x86.R_BP = pop_word(); 
 | 
    M.x86.R_SP += 2;           /* skip SP */ 
 | 
    M.x86.R_BX = pop_word(); 
 | 
    M.x86.R_DX = pop_word(); 
 | 
    M.x86.R_CX = pop_word(); 
 | 
    M.x86.R_AX = pop_word(); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/*opcode 0x62    ILLEGAL OP, calls x86emuOp_illegal_op() */ 
 | 
/*opcode 0x63    ILLEGAL OP, calls x86emuOp_illegal_op() */ 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x64 
 | 
****************************************************************************/ 
 | 
void x86emuOp_segovr_FS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("FS:\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_SEGOVR_FS; 
 | 
    /* 
 | 
     * note the lack of DECODE_CLEAR_SEGOVR(r) since, here is one of 4 
 | 
     * opcode subroutines we do not want to do this. 
 | 
     */ 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x65 
 | 
****************************************************************************/ 
 | 
void x86emuOp_segovr_GS(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("GS:\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_SEGOVR_GS; 
 | 
    /* 
 | 
     * note the lack of DECODE_CLEAR_SEGOVR(r) since, here is one of 4 
 | 
     * opcode subroutines we do not want to do this. 
 | 
     */ 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x66 - prefix for 32-bit register 
 | 
****************************************************************************/ 
 | 
void x86emuOp_prefix_data(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("DATA:\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_PREFIX_DATA; 
 | 
    /* note no DECODE_CLEAR_SEGOVR here. */ 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x67 - prefix for 32-bit address 
 | 
****************************************************************************/ 
 | 
void x86emuOp_prefix_addr(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("ADDR:\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_PREFIX_ADDR; 
 | 
    /* note no DECODE_CLEAR_SEGOVR here. */ 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x68 
 | 
****************************************************************************/ 
 | 
void x86emuOp_push_word_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u32 imm; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    imm = fetch_long_imm(); 
 | 
    } else { 
 | 
    imm = fetch_word_imm(); 
 | 
    } 
 | 
    DECODE_PRINTF2("PUSH\t%x\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    push_long(imm); 
 | 
    } else { 
 | 
    push_word((u16)imm); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x69 
 | 
****************************************************************************/ 
 | 
void x86emuOp_imul_word_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint srcoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("IMUL\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    srcoffset = decode_rmXX_address(mod, rl); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
        u32 srcval; 
 | 
        u32 res_lo,res_hi; 
 | 
        s32 imm; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_data_long(srcoffset); 
 | 
        imm = fetch_long_imm(); 
 | 
        DECODE_PRINTF2(",%d\n", (s32)imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_long_direct(&res_lo,&res_hi,(s32)srcval,(s32)imm); 
 | 
        if ((((res_lo & 0x80000000) == 0) && (res_hi == 0x00000000)) || 
 | 
        (((res_lo & 0x80000000) != 0) && (res_hi == 0xFFFFFFFF))) { 
 | 
        CLEAR_FLAG(F_CF); 
 | 
        CLEAR_FLAG(F_OF); 
 | 
        } else { 
 | 
        SET_FLAG(F_CF); 
 | 
        SET_FLAG(F_OF); 
 | 
        } 
 | 
        *destreg = (u32)res_lo; 
 | 
    } else { 
 | 
        u16 *destreg; 
 | 
        u16 srcval; 
 | 
        u32 res; 
 | 
        s16 imm; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_data_word(srcoffset); 
 | 
        imm = fetch_word_imm(); 
 | 
        DECODE_PRINTF2(",%d\n", (s32)imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        res = (s16)srcval * (s16)imm; 
 | 
        if ((((res & 0x8000) == 0) && ((res >> 16) == 0x0000)) || 
 | 
        (((res & 0x8000) != 0) && ((res >> 16) == 0xFFFF))) { 
 | 
        CLEAR_FLAG(F_CF); 
 | 
        CLEAR_FLAG(F_OF); 
 | 
        } else { 
 | 
        SET_FLAG(F_CF); 
 | 
        SET_FLAG(F_OF); 
 | 
        } 
 | 
        *destreg = (u16)res; 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg,*srcreg; 
 | 
        u32 res_lo,res_hi; 
 | 
        s32 imm; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        imm = fetch_long_imm(); 
 | 
        DECODE_PRINTF2(",%d\n", (s32)imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_long_direct(&res_lo,&res_hi,(s32)*srcreg,(s32)imm); 
 | 
        if ((((res_lo & 0x80000000) == 0) && (res_hi == 0x00000000)) || 
 | 
        (((res_lo & 0x80000000) != 0) && (res_hi == 0xFFFFFFFF))) { 
 | 
        CLEAR_FLAG(F_CF); 
 | 
        CLEAR_FLAG(F_OF); 
 | 
        } else { 
 | 
        SET_FLAG(F_CF); 
 | 
        SET_FLAG(F_OF); 
 | 
        } 
 | 
        *destreg = (u32)res_lo; 
 | 
    } else { 
 | 
        u16 *destreg,*srcreg; 
 | 
        u32 res; 
 | 
        s16 imm; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        imm = fetch_word_imm(); 
 | 
        DECODE_PRINTF2(",%d\n", (s32)imm); 
 | 
        res = (s16)*srcreg * (s16)imm; 
 | 
        if ((((res & 0x8000) == 0) && ((res >> 16) == 0x0000)) || 
 | 
        (((res & 0x8000) != 0) && ((res >> 16) == 0xFFFF))) { 
 | 
        CLEAR_FLAG(F_CF); 
 | 
        CLEAR_FLAG(F_OF); 
 | 
        } else { 
 | 
        SET_FLAG(F_CF); 
 | 
        SET_FLAG(F_OF); 
 | 
        } 
 | 
        *destreg = (u16)res; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x6a 
 | 
****************************************************************************/ 
 | 
void x86emuOp_push_byte_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    s16 imm; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    imm = (s8)fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("PUSH\t%d\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    push_word(imm); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x6b 
 | 
****************************************************************************/ 
 | 
void x86emuOp_imul_byte_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint srcoffset; 
 | 
    s8    imm; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("IMUL\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    srcoffset = decode_rmXX_address(mod, rl); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
        u32 srcval; 
 | 
        u32 res_lo,res_hi; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_data_long(srcoffset); 
 | 
        imm = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%d\n", (s32)imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_long_direct(&res_lo,&res_hi,(s32)srcval,(s32)imm); 
 | 
        if ((((res_lo & 0x80000000) == 0) && (res_hi == 0x00000000)) || 
 | 
        (((res_lo & 0x80000000) != 0) && (res_hi == 0xFFFFFFFF))) { 
 | 
        CLEAR_FLAG(F_CF); 
 | 
        CLEAR_FLAG(F_OF); 
 | 
        } else { 
 | 
        SET_FLAG(F_CF); 
 | 
        SET_FLAG(F_OF); 
 | 
        } 
 | 
        *destreg = (u32)res_lo; 
 | 
    } else { 
 | 
        u16 *destreg; 
 | 
        u16 srcval; 
 | 
        u32 res; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_data_word(srcoffset); 
 | 
        imm = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%d\n", (s32)imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        res = (s16)srcval * (s16)imm; 
 | 
        if ((((res & 0x8000) == 0) && ((res >> 16) == 0x0000)) || 
 | 
        (((res & 0x8000) != 0) && ((res >> 16) == 0xFFFF))) { 
 | 
        CLEAR_FLAG(F_CF); 
 | 
        CLEAR_FLAG(F_OF); 
 | 
        } else { 
 | 
        SET_FLAG(F_CF); 
 | 
        SET_FLAG(F_OF); 
 | 
        } 
 | 
        *destreg = (u16)res; 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg,*srcreg; 
 | 
        u32 res_lo,res_hi; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        imm = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%d\n", (s32)imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_long_direct(&res_lo,&res_hi,(s32)*srcreg,(s32)imm); 
 | 
        if ((((res_lo & 0x80000000) == 0) && (res_hi == 0x00000000)) || 
 | 
        (((res_lo & 0x80000000) != 0) && (res_hi == 0xFFFFFFFF))) { 
 | 
        CLEAR_FLAG(F_CF); 
 | 
        CLEAR_FLAG(F_OF); 
 | 
        } else { 
 | 
        SET_FLAG(F_CF); 
 | 
        SET_FLAG(F_OF); 
 | 
        } 
 | 
        *destreg = (u32)res_lo; 
 | 
    } else { 
 | 
        u16 *destreg,*srcreg; 
 | 
        u32 res; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        imm = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%d\n", (s32)imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        res = (s16)*srcreg * (s16)imm; 
 | 
        if ((((res & 0x8000) == 0) && ((res >> 16) == 0x0000)) || 
 | 
        (((res & 0x8000) != 0) && ((res >> 16) == 0xFFFF))) { 
 | 
        CLEAR_FLAG(F_CF); 
 | 
        CLEAR_FLAG(F_OF); 
 | 
        } else { 
 | 
        SET_FLAG(F_CF); 
 | 
        SET_FLAG(F_OF); 
 | 
        } 
 | 
        *destreg = (u16)res; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x6c 
 | 
****************************************************************************/ 
 | 
void x86emuOp_ins_byte(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("INSB\n"); 
 | 
    ins(1); 
 | 
    TRACE_AND_STEP(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x6d 
 | 
****************************************************************************/ 
 | 
void x86emuOp_ins_word(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("INSD\n"); 
 | 
    ins(4); 
 | 
    } else { 
 | 
    DECODE_PRINTF("INSW\n"); 
 | 
    ins(2); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x6e 
 | 
****************************************************************************/ 
 | 
void x86emuOp_outs_byte(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("OUTSB\n"); 
 | 
    outs(1); 
 | 
    TRACE_AND_STEP(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x6f 
 | 
****************************************************************************/ 
 | 
void x86emuOp_outs_word(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("OUTSD\n"); 
 | 
    outs(4); 
 | 
    } else { 
 | 
    DECODE_PRINTF("OUTSW\n"); 
 | 
    outs(2); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x70 - 0x7F 
 | 
****************************************************************************/ 
 | 
int x86emu_check_jump_condition(u8 op); 
 | 
  
 | 
void x86emuOp_jump_near_cond(u8 op1) 
 | 
{ 
 | 
    s8 offset; 
 | 
    u16 target; 
 | 
    int cond; 
 | 
  
 | 
    /* jump to byte offset if overflow flag is set */ 
 | 
    START_OF_INSTR(); 
 | 
    cond = x86emu_check_jump_condition(op1 & 0xF); 
 | 
    offset = (s8)fetch_byte_imm(); 
 | 
    target = (u16)(M.x86.R_IP + (s16)offset); 
 | 
    DECODE_PRINTF2("%x\n", target); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (cond) 
 | 
    M.x86.R_IP = target; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x80 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opc80_byte_RM_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg; 
 | 
    uint destoffset; 
 | 
    u8 imm; 
 | 
    u8 destval; 
 | 
  
 | 
    /* 
 | 
     * Weirdo special case instruction format.    Part of the opcode 
 | 
     * held below in "RH".  Doubly nested case would result, except 
 | 
     * that the decoded instruction 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
  
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ADD\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("OR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("ADC\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("SBB\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("AND\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SUB\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("XOR\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("CMP\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* know operation, decode the mod byte to find the addressing 
 | 
       mode. */ 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("BYTE PTR "); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
    imm = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("%x\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*genop_byte_operation[rh]) (destval, imm); 
 | 
    if (rh != 7) 
 | 
        store_data_byte(destoffset, destval); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    imm = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("%x\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*genop_byte_operation[rh]) (*destreg, imm); 
 | 
    if (rh != 7) 
 | 
        *destreg = destval; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x81 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opc81_word_RM_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    /* 
 | 
     * Weirdo special case instruction format.    Part of the opcode 
 | 
     * held below in "RH".  Doubly nested case would result, except 
 | 
     * that the decoded instruction 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
  
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ADD\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("OR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("ADC\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("SBB\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("AND\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SUB\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("XOR\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("CMP\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* 
 | 
     * Know operation, decode the mod byte to find the addressing 
 | 
     * mode. 
 | 
     */ 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("DWORD PTR "); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval,imm; 
 | 
  
 | 
        DECODE_PRINTF(","); 
 | 
        destval = fetch_data_long(destoffset); 
 | 
        imm = fetch_long_imm(); 
 | 
        DECODE_PRINTF2("%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*genop_long_operation[rh]) (destval, imm); 
 | 
        if (rh != 7) 
 | 
        store_data_long(destoffset, destval); 
 | 
    } else { 
 | 
        u16 destval,imm; 
 | 
  
 | 
        DECODE_PRINTF(","); 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        imm = fetch_word_imm(); 
 | 
        DECODE_PRINTF2("%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*genop_word_operation[rh]) (destval, imm); 
 | 
        if (rh != 7) 
 | 
        store_data_word(destoffset, destval); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
        u32 destval,imm; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        imm = fetch_long_imm(); 
 | 
        DECODE_PRINTF2("%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*genop_long_operation[rh]) (*destreg, imm); 
 | 
        if (rh != 7) 
 | 
        *destreg = destval; 
 | 
    } else { 
 | 
        u16 *destreg; 
 | 
        u16 destval,imm; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        imm = fetch_word_imm(); 
 | 
        DECODE_PRINTF2("%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*genop_word_operation[rh]) (*destreg, imm); 
 | 
        if (rh != 7) 
 | 
        *destreg = destval; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x82 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opc82_byte_RM_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg; 
 | 
    uint destoffset; 
 | 
    u8 imm; 
 | 
    u8 destval; 
 | 
  
 | 
    /* 
 | 
     * Weirdo special case instruction format.    Part of the opcode 
 | 
     * held below in "RH".  Doubly nested case would result, except 
 | 
     * that the decoded instruction Similar to opcode 81, except that 
 | 
     * the immediate byte is sign extended to a word length. 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ADD\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("OR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("ADC\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("SBB\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("AND\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SUB\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("XOR\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("CMP\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* know operation, decode the mod byte to find the addressing 
 | 
       mode. */ 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("BYTE PTR "); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
    imm = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2(",%x\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*genop_byte_operation[rh]) (destval, imm); 
 | 
    if (rh != 7) 
 | 
        store_data_byte(destoffset, destval); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    imm = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2(",%x\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*genop_byte_operation[rh]) (*destreg, imm); 
 | 
    if (rh != 7) 
 | 
        *destreg = destval; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x83 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opc83_word_RM_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    /* 
 | 
     * Weirdo special case instruction format.    Part of the opcode 
 | 
     * held below in "RH".  Doubly nested case would result, except 
 | 
     * that the decoded instruction Similar to opcode 81, except that 
 | 
     * the immediate byte is sign extended to a word length. 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
       switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ADD\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("OR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("ADC\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("SBB\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("AND\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SUB\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("XOR\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("CMP\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* know operation, decode the mod byte to find the addressing 
 | 
       mode. */ 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("DWORD PTR "); 
 | 
    destoffset = decode_rmXX_address(mod,rl); 
 | 
  
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval,imm; 
 | 
  
 | 
        destval = fetch_data_long(destoffset); 
 | 
        imm = (s8) fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*genop_long_operation[rh]) (destval, imm); 
 | 
        if (rh != 7) 
 | 
        store_data_long(destoffset, destval); 
 | 
    } else { 
 | 
        u16 destval,imm; 
 | 
  
 | 
        destval = fetch_data_word(destoffset); 
 | 
        imm = (s8) fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*genop_word_operation[rh]) (destval, imm); 
 | 
        if (rh != 7) 
 | 
        store_data_word(destoffset, destval); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
        u32 destval,imm; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        imm = (s8) fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*genop_long_operation[rh]) (*destreg, imm); 
 | 
        if (rh != 7) 
 | 
        *destreg = destval; 
 | 
    } else { 
 | 
        u16 *destreg; 
 | 
        u16 destval,imm; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        imm = (s8) fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*genop_word_operation[rh]) (*destreg, imm); 
 | 
        if (rh != 7) 
 | 
        *destreg = destval; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x84 
 | 
****************************************************************************/ 
 | 
void x86emuOp_test_byte_RM_R(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg, *srcreg; 
 | 
    uint destoffset; 
 | 
    u8 destval; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("TEST\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    test_byte(destval, *srcreg); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    test_byte(*destreg, *srcreg); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x85 
 | 
****************************************************************************/ 
 | 
void x86emuOp_test_word_RM_R(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("TEST\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval; 
 | 
        u32 *srcreg; 
 | 
  
 | 
        DECODE_PRINTF(","); 
 | 
        destval = fetch_data_long(destoffset); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_long(destval, *srcreg); 
 | 
    } else { 
 | 
        u16 destval; 
 | 
        u16 *srcreg; 
 | 
  
 | 
        DECODE_PRINTF(","); 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_word(destval, *srcreg); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg,*srcreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_long(*destreg, *srcreg); 
 | 
    } else { 
 | 
        u16 *destreg,*srcreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_word(*destreg, *srcreg); 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x86 
 | 
****************************************************************************/ 
 | 
void x86emuOp_xchg_byte_RM_R(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg, *srcreg; 
 | 
    uint destoffset; 
 | 
    u8 destval; 
 | 
    u8 tmp; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("XCHG\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    tmp = *srcreg; 
 | 
    *srcreg = destval; 
 | 
    destval = tmp; 
 | 
    store_data_byte(destoffset, destval); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    tmp = *srcreg; 
 | 
    *srcreg = *destreg; 
 | 
    *destreg = tmp; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x87 
 | 
****************************************************************************/ 
 | 
void x86emuOp_xchg_word_RM_R(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("XCHG\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *srcreg; 
 | 
        u32 destval,tmp; 
 | 
  
 | 
        destval = fetch_data_long(destoffset); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        tmp = *srcreg; 
 | 
        *srcreg = destval; 
 | 
        destval = tmp; 
 | 
        store_data_long(destoffset, destval); 
 | 
    } else { 
 | 
        u16 *srcreg; 
 | 
        u16 destval,tmp; 
 | 
  
 | 
        destval = fetch_data_word(destoffset); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        tmp = *srcreg; 
 | 
        *srcreg = destval; 
 | 
        destval = tmp; 
 | 
        store_data_word(destoffset, destval); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg,*srcreg; 
 | 
        u32 tmp; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        tmp = *srcreg; 
 | 
        *srcreg = *destreg; 
 | 
        *destreg = tmp; 
 | 
    } else { 
 | 
        u16 *destreg,*srcreg; 
 | 
        u16 tmp; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        tmp = *srcreg; 
 | 
        *srcreg = *destreg; 
 | 
        *destreg = tmp; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x88 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_byte_RM_R(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg, *srcreg; 
 | 
    uint destoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    store_data_byte(destoffset, *srcreg); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *destreg = *srcreg; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x89 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_word_RM_R(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *srcreg; 
 | 
  
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        store_data_long(destoffset, *srcreg); 
 | 
    } else { 
 | 
        u16 *srcreg; 
 | 
  
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        store_data_word(destoffset, *srcreg); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg,*srcreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = *srcreg; 
 | 
    } else { 
 | 
        u16 *destreg,*srcreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = *srcreg; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x8a 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_byte_R_RM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg, *srcreg; 
 | 
    uint srcoffset; 
 | 
    u8 srcval; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcoffset = decode_rmXX_address(mod, rl); 
 | 
    srcval = fetch_data_byte(srcoffset); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *destreg = srcval; 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rh); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *destreg = *srcreg; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x8b 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_word_R_RM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint srcoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
        u32 srcval; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcoffset = decode_rmXX_address(mod, rl); 
 | 
        srcval = fetch_data_long(srcoffset); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = srcval; 
 | 
    } else { 
 | 
        u16 *destreg; 
 | 
        u16 srcval; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcoffset = decode_rmXX_address(mod, rl); 
 | 
        srcval = fetch_data_word(srcoffset); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = srcval; 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg, *srcreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = *srcreg; 
 | 
    } else { 
 | 
        u16 *destreg, *srcreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
        DECODE_PRINTF(","); 
 | 
        srcreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = *srcreg; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x8c 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_word_RM_SR(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u16 *destreg, *srcreg; 
 | 
    uint destoffset; 
 | 
    u16 destval; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = decode_rm_seg_register(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = *srcreg; 
 | 
    store_data_word(destoffset, destval); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = decode_rm_seg_register(rh); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *destreg = *srcreg; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x8d 
 | 
****************************************************************************/ 
 | 
void x86emuOp_lea_word_R_M(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u16 *srcreg; 
 | 
    uint destoffset; 
 | 
  
 | 
/* 
 | 
 * TODO: Need to handle address size prefix! 
 | 
 * 
 | 
 * lea    eax,[eax+ebx*2] ?? 
 | 
 */ 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LEA\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    srcreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
    DECODE_PRINTF(","); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *srcreg = (u16)destoffset; 
 | 
    } 
 | 
    /* } else { undefined.  Do nothing. } */ 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x8e 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_word_SR_RM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u16 *destreg, *srcreg; 
 | 
    uint srcoffset; 
 | 
    u16 srcval; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    destreg = decode_rm_seg_register(rh); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcoffset = decode_rmXX_address(mod, rl); 
 | 
    srcval = fetch_data_word(srcoffset); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *destreg = srcval; 
 | 
    } else {             /* register to register */ 
 | 
    destreg = decode_rm_seg_register(rh); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *destreg = *srcreg; 
 | 
    } 
 | 
    /* 
 | 
     * Clean up, and reset all the R_xSP pointers to the correct 
 | 
     * locations.  This is about 3x too much overhead (doing all the 
 | 
     * segreg ptrs when only one is needed, but this instruction 
 | 
     * *cannot* be that common, and this isn't too much work anyway. 
 | 
     */ 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x8f 
 | 
****************************************************************************/ 
 | 
void x86emuOp_pop_RM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("POP\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (rh != 0) { 
 | 
    ERR_PRINTF("ILLEGAL DECODE OF OPCODE 8F\n"); 
 | 
    HALT_SYS(); 
 | 
    } 
 | 
    if (mod < 3) { 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval; 
 | 
  
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = pop_long(); 
 | 
        store_data_long(destoffset, destval); 
 | 
    } else { 
 | 
        u16 destval; 
 | 
  
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = pop_word(); 
 | 
        store_data_word(destoffset, destval); 
 | 
    } 
 | 
    } else {            /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = pop_long(); 
 | 
    } else { 
 | 
        u16 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = pop_word(); 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x90 
 | 
****************************************************************************/ 
 | 
void x86emuOp_nop(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("NOP\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x91-0x97 
 | 
****************************************************************************/ 
 | 
void x86emuOp_xchg_word_AX_register(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u32 tmp; 
 | 
  
 | 
    op1 &= 0x7; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
  
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    u32 *reg32; 
 | 
    DECODE_PRINTF("XCHG\tEAX,"); 
 | 
    reg32 = DECODE_RM_LONG_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    tmp = M.x86.R_EAX; 
 | 
    M.x86.R_EAX = *reg32; 
 | 
    *reg32 = tmp; 
 | 
    } else { 
 | 
    u16 *reg16; 
 | 
    DECODE_PRINTF("XCHG\tAX,"); 
 | 
    reg16 = DECODE_RM_WORD_REGISTER(op1); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    tmp = M.x86.R_AX; 
 | 
    M.x86.R_EAX = *reg16; 
 | 
    *reg16 = (u16)tmp; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x98 
 | 
****************************************************************************/ 
 | 
void x86emuOp_cbw(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("CWDE\n"); 
 | 
    } else { 
 | 
    DECODE_PRINTF("CBW\n"); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    if (M.x86.R_AX & 0x8000) { 
 | 
        M.x86.R_EAX |= 0xffff0000; 
 | 
    } else { 
 | 
        M.x86.R_EAX &= 0x0000ffff; 
 | 
    } 
 | 
    } else { 
 | 
    if (M.x86.R_AL & 0x80) { 
 | 
        M.x86.R_AH = 0xff; 
 | 
    } else { 
 | 
        M.x86.R_AH = 0x0; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x99 
 | 
****************************************************************************/ 
 | 
void x86emuOp_cwd(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("CDQ\n"); 
 | 
    } else { 
 | 
    DECODE_PRINTF("CWD\n"); 
 | 
    } 
 | 
    DECODE_PRINTF("CWD\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    if (M.x86.R_EAX & 0x80000000) { 
 | 
        M.x86.R_EDX = 0xffffffff; 
 | 
    } else { 
 | 
        M.x86.R_EDX = 0x0; 
 | 
    } 
 | 
    } else { 
 | 
    if (M.x86.R_AX & 0x8000) { 
 | 
        M.x86.R_DX = 0xffff; 
 | 
    } else { 
 | 
        M.x86.R_DX = 0x0; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x9a 
 | 
****************************************************************************/ 
 | 
void x86emuOp_call_far_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 farseg, faroff; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("CALL\t"); 
 | 
    faroff = fetch_word_imm(); 
 | 
    farseg = fetch_word_imm(); 
 | 
    DECODE_PRINTF2("%04x:", farseg); 
 | 
    DECODE_PRINTF2("%04x\n", faroff); 
 | 
    CALL_TRACE(M.x86.saved_cs, M.x86.saved_ip, farseg, faroff, "FAR "); 
 | 
  
 | 
    /* XXX 
 | 
     * 
 | 
     * Hooked interrupt vectors calling into our "BIOS" will cause 
 | 
     * problems unless all intersegment stuff is checked for BIOS 
 | 
     * access.    Check needed here.  For moment, let it alone. 
 | 
     */ 
 | 
    TRACE_AND_STEP(); 
 | 
    push_word(M.x86.R_CS); 
 | 
    M.x86.R_CS = farseg; 
 | 
    push_word(M.x86.R_IP); 
 | 
    M.x86.R_IP = faroff; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x9b 
 | 
****************************************************************************/ 
 | 
void x86emuOp_wait(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("WAIT"); 
 | 
    TRACE_AND_STEP(); 
 | 
    /* NADA.  */ 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x9c 
 | 
****************************************************************************/ 
 | 
void x86emuOp_pushf_word(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u32 flags; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("PUSHFD\n"); 
 | 
    } else { 
 | 
    DECODE_PRINTF("PUSHF\n"); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
  
 | 
    /* clear out *all* bits not representing flags, and turn on real bits */ 
 | 
    flags = (M.x86.R_EFLG & F_MSK) | F_ALWAYS_ON; 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    push_long(flags); 
 | 
    } else { 
 | 
    push_word((u16)flags); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x9d 
 | 
****************************************************************************/ 
 | 
void x86emuOp_popf_word(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("POPFD\n"); 
 | 
    } else { 
 | 
    DECODE_PRINTF("POPF\n"); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    M.x86.R_EFLG = pop_long(); 
 | 
    } else { 
 | 
    M.x86.R_FLG = pop_word(); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x9e 
 | 
****************************************************************************/ 
 | 
void x86emuOp_sahf(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("SAHF\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    /* clear the lower bits of the flag register */ 
 | 
    M.x86.R_FLG &= 0xffffff00; 
 | 
    /* or in the AH register into the flags register */ 
 | 
    M.x86.R_FLG |= M.x86.R_AH; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0x9f 
 | 
****************************************************************************/ 
 | 
void x86emuOp_lahf(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LAHF\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AH = (u8)(M.x86.R_FLG & 0xff); 
 | 
    /*undocumented TC++ behavior??? Nope.  It's documented, but 
 | 
       you have too look real hard to notice it. */ 
 | 
    M.x86.R_AH |= 0x2; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa0 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_AL_M_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 offset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\tAL,"); 
 | 
    offset = fetch_word_imm(); 
 | 
    DECODE_PRINTF2("[%04x]\n", offset); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AL = fetch_data_byte(offset); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa1 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_AX_M_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 offset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    offset = fetch_word_imm(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF2("MOV\tEAX,[%04x]\n", offset); 
 | 
    } else { 
 | 
    DECODE_PRINTF2("MOV\tAX,[%04x]\n", offset); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    M.x86.R_EAX = fetch_data_long(offset); 
 | 
    } else { 
 | 
    M.x86.R_AX = fetch_data_word(offset); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa2 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_M_AL_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 offset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    offset = fetch_word_imm(); 
 | 
    DECODE_PRINTF2("[%04x],AL\n", offset); 
 | 
    TRACE_AND_STEP(); 
 | 
    store_data_byte(offset, M.x86.R_AL); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa3 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_M_AX_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 offset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    offset = fetch_word_imm(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF2("MOV\t[%04x],EAX\n", offset); 
 | 
    } else { 
 | 
    DECODE_PRINTF2("MOV\t[%04x],AX\n", offset); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    store_data_long(offset, M.x86.R_EAX); 
 | 
    } else { 
 | 
    store_data_word(offset, M.x86.R_AX); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa4 
 | 
****************************************************************************/ 
 | 
void x86emuOp_movs_byte(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u8    val; 
 | 
    u32 count; 
 | 
    int inc; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOVS\tBYTE\n"); 
 | 
    if (ACCESS_FLAG(F_DF))   /* down */ 
 | 
    inc = -1; 
 | 
    else 
 | 
    inc = 1; 
 | 
    TRACE_AND_STEP(); 
 | 
    count = 1; 
 | 
    if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) { 
 | 
    /* dont care whether REPE or REPNE */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    count = M.x86.R_CX; 
 | 
    M.x86.R_CX = 0; 
 | 
    M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE); 
 | 
    } 
 | 
    while (count--) { 
 | 
    val = fetch_data_byte(M.x86.R_SI); 
 | 
    store_data_byte_abs(M.x86.R_ES, M.x86.R_DI, val); 
 | 
    M.x86.R_SI += inc; 
 | 
    M.x86.R_DI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa5 
 | 
****************************************************************************/ 
 | 
void x86emuOp_movs_word(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u32 val; 
 | 
    int inc; 
 | 
    u32 count; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("MOVS\tDWORD\n"); 
 | 
    if (ACCESS_FLAG(F_DF))        /* down */ 
 | 
        inc = -4; 
 | 
    else 
 | 
        inc = 4; 
 | 
    } else { 
 | 
    DECODE_PRINTF("MOVS\tWORD\n"); 
 | 
    if (ACCESS_FLAG(F_DF))        /* down */ 
 | 
        inc = -2; 
 | 
    else 
 | 
        inc = 2; 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    count = 1; 
 | 
    if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) { 
 | 
    /* dont care whether REPE or REPNE */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    count = M.x86.R_CX; 
 | 
    M.x86.R_CX = 0; 
 | 
    M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE); 
 | 
    } 
 | 
    while (count--) { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        val = fetch_data_long(M.x86.R_SI); 
 | 
        store_data_long_abs(M.x86.R_ES, M.x86.R_DI, val); 
 | 
    } else { 
 | 
        val = fetch_data_word(M.x86.R_SI); 
 | 
        store_data_word_abs(M.x86.R_ES, M.x86.R_DI, (u16)val); 
 | 
    } 
 | 
    M.x86.R_SI += inc; 
 | 
    M.x86.R_DI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa6 
 | 
****************************************************************************/ 
 | 
void x86emuOp_cmps_byte(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    s8 val1, val2; 
 | 
    int inc; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("CMPS\tBYTE\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (ACCESS_FLAG(F_DF))   /* down */ 
 | 
    inc = -1; 
 | 
    else 
 | 
    inc = 1; 
 | 
  
 | 
    if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) { 
 | 
    /* REPE     */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    while (M.x86.R_CX != 0) { 
 | 
        val1 = fetch_data_byte(M.x86.R_SI); 
 | 
        val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
             cmp_byte(val1, val2); 
 | 
        M.x86.R_CX -= 1; 
 | 
        M.x86.R_SI += inc; 
 | 
        M.x86.R_DI += inc; 
 | 
        if ( (M.x86.mode & SYSMODE_PREFIX_REPE) && (ACCESS_FLAG(F_ZF) == 0) ) break; 
 | 
        if ( (M.x86.mode & SYSMODE_PREFIX_REPNE) && ACCESS_FLAG(F_ZF) ) break; 
 | 
    } 
 | 
    M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE); 
 | 
    } else { 
 | 
    val1 = fetch_data_byte(M.x86.R_SI); 
 | 
    val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
    cmp_byte(val1, val2); 
 | 
    M.x86.R_SI += inc; 
 | 
    M.x86.R_DI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa7 
 | 
****************************************************************************/ 
 | 
void x86emuOp_cmps_word(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u32 val1,val2; 
 | 
    int inc; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("CMPS\tDWORD\n"); 
 | 
    inc = 4; 
 | 
    } else { 
 | 
    DECODE_PRINTF("CMPS\tWORD\n"); 
 | 
    inc = 2; 
 | 
    } 
 | 
    if (ACCESS_FLAG(F_DF))   /* down */ 
 | 
    inc = -inc; 
 | 
  
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) { 
 | 
    /* REPE     */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    while (M.x86.R_CX != 0) { 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        val1 = fetch_data_long(M.x86.R_SI); 
 | 
        val2 = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_long(val1, val2); 
 | 
        } else { 
 | 
        val1 = fetch_data_word(M.x86.R_SI); 
 | 
        val2 = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_word((u16)val1, (u16)val2); 
 | 
        } 
 | 
        M.x86.R_CX -= 1; 
 | 
        M.x86.R_SI += inc; 
 | 
        M.x86.R_DI += inc; 
 | 
        if ( (M.x86.mode & SYSMODE_PREFIX_REPE) && ACCESS_FLAG(F_ZF) == 0 ) break; 
 | 
        if ( (M.x86.mode & SYSMODE_PREFIX_REPNE) && ACCESS_FLAG(F_ZF) ) break; 
 | 
    } 
 | 
    M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE); 
 | 
    } else { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        val1 = fetch_data_long(M.x86.R_SI); 
 | 
        val2 = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_long(val1, val2); 
 | 
    } else { 
 | 
        val1 = fetch_data_word(M.x86.R_SI); 
 | 
        val2 = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_word((u16)val1, (u16)val2); 
 | 
    } 
 | 
    M.x86.R_SI += inc; 
 | 
    M.x86.R_DI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa8 
 | 
****************************************************************************/ 
 | 
void x86emuOp_test_AL_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int imm; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("TEST\tAL,"); 
 | 
    imm = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("%04x\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    test_byte(M.x86.R_AL, (u8)imm); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xa9 
 | 
****************************************************************************/ 
 | 
void x86emuOp_test_AX_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u32 srcval; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("TEST\tEAX,"); 
 | 
    srcval = fetch_long_imm(); 
 | 
    } else { 
 | 
    DECODE_PRINTF("TEST\tAX,"); 
 | 
    srcval = fetch_word_imm(); 
 | 
    } 
 | 
    DECODE_PRINTF2("%x\n", srcval); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    test_long(M.x86.R_EAX, srcval); 
 | 
    } else { 
 | 
    test_word(M.x86.R_AX, (u16)srcval); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xaa 
 | 
****************************************************************************/ 
 | 
void x86emuOp_stos_byte(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int inc; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("STOS\tBYTE\n"); 
 | 
    if (ACCESS_FLAG(F_DF))   /* down */ 
 | 
    inc = -1; 
 | 
    else 
 | 
    inc = 1; 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) { 
 | 
    /* dont care whether REPE or REPNE */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    while (M.x86.R_CX != 0) { 
 | 
        store_data_byte_abs(M.x86.R_ES, M.x86.R_DI, M.x86.R_AL); 
 | 
        M.x86.R_CX -= 1; 
 | 
        M.x86.R_DI += inc; 
 | 
    } 
 | 
    M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE); 
 | 
    } else { 
 | 
    store_data_byte_abs(M.x86.R_ES, M.x86.R_DI, M.x86.R_AL); 
 | 
    M.x86.R_DI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xab 
 | 
****************************************************************************/ 
 | 
void x86emuOp_stos_word(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int inc; 
 | 
    u32 count; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("STOS\tDWORD\n"); 
 | 
    if (ACCESS_FLAG(F_DF))     /* down */ 
 | 
        inc = -4; 
 | 
    else 
 | 
        inc = 4; 
 | 
    } else { 
 | 
    DECODE_PRINTF("STOS\tWORD\n"); 
 | 
    if (ACCESS_FLAG(F_DF))     /* down */ 
 | 
        inc = -2; 
 | 
    else 
 | 
        inc = 2; 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    count = 1; 
 | 
    if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) { 
 | 
    /* dont care whether REPE or REPNE */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    count = M.x86.R_CX; 
 | 
    M.x86.R_CX = 0; 
 | 
    M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE); 
 | 
    } 
 | 
    while (count--) { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        store_data_long_abs(M.x86.R_ES, M.x86.R_DI, M.x86.R_EAX); 
 | 
    } else { 
 | 
        store_data_word_abs(M.x86.R_ES, M.x86.R_DI, M.x86.R_AX); 
 | 
    } 
 | 
    M.x86.R_DI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xac 
 | 
****************************************************************************/ 
 | 
void x86emuOp_lods_byte(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int inc; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LODS\tBYTE\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (ACCESS_FLAG(F_DF))   /* down */ 
 | 
    inc = -1; 
 | 
    else 
 | 
    inc = 1; 
 | 
    if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) { 
 | 
    /* dont care whether REPE or REPNE */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    while (M.x86.R_CX != 0) { 
 | 
        M.x86.R_AL = fetch_data_byte(M.x86.R_SI); 
 | 
        M.x86.R_CX -= 1; 
 | 
        M.x86.R_SI += inc; 
 | 
    } 
 | 
    M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE); 
 | 
    } else { 
 | 
    M.x86.R_AL = fetch_data_byte(M.x86.R_SI); 
 | 
    M.x86.R_SI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xad 
 | 
****************************************************************************/ 
 | 
void x86emuOp_lods_word(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int inc; 
 | 
    u32 count; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("LODS\tDWORD\n"); 
 | 
    if (ACCESS_FLAG(F_DF))     /* down */ 
 | 
        inc = -4; 
 | 
    else 
 | 
        inc = 4; 
 | 
    } else { 
 | 
    DECODE_PRINTF("LODS\tWORD\n"); 
 | 
    if (ACCESS_FLAG(F_DF))     /* down */ 
 | 
        inc = -2; 
 | 
    else 
 | 
        inc = 2; 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    count = 1; 
 | 
    if (M.x86.mode & (SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE)) { 
 | 
    /* dont care whether REPE or REPNE */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    count = M.x86.R_CX; 
 | 
    M.x86.R_CX = 0; 
 | 
    M.x86.mode &= ~(SYSMODE_PREFIX_REPE | SYSMODE_PREFIX_REPNE); 
 | 
    } 
 | 
    while (count--) { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        M.x86.R_EAX = fetch_data_long(M.x86.R_SI); 
 | 
    } else { 
 | 
        M.x86.R_AX = fetch_data_word(M.x86.R_SI); 
 | 
    } 
 | 
    M.x86.R_SI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xae 
 | 
****************************************************************************/ 
 | 
void x86emuOp_scas_byte(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    s8 val2; 
 | 
    int inc; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("SCAS\tBYTE\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (ACCESS_FLAG(F_DF))   /* down */ 
 | 
    inc = -1; 
 | 
    else 
 | 
    inc = 1; 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_REPE) { 
 | 
    /* REPE     */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    while (M.x86.R_CX != 0) { 
 | 
        val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_byte(M.x86.R_AL, val2); 
 | 
        M.x86.R_CX -= 1; 
 | 
        M.x86.R_DI += inc; 
 | 
        if (ACCESS_FLAG(F_ZF) == 0) 
 | 
        break; 
 | 
    } 
 | 
    M.x86.mode &= ~SYSMODE_PREFIX_REPE; 
 | 
    } else if (M.x86.mode & SYSMODE_PREFIX_REPNE) { 
 | 
    /* REPNE  */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    while (M.x86.R_CX != 0) { 
 | 
        val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_byte(M.x86.R_AL, val2); 
 | 
        M.x86.R_CX -= 1; 
 | 
        M.x86.R_DI += inc; 
 | 
        if (ACCESS_FLAG(F_ZF)) 
 | 
        break;        /* zero flag set means equal */ 
 | 
    } 
 | 
    M.x86.mode &= ~SYSMODE_PREFIX_REPNE; 
 | 
    } else { 
 | 
    val2 = fetch_data_byte_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
    cmp_byte(M.x86.R_AL, val2); 
 | 
    M.x86.R_DI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xaf 
 | 
****************************************************************************/ 
 | 
void x86emuOp_scas_word(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int inc; 
 | 
    u32 val; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("SCAS\tDWORD\n"); 
 | 
    if (ACCESS_FLAG(F_DF))     /* down */ 
 | 
        inc = -4; 
 | 
    else 
 | 
        inc = 4; 
 | 
    } else { 
 | 
    DECODE_PRINTF("SCAS\tWORD\n"); 
 | 
    if (ACCESS_FLAG(F_DF))     /* down */ 
 | 
        inc = -2; 
 | 
    else 
 | 
        inc = 2; 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_REPE) { 
 | 
    /* REPE     */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    while (M.x86.R_CX != 0) { 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        val = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_long(M.x86.R_EAX, val); 
 | 
        } else { 
 | 
        val = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_word(M.x86.R_AX, (u16)val); 
 | 
        } 
 | 
        M.x86.R_CX -= 1; 
 | 
        M.x86.R_DI += inc; 
 | 
        if (ACCESS_FLAG(F_ZF) == 0) 
 | 
        break; 
 | 
    } 
 | 
    M.x86.mode &= ~SYSMODE_PREFIX_REPE; 
 | 
    } else if (M.x86.mode & SYSMODE_PREFIX_REPNE) { 
 | 
    /* REPNE  */ 
 | 
    /* move them until CX is ZERO. */ 
 | 
    while (M.x86.R_CX != 0) { 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        val = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_long(M.x86.R_EAX, val); 
 | 
        } else { 
 | 
        val = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_word(M.x86.R_AX, (u16)val); 
 | 
        } 
 | 
        M.x86.R_CX -= 1; 
 | 
        M.x86.R_DI += inc; 
 | 
        if (ACCESS_FLAG(F_ZF)) 
 | 
        break;        /* zero flag set means equal */ 
 | 
    } 
 | 
    M.x86.mode &= ~SYSMODE_PREFIX_REPNE; 
 | 
    } else { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        val = fetch_data_long_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_long(M.x86.R_EAX, val); 
 | 
    } else { 
 | 
        val = fetch_data_word_abs(M.x86.R_ES, M.x86.R_DI); 
 | 
        cmp_word(M.x86.R_AX, (u16)val); 
 | 
    } 
 | 
    M.x86.R_DI += inc; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xb0 - 0xb7 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_byte_register_IMM(u8 op1) 
 | 
{ 
 | 
    u8 imm, *ptr; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    ptr = DECODE_RM_BYTE_REGISTER(op1 & 0x7); 
 | 
    DECODE_PRINTF(","); 
 | 
    imm = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("%x\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    *ptr = imm; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xb8 - 0xbf 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_word_register_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u32 srcval; 
 | 
  
 | 
    op1 &= 0x7; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    u32 *reg32; 
 | 
    reg32 = DECODE_RM_LONG_REGISTER(op1); 
 | 
    srcval = fetch_long_imm(); 
 | 
    DECODE_PRINTF2(",%x\n", srcval); 
 | 
    TRACE_AND_STEP(); 
 | 
    *reg32 = srcval; 
 | 
    } else { 
 | 
    u16 *reg16; 
 | 
    reg16 = DECODE_RM_WORD_REGISTER(op1); 
 | 
    srcval = fetch_word_imm(); 
 | 
    DECODE_PRINTF2(",%x\n", srcval); 
 | 
    TRACE_AND_STEP(); 
 | 
    *reg16 = (u16)srcval; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc0 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcC0_byte_RM_MEM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg; 
 | 
    uint destoffset; 
 | 
    u8 destval; 
 | 
    u8 amt; 
 | 
  
 | 
    /* 
 | 
     * Yet another weirdo special case instruction format.  Part of 
 | 
     * the opcode held below in "RH".  Doubly nested case would 
 | 
     * result, except that the decoded instruction 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
  
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ROL\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("ROR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("RCL\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("RCR\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("SHL\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SHR\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("SAL\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("SAR\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* know operation, decode the mod byte to find the addressing 
 | 
       mode. */ 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("BYTE PTR "); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    amt = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2(",%x\n", amt); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*opcD0_byte_operation[rh]) (destval, amt); 
 | 
    store_data_byte(destoffset, destval); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    amt = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2(",%x\n", amt); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*opcD0_byte_operation[rh]) (*destreg, amt); 
 | 
    *destreg = destval; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc1 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcC1_word_RM_MEM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
    u8 amt; 
 | 
  
 | 
    /* 
 | 
     * Yet another weirdo special case instruction format.  Part of 
 | 
     * the opcode held below in "RH".  Doubly nested case would 
 | 
     * result, except that the decoded instruction 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
  
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ROL\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("ROR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("RCL\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("RCR\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("SHL\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SHR\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("SAL\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("SAR\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* know operation, decode the mod byte to find the addressing 
 | 
       mode. */ 
 | 
    if (mod < 3) { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval; 
 | 
  
 | 
        DECODE_PRINTF("DWORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        amt = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", amt); 
 | 
        destval = fetch_data_long(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*opcD1_long_operation[rh]) (destval, amt); 
 | 
        store_data_long(destoffset, destval); 
 | 
    } else { 
 | 
        u16 destval; 
 | 
  
 | 
        DECODE_PRINTF("WORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        amt = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", amt); 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*opcD1_word_operation[rh]) (destval, amt); 
 | 
        store_data_word(destoffset, destval); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        amt = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", amt); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = (*opcD1_long_operation[rh]) (*destreg, amt); 
 | 
    } else { 
 | 
        u16 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        amt = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", amt); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = (*opcD1_word_operation[rh]) (*destreg, amt); 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc2 
 | 
****************************************************************************/ 
 | 
void x86emuOp_ret_near_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 imm; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("RET\t"); 
 | 
    imm = fetch_word_imm(); 
 | 
    DECODE_PRINTF2("%x\n", imm); 
 | 
    RETURN_TRACE("RET",M.x86.saved_cs,M.x86.saved_ip); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_IP = pop_word(); 
 | 
    M.x86.R_SP += imm; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc3 
 | 
****************************************************************************/ 
 | 
void x86emuOp_ret_near(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("RET\n"); 
 | 
    RETURN_TRACE("RET",M.x86.saved_cs,M.x86.saved_ip); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_IP = pop_word(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc4 
 | 
****************************************************************************/ 
 | 
void x86emuOp_les_R_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rh, rl; 
 | 
    u16 *dstreg; 
 | 
    uint srcoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LES\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    dstreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *dstreg = fetch_data_word(srcoffset); 
 | 
    M.x86.R_ES = fetch_data_word(srcoffset + 2); 
 | 
    } 
 | 
    /* else UNDEFINED!             register to register */ 
 | 
  
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc5 
 | 
****************************************************************************/ 
 | 
void x86emuOp_lds_R_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rh, rl; 
 | 
    u16 *dstreg; 
 | 
    uint srcoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LDS\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (mod < 3) { 
 | 
    dstreg = DECODE_RM_WORD_REGISTER(rh); 
 | 
    DECODE_PRINTF(","); 
 | 
    srcoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    *dstreg = fetch_data_word(srcoffset); 
 | 
    M.x86.R_DS = fetch_data_word(srcoffset + 2); 
 | 
    } 
 | 
    /* else UNDEFINED! */ 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc6 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_byte_RM_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg; 
 | 
    uint destoffset; 
 | 
    u8 imm; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (rh != 0) { 
 | 
    ERR_PRINTF("ILLEGAL DECODE OF OPCODE c6\n"); 
 | 
    HALT_SYS(); 
 | 
    } 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("BYTE PTR "); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    imm = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2(",%2x\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    store_data_byte(destoffset, imm); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    imm = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2(",%2x\n", imm); 
 | 
    TRACE_AND_STEP(); 
 | 
    *destreg = imm; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc7 
 | 
****************************************************************************/ 
 | 
void x86emuOp_mov_word_RM_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("MOV\t"); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    if (rh != 0) { 
 | 
    ERR_PRINTF("ILLEGAL DECODE OF OPCODE 8F\n"); 
 | 
    HALT_SYS(); 
 | 
    } 
 | 
    if (mod < 3) { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 imm; 
 | 
  
 | 
        DECODE_PRINTF("DWORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        imm = fetch_long_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        store_data_long(destoffset, imm); 
 | 
    } else { 
 | 
        u16 imm; 
 | 
  
 | 
        DECODE_PRINTF("WORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        imm = fetch_word_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        store_data_word(destoffset, imm); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
            u32 *destreg; 
 | 
            u32 imm; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        imm = fetch_long_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = imm; 
 | 
    } else { 
 | 
            u16 *destreg; 
 | 
            u16 imm; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        imm = fetch_word_imm(); 
 | 
        DECODE_PRINTF2(",%x\n", imm); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = imm; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc8 
 | 
****************************************************************************/ 
 | 
void x86emuOp_enter(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 local,frame_pointer; 
 | 
    u8    nesting; 
 | 
    int i; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    local = fetch_word_imm(); 
 | 
    nesting = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("ENTER %x\n", local); 
 | 
    DECODE_PRINTF2(",%x\n", nesting); 
 | 
    TRACE_AND_STEP(); 
 | 
    push_word(M.x86.R_BP); 
 | 
    frame_pointer = M.x86.R_SP; 
 | 
    if (nesting > 0) { 
 | 
    for (i = 1; i < nesting; i++) { 
 | 
        M.x86.R_BP -= 2; 
 | 
        push_word(fetch_data_word_abs(M.x86.R_SS, M.x86.R_BP)); 
 | 
        } 
 | 
    push_word(frame_pointer); 
 | 
    } 
 | 
    M.x86.R_BP = frame_pointer; 
 | 
    M.x86.R_SP = (u16)(M.x86.R_SP - local); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xc9 
 | 
****************************************************************************/ 
 | 
void x86emuOp_leave(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LEAVE\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_SP = M.x86.R_BP; 
 | 
    M.x86.R_BP = pop_word(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xca 
 | 
****************************************************************************/ 
 | 
void x86emuOp_ret_far_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 imm; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("RETF\t"); 
 | 
    imm = fetch_word_imm(); 
 | 
    DECODE_PRINTF2("%x\n", imm); 
 | 
    RETURN_TRACE("RETF",M.x86.saved_cs,M.x86.saved_ip); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_IP = pop_word(); 
 | 
    M.x86.R_CS = pop_word(); 
 | 
    M.x86.R_SP += imm; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xcb 
 | 
****************************************************************************/ 
 | 
void x86emuOp_ret_far(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("RETF\n"); 
 | 
    RETURN_TRACE("RETF",M.x86.saved_cs,M.x86.saved_ip); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_IP = pop_word(); 
 | 
    M.x86.R_CS = pop_word(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xcc 
 | 
****************************************************************************/ 
 | 
void x86emuOp_int3(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("INT 3\n"); 
 | 
    (void)mem_access_word(3 * 4 + 2); 
 | 
    /* access the segment register */ 
 | 
    TRACE_AND_STEP(); 
 | 
    if (_X86EMU_intrTab[3]) { 
 | 
        (*_X86EMU_intrTab[3])(3); 
 | 
    } else { 
 | 
    push_word((u16)M.x86.R_FLG); 
 | 
    CLEAR_FLAG(F_IF); 
 | 
    CLEAR_FLAG(F_TF); 
 | 
    push_word(M.x86.R_CS); 
 | 
    M.x86.R_CS = mem_access_word(3 * 4 + 2); 
 | 
    push_word(M.x86.R_IP); 
 | 
    M.x86.R_IP = mem_access_word(3 * 4); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xcd 
 | 
****************************************************************************/ 
 | 
void x86emuOp_int_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u8 intnum; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("INT\t"); 
 | 
    intnum = fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("%x\n", intnum); 
 | 
    (void)mem_access_word(intnum * 4 + 2); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (_X86EMU_intrTab[intnum]) { 
 | 
        (*_X86EMU_intrTab[intnum])(intnum); 
 | 
    } else { 
 | 
    push_word((u16)M.x86.R_FLG); 
 | 
    CLEAR_FLAG(F_IF); 
 | 
    CLEAR_FLAG(F_TF); 
 | 
    push_word(M.x86.R_CS); 
 | 
    M.x86.R_CS = mem_access_word(intnum * 4 + 2); 
 | 
    push_word(M.x86.R_IP); 
 | 
    M.x86.R_IP = mem_access_word(intnum * 4); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xce 
 | 
****************************************************************************/ 
 | 
void x86emuOp_into(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("INTO\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (ACCESS_FLAG(F_OF)) { 
 | 
    (void)mem_access_word(4 * 4 + 2); 
 | 
        if (_X86EMU_intrTab[4]) { 
 | 
            (*_X86EMU_intrTab[4])(4); 
 | 
    } else { 
 | 
        push_word((u16)M.x86.R_FLG); 
 | 
        CLEAR_FLAG(F_IF); 
 | 
        CLEAR_FLAG(F_TF); 
 | 
        push_word(M.x86.R_CS); 
 | 
        M.x86.R_CS = mem_access_word(4 * 4 + 2); 
 | 
        push_word(M.x86.R_IP); 
 | 
        M.x86.R_IP = mem_access_word(4 * 4); 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xcf 
 | 
****************************************************************************/ 
 | 
void x86emuOp_iret(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("IRET\n"); 
 | 
  
 | 
    TRACE_AND_STEP(); 
 | 
  
 | 
    M.x86.R_IP = pop_word(); 
 | 
    M.x86.R_CS = pop_word(); 
 | 
    M.x86.R_FLG = pop_word(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xd0 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcD0_byte_RM_1(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg; 
 | 
    uint destoffset; 
 | 
    u8 destval; 
 | 
  
 | 
    /* 
 | 
     * Yet another weirdo special case instruction format.  Part of 
 | 
     * the opcode held below in "RH".  Doubly nested case would 
 | 
     * result, except that the decoded instruction 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ROL\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("ROR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("RCL\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("RCR\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("SHL\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SHR\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("SAL\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("SAR\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* know operation, decode the mod byte to find the addressing 
 | 
       mode. */ 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("BYTE PTR "); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF(",1\n"); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*opcD0_byte_operation[rh]) (destval, 1); 
 | 
    store_data_byte(destoffset, destval); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    DECODE_PRINTF(",1\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*opcD0_byte_operation[rh]) (*destreg, 1); 
 | 
    *destreg = destval; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xd1 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcD1_word_RM_1(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    /* 
 | 
     * Yet another weirdo special case instruction format.  Part of 
 | 
     * the opcode held below in "RH".  Doubly nested case would 
 | 
     * result, except that the decoded instruction 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ROL\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("ROR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("RCL\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("RCR\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("SHL\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SHR\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("SAL\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("SAR\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* know operation, decode the mod byte to find the addressing 
 | 
       mode. */ 
 | 
    if (mod < 3) { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval; 
 | 
  
 | 
        DECODE_PRINTF("DWORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        DECODE_PRINTF(",1\n"); 
 | 
        destval = fetch_data_long(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*opcD1_long_operation[rh]) (destval, 1); 
 | 
        store_data_long(destoffset, destval); 
 | 
    } else { 
 | 
        u16 destval; 
 | 
  
 | 
        DECODE_PRINTF("WORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        DECODE_PRINTF(",1\n"); 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*opcD1_word_operation[rh]) (destval, 1); 
 | 
        store_data_word(destoffset, destval); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
            u32 destval; 
 | 
            u32 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF(",1\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*opcD1_long_operation[rh]) (*destreg, 1); 
 | 
        *destreg = destval; 
 | 
    } else { 
 | 
            u16 destval; 
 | 
            u16 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF(",1\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*opcD1_word_operation[rh]) (*destreg, 1); 
 | 
        *destreg = destval; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xd2 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcD2_byte_RM_CL(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg; 
 | 
    uint destoffset; 
 | 
    u8 destval; 
 | 
    u8 amt; 
 | 
  
 | 
    /* 
 | 
     * Yet another weirdo special case instruction format.  Part of 
 | 
     * the opcode held below in "RH".  Doubly nested case would 
 | 
     * result, except that the decoded instruction 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ROL\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("ROR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("RCL\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("RCR\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("SHL\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SHR\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("SAL\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("SAR\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* know operation, decode the mod byte to find the addressing 
 | 
       mode. */ 
 | 
    amt = M.x86.R_CL; 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("BYTE PTR "); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF(",CL\n"); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*opcD0_byte_operation[rh]) (destval, amt); 
 | 
    store_data_byte(destoffset, destval); 
 | 
    } else {             /* register to register */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    DECODE_PRINTF(",CL\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    destval = (*opcD0_byte_operation[rh]) (*destreg, amt); 
 | 
    *destreg = destval; 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xd3 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcD3_word_RM_CL(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
    u8 amt; 
 | 
  
 | 
    /* 
 | 
     * Yet another weirdo special case instruction format.  Part of 
 | 
     * the opcode held below in "RH".  Doubly nested case would 
 | 
     * result, except that the decoded instruction 
 | 
     */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("ROL\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("ROR\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("RCL\t"); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("RCR\t"); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("SHL\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("SHR\t"); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("SAL\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        DECODE_PRINTF("SAR\t"); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    /* know operation, decode the mod byte to find the addressing 
 | 
       mode. */ 
 | 
    amt = M.x86.R_CL; 
 | 
    if (mod < 3) { 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval; 
 | 
  
 | 
        DECODE_PRINTF("DWORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        DECODE_PRINTF(",CL\n"); 
 | 
        destval = fetch_data_long(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*opcD1_long_operation[rh]) (destval, amt); 
 | 
        store_data_long(destoffset, destval); 
 | 
    } else { 
 | 
        u16 destval; 
 | 
  
 | 
        DECODE_PRINTF("WORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        DECODE_PRINTF(",CL\n"); 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = (*opcD1_word_operation[rh]) (destval, amt); 
 | 
        store_data_word(destoffset, destval); 
 | 
    } 
 | 
    } else {             /* register to register */ 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF(",CL\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = (*opcD1_long_operation[rh]) (*destreg, amt); 
 | 
    } else { 
 | 
        u16 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF(",CL\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = (*opcD1_word_operation[rh]) (*destreg, amt); 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xd4 
 | 
****************************************************************************/ 
 | 
void x86emuOp_aam(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u8 a; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("AAM\n"); 
 | 
    a = fetch_byte_imm();      /* this is a stupid encoding. */ 
 | 
    if (a != 10) { 
 | 
    ERR_PRINTF("ERROR DECODING AAM\n"); 
 | 
    TRACE_REGS(); 
 | 
    HALT_SYS(); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    /* note the type change here --- returning AL and AH in AX. */ 
 | 
    M.x86.R_AX = aam_word(M.x86.R_AL); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xd5 
 | 
****************************************************************************/ 
 | 
void x86emuOp_aad(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("AAD\n"); 
 | 
    (void)fetch_byte_imm(); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AX = aad_word(M.x86.R_AX); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/* opcode 0xd6 ILLEGAL OPCODE */ 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xd7 
 | 
****************************************************************************/ 
 | 
void x86emuOp_xlat(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 addr; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("XLAT\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    addr = (u16)(M.x86.R_BX + (u8)M.x86.R_AL); 
 | 
    M.x86.R_AL = fetch_data_byte(addr); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/* instuctions    D8 .. DF are in i87_ops.c */ 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe0 
 | 
****************************************************************************/ 
 | 
void x86emuOp_loopne(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    s16 ip; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LOOPNE\t"); 
 | 
    ip = (s8) fetch_byte_imm(); 
 | 
    ip += (s16) M.x86.R_IP; 
 | 
    DECODE_PRINTF2("%04x\n", ip); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_CX -= 1; 
 | 
    if (M.x86.R_CX != 0 && !ACCESS_FLAG(F_ZF))        /* CX != 0 and !ZF */ 
 | 
    M.x86.R_IP = ip; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe1 
 | 
****************************************************************************/ 
 | 
void x86emuOp_loope(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    s16 ip; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LOOPE\t"); 
 | 
    ip = (s8) fetch_byte_imm(); 
 | 
    ip += (s16) M.x86.R_IP; 
 | 
    DECODE_PRINTF2("%04x\n", ip); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_CX -= 1; 
 | 
    if (M.x86.R_CX != 0 && ACCESS_FLAG(F_ZF))        /* CX != 0 and ZF */ 
 | 
    M.x86.R_IP = ip; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe2 
 | 
****************************************************************************/ 
 | 
void x86emuOp_loop(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    s16 ip; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LOOP\t"); 
 | 
    ip = (s8) fetch_byte_imm(); 
 | 
    ip += (s16) M.x86.R_IP; 
 | 
    DECODE_PRINTF2("%04x\n", ip); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_CX -= 1; 
 | 
    if (M.x86.R_CX != 0) 
 | 
    M.x86.R_IP = ip; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe3 
 | 
****************************************************************************/ 
 | 
void x86emuOp_jcxz(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 target; 
 | 
    s8    offset; 
 | 
  
 | 
    /* jump to byte offset if overflow flag is set */ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("JCXZ\t"); 
 | 
    offset = (s8)fetch_byte_imm(); 
 | 
    target = (u16)(M.x86.R_IP + offset); 
 | 
    DECODE_PRINTF2("%x\n", target); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.R_CX == 0) 
 | 
    M.x86.R_IP = target; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe4 
 | 
****************************************************************************/ 
 | 
void x86emuOp_in_byte_AL_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u8 port; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("IN\t"); 
 | 
    port = (u8) fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("%x,AL\n", port); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AL = (*sys_inb)(port); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe5 
 | 
****************************************************************************/ 
 | 
void x86emuOp_in_word_AX_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u8 port; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("IN\t"); 
 | 
    port = (u8) fetch_byte_imm(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF2("EAX,%x\n", port); 
 | 
    } else { 
 | 
    DECODE_PRINTF2("AX,%x\n", port); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    M.x86.R_EAX = (*sys_inl)(port); 
 | 
    } else { 
 | 
    M.x86.R_AX = (*sys_inw)(port); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe6 
 | 
****************************************************************************/ 
 | 
void x86emuOp_out_byte_IMM_AL(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u8 port; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("OUT\t"); 
 | 
    port = (u8) fetch_byte_imm(); 
 | 
    DECODE_PRINTF2("%x,AL\n", port); 
 | 
    TRACE_AND_STEP(); 
 | 
    (*sys_outb)(port, M.x86.R_AL); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe7 
 | 
****************************************************************************/ 
 | 
void x86emuOp_out_word_IMM_AX(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u8 port; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("OUT\t"); 
 | 
    port = (u8) fetch_byte_imm(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF2("%x,EAX\n", port); 
 | 
    } else { 
 | 
    DECODE_PRINTF2("%x,AX\n", port); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    (*sys_outl)(port, M.x86.R_EAX); 
 | 
    } else { 
 | 
    (*sys_outw)(port, M.x86.R_AX); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe8 
 | 
****************************************************************************/ 
 | 
void x86emuOp_call_near_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    s16 ip; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("CALL\t"); 
 | 
    ip = (s16) fetch_word_imm(); 
 | 
    ip += (s16) M.x86.R_IP;       /* CHECK SIGN */ 
 | 
    DECODE_PRINTF2("%04x\n", ip); 
 | 
    CALL_TRACE(M.x86.saved_cs, M.x86.saved_ip, M.x86.R_CS, ip, ""); 
 | 
    TRACE_AND_STEP(); 
 | 
    push_word(M.x86.R_IP); 
 | 
    M.x86.R_IP = ip; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xe9 
 | 
****************************************************************************/ 
 | 
void x86emuOp_jump_near_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int ip; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("JMP\t"); 
 | 
    ip = (s16)fetch_word_imm(); 
 | 
    ip += (s16)M.x86.R_IP; 
 | 
    DECODE_PRINTF2("%04x\n", ip); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_IP = (u16)ip; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xea 
 | 
****************************************************************************/ 
 | 
void x86emuOp_jump_far_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 cs, ip; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("JMP\tFAR "); 
 | 
    ip = fetch_word_imm(); 
 | 
    cs = fetch_word_imm(); 
 | 
    DECODE_PRINTF2("%04x:", cs); 
 | 
    DECODE_PRINTF2("%04x\n", ip); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_IP = ip; 
 | 
    M.x86.R_CS = cs; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xeb 
 | 
****************************************************************************/ 
 | 
void x86emuOp_jump_byte_IMM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    u16 target; 
 | 
    s8 offset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("JMP\t"); 
 | 
    offset = (s8)fetch_byte_imm(); 
 | 
    target = (u16)(M.x86.R_IP + offset); 
 | 
    DECODE_PRINTF2("%x\n", target); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_IP = target; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xec 
 | 
****************************************************************************/ 
 | 
void x86emuOp_in_byte_AL_DX(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("IN\tAL,DX\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.R_AL = (*sys_inb)(M.x86.R_DX); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xed 
 | 
****************************************************************************/ 
 | 
void x86emuOp_in_word_AX_DX(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("IN\tEAX,DX\n"); 
 | 
    } else { 
 | 
    DECODE_PRINTF("IN\tAX,DX\n"); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    M.x86.R_EAX = (*sys_inl)(M.x86.R_DX); 
 | 
    } else { 
 | 
    M.x86.R_AX = (*sys_inw)(M.x86.R_DX); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xee 
 | 
****************************************************************************/ 
 | 
void x86emuOp_out_byte_DX_AL(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("OUT\tDX,AL\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    (*sys_outb)(M.x86.R_DX, M.x86.R_AL); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xef 
 | 
****************************************************************************/ 
 | 
void x86emuOp_out_word_DX_AX(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    DECODE_PRINTF("OUT\tDX,EAX\n"); 
 | 
    } else { 
 | 
    DECODE_PRINTF("OUT\tDX,AX\n"); 
 | 
    } 
 | 
    TRACE_AND_STEP(); 
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
    (*sys_outl)(M.x86.R_DX, M.x86.R_EAX); 
 | 
    } else { 
 | 
    (*sys_outw)(M.x86.R_DX, M.x86.R_AX); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xf0 
 | 
****************************************************************************/ 
 | 
void x86emuOp_lock(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("LOCK:\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/*opcode 0xf1 ILLEGAL OPERATION */ 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xf2 
 | 
****************************************************************************/ 
 | 
void x86emuOp_repne(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("REPNE\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_PREFIX_REPNE; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xf3 
 | 
****************************************************************************/ 
 | 
void x86emuOp_repe(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("REPE\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    M.x86.mode |= SYSMODE_PREFIX_REPE; 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xf4 
 | 
****************************************************************************/ 
 | 
void x86emuOp_halt(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("HALT\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    HALT_SYS(); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xf5 
 | 
****************************************************************************/ 
 | 
void x86emuOp_cmc(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    /* complement the carry flag. */ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("CMC\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    TOGGLE_FLAG(F_CF); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xf6 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcF6_byte_RM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    u8 *destreg; 
 | 
    uint destoffset; 
 | 
    u8 destval, srcval; 
 | 
  
 | 
    /* long, drawn out code follows.  Double switch for a total 
 | 
       of 32 cases.  */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    DECODE_PRINTF(opF6_names[rh]); 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("BYTE PTR "); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
  
 | 
    switch (rh) { 
 | 
    case 0:        /* test byte imm */ 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2("%02x\n", srcval); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_byte(destval, srcval); 
 | 
        break; 
 | 
    case 1: 
 | 
        ERR_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F6\n"); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = not_byte(destval); 
 | 
        store_data_byte(destoffset, destval); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = neg_byte(destval); 
 | 
        store_data_byte(destoffset, destval); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        mul_byte(destval); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_byte(destval); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        div_byte(destval); 
 | 
        break; 
 | 
    default: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        idiv_byte(destval); 
 | 
        break; 
 | 
    } 
 | 
    } else {             /* mod=11 */ 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    switch (rh) { 
 | 
    case 0:        /* test byte imm */ 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_byte_imm(); 
 | 
        DECODE_PRINTF2("%02x\n", srcval); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_byte(*destreg, srcval); 
 | 
        break; 
 | 
    case 1: 
 | 
        ERR_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F6\n"); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = not_byte(*destreg); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = neg_byte(*destreg); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        mul_byte(*destreg);         /*!!!  */ 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_byte(*destreg); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        div_byte(*destreg); 
 | 
        break; 
 | 
    default: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        idiv_byte(*destreg); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xf7 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcF7_word_RM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rl, rh; 
 | 
    uint destoffset; 
 | 
  
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
    DECODE_PRINTF(opF6_names[rh]); 
 | 
    if (mod < 3) { 
 | 
  
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval, srcval; 
 | 
  
 | 
        DECODE_PRINTF("DWORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        destval = fetch_data_long(destoffset); 
 | 
  
 | 
        switch (rh) { 
 | 
        case 0: 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_long_imm(); 
 | 
        DECODE_PRINTF2("%x\n", srcval); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_long(destval, srcval); 
 | 
        break; 
 | 
        case 1: 
 | 
        ERR_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F7\n"); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
        case 2: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = not_long(destval); 
 | 
        store_data_long(destoffset, destval); 
 | 
        break; 
 | 
        case 3: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = neg_long(destval); 
 | 
        store_data_long(destoffset, destval); 
 | 
        break; 
 | 
        case 4: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        mul_long(destval); 
 | 
        break; 
 | 
        case 5: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_long(destval); 
 | 
        break; 
 | 
        case 6: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        div_long(destval); 
 | 
        break; 
 | 
        case 7: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        idiv_long(destval); 
 | 
        break; 
 | 
        } 
 | 
    } else { 
 | 
        u16 destval, srcval; 
 | 
  
 | 
        DECODE_PRINTF("WORD PTR "); 
 | 
        destoffset = decode_rmXX_address(mod, rl); 
 | 
        destval = fetch_data_word(destoffset); 
 | 
  
 | 
        switch (rh) { 
 | 
        case 0:        /* test word imm */ 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_word_imm(); 
 | 
        DECODE_PRINTF2("%x\n", srcval); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_word(destval, srcval); 
 | 
        break; 
 | 
        case 1: 
 | 
        ERR_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F7\n"); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
        case 2: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = not_word(destval); 
 | 
        store_data_word(destoffset, destval); 
 | 
        break; 
 | 
        case 3: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = neg_word(destval); 
 | 
        store_data_word(destoffset, destval); 
 | 
        break; 
 | 
        case 4: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        mul_word(destval); 
 | 
        break; 
 | 
        case 5: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_word(destval); 
 | 
        break; 
 | 
        case 6: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        div_word(destval); 
 | 
        break; 
 | 
        case 7: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        idiv_word(destval); 
 | 
        break; 
 | 
        } 
 | 
    } 
 | 
  
 | 
    } else {             /* mod=11 */ 
 | 
  
 | 
    if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
        u32 srcval; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
  
 | 
        switch (rh) { 
 | 
        case 0:        /* test word imm */ 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_long_imm(); 
 | 
        DECODE_PRINTF2("%x\n", srcval); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_long(*destreg, srcval); 
 | 
        break; 
 | 
        case 1: 
 | 
        ERR_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F6\n"); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
        case 2: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = not_long(*destreg); 
 | 
        break; 
 | 
        case 3: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = neg_long(*destreg); 
 | 
        break; 
 | 
        case 4: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        mul_long(*destreg);     /*!!!    */ 
 | 
        break; 
 | 
        case 5: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_long(*destreg); 
 | 
        break; 
 | 
        case 6: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        div_long(*destreg); 
 | 
        break; 
 | 
        case 7: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        idiv_long(*destreg); 
 | 
        break; 
 | 
        } 
 | 
    } else { 
 | 
        u16 *destreg; 
 | 
        u16 srcval; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
  
 | 
        switch (rh) { 
 | 
        case 0:        /* test word imm */ 
 | 
        DECODE_PRINTF(","); 
 | 
        srcval = fetch_word_imm(); 
 | 
        DECODE_PRINTF2("%x\n", srcval); 
 | 
        TRACE_AND_STEP(); 
 | 
        test_word(*destreg, srcval); 
 | 
        break; 
 | 
        case 1: 
 | 
        ERR_PRINTF("ILLEGAL OP MOD=00 RH=01 OP=F6\n"); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
        case 2: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = not_word(*destreg); 
 | 
        break; 
 | 
        case 3: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = neg_word(*destreg); 
 | 
        break; 
 | 
        case 4: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        mul_word(*destreg);     /*!!!    */ 
 | 
        break; 
 | 
        case 5: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        imul_word(*destreg); 
 | 
        break; 
 | 
        case 6: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        div_word(*destreg); 
 | 
        break; 
 | 
        case 7: 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        idiv_word(*destreg); 
 | 
        break; 
 | 
        } 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xf8 
 | 
****************************************************************************/ 
 | 
void x86emuOp_clc(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    /* clear the carry flag. */ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("CLC\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    CLEAR_FLAG(F_CF); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xf9 
 | 
****************************************************************************/ 
 | 
void x86emuOp_stc(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    /* set the carry flag. */ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("STC\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    SET_FLAG(F_CF); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xfa 
 | 
****************************************************************************/ 
 | 
void x86emuOp_cli(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    /* clear interrupts. */ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("CLI\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    CLEAR_FLAG(F_IF); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xfb 
 | 
****************************************************************************/ 
 | 
void x86emuOp_sti(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    /* enable  interrupts. */ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("STI\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    SET_FLAG(F_IF); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xfc 
 | 
****************************************************************************/ 
 | 
void x86emuOp_cld(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    /* clear interrupts. */ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("CLD\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    CLEAR_FLAG(F_DF); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xfd 
 | 
****************************************************************************/ 
 | 
void x86emuOp_std(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    /* clear interrupts. */ 
 | 
    START_OF_INSTR(); 
 | 
    DECODE_PRINTF("STD\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    SET_FLAG(F_DF); 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xfe 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcFE_byte_RM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rh, rl; 
 | 
    u8 destval; 
 | 
    uint destoffset; 
 | 
    u8 *destreg; 
 | 
  
 | 
    /* Yet another special case instruction. */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
  
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        DECODE_PRINTF("INC\t"); 
 | 
        break; 
 | 
    case 1: 
 | 
        DECODE_PRINTF("DEC\t"); 
 | 
        break; 
 | 
    case 2: 
 | 
    case 3: 
 | 
    case 4: 
 | 
    case 5: 
 | 
    case 6: 
 | 
    case 7: 
 | 
        ERR_PRINTF2("ILLEGAL OP MAJOR OP 0xFE MINOR OP %x\n", mod); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    if (mod < 3) { 
 | 
    DECODE_PRINTF("BYTE PTR "); 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    destval = fetch_data_byte(destoffset); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (rh == 0) 
 | 
      destval = inc_byte(destval); 
 | 
    else 
 | 
      destval = dec_byte(destval); 
 | 
    store_data_byte(destoffset, destval); 
 | 
    } else { 
 | 
    destreg = DECODE_RM_BYTE_REGISTER(rl); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    TRACE_AND_STEP(); 
 | 
    if (rh == 0) 
 | 
      *destreg = inc_byte(*destreg); 
 | 
    else 
 | 
      *destreg = dec_byte(*destreg); 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/**************************************************************************** 
 | 
REMARKS: 
 | 
Handles opcode 0xff 
 | 
****************************************************************************/ 
 | 
void x86emuOp_opcFF_word_RM(u8 X86EMU_UNUSED(op1)) 
 | 
{ 
 | 
    int mod, rh, rl; 
 | 
    uint destoffset = 0; 
 | 
    u16 *destreg; 
 | 
    u16 destval,destval2; 
 | 
  
 | 
    /* Yet another special case instruction. */ 
 | 
    START_OF_INSTR(); 
 | 
    FETCH_DECODE_MODRM(mod, rh, rl); 
 | 
#ifdef CONFIG_X86EMU_DEBUG 
 | 
    if (DEBUG_DECODE()) { 
 | 
    /* XXX DECODE_PRINTF may be changed to something more 
 | 
       general, so that it is important to leave the strings 
 | 
       in the same format, even though the result is that the 
 | 
       above test is done twice. */ 
 | 
  
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        DECODE_PRINTF("INC\tDWORD PTR "); 
 | 
        } else { 
 | 
        DECODE_PRINTF("INC\tWORD PTR "); 
 | 
        } 
 | 
        break; 
 | 
    case 1: 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        DECODE_PRINTF("DEC\tDWORD PTR "); 
 | 
        } else { 
 | 
        DECODE_PRINTF("DEC\tWORD PTR "); 
 | 
        } 
 | 
        break; 
 | 
    case 2: 
 | 
        DECODE_PRINTF("CALL\t "); 
 | 
        break; 
 | 
    case 3: 
 | 
        DECODE_PRINTF("CALL\tFAR "); 
 | 
        break; 
 | 
    case 4: 
 | 
        DECODE_PRINTF("JMP\t"); 
 | 
        break; 
 | 
    case 5: 
 | 
        DECODE_PRINTF("JMP\tFAR "); 
 | 
        break; 
 | 
    case 6: 
 | 
        DECODE_PRINTF("PUSH\t"); 
 | 
        break; 
 | 
    case 7: 
 | 
        ERR_PRINTF("ILLEGAL DECODING OF OPCODE FF\t"); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
#endif 
 | 
    if (mod < 3) { 
 | 
    destoffset = decode_rmXX_address(mod, rl); 
 | 
    DECODE_PRINTF("\n"); 
 | 
    switch (rh) { 
 | 
    case 0:        /* inc word ptr ... */ 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval; 
 | 
  
 | 
        destval = fetch_data_long(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = inc_long(destval); 
 | 
        store_data_long(destoffset, destval); 
 | 
        } else { 
 | 
        u16 destval; 
 | 
  
 | 
        destval = fetch_data_word(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = inc_word(destval); 
 | 
        store_data_word(destoffset, destval); 
 | 
        } 
 | 
        break; 
 | 
    case 1:        /* dec word ptr ... */ 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval; 
 | 
  
 | 
        destval = fetch_data_long(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = dec_long(destval); 
 | 
        store_data_long(destoffset, destval); 
 | 
        } else { 
 | 
        u16 destval; 
 | 
  
 | 
        destval = fetch_data_word(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        destval = dec_word(destval); 
 | 
        store_data_word(destoffset, destval); 
 | 
        } 
 | 
        break; 
 | 
    case 2:        /* call word ptr ... */ 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        push_word(M.x86.R_IP); 
 | 
        M.x86.R_IP = destval; 
 | 
        break; 
 | 
    case 3:        /* call far ptr ... */ 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        destval2 = fetch_data_word(destoffset + 2); 
 | 
        TRACE_AND_STEP(); 
 | 
        push_word(M.x86.R_CS); 
 | 
        M.x86.R_CS = destval2; 
 | 
        push_word(M.x86.R_IP); 
 | 
        M.x86.R_IP = destval; 
 | 
        break; 
 | 
    case 4:        /* jmp word ptr ... */ 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        M.x86.R_IP = destval; 
 | 
        break; 
 | 
    case 5:        /* jmp far ptr ... */ 
 | 
        destval = fetch_data_word(destoffset); 
 | 
        destval2 = fetch_data_word(destoffset + 2); 
 | 
        TRACE_AND_STEP(); 
 | 
        M.x86.R_IP = destval; 
 | 
        M.x86.R_CS = destval2; 
 | 
        break; 
 | 
    case 6:        /*  push word ptr ... */ 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 destval; 
 | 
  
 | 
        destval = fetch_data_long(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        push_long(destval); 
 | 
        } else { 
 | 
        u16 destval; 
 | 
  
 | 
        destval = fetch_data_word(destoffset); 
 | 
        TRACE_AND_STEP(); 
 | 
        push_word(destval); 
 | 
        } 
 | 
        break; 
 | 
    } 
 | 
    } else { 
 | 
    switch (rh) { 
 | 
    case 0: 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = inc_long(*destreg); 
 | 
        } else { 
 | 
        u16 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = inc_word(*destreg); 
 | 
        } 
 | 
        break; 
 | 
    case 1: 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = dec_long(*destreg); 
 | 
        } else { 
 | 
        u16 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        *destreg = dec_word(*destreg); 
 | 
        } 
 | 
        break; 
 | 
    case 2:        /* call word ptr ... */ 
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        push_word(M.x86.R_IP); 
 | 
        M.x86.R_IP = *destreg; 
 | 
        break; 
 | 
    case 3:        /* jmp far ptr ... */ 
 | 
        ERR_PRINTF("OPERATION UNDEFINED 0XFF\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
  
 | 
    case 4:        /* jmp    ... */ 
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        M.x86.R_IP = (u16) (*destreg); 
 | 
        break; 
 | 
    case 5:        /* jmp far ptr ... */ 
 | 
        ERR_PRINTF("OPERATION UNDEFINED 0XFF\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        HALT_SYS(); 
 | 
        break; 
 | 
    case 6: 
 | 
        if (M.x86.mode & SYSMODE_PREFIX_DATA) { 
 | 
        u32 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_LONG_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        push_long(*destreg); 
 | 
        } else { 
 | 
        u16 *destreg; 
 | 
  
 | 
        destreg = DECODE_RM_WORD_REGISTER(rl); 
 | 
        DECODE_PRINTF("\n"); 
 | 
        TRACE_AND_STEP(); 
 | 
        push_word(*destreg); 
 | 
        } 
 | 
        break; 
 | 
    } 
 | 
    } 
 | 
    DECODE_CLEAR_SEGOVR(); 
 | 
    END_OF_INSTR(); 
 | 
} 
 | 
  
 | 
/*************************************************************************** 
 | 
 * Single byte operation code table: 
 | 
 **************************************************************************/ 
 | 
void (*x86emu_optab[256])(u8) = 
 | 
{ 
 | 
/*  0x00 */ x86emuOp_genop_byte_RM_R, 
 | 
/*  0x01 */ x86emuOp_genop_word_RM_R, 
 | 
/*  0x02 */ x86emuOp_genop_byte_R_RM, 
 | 
/*  0x03 */ x86emuOp_genop_word_R_RM, 
 | 
/*  0x04 */ x86emuOp_genop_byte_AL_IMM, 
 | 
/*  0x05 */ x86emuOp_genop_word_AX_IMM, 
 | 
/*  0x06 */ x86emuOp_push_ES, 
 | 
/*  0x07 */ x86emuOp_pop_ES, 
 | 
  
 | 
/*  0x08 */ x86emuOp_genop_byte_RM_R, 
 | 
/*  0x09 */ x86emuOp_genop_word_RM_R, 
 | 
/*  0x0a */ x86emuOp_genop_byte_R_RM, 
 | 
/*  0x0b */ x86emuOp_genop_word_R_RM, 
 | 
/*  0x0c */ x86emuOp_genop_byte_AL_IMM, 
 | 
/*  0x0d */ x86emuOp_genop_word_AX_IMM, 
 | 
/*  0x0e */ x86emuOp_push_CS, 
 | 
/*  0x0f */ x86emuOp_two_byte, 
 | 
  
 | 
/*  0x10 */ x86emuOp_genop_byte_RM_R, 
 | 
/*  0x11 */ x86emuOp_genop_word_RM_R, 
 | 
/*  0x12 */ x86emuOp_genop_byte_R_RM, 
 | 
/*  0x13 */ x86emuOp_genop_word_R_RM, 
 | 
/*  0x14 */ x86emuOp_genop_byte_AL_IMM, 
 | 
/*  0x15 */ x86emuOp_genop_word_AX_IMM, 
 | 
/*  0x16 */ x86emuOp_push_SS, 
 | 
/*  0x17 */ x86emuOp_pop_SS, 
 | 
  
 | 
/*  0x18 */ x86emuOp_genop_byte_RM_R, 
 | 
/*  0x19 */ x86emuOp_genop_word_RM_R, 
 | 
/*  0x1a */ x86emuOp_genop_byte_R_RM, 
 | 
/*  0x1b */ x86emuOp_genop_word_R_RM, 
 | 
/*  0x1c */ x86emuOp_genop_byte_AL_IMM, 
 | 
/*  0x1d */ x86emuOp_genop_word_AX_IMM, 
 | 
/*  0x1e */ x86emuOp_push_DS, 
 | 
/*  0x1f */ x86emuOp_pop_DS, 
 | 
  
 | 
/*  0x20 */ x86emuOp_genop_byte_RM_R, 
 | 
/*  0x21 */ x86emuOp_genop_word_RM_R, 
 | 
/*  0x22 */ x86emuOp_genop_byte_R_RM, 
 | 
/*  0x23 */ x86emuOp_genop_word_R_RM, 
 | 
/*  0x24 */ x86emuOp_genop_byte_AL_IMM, 
 | 
/*  0x25 */ x86emuOp_genop_word_AX_IMM, 
 | 
/*  0x26 */ x86emuOp_segovr_ES, 
 | 
/*  0x27 */ x86emuOp_daa, 
 | 
  
 | 
/*  0x28 */ x86emuOp_genop_byte_RM_R, 
 | 
/*  0x29 */ x86emuOp_genop_word_RM_R, 
 | 
/*  0x2a */ x86emuOp_genop_byte_R_RM, 
 | 
/*  0x2b */ x86emuOp_genop_word_R_RM, 
 | 
/*  0x2c */ x86emuOp_genop_byte_AL_IMM, 
 | 
/*  0x2d */ x86emuOp_genop_word_AX_IMM, 
 | 
/*  0x2e */ x86emuOp_segovr_CS, 
 | 
/*  0x2f */ x86emuOp_das, 
 | 
  
 | 
/*  0x30 */ x86emuOp_genop_byte_RM_R, 
 | 
/*  0x31 */ x86emuOp_genop_word_RM_R, 
 | 
/*  0x32 */ x86emuOp_genop_byte_R_RM, 
 | 
/*  0x33 */ x86emuOp_genop_word_R_RM, 
 | 
/*  0x34 */ x86emuOp_genop_byte_AL_IMM, 
 | 
/*  0x35 */ x86emuOp_genop_word_AX_IMM, 
 | 
/*  0x36 */ x86emuOp_segovr_SS, 
 | 
/*  0x37 */ x86emuOp_aaa, 
 | 
  
 | 
/*  0x38 */ x86emuOp_genop_byte_RM_R, 
 | 
/*  0x39 */ x86emuOp_genop_word_RM_R, 
 | 
/*  0x3a */ x86emuOp_genop_byte_R_RM, 
 | 
/*  0x3b */ x86emuOp_genop_word_R_RM, 
 | 
/*  0x3c */ x86emuOp_genop_byte_AL_IMM, 
 | 
/*  0x3d */ x86emuOp_genop_word_AX_IMM, 
 | 
/*  0x3e */ x86emuOp_segovr_DS, 
 | 
/*  0x3f */ x86emuOp_aas, 
 | 
  
 | 
/*  0x40 */ x86emuOp_inc_register, 
 | 
/*  0x41 */ x86emuOp_inc_register, 
 | 
/*  0x42 */ x86emuOp_inc_register, 
 | 
/*  0x43 */ x86emuOp_inc_register, 
 | 
/*  0x44 */ x86emuOp_inc_register, 
 | 
/*  0x45 */ x86emuOp_inc_register, 
 | 
/*  0x46 */ x86emuOp_inc_register, 
 | 
/*  0x47 */ x86emuOp_inc_register, 
 | 
  
 | 
/*  0x48 */ x86emuOp_dec_register, 
 | 
/*  0x49 */ x86emuOp_dec_register, 
 | 
/*  0x4a */ x86emuOp_dec_register, 
 | 
/*  0x4b */ x86emuOp_dec_register, 
 | 
/*  0x4c */ x86emuOp_dec_register, 
 | 
/*  0x4d */ x86emuOp_dec_register, 
 | 
/*  0x4e */ x86emuOp_dec_register, 
 | 
/*  0x4f */ x86emuOp_dec_register, 
 | 
  
 | 
/*  0x50 */ x86emuOp_push_register, 
 | 
/*  0x51 */ x86emuOp_push_register, 
 | 
/*  0x52 */ x86emuOp_push_register, 
 | 
/*  0x53 */ x86emuOp_push_register, 
 | 
/*  0x54 */ x86emuOp_push_register, 
 | 
/*  0x55 */ x86emuOp_push_register, 
 | 
/*  0x56 */ x86emuOp_push_register, 
 | 
/*  0x57 */ x86emuOp_push_register, 
 | 
  
 | 
/*  0x58 */ x86emuOp_pop_register, 
 | 
/*  0x59 */ x86emuOp_pop_register, 
 | 
/*  0x5a */ x86emuOp_pop_register, 
 | 
/*  0x5b */ x86emuOp_pop_register, 
 | 
/*  0x5c */ x86emuOp_pop_register, 
 | 
/*  0x5d */ x86emuOp_pop_register, 
 | 
/*  0x5e */ x86emuOp_pop_register, 
 | 
/*  0x5f */ x86emuOp_pop_register, 
 | 
  
 | 
/*  0x60 */ x86emuOp_push_all, 
 | 
/*  0x61 */ x86emuOp_pop_all, 
 | 
/*  0x62 */ x86emuOp_illegal_op,   /* bound */ 
 | 
/*  0x63 */ x86emuOp_illegal_op,   /* arpl */ 
 | 
/*  0x64 */ x86emuOp_segovr_FS, 
 | 
/*  0x65 */ x86emuOp_segovr_GS, 
 | 
/*  0x66 */ x86emuOp_prefix_data, 
 | 
/*  0x67 */ x86emuOp_prefix_addr, 
 | 
  
 | 
/*  0x68 */ x86emuOp_push_word_IMM, 
 | 
/*  0x69 */ x86emuOp_imul_word_IMM, 
 | 
/*  0x6a */ x86emuOp_push_byte_IMM, 
 | 
/*  0x6b */ x86emuOp_imul_byte_IMM, 
 | 
/*  0x6c */ x86emuOp_ins_byte, 
 | 
/*  0x6d */ x86emuOp_ins_word, 
 | 
/*  0x6e */ x86emuOp_outs_byte, 
 | 
/*  0x6f */ x86emuOp_outs_word, 
 | 
  
 | 
/*  0x70 */ x86emuOp_jump_near_cond, 
 | 
/*  0x71 */ x86emuOp_jump_near_cond, 
 | 
/*  0x72 */ x86emuOp_jump_near_cond, 
 | 
/*  0x73 */ x86emuOp_jump_near_cond, 
 | 
/*  0x74 */ x86emuOp_jump_near_cond, 
 | 
/*  0x75 */ x86emuOp_jump_near_cond, 
 | 
/*  0x76 */ x86emuOp_jump_near_cond, 
 | 
/*  0x77 */ x86emuOp_jump_near_cond, 
 | 
  
 | 
/*  0x78 */ x86emuOp_jump_near_cond, 
 | 
/*  0x79 */ x86emuOp_jump_near_cond, 
 | 
/*  0x7a */ x86emuOp_jump_near_cond, 
 | 
/*  0x7b */ x86emuOp_jump_near_cond, 
 | 
/*  0x7c */ x86emuOp_jump_near_cond, 
 | 
/*  0x7d */ x86emuOp_jump_near_cond, 
 | 
/*  0x7e */ x86emuOp_jump_near_cond, 
 | 
/*  0x7f */ x86emuOp_jump_near_cond, 
 | 
  
 | 
/*  0x80 */ x86emuOp_opc80_byte_RM_IMM, 
 | 
/*  0x81 */ x86emuOp_opc81_word_RM_IMM, 
 | 
/*  0x82 */ x86emuOp_opc82_byte_RM_IMM, 
 | 
/*  0x83 */ x86emuOp_opc83_word_RM_IMM, 
 | 
/*  0x84 */ x86emuOp_test_byte_RM_R, 
 | 
/*  0x85 */ x86emuOp_test_word_RM_R, 
 | 
/*  0x86 */ x86emuOp_xchg_byte_RM_R, 
 | 
/*  0x87 */ x86emuOp_xchg_word_RM_R, 
 | 
  
 | 
/*  0x88 */ x86emuOp_mov_byte_RM_R, 
 | 
/*  0x89 */ x86emuOp_mov_word_RM_R, 
 | 
/*  0x8a */ x86emuOp_mov_byte_R_RM, 
 | 
/*  0x8b */ x86emuOp_mov_word_R_RM, 
 | 
/*  0x8c */ x86emuOp_mov_word_RM_SR, 
 | 
/*  0x8d */ x86emuOp_lea_word_R_M, 
 | 
/*  0x8e */ x86emuOp_mov_word_SR_RM, 
 | 
/*  0x8f */ x86emuOp_pop_RM, 
 | 
  
 | 
/*  0x90 */ x86emuOp_nop, 
 | 
/*  0x91 */ x86emuOp_xchg_word_AX_register, 
 | 
/*  0x92 */ x86emuOp_xchg_word_AX_register, 
 | 
/*  0x93 */ x86emuOp_xchg_word_AX_register, 
 | 
/*  0x94 */ x86emuOp_xchg_word_AX_register, 
 | 
/*  0x95 */ x86emuOp_xchg_word_AX_register, 
 | 
/*  0x96 */ x86emuOp_xchg_word_AX_register, 
 | 
/*  0x97 */ x86emuOp_xchg_word_AX_register, 
 | 
  
 | 
/*  0x98 */ x86emuOp_cbw, 
 | 
/*  0x99 */ x86emuOp_cwd, 
 | 
/*  0x9a */ x86emuOp_call_far_IMM, 
 | 
/*  0x9b */ x86emuOp_wait, 
 | 
/*  0x9c */ x86emuOp_pushf_word, 
 | 
/*  0x9d */ x86emuOp_popf_word, 
 | 
/*  0x9e */ x86emuOp_sahf, 
 | 
/*  0x9f */ x86emuOp_lahf, 
 | 
  
 | 
/*  0xa0 */ x86emuOp_mov_AL_M_IMM, 
 | 
/*  0xa1 */ x86emuOp_mov_AX_M_IMM, 
 | 
/*  0xa2 */ x86emuOp_mov_M_AL_IMM, 
 | 
/*  0xa3 */ x86emuOp_mov_M_AX_IMM, 
 | 
/*  0xa4 */ x86emuOp_movs_byte, 
 | 
/*  0xa5 */ x86emuOp_movs_word, 
 | 
/*  0xa6 */ x86emuOp_cmps_byte, 
 | 
/*  0xa7 */ x86emuOp_cmps_word, 
 | 
/*  0xa8 */ x86emuOp_test_AL_IMM, 
 | 
/*  0xa9 */ x86emuOp_test_AX_IMM, 
 | 
/*  0xaa */ x86emuOp_stos_byte, 
 | 
/*  0xab */ x86emuOp_stos_word, 
 | 
/*  0xac */ x86emuOp_lods_byte, 
 | 
/*  0xad */ x86emuOp_lods_word, 
 | 
/*  0xac */ x86emuOp_scas_byte, 
 | 
/*  0xad */ x86emuOp_scas_word, 
 | 
  
 | 
/*  0xb0 */ x86emuOp_mov_byte_register_IMM, 
 | 
/*  0xb1 */ x86emuOp_mov_byte_register_IMM, 
 | 
/*  0xb2 */ x86emuOp_mov_byte_register_IMM, 
 | 
/*  0xb3 */ x86emuOp_mov_byte_register_IMM, 
 | 
/*  0xb4 */ x86emuOp_mov_byte_register_IMM, 
 | 
/*  0xb5 */ x86emuOp_mov_byte_register_IMM, 
 | 
/*  0xb6 */ x86emuOp_mov_byte_register_IMM, 
 | 
/*  0xb7 */ x86emuOp_mov_byte_register_IMM, 
 | 
  
 | 
/*  0xb8 */ x86emuOp_mov_word_register_IMM, 
 | 
/*  0xb9 */ x86emuOp_mov_word_register_IMM, 
 | 
/*  0xba */ x86emuOp_mov_word_register_IMM, 
 | 
/*  0xbb */ x86emuOp_mov_word_register_IMM, 
 | 
/*  0xbc */ x86emuOp_mov_word_register_IMM, 
 | 
/*  0xbd */ x86emuOp_mov_word_register_IMM, 
 | 
/*  0xbe */ x86emuOp_mov_word_register_IMM, 
 | 
/*  0xbf */ x86emuOp_mov_word_register_IMM, 
 | 
  
 | 
/*  0xc0 */ x86emuOp_opcC0_byte_RM_MEM, 
 | 
/*  0xc1 */ x86emuOp_opcC1_word_RM_MEM, 
 | 
/*  0xc2 */ x86emuOp_ret_near_IMM, 
 | 
/*  0xc3 */ x86emuOp_ret_near, 
 | 
/*  0xc4 */ x86emuOp_les_R_IMM, 
 | 
/*  0xc5 */ x86emuOp_lds_R_IMM, 
 | 
/*  0xc6 */ x86emuOp_mov_byte_RM_IMM, 
 | 
/*  0xc7 */ x86emuOp_mov_word_RM_IMM, 
 | 
/*  0xc8 */ x86emuOp_enter, 
 | 
/*  0xc9 */ x86emuOp_leave, 
 | 
/*  0xca */ x86emuOp_ret_far_IMM, 
 | 
/*  0xcb */ x86emuOp_ret_far, 
 | 
/*  0xcc */ x86emuOp_int3, 
 | 
/*  0xcd */ x86emuOp_int_IMM, 
 | 
/*  0xce */ x86emuOp_into, 
 | 
/*  0xcf */ x86emuOp_iret, 
 | 
  
 | 
/*  0xd0 */ x86emuOp_opcD0_byte_RM_1, 
 | 
/*  0xd1 */ x86emuOp_opcD1_word_RM_1, 
 | 
/*  0xd2 */ x86emuOp_opcD2_byte_RM_CL, 
 | 
/*  0xd3 */ x86emuOp_opcD3_word_RM_CL, 
 | 
/*  0xd4 */ x86emuOp_aam, 
 | 
/*  0xd5 */ x86emuOp_aad, 
 | 
/*  0xd6 */ x86emuOp_illegal_op,   /* Undocumented SETALC instruction */ 
 | 
/*  0xd7 */ x86emuOp_xlat, 
 | 
/*  0xd8 */ NULL, /*x86emuOp_esc_coprocess_d8,*/ 
 | 
/*  0xd9 */ NULL, /*x86emuOp_esc_coprocess_d9,*/ 
 | 
/*  0xda */ NULL, /*x86emuOp_esc_coprocess_da,*/ 
 | 
/*  0xdb */ NULL, /*x86emuOp_esc_coprocess_db,*/ 
 | 
/*  0xdc */ NULL, /*x86emuOp_esc_coprocess_dc,*/ 
 | 
/*  0xdd */ NULL, /*x86emuOp_esc_coprocess_dd,*/ 
 | 
/*  0xde */ NULL, /*x86emuOp_esc_coprocess_de,*/ 
 | 
/*  0xdf */ NULL, /*x86emuOp_esc_coprocess_df,*/ 
 | 
  
 | 
/*  0xe0 */ x86emuOp_loopne, 
 | 
/*  0xe1 */ x86emuOp_loope, 
 | 
/*  0xe2 */ x86emuOp_loop, 
 | 
/*  0xe3 */ x86emuOp_jcxz, 
 | 
/*  0xe4 */ x86emuOp_in_byte_AL_IMM, 
 | 
/*  0xe5 */ x86emuOp_in_word_AX_IMM, 
 | 
/*  0xe6 */ x86emuOp_out_byte_IMM_AL, 
 | 
/*  0xe7 */ x86emuOp_out_word_IMM_AX, 
 | 
  
 | 
/*  0xe8 */ x86emuOp_call_near_IMM, 
 | 
/*  0xe9 */ x86emuOp_jump_near_IMM, 
 | 
/*  0xea */ x86emuOp_jump_far_IMM, 
 | 
/*  0xeb */ x86emuOp_jump_byte_IMM, 
 | 
/*  0xec */ x86emuOp_in_byte_AL_DX, 
 | 
/*  0xed */ x86emuOp_in_word_AX_DX, 
 | 
/*  0xee */ x86emuOp_out_byte_DX_AL, 
 | 
/*  0xef */ x86emuOp_out_word_DX_AX, 
 | 
  
 | 
/*  0xf0 */ x86emuOp_lock, 
 | 
/*  0xf1 */ x86emuOp_illegal_op, 
 | 
/*  0xf2 */ x86emuOp_repne, 
 | 
/*  0xf3 */ x86emuOp_repe, 
 | 
/*  0xf4 */ x86emuOp_halt, 
 | 
/*  0xf5 */ x86emuOp_cmc, 
 | 
/*  0xf6 */ x86emuOp_opcF6_byte_RM, 
 | 
/*  0xf7 */ x86emuOp_opcF7_word_RM, 
 | 
  
 | 
/*  0xf8 */ x86emuOp_clc, 
 | 
/*  0xf9 */ x86emuOp_stc, 
 | 
/*  0xfa */ x86emuOp_cli, 
 | 
/*  0xfb */ x86emuOp_sti, 
 | 
/*  0xfc */ x86emuOp_cld, 
 | 
/*  0xfd */ x86emuOp_std, 
 | 
/*  0xfe */ x86emuOp_opcFE_byte_RM, 
 | 
/*  0xff */ x86emuOp_opcFF_word_RM, 
 | 
}; 
 |