| /**************************************************************************** |
| * |
| * Realmode X86 Emulator Library |
| * |
| * Copyright (C) 1996-1999 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 preceeding 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 "x86emu/x86emui.h" |
| |
| /*----------------------------- Implementation ----------------------------*/ |
| |
| /**************************************************************************** |
| PARAMETERS: |
| op1 - Instruction op code |
| |
| REMARKS: |
| Handles illegal opcodes. |
| ****************************************************************************/ |
| void x86emuOp_illegal_op( |
| u8 op1) |
| { |
| START_OF_INSTR(); |
| DECODE_PRINTF("ILLEGAL X86 OPCODE\n"); |
| TRACE_REGS(); |
| printk("%04x:%04x: %02X ILLEGAL X86 OPCODE!\n", |
| M.x86.R_CS, M.x86.R_IP-1,op1); |
| HALT_SYS(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x00 |
| ****************************************************************************/ |
| void x86emuOp_add_byte_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| u8 *destreg, *srcreg; |
| u8 destval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADD\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = add_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 1: |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = add_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 2: |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = add_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x01 |
| ****************************************************************************/ |
| void x86emuOp_add_word_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADD\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = add_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = add_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = add_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = add_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = add_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = add_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 3: /* 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 = add_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(); |
| *destreg = add_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x02 |
| ****************************************************************************/ |
| void x86emuOp_add_byte_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint srcoffset; |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADD\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_byte(*destreg, srcval); |
| break; |
| case 1: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_byte(*destreg, srcval); |
| break; |
| case 2: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_byte(*destreg, srcval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x03 |
| ****************************************************************************/ |
| void x86emuOp_add_word_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint srcoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADD\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_word(*destreg, srcval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_word(*destreg, srcval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = add_word(*destreg, srcval); |
| } |
| break; |
| case 3: /* 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 = add_long(*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 = add_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x04 |
| ****************************************************************************/ |
| void x86emuOp_add_byte_AL_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADD\tAL,"); |
| srcval = fetch_byte_imm(); |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| M.x86.R_AL = add_byte(M.x86.R_AL, srcval); |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x05 |
| ****************************************************************************/ |
| void x86emuOp_add_word_AX_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u32 srcval; |
| |
| START_OF_INSTR(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| DECODE_PRINTF("ADD\tEAX,"); |
| srcval = fetch_long_imm(); |
| } else { |
| DECODE_PRINTF("ADD\tAX,"); |
| srcval = fetch_word_imm(); |
| } |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| M.x86.R_EAX = add_long(M.x86.R_EAX, srcval); |
| } else { |
| M.x86.R_AX = add_word(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 0x08 |
| ****************************************************************************/ |
| void x86emuOp_or_byte_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint destoffset; |
| u8 destval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("OR\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = or_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 1: |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = or_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 2: |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = or_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x09 |
| ****************************************************************************/ |
| void x86emuOp_or_word_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("OR\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = or_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = or_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = or_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = or_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = or_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = or_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 3: /* 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 = or_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(); |
| *destreg = or_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x0a |
| ****************************************************************************/ |
| void x86emuOp_or_byte_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint srcoffset; |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("OR\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_byte(*destreg, srcval); |
| break; |
| case 1: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_byte(*destreg, srcval); |
| break; |
| case 2: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_byte(*destreg, srcval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x0b |
| ****************************************************************************/ |
| void x86emuOp_or_word_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint srcoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("OR\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_word(*destreg, srcval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_word(*destreg, srcval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = or_word(*destreg, srcval); |
| } |
| break; |
| case 3: /* 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 = or_long(*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 = or_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x0c |
| ****************************************************************************/ |
| void x86emuOp_or_byte_AL_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("OR\tAL,"); |
| srcval = fetch_byte_imm(); |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| M.x86.R_AL = or_byte(M.x86.R_AL, srcval); |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x0d |
| ****************************************************************************/ |
| void x86emuOp_or_word_AX_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u32 srcval; |
| |
| START_OF_INSTR(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| DECODE_PRINTF("OR\tEAX,"); |
| srcval = fetch_long_imm(); |
| } else { |
| DECODE_PRINTF("OR\tAX,"); |
| srcval = fetch_word_imm(); |
| } |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| M.x86.R_EAX = or_long(M.x86.R_EAX, srcval); |
| } else { |
| M.x86.R_AX = or_word(M.x86.R_AX, (u16)srcval); |
| } |
| 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 0x10 |
| ****************************************************************************/ |
| void x86emuOp_adc_byte_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint destoffset; |
| u8 destval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADC\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = adc_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 1: |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = adc_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 2: |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = adc_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x11 |
| ****************************************************************************/ |
| void x86emuOp_adc_word_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADC\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = adc_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = adc_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = adc_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = adc_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = adc_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = adc_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 3: /* 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 = adc_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(); |
| *destreg = adc_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x12 |
| ****************************************************************************/ |
| void x86emuOp_adc_byte_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint srcoffset; |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADC\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_byte(*destreg, srcval); |
| break; |
| case 1: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_byte(*destreg, srcval); |
| break; |
| case 2: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_byte(*destreg, srcval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x13 |
| ****************************************************************************/ |
| void x86emuOp_adc_word_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint srcoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADC\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_word(*destreg, srcval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_word(*destreg, srcval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = adc_word(*destreg, srcval); |
| } |
| break; |
| case 3: /* 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 = adc_long(*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 = adc_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x14 |
| ****************************************************************************/ |
| void x86emuOp_adc_byte_AL_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("ADC\tAL,"); |
| srcval = fetch_byte_imm(); |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| M.x86.R_AL = adc_byte(M.x86.R_AL, srcval); |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x15 |
| ****************************************************************************/ |
| void x86emuOp_adc_word_AX_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u32 srcval; |
| |
| START_OF_INSTR(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| DECODE_PRINTF("ADC\tEAX,"); |
| srcval = fetch_long_imm(); |
| } else { |
| DECODE_PRINTF("ADC\tAX,"); |
| srcval = fetch_word_imm(); |
| } |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| M.x86.R_EAX = adc_long(M.x86.R_EAX, srcval); |
| } else { |
| M.x86.R_AX = adc_word(M.x86.R_AX, (u16)srcval); |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| 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 0x18 |
| ****************************************************************************/ |
| void x86emuOp_sbb_byte_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint destoffset; |
| u8 destval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SBB\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sbb_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 1: |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sbb_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 2: |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sbb_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x19 |
| ****************************************************************************/ |
| void x86emuOp_sbb_word_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SBB\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sbb_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sbb_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sbb_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sbb_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sbb_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sbb_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 3: /* 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 = sbb_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(); |
| *destreg = sbb_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x1a |
| ****************************************************************************/ |
| void x86emuOp_sbb_byte_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint srcoffset; |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SBB\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_byte(*destreg, srcval); |
| break; |
| case 1: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_byte(*destreg, srcval); |
| break; |
| case 2: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_byte(*destreg, srcval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x1b |
| ****************************************************************************/ |
| void x86emuOp_sbb_word_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint srcoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SBB\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_word(*destreg, srcval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_word(*destreg, srcval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sbb_word(*destreg, srcval); |
| } |
| break; |
| case 3: /* 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 = sbb_long(*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 = sbb_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x1c |
| ****************************************************************************/ |
| void x86emuOp_sbb_byte_AL_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SBB\tAL,"); |
| srcval = fetch_byte_imm(); |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| M.x86.R_AL = sbb_byte(M.x86.R_AL, srcval); |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x1d |
| ****************************************************************************/ |
| void x86emuOp_sbb_word_AX_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u32 srcval; |
| |
| START_OF_INSTR(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| DECODE_PRINTF("SBB\tEAX,"); |
| srcval = fetch_long_imm(); |
| } else { |
| DECODE_PRINTF("SBB\tAX,"); |
| srcval = fetch_word_imm(); |
| } |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| M.x86.R_EAX = sbb_long(M.x86.R_EAX, srcval); |
| } else { |
| M.x86.R_AX = sbb_word(M.x86.R_AX, (u16)srcval); |
| } |
| 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 0x20 |
| ****************************************************************************/ |
| void x86emuOp_and_byte_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint destoffset; |
| u8 destval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("AND\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| |
| switch (mod) { |
| case 0: |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = and_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| |
| case 1: |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = and_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| |
| case 2: |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = and_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x21 |
| ****************************************************************************/ |
| void x86emuOp_and_word_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("AND\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = and_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = and_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = and_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = and_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = and_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = and_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 3: /* 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 = and_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(); |
| *destreg = and_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x22 |
| ****************************************************************************/ |
| void x86emuOp_and_byte_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint srcoffset; |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("AND\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_byte(*destreg, srcval); |
| break; |
| case 1: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_byte(*destreg, srcval); |
| break; |
| case 2: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_byte(*destreg, srcval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x23 |
| ****************************************************************************/ |
| void x86emuOp_and_word_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint srcoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("AND\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_word(*destreg, srcval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_long(*destreg, srcval); |
| break; |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_word(*destreg, srcval); |
| break; |
| } |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = and_word(*destreg, srcval); |
| } |
| break; |
| case 3: /* 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 = and_long(*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 = and_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x24 |
| ****************************************************************************/ |
| void x86emuOp_and_byte_AL_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("AND\tAL,"); |
| srcval = fetch_byte_imm(); |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| M.x86.R_AL = and_byte(M.x86.R_AL, srcval); |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x25 |
| ****************************************************************************/ |
| void x86emuOp_and_word_AX_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u32 srcval; |
| |
| START_OF_INSTR(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| DECODE_PRINTF("AND\tEAX,"); |
| srcval = fetch_long_imm(); |
| } else { |
| DECODE_PRINTF("AND\tAX,"); |
| srcval = fetch_word_imm(); |
| } |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| M.x86.R_EAX = and_long(M.x86.R_EAX, srcval); |
| } else { |
| M.x86.R_AX = and_word(M.x86.R_AX, (u16)srcval); |
| } |
| 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 0x28 |
| ****************************************************************************/ |
| void x86emuOp_sub_byte_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint destoffset; |
| u8 destval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SUB\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sub_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 1: |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sub_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 2: |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sub_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x29 |
| ****************************************************************************/ |
| void x86emuOp_sub_word_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SUB\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sub_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sub_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sub_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sub_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sub_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = sub_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 3: /* 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 = sub_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(); |
| *destreg = sub_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x2a |
| ****************************************************************************/ |
| void x86emuOp_sub_byte_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint srcoffset; |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SUB\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_byte(*destreg, srcval); |
| break; |
| case 1: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_byte(*destreg, srcval); |
| break; |
| case 2: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_byte(*destreg, srcval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x2b |
| ****************************************************************************/ |
| void x86emuOp_sub_word_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint srcoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SUB\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_word(*destreg, srcval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_word(*destreg, srcval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = sub_word(*destreg, srcval); |
| } |
| break; |
| case 3: /* 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 = sub_long(*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 = sub_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x2c |
| ****************************************************************************/ |
| void x86emuOp_sub_byte_AL_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("SUB\tAL,"); |
| srcval = fetch_byte_imm(); |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| M.x86.R_AL = sub_byte(M.x86.R_AL, srcval); |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x2d |
| ****************************************************************************/ |
| void x86emuOp_sub_word_AX_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u32 srcval; |
| |
| START_OF_INSTR(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| DECODE_PRINTF("SUB\tEAX,"); |
| srcval = fetch_long_imm(); |
| } else { |
| DECODE_PRINTF("SUB\tAX,"); |
| srcval = fetch_word_imm(); |
| } |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| M.x86.R_EAX = sub_long(M.x86.R_EAX, srcval); |
| } else { |
| M.x86.R_AX = sub_word(M.x86.R_AX, (u16)srcval); |
| } |
| 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 0x30 |
| ****************************************************************************/ |
| void x86emuOp_xor_byte_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint destoffset; |
| u8 destval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("XOR\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = xor_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 1: |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = xor_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 2: |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = xor_byte(destval, *srcreg); |
| store_data_byte(destoffset, destval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x31 |
| ****************************************************************************/ |
| void x86emuOp_xor_word_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("XOR\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = xor_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = xor_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = xor_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = xor_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = xor_long(destval, *srcreg); |
| store_data_long(destoffset, destval); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| destval = xor_word(destval, *srcreg); |
| store_data_word(destoffset, destval); |
| } |
| break; |
| case 3: /* 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 = xor_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(); |
| *destreg = xor_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x32 |
| ****************************************************************************/ |
| void x86emuOp_xor_byte_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint srcoffset; |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("XOR\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_byte(*destreg, srcval); |
| break; |
| case 1: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_byte(*destreg, srcval); |
| break; |
| case 2: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_byte(*destreg, srcval); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x33 |
| ****************************************************************************/ |
| void x86emuOp_xor_word_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint srcoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("XOR\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_word(*destreg, srcval); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_word(*destreg, srcval); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 *destreg; |
| u32 srcval; |
| |
| destreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_long(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_long(*destreg, srcval); |
| } else { |
| u16 *destreg; |
| u16 srcval; |
| |
| destreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm10_address(rl); |
| srcval = fetch_data_word(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| *destreg = xor_word(*destreg, srcval); |
| } |
| break; |
| case 3: /* 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 = xor_long(*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 = xor_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x34 |
| ****************************************************************************/ |
| void x86emuOp_xor_byte_AL_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("XOR\tAL,"); |
| srcval = fetch_byte_imm(); |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| M.x86.R_AL = xor_byte(M.x86.R_AL, srcval); |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x35 |
| ****************************************************************************/ |
| void x86emuOp_xor_word_AX_IMM(u8 X86EMU_UNUSED(op1)) |
| { |
| u32 srcval; |
| |
| START_OF_INSTR(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| DECODE_PRINTF("XOR\tEAX,"); |
| srcval = fetch_long_imm(); |
| } else { |
| DECODE_PRINTF("XOR\tAX,"); |
| srcval = fetch_word_imm(); |
| } |
| DECODE_PRINTF2("%x\n", srcval); |
| TRACE_AND_STEP(); |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| M.x86.R_EAX = xor_long(M.x86.R_EAX, srcval); |
| } else { |
| M.x86.R_AX = xor_word(M.x86.R_AX, (u16)srcval); |
| } |
| 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 0x38 |
| ****************************************************************************/ |
| void x86emuOp_cmp_byte_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| u8 *destreg, *srcreg; |
| u8 destval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("CMP\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_byte(destval, *srcreg); |
| break; |
| case 1: |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_byte(destval, *srcreg); |
| break; |
| case 2: |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_byte(destoffset); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_byte(destval, *srcreg); |
| break; |
| case 3: /* register to register */ |
| destreg = DECODE_RM_BYTE_REGISTER(rl); |
| DECODE_PRINTF(","); |
| srcreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_byte(*destreg, *srcreg); |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x39 |
| ****************************************************************************/ |
| void x86emuOp_cmp_word_RM_R(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| uint destoffset; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("CMP\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_long(destval, *srcreg); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm00_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_word(destval, *srcreg); |
| } |
| break; |
| case 1: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_long(destval, *srcreg); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm01_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_word(destval, *srcreg); |
| } |
| break; |
| case 2: |
| if (M.x86.mode & SYSMODE_PREFIX_DATA) { |
| u32 destval; |
| u32 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_long(destoffset); |
| srcreg = DECODE_RM_LONG_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_long(destval, *srcreg); |
| } else { |
| u16 destval; |
| u16 *srcreg; |
| |
| destoffset = decode_rm10_address(rl); |
| DECODE_PRINTF(","); |
| destval = fetch_data_word(destoffset); |
| srcreg = DECODE_RM_WORD_REGISTER(rh); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_word(destval, *srcreg); |
| } |
| break; |
| case 3: /* 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(); |
| cmp_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(); |
| cmp_word(*destreg, *srcreg); |
| } |
| break; |
| } |
| DECODE_CLEAR_SEGOVR(); |
| END_OF_INSTR(); |
| } |
| |
| /**************************************************************************** |
| REMARKS: |
| Handles opcode 0x3a |
| ****************************************************************************/ |
| void x86emuOp_cmp_byte_R_RM(u8 X86EMU_UNUSED(op1)) |
| { |
| int mod, rl, rh; |
| u8 *destreg, *srcreg; |
| uint srcoffset; |
| u8 srcval; |
| |
| START_OF_INSTR(); |
| DECODE_PRINTF("CMP\t"); |
| FETCH_DECODE_MODRM(mod, rh, rl); |
| switch (mod) { |
| case 0: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm00_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_byte(*destreg, srcval); |
| break; |
| case 1: |
| destreg = DECODE_RM_BYTE_REGISTER(rh); |
| DECODE_PRINTF(","); |
| srcoffset = decode_rm01_address(rl); |
| srcval = fetch_data_byte(srcoffset); |
| DECODE_PRINTF("\n"); |
| TRACE_AND_STEP(); |
| cmp_byte(*destreg, srcval); |
| break; |
| case 2 |