| "use strict"; |
| |
| var t = []; |
| var t16 = []; |
| var t32 = []; |
| |
| t[0x00] = cpu => { cpu.read_modrm_byte(); cpu.write_e8(cpu.add8(cpu.read_write_e8(), cpu.read_g8())); }; |
| t16[0x01] = cpu => { cpu.read_modrm_byte(); cpu.write_e16(cpu.add16(cpu.read_write_e16(), cpu.read_g16())); }; |
| t32[0x01] = cpu => { cpu.read_modrm_byte(); cpu.write_e32(cpu.add32(cpu.read_write_e32(), cpu.read_g32s())); }; |
| t[0x02] = cpu => { cpu.read_modrm_byte(); cpu.write_g8(cpu.add8(cpu.read_g8(), cpu.read_e8())); }; |
| t16[0x03] = cpu => { cpu.read_modrm_byte(); cpu.write_g16(cpu.add16(cpu.read_g16(), cpu.read_e16())); }; |
| t32[0x03] = cpu => { cpu.read_modrm_byte(); cpu.write_g32(cpu.add32(cpu.read_g32s(), cpu.read_e32s())); }; |
| t[0x04] = cpu => { cpu.reg8[reg_al] = cpu.add8(cpu.reg8[reg_al], cpu.read_op8()); }; |
| t16[0x05] = cpu => { cpu.reg16[reg_ax] = cpu.add16(cpu.reg16[reg_ax], cpu.read_op16()); }; |
| t32[0x05] = cpu => { cpu.reg32s[reg_eax] = cpu.add32(cpu.reg32s[reg_eax], cpu.read_op32s()); }; |
| |
| t16[0x06] = cpu => { cpu.push16(cpu.sreg[reg_es]); }; |
| t32[0x06] = cpu => { cpu.push32(cpu.sreg[reg_es]); }; |
| t16[0x07] = cpu => { |
| cpu.switch_seg(reg_es, cpu.safe_read16(cpu.get_stack_pointer(0))); |
| cpu.adjust_stack_reg(2); |
| }; |
| t32[0x07] = cpu => { |
| cpu.switch_seg(reg_es, cpu.safe_read32s(cpu.get_stack_pointer(0)) & 0xFFFF); |
| cpu.adjust_stack_reg(4); |
| }; |
| |
| t[0x08] = cpu => { cpu.read_modrm_byte(); cpu.write_e8(cpu.or8(cpu.read_write_e8(), cpu.read_g8())); }; |
| t16[0x09] = cpu => { cpu.read_modrm_byte(); cpu.write_e16(cpu.or16(cpu.read_write_e16(), cpu.read_g16())); }; |
| t32[0x09] = cpu => { cpu.read_modrm_byte(); cpu.write_e32(cpu.or32(cpu.read_write_e32(), cpu.read_g32s())); }; |
| t[0x0a] = cpu => { cpu.read_modrm_byte(); cpu.write_g8(cpu.or8(cpu.read_g8(), cpu.read_e8())); }; |
| t16[0x0b] = cpu => { cpu.read_modrm_byte(); cpu.write_g16(cpu.or16(cpu.read_g16(), cpu.read_e16())); }; |
| t32[0x0b] = cpu => { cpu.read_modrm_byte(); cpu.write_g32(cpu.or32(cpu.read_g32s(), cpu.read_e32s())); } |
| t[0x0c] = cpu => { cpu.reg8[reg_al] = cpu.or8(cpu.reg8[reg_al], cpu.read_op8()); }; |
| t16[0x0d] = cpu => { cpu.reg16[reg_ax] = cpu.or16(cpu.reg16[reg_ax], cpu.read_op16()); }; |
| t32[0x0d] = cpu => { cpu.reg32s[reg_eax] = cpu.or32(cpu.reg32s[reg_eax], cpu.read_op32s()); } |
| |
| |
| t16[0x0E] = cpu => { cpu.push16(cpu.sreg[reg_cs]); }; |
| t32[0x0E] = cpu => { cpu.push32(cpu.sreg[reg_cs]); }; |
| t16[0x0F] = cpu => { |
| cpu.table0F_16[cpu.read_op0F()](cpu); |
| }; |
| t32[0x0F] = cpu => { |
| cpu.table0F_32[cpu.read_op0F()](cpu); |
| }; |
| |
| t[0x10] = cpu => { cpu.read_modrm_byte(); cpu.write_e8(cpu.adc8(cpu.read_write_e8(), cpu.read_g8())); }; |
| t16[0x11] = cpu => { cpu.read_modrm_byte(); cpu.write_e16(cpu.adc16(cpu.read_write_e16(), cpu.read_g16())); }; |
| t32[0x11] = cpu => { cpu.read_modrm_byte(); cpu.write_e32(cpu.adc32(cpu.read_write_e32(), cpu.read_g32s())); } |
| t[0x12] = cpu => { cpu.read_modrm_byte(); cpu.write_g8(cpu.adc8(cpu.read_g8(), cpu.read_e8())); }; |
| t16[0x13] = cpu => { cpu.read_modrm_byte(); cpu.write_g16(cpu.adc16(cpu.read_g16(), cpu.read_e16())); }; |
| t32[0x13] = cpu => { cpu.read_modrm_byte(); cpu.write_g32(cpu.adc32(cpu.read_g32s(), cpu.read_e32s())); } |
| t[0x14] = cpu => { cpu.reg8[reg_al] = cpu.adc8(cpu.reg8[reg_al], cpu.read_op8()); }; |
| t16[0x15] = cpu => { cpu.reg16[reg_ax] = cpu.adc16(cpu.reg16[reg_ax], cpu.read_op16()); }; |
| t32[0x15] = cpu => { cpu.reg32s[reg_eax] = cpu.adc32(cpu.reg32s[reg_eax], cpu.read_op32s()); } |
| |
| t16[0x16] = cpu => { cpu.push16(cpu.sreg[reg_ss]); }; |
| t32[0x16] = cpu => { cpu.push32(cpu.sreg[reg_ss]); }; |
| t16[0x17] = cpu => { |
| cpu.switch_seg(reg_ss, cpu.safe_read16(cpu.get_stack_pointer(0))); |
| cpu.adjust_stack_reg(2); |
| cpu.clear_prefixes(); |
| cpu.cycle_internal(); |
| }; |
| t32[0x17] = cpu => { |
| cpu.switch_seg(reg_ss, cpu.safe_read32s(cpu.get_stack_pointer(0)) & 0xFFFF); |
| cpu.adjust_stack_reg(4); |
| cpu.clear_prefixes(); |
| cpu.cycle_internal(); |
| }; |
| |
| t[0x18] = cpu => { cpu.read_modrm_byte(); cpu.write_e8(cpu.sbb8(cpu.read_write_e8(), cpu.read_g8())); }; |
| t16[0x19] = cpu => { cpu.read_modrm_byte(); cpu.write_e16(cpu.sbb16(cpu.read_write_e16(), cpu.read_g16())); }; |
| t32[0x19] = cpu => { cpu.read_modrm_byte(); cpu.write_e32(cpu.sbb32(cpu.read_write_e32(), cpu.read_g32s())); } |
| t[0x1a] = cpu => { cpu.read_modrm_byte(); cpu.write_g8(cpu.sbb8(cpu.read_g8(), cpu.read_e8())); }; |
| t16[0x1b] = cpu => { cpu.read_modrm_byte(); cpu.write_g16(cpu.sbb16(cpu.read_g16(), cpu.read_e16())); }; |
| t32[0x1b] = cpu => { cpu.read_modrm_byte(); cpu.write_g32(cpu.sbb32(cpu.read_g32s(), cpu.read_e32s())); } |
| t[0x1c] = cpu => { cpu.reg8[reg_al] = cpu.sbb8(cpu.reg8[reg_al], cpu.read_op8()); }; |
| t16[0x1d] = cpu => { cpu.reg16[reg_ax] = cpu.sbb16(cpu.reg16[reg_ax], cpu.read_op16()); }; |
| t32[0x1d] = cpu => { cpu.reg32s[reg_eax] = cpu.sbb32(cpu.reg32s[reg_eax], cpu.read_op32s()); } |
| |
| |
| t16[0x1E] = cpu => { cpu.push16(cpu.sreg[reg_ds]); }; |
| t32[0x1E] = cpu => { cpu.push32(cpu.sreg[reg_ds]); }; |
| t16[0x1F] = cpu => { |
| cpu.switch_seg(reg_ds, cpu.safe_read16(cpu.get_stack_pointer(0))); |
| cpu.adjust_stack_reg(2); |
| }; |
| t32[0x1F] = cpu => { |
| cpu.switch_seg(reg_ds, cpu.safe_read32s(cpu.get_stack_pointer(0)) & 0xFFFF); |
| cpu.adjust_stack_reg(4); |
| }; |
| |
| t[0x20] = cpu => { cpu.read_modrm_byte(); cpu.write_e8(cpu.and8(cpu.read_write_e8(), cpu.read_g8())); }; |
| t16[0x21] = cpu => { cpu.read_modrm_byte(); cpu.write_e16(cpu.and16(cpu.read_write_e16(), cpu.read_g16())); }; |
| t32[0x21] = cpu => { cpu.read_modrm_byte(); cpu.write_e32(cpu.and32(cpu.read_write_e32(), cpu.read_g32s())); } |
| t[0x22] = cpu => { cpu.read_modrm_byte(); cpu.write_g8(cpu.and8(cpu.read_g8(), cpu.read_e8())); }; |
| t16[0x23] = cpu => { cpu.read_modrm_byte(); cpu.write_g16(cpu.and16(cpu.read_g16(), cpu.read_e16())); }; |
| t32[0x23] = cpu => { cpu.read_modrm_byte(); cpu.write_g32(cpu.and32(cpu.read_g32s(), cpu.read_e32s())); } |
| t[0x24] = cpu => { cpu.reg8[reg_al] = cpu.and8(cpu.reg8[reg_al], cpu.read_op8()); }; |
| t16[0x25] = cpu => { cpu.reg16[reg_ax] = cpu.and16(cpu.reg16[reg_ax], cpu.read_op16()); }; |
| t32[0x25] = cpu => { cpu.reg32s[reg_eax] = cpu.and32(cpu.reg32s[reg_eax], cpu.read_op32s()); } |
| |
| |
| t[0x26] = cpu => { cpu.segment_prefix_op(reg_es); }; |
| t[0x27] = cpu => { cpu.bcd_daa(); }; |
| |
| t[0x28] = cpu => { cpu.read_modrm_byte(); cpu.write_e8(cpu.sub8(cpu.read_write_e8(), cpu.read_g8())); }; |
| t16[0x29] = cpu => { cpu.read_modrm_byte(); cpu.write_e16(cpu.sub16(cpu.read_write_e16(), cpu.read_g16())); }; |
| t32[0x29] = cpu => { cpu.read_modrm_byte(); cpu.write_e32(cpu.sub32(cpu.read_write_e32(), cpu.read_g32s())); } |
| t[0x2a] = cpu => { cpu.read_modrm_byte(); cpu.write_g8(cpu.sub8(cpu.read_g8(), cpu.read_e8())); }; |
| t16[0x2b] = cpu => { cpu.read_modrm_byte(); cpu.write_g16(cpu.sub16(cpu.read_g16(), cpu.read_e16())); }; |
| t32[0x2b] = cpu => { cpu.read_modrm_byte(); cpu.write_g32(cpu.sub32(cpu.read_g32s(), cpu.read_e32s())); } |
| t[0x2c] = cpu => { cpu.reg8[reg_al] = cpu.sub8(cpu.reg8[reg_al], cpu.read_op8()); }; |
| t16[0x2d] = cpu => { cpu.reg16[reg_ax] = cpu.sub16(cpu.reg16[reg_ax], cpu.read_op16()); }; |
| t32[0x2d] = cpu => { cpu.reg32s[reg_eax] = cpu.sub32(cpu.reg32s[reg_eax], cpu.read_op32s()); } |
| |
| t[0x2E] = cpu => { cpu.segment_prefix_op(reg_cs); }; |
| t[0x2F] = cpu => { cpu.bcd_das(); }; |
| |
| t[0x30] = cpu => { cpu.read_modrm_byte(); cpu.write_e8(cpu.xor8(cpu.read_write_e8(), cpu.read_g8())); }; |
| t16[0x31] = cpu => { cpu.read_modrm_byte(); cpu.write_e16(cpu.xor16(cpu.read_write_e16(), cpu.read_g16())); }; |
| t32[0x31] = cpu => { cpu.read_modrm_byte(); cpu.write_e32(cpu.xor32(cpu.read_write_e32(), cpu.read_g32s())); } |
| t[0x32] = cpu => { cpu.read_modrm_byte(); cpu.write_g8(cpu.xor8(cpu.read_g8(), cpu.read_e8())); }; |
| t16[0x33] = cpu => { cpu.read_modrm_byte(); cpu.write_g16(cpu.xor16(cpu.read_g16(), cpu.read_e16())); }; |
| t32[0x33] = cpu => { cpu.read_modrm_byte(); cpu.write_g32(cpu.xor32(cpu.read_g32s(), cpu.read_e32s())); } |
| t[0x34] = cpu => { cpu.reg8[reg_al] = cpu.xor8(cpu.reg8[reg_al], cpu.read_op8()); }; |
| t16[0x35] = cpu => { cpu.reg16[reg_ax] = cpu.xor16(cpu.reg16[reg_ax], cpu.read_op16()); }; |
| t32[0x35] = cpu => { cpu.reg32s[reg_eax] = cpu.xor32(cpu.reg32s[reg_eax], cpu.read_op32s()); } |
| |
| t[0x36] = cpu => { cpu.segment_prefix_op(reg_ss); }; |
| t[0x37] = cpu => { cpu.bcd_aaa(); }; |
| |
| t[0x38] = cpu => { cpu.read_modrm_byte(); cpu.cmp8(cpu.read_e8(), cpu.read_g8()); }; |
| t16[0x39] = cpu => { cpu.read_modrm_byte(); cpu.cmp16(cpu.read_e16(), cpu.read_g16()); }; |
| t32[0x39] = cpu => { cpu.read_modrm_byte(); cpu.cmp32(cpu.read_e32s(), cpu.read_g32s()); } |
| t[0x3A] = cpu => { cpu.read_modrm_byte(); cpu.cmp8(cpu.read_g8(), cpu.read_e8()); }; |
| t16[0x3B] = cpu => { cpu.read_modrm_byte(); cpu.cmp16(cpu.read_g16(), cpu.read_e16()); }; |
| t32[0x3B] = cpu => { cpu.read_modrm_byte(); cpu.cmp32(cpu.read_g32s(), cpu.read_e32s()); } |
| t[0x3C] = cpu => { cpu.cmp8(cpu.reg8[reg_al], cpu.read_op8()); }; |
| t16[0x3D] = cpu => { cpu.cmp16(cpu.reg16[reg_ax], cpu.read_op16()); }; |
| t32[0x3D] = cpu => { cpu.cmp32(cpu.reg32s[reg_eax], cpu.read_op32s()); } |
| |
| t[0x3E] = cpu => { cpu.segment_prefix_op(reg_ds); }; |
| t[0x3F] = cpu => { cpu.bcd_aas(); }; |
| |
| |
| t16[0x40] = cpu => { cpu.reg16[reg_ax] = cpu.inc16(cpu.reg16[reg_ax]); }; |
| t32[0x40] = cpu => { cpu.reg32s[reg_eax] = cpu.inc32(cpu.reg32s[reg_eax]); }; |
| t16[0x41] = cpu => { cpu.reg16[reg_cx] = cpu.inc16(cpu.reg16[reg_cx]); }; |
| t32[0x41] = cpu => { cpu.reg32s[reg_ecx] = cpu.inc32(cpu.reg32s[reg_ecx]); }; |
| t16[0x42] = cpu => { cpu.reg16[reg_dx] = cpu.inc16(cpu.reg16[reg_dx]); }; |
| t32[0x42] = cpu => { cpu.reg32s[reg_edx] = cpu.inc32(cpu.reg32s[reg_edx]); }; |
| t16[0x43] = cpu => { cpu.reg16[reg_bx] = cpu.inc16(cpu.reg16[reg_bx]); }; |
| t32[0x43] = cpu => { cpu.reg32s[reg_ebx] = cpu.inc32(cpu.reg32s[reg_ebx]); }; |
| t16[0x44] = cpu => { cpu.reg16[reg_sp] = cpu.inc16(cpu.reg16[reg_sp]); }; |
| t32[0x44] = cpu => { cpu.reg32s[reg_esp] = cpu.inc32(cpu.reg32s[reg_esp]); }; |
| t16[0x45] = cpu => { cpu.reg16[reg_bp] = cpu.inc16(cpu.reg16[reg_bp]); }; |
| t32[0x45] = cpu => { cpu.reg32s[reg_ebp] = cpu.inc32(cpu.reg32s[reg_ebp]); }; |
| t16[0x46] = cpu => { cpu.reg16[reg_si] = cpu.inc16(cpu.reg16[reg_si]); }; |
| t32[0x46] = cpu => { cpu.reg32s[reg_esi] = cpu.inc32(cpu.reg32s[reg_esi]); }; |
| t16[0x47] = cpu => { cpu.reg16[reg_di] = cpu.inc16(cpu.reg16[reg_di]); }; |
| t32[0x47] = cpu => { cpu.reg32s[reg_edi] = cpu.inc32(cpu.reg32s[reg_edi]); }; |
| |
| |
| t16[0x48] = cpu => { cpu.reg16[reg_ax] = cpu.dec16(cpu.reg16[reg_ax]); }; |
| t32[0x48] = cpu => { cpu.reg32s[reg_eax] = cpu.dec32(cpu.reg32s[reg_eax]); }; |
| t16[0x49] = cpu => { cpu.reg16[reg_cx] = cpu.dec16(cpu.reg16[reg_cx]); }; |
| t32[0x49] = cpu => { cpu.reg32s[reg_ecx] = cpu.dec32(cpu.reg32s[reg_ecx]); }; |
| t16[0x4A] = cpu => { cpu.reg16[reg_dx] = cpu.dec16(cpu.reg16[reg_dx]); }; |
| t32[0x4A] = cpu => { cpu.reg32s[reg_edx] = cpu.dec32(cpu.reg32s[reg_edx]); }; |
| t16[0x4B] = cpu => { cpu.reg16[reg_bx] = cpu.dec16(cpu.reg16[reg_bx]); }; |
| t32[0x4B] = cpu => { cpu.reg32s[reg_ebx] = cpu.dec32(cpu.reg32s[reg_ebx]); }; |
| t16[0x4C] = cpu => { cpu.reg16[reg_sp] = cpu.dec16(cpu.reg16[reg_sp]); }; |
| t32[0x4C] = cpu => { cpu.reg32s[reg_esp] = cpu.dec32(cpu.reg32s[reg_esp]); }; |
| t16[0x4D] = cpu => { cpu.reg16[reg_bp] = cpu.dec16(cpu.reg16[reg_bp]); }; |
| t32[0x4D] = cpu => { cpu.reg32s[reg_ebp] = cpu.dec32(cpu.reg32s[reg_ebp]); }; |
| t16[0x4E] = cpu => { cpu.reg16[reg_si] = cpu.dec16(cpu.reg16[reg_si]); }; |
| t32[0x4E] = cpu => { cpu.reg32s[reg_esi] = cpu.dec32(cpu.reg32s[reg_esi]); }; |
| t16[0x4F] = cpu => { cpu.reg16[reg_di] = cpu.dec16(cpu.reg16[reg_di]); }; |
| t32[0x4F] = cpu => { cpu.reg32s[reg_edi] = cpu.dec32(cpu.reg32s[reg_edi]); }; |
| |
| |
| t16[0x50] = cpu => { cpu.push16(cpu.reg16[reg_ax]); }; |
| t32[0x50] = cpu => { cpu.push32(cpu.reg32s[reg_eax]); } |
| t16[0x51] = cpu => { cpu.push16(cpu.reg16[reg_cx]); }; |
| t32[0x51] = cpu => { cpu.push32(cpu.reg32s[reg_ecx]); } |
| t16[0x52] = cpu => { cpu.push16(cpu.reg16[reg_dx]); }; |
| t32[0x52] = cpu => { cpu.push32(cpu.reg32s[reg_edx]); } |
| t16[0x53] = cpu => { cpu.push16(cpu.reg16[reg_bx]); }; |
| t32[0x53] = cpu => { cpu.push32(cpu.reg32s[reg_ebx]); } |
| t16[0x54] = cpu => { cpu.push16(cpu.reg16[reg_sp]); }; |
| t32[0x54] = cpu => { cpu.push32(cpu.reg32s[reg_esp]); } |
| t16[0x55] = cpu => { cpu.push16(cpu.reg16[reg_bp]); }; |
| t32[0x55] = cpu => { cpu.push32(cpu.reg32s[reg_ebp]); } |
| t16[0x56] = cpu => { cpu.push16(cpu.reg16[reg_si]); }; |
| t32[0x56] = cpu => { cpu.push32(cpu.reg32s[reg_esi]); } |
| t16[0x57] = cpu => { cpu.push16(cpu.reg16[reg_di]); }; |
| t32[0x57] = cpu => { cpu.push32(cpu.reg32s[reg_edi]); } |
| |
| t16[0x58] = cpu => { cpu.reg16[reg_ax] = cpu.pop16(); }; |
| t32[0x58] = cpu => { cpu.reg32s[reg_eax] = cpu.pop32s(); } |
| t16[0x59] = cpu => { cpu.reg16[reg_cx] = cpu.pop16(); }; |
| t32[0x59] = cpu => { cpu.reg32s[reg_ecx] = cpu.pop32s(); } |
| t16[0x5A] = cpu => { cpu.reg16[reg_dx] = cpu.pop16(); }; |
| t32[0x5A] = cpu => { cpu.reg32s[reg_edx] = cpu.pop32s(); } |
| t16[0x5B] = cpu => { cpu.reg16[reg_bx] = cpu.pop16(); }; |
| t32[0x5B] = cpu => { cpu.reg32s[reg_ebx] = cpu.pop32s(); } |
| t16[0x5C] = cpu => { cpu.reg16[reg_sp] = cpu.pop16(); }; |
| t32[0x5C] = cpu => { cpu.reg32s[reg_esp] = cpu.pop32s(); } |
| t16[0x5D] = cpu => { cpu.reg16[reg_bp] = cpu.pop16(); }; |
| t32[0x5D] = cpu => { cpu.reg32s[reg_ebp] = cpu.pop32s(); } |
| t16[0x5E] = cpu => { cpu.reg16[reg_si] = cpu.pop16(); }; |
| t32[0x5E] = cpu => { cpu.reg32s[reg_esi] = cpu.pop32s(); } |
| t16[0x5F] = cpu => { cpu.reg16[reg_di] = cpu.pop16(); }; |
| t32[0x5F] = cpu => { cpu.reg32s[reg_edi] = cpu.pop32s(); } |
| |
| |
| t16[0x60] = cpu => { cpu.pusha16(); }; |
| t32[0x60] = cpu => { cpu.pusha32(); }; |
| t16[0x61] = cpu => { cpu.popa16(); }; |
| t32[0x61] = cpu => { cpu.popa32(); }; |
| |
| t[0x62] = cpu => { |
| // bound |
| dbg_log("Unimplemented BOUND instruction", LOG_CPU); |
| dbg_assert(false); |
| }; |
| t[0x63] = cpu => { cpu.read_modrm_byte(); |
| // arpl |
| //dbg_log("arpl", LOG_CPU); |
| if(cpu.protected_mode && !cpu.vm86_mode()) |
| { |
| cpu.write_e16(cpu.arpl(cpu.read_write_e16(), cpu.modrm_byte >> 2 & 14)); |
| } |
| else |
| { |
| dbg_log("arpl #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| }; |
| |
| t[0x64] = cpu => { cpu.segment_prefix_op(reg_fs); }; |
| t[0x65] = cpu => { cpu.segment_prefix_op(reg_gs); }; |
| |
| t[0x66] = cpu => { |
| // Operand-size override prefix |
| cpu.prefixes |= PREFIX_MASK_OPSIZE; |
| cpu.run_prefix_instruction(); |
| cpu.prefixes = 0; |
| }; |
| |
| t[0x67] = cpu => { |
| // Address-size override prefix |
| dbg_assert(cpu.is_asize_32() === cpu.is_32); |
| |
| cpu.prefixes |= PREFIX_MASK_ADDRSIZE; |
| cpu.run_prefix_instruction(); |
| cpu.prefixes = 0; |
| }; |
| |
| t16[0x68] = cpu => { cpu.push16(cpu.read_op16()); }; |
| t32[0x68] = cpu => { cpu.push32(cpu.read_op32s()); }; |
| |
| t16[0x69] = cpu => { cpu.read_modrm_byte(); |
| cpu.write_g16(cpu.imul_reg16(cpu.read_e16s(), cpu.read_op16() << 16 >> 16)); |
| }; |
| t32[0x69] = cpu => { cpu.read_modrm_byte(); |
| cpu.write_g32(cpu.imul_reg32(cpu.read_e32s(), cpu.read_op32s())); |
| }; |
| |
| t16[0x6A] = cpu => { cpu.push16(cpu.read_op8s()); }; |
| t32[0x6A] = cpu => { cpu.push32(cpu.read_op8s()); }; |
| |
| t16[0x6B] = cpu => { cpu.read_modrm_byte(); |
| cpu.write_g16(cpu.imul_reg16(cpu.read_e16s(), cpu.read_op8s())); |
| }; |
| t32[0x6B] = cpu => { cpu.read_modrm_byte(); |
| cpu.write_g32(cpu.imul_reg32(cpu.read_e32s(), cpu.read_op8s())); |
| }; |
| |
| t[0x6C] = cpu => { insb(cpu); }; |
| t16[0x6D] = cpu => { insw(cpu); }; |
| t32[0x6D] = cpu => { insd(cpu); }; |
| t[0x6E] = cpu => { outsb(cpu); }; |
| t16[0x6F] = cpu => { outsw(cpu); }; |
| t32[0x6F] = cpu => { outsd(cpu); }; |
| |
| t[0x70] = cpu => { cpu.jmpcc8( cpu.test_o()); }; |
| t[0x71] = cpu => { cpu.jmpcc8(!cpu.test_o()); }; |
| t[0x72] = cpu => { cpu.jmpcc8( cpu.test_b()); }; |
| t[0x73] = cpu => { cpu.jmpcc8(!cpu.test_b()); }; |
| t[0x74] = cpu => { cpu.jmpcc8( cpu.test_z()); }; |
| t[0x75] = cpu => { cpu.jmpcc8(!cpu.test_z()); }; |
| t[0x76] = cpu => { cpu.jmpcc8( cpu.test_be()); }; |
| t[0x77] = cpu => { cpu.jmpcc8(!cpu.test_be()); }; |
| t[0x78] = cpu => { cpu.jmpcc8( cpu.test_s()); }; |
| t[0x79] = cpu => { cpu.jmpcc8(!cpu.test_s()); }; |
| t[0x7A] = cpu => { cpu.jmpcc8( cpu.test_p()); }; |
| t[0x7B] = cpu => { cpu.jmpcc8(!cpu.test_p()); }; |
| t[0x7C] = cpu => { cpu.jmpcc8( cpu.test_l()); }; |
| t[0x7D] = cpu => { cpu.jmpcc8(!cpu.test_l()); }; |
| t[0x7E] = cpu => { cpu.jmpcc8( cpu.test_le()); }; |
| t[0x7F] = cpu => { cpu.jmpcc8(!cpu.test_le()); }; |
| |
| t[0x80] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: cpu.write_e8(cpu.add8(cpu.read_write_e8(), cpu.read_op8())); break; |
| case 1: cpu.write_e8(cpu. or8(cpu.read_write_e8(), cpu.read_op8())); break; |
| case 2: cpu.write_e8(cpu.adc8(cpu.read_write_e8(), cpu.read_op8())); break; |
| case 3: cpu.write_e8(cpu.sbb8(cpu.read_write_e8(), cpu.read_op8())); break; |
| case 4: cpu.write_e8(cpu.and8(cpu.read_write_e8(), cpu.read_op8())); break; |
| case 5: cpu.write_e8(cpu.sub8(cpu.read_write_e8(), cpu.read_op8())); break; |
| case 6: cpu.write_e8(cpu.xor8(cpu.read_write_e8(), cpu.read_op8())); break; |
| case 7: cpu.cmp8(cpu.read_e8(), cpu.read_op8()); break; |
| } |
| }; |
| t16[0x81] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: cpu.write_e16(cpu.add16(cpu.read_write_e16(), cpu.read_op16())); break; |
| case 1: cpu.write_e16(cpu. or16(cpu.read_write_e16(), cpu.read_op16())); break; |
| case 2: cpu.write_e16(cpu.adc16(cpu.read_write_e16(), cpu.read_op16())); break; |
| case 3: cpu.write_e16(cpu.sbb16(cpu.read_write_e16(), cpu.read_op16())); break; |
| case 4: cpu.write_e16(cpu.and16(cpu.read_write_e16(), cpu.read_op16())); break; |
| case 5: cpu.write_e16(cpu.sub16(cpu.read_write_e16(), cpu.read_op16())); break; |
| case 6: cpu.write_e16(cpu.xor16(cpu.read_write_e16(), cpu.read_op16())); break; |
| case 7: cpu.cmp16(cpu.read_e16(), cpu.read_op16()); break; |
| } |
| }; |
| t32[0x81] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: cpu.write_e32(cpu.add32(cpu.read_write_e32(), cpu.read_op32s())); break; |
| case 1: cpu.write_e32(cpu. or32(cpu.read_write_e32(), cpu.read_op32s())); break; |
| case 2: cpu.write_e32(cpu.adc32(cpu.read_write_e32(), cpu.read_op32s())); break; |
| case 3: cpu.write_e32(cpu.sbb32(cpu.read_write_e32(), cpu.read_op32s())); break; |
| case 4: cpu.write_e32(cpu.and32(cpu.read_write_e32(), cpu.read_op32s())); break; |
| case 5: cpu.write_e32(cpu.sub32(cpu.read_write_e32(), cpu.read_op32s())); break; |
| case 6: cpu.write_e32(cpu.xor32(cpu.read_write_e32(), cpu.read_op32s())); break; |
| case 7: cpu.cmp32(cpu.read_e32s(), cpu.read_op32s()); break; |
| } |
| }; |
| t[0x82] = t[0x80]; // alias |
| t16[0x83] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: cpu.write_e16(cpu.add16(cpu.read_write_e16(), cpu.read_op8s())); break; |
| case 1: cpu.write_e16(cpu. or16(cpu.read_write_e16(), cpu.read_op8s())); break; |
| case 2: cpu.write_e16(cpu.adc16(cpu.read_write_e16(), cpu.read_op8s())); break; |
| case 3: cpu.write_e16(cpu.sbb16(cpu.read_write_e16(), cpu.read_op8s())); break; |
| case 4: cpu.write_e16(cpu.and16(cpu.read_write_e16(), cpu.read_op8s())); break; |
| case 5: cpu.write_e16(cpu.sub16(cpu.read_write_e16(), cpu.read_op8s())); break; |
| case 6: cpu.write_e16(cpu.xor16(cpu.read_write_e16(), cpu.read_op8s())); break; |
| case 7: cpu.cmp16(cpu.read_e16(), cpu.read_op8s()); break; |
| } |
| }; |
| t32[0x83] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: cpu.write_e32(cpu.add32(cpu.read_write_e32(), cpu.read_op8s())); break; |
| case 1: cpu.write_e32(cpu. or32(cpu.read_write_e32(), cpu.read_op8s())); break; |
| case 2: cpu.write_e32(cpu.adc32(cpu.read_write_e32(), cpu.read_op8s())); break; |
| case 3: cpu.write_e32(cpu.sbb32(cpu.read_write_e32(), cpu.read_op8s())); break; |
| case 4: cpu.write_e32(cpu.and32(cpu.read_write_e32(), cpu.read_op8s())); break; |
| case 5: cpu.write_e32(cpu.sub32(cpu.read_write_e32(), cpu.read_op8s())); break; |
| case 6: cpu.write_e32(cpu.xor32(cpu.read_write_e32(), cpu.read_op8s())); break; |
| case 7: cpu.cmp32(cpu.read_e32s(), cpu.read_op8s()); break; |
| } |
| }; |
| |
| t[0x84] = cpu => { cpu.read_modrm_byte(); var data = cpu.read_e8(); cpu.test8(data, cpu.read_g8()); }; |
| t16[0x85] = cpu => { cpu.read_modrm_byte(); var data = cpu.read_e16(); cpu.test16(data, cpu.read_g16()); }; |
| t32[0x85] = cpu => { cpu.read_modrm_byte(); var data = cpu.read_e32s(); cpu.test32(data, cpu.read_g32s()); } |
| |
| |
| t[0x86] = cpu => { cpu.read_modrm_byte(); var data = cpu.read_write_e8(); cpu.write_e8(cpu.xchg8(data, cpu.modrm_byte)); }; |
| t16[0x87] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e16(); cpu.write_e16(cpu.xchg16(data, cpu.modrm_byte)); |
| }; |
| t32[0x87] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e32(); cpu.write_e32(cpu.xchg32(data, cpu.modrm_byte)); |
| }; |
| |
| t[0x88] = cpu => { cpu.read_modrm_byte(); cpu.set_e8(cpu.read_g8()); }; |
| t16[0x89] = cpu => { cpu.read_modrm_byte(); cpu.set_e16(cpu.read_g16()); }; |
| t32[0x89] = cpu => { cpu.read_modrm_byte(); cpu.set_e32(cpu.read_g32s()); } |
| |
| t[0x8A] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e8(); |
| cpu.write_g8(data); |
| }; |
| t16[0x8B] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e16(); |
| cpu.write_g16(data); |
| }; |
| t32[0x8B] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e32s(); |
| cpu.write_g32(data); |
| }; |
| |
| t16[0x8C] = cpu => { cpu.read_modrm_byte(); |
| cpu.set_e16(cpu.sreg[cpu.modrm_byte >> 3 & 7]); |
| }; |
| t32[0x8C] = cpu => { cpu.read_modrm_byte(); |
| cpu.set_e32(cpu.sreg[cpu.modrm_byte >> 3 & 7]); |
| }; |
| |
| t16[0x8D] = cpu => { cpu.read_modrm_byte(); |
| // lea |
| if(cpu.modrm_byte >= 0xC0) |
| { |
| dbg_log("lea #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| var mod = cpu.modrm_byte >> 3 & 7; |
| |
| // override prefix, so modrm_resolve does not return the segment part |
| cpu.prefixes |= SEG_PREFIX_ZERO; |
| cpu.reg16[mod << 1] = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.prefixes = 0; |
| }; |
| t32[0x8D] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte >= 0xC0) |
| { |
| dbg_log("lea #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| var mod = cpu.modrm_byte >> 3 & 7; |
| |
| cpu.prefixes |= SEG_PREFIX_ZERO; |
| cpu.reg32s[mod] = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.prefixes = 0; |
| }; |
| |
| t[0x8E] = cpu => { cpu.read_modrm_byte(); |
| var mod = cpu.modrm_byte >> 3 & 7; |
| var data = cpu.read_e16(); |
| cpu.switch_seg(mod, data); |
| |
| if(mod === reg_ss) |
| { |
| // run next instruction, so no interrupts are handled |
| cpu.clear_prefixes(); |
| cpu.cycle_internal(); |
| } |
| }; |
| |
| t16[0x8F] = cpu => { cpu.read_modrm_byte(); |
| // pop |
| var sp = cpu.safe_read16(cpu.get_stack_pointer(0)); |
| |
| cpu.adjust_stack_reg(2); |
| |
| if(cpu.modrm_byte < 0xC0) { |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.adjust_stack_reg(-2); |
| cpu.safe_write16(addr, sp); |
| cpu.adjust_stack_reg(2); |
| } else { |
| cpu.write_reg_e16(sp); |
| } |
| }; |
| t32[0x8F] = cpu => { cpu.read_modrm_byte(); |
| var sp = cpu.safe_read32s(cpu.get_stack_pointer(0)); |
| |
| // change esp first, then resolve modrm address |
| cpu.adjust_stack_reg(4); |
| |
| if(cpu.modrm_byte < 0xC0) { |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| |
| // Before attempting a write that might cause a page fault, |
| // we must set esp to the old value. Fuck Intel. |
| cpu.adjust_stack_reg(-4); |
| cpu.safe_write32(addr, sp); |
| cpu.adjust_stack_reg(4); |
| } else { |
| cpu.write_reg_e32(sp); |
| } |
| }; |
| |
| t[0x90] = cpu => { }; |
| t16[0x91] = cpu => { cpu.xchg16r(reg_cx) }; |
| t32[0x91] = cpu => { cpu.xchg32r(reg_ecx) }; |
| t16[0x92] = cpu => { cpu.xchg16r(reg_dx) }; |
| t32[0x92] = cpu => { cpu.xchg32r(reg_edx) }; |
| t16[0x93] = cpu => { cpu.xchg16r(reg_bx) }; |
| t32[0x93] = cpu => { cpu.xchg32r(reg_ebx) }; |
| t16[0x94] = cpu => { cpu.xchg16r(reg_sp) }; |
| t32[0x94] = cpu => { cpu.xchg32r(reg_esp) }; |
| t16[0x95] = cpu => { cpu.xchg16r(reg_bp) }; |
| t32[0x95] = cpu => { cpu.xchg32r(reg_ebp) }; |
| t16[0x96] = cpu => { cpu.xchg16r(reg_si) }; |
| t32[0x96] = cpu => { cpu.xchg32r(reg_esi) }; |
| t16[0x97] = cpu => { cpu.xchg16r(reg_di) }; |
| t32[0x97] = cpu => { cpu.xchg32r(reg_edi) }; |
| |
| t16[0x98] = cpu => { /* cbw */ cpu.reg16[reg_ax] = cpu.reg8s[reg_al]; }; |
| t32[0x98] = cpu => { /* cwde */ cpu.reg32s[reg_eax] = cpu.reg16s[reg_ax]; }; |
| t16[0x99] = cpu => { /* cwd */ cpu.reg16[reg_dx] = cpu.reg16s[reg_ax] >> 15; }; |
| t32[0x99] = cpu => { /* cdq */ cpu.reg32s[reg_edx] = cpu.reg32s[reg_eax] >> 31; }; |
| |
| t16[0x9A] = cpu => { |
| // callf |
| var new_ip = cpu.read_op16(); |
| var new_cs = cpu.read_disp16(); |
| |
| cpu.far_jump(new_ip, new_cs, true); |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| t32[0x9A] = cpu => { |
| var new_ip = cpu.read_op32s(); |
| var new_cs = cpu.read_disp16(); |
| |
| if(!cpu.protected_mode || cpu.vm86_mode()) |
| { |
| if(new_ip & 0xFFFF0000) |
| { |
| throw cpu.debug.unimpl("#GP handler"); |
| } |
| } |
| |
| cpu.far_jump(new_ip, new_cs, true); |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| |
| t[0x9B] = cpu => { |
| // fwait: check for pending fpu exceptions |
| if((cpu.cr[0] & (CR0_MP | CR0_TS)) === (CR0_MP | CR0_TS)) |
| { |
| // task switched and MP bit is set |
| cpu.trigger_nm(); |
| } |
| else |
| { |
| if(cpu.fpu) |
| { |
| cpu.fpu.fwait(); |
| } |
| else |
| { |
| // EM bit isn't checked |
| // If there's no FPU, do nothing |
| } |
| } |
| }; |
| t16[0x9C] = cpu => { |
| // pushf |
| if((cpu.flags & flag_vm) && cpu.getiopl() < 3) |
| { |
| dbg_assert(cpu.protected_mode); |
| dbg_log("pushf #gp", LOG_CPU); |
| cpu.trigger_gp(0); |
| } |
| else |
| { |
| cpu.push16(cpu.get_eflags()); |
| } |
| }; |
| t32[0x9C] = cpu => { |
| // pushf |
| if((cpu.flags & flag_vm) && cpu.getiopl() < 3) |
| { |
| // trap to virtual 8086 monitor |
| dbg_assert(cpu.protected_mode); |
| dbg_log("pushf #gp", LOG_CPU); |
| cpu.trigger_gp(0); |
| } |
| else |
| { |
| // vm and rf flag are cleared in image stored on the stack |
| cpu.push32(cpu.get_eflags() & 0x00FCFFFF); |
| } |
| }; |
| t16[0x9D] = cpu => { |
| // popf |
| if((cpu.flags & flag_vm) && cpu.getiopl() < 3) |
| { |
| dbg_log("popf #gp", LOG_CPU); |
| cpu.trigger_gp(0); |
| } |
| |
| cpu.update_eflags((cpu.flags & ~0xFFFF) | cpu.pop16()); |
| |
| if(cpu.flags & flag_trap) |
| { |
| // XXX: Problems with fdgame |
| //cpu.clear_prefixes(); |
| //cpu.cycle_internal(); |
| cpu.flags &= ~flag_trap; |
| //cpu.instruction_pointer = cpu.previous_ip; |
| //cpu.raise_exception(1); |
| } |
| else |
| { |
| cpu.handle_irqs(); |
| } |
| }; |
| t32[0x9D] = cpu => { |
| // popf |
| if((cpu.flags & flag_vm) && cpu.getiopl() < 3) |
| { |
| dbg_log("popf #gp", LOG_CPU); |
| cpu.trigger_gp(0); |
| } |
| |
| cpu.update_eflags(cpu.pop32s()); |
| cpu.handle_irqs(); |
| }; |
| t[0x9E] = cpu => { |
| // sahf |
| cpu.flags = (cpu.flags & ~0xFF) | cpu.reg8[reg_ah]; |
| cpu.flags = (cpu.flags & flags_mask) | flags_default; |
| cpu.flags_changed = 0; |
| }; |
| t[0x9F] = cpu => { |
| // lahf |
| cpu.reg8[reg_ah] = cpu.get_eflags(); |
| }; |
| |
| t[0xA0] = cpu => { |
| // mov |
| var data = cpu.safe_read8(cpu.read_moffs()); |
| cpu.reg8[reg_al] = data; |
| }; |
| t16[0xA1] = cpu => { |
| // mov |
| var data = cpu.safe_read16(cpu.read_moffs()); |
| cpu.reg16[reg_ax] = data; |
| }; |
| t32[0xA1] = cpu => { |
| var data = cpu.safe_read32s(cpu.read_moffs()); |
| cpu.reg32s[reg_eax] = data; |
| }; |
| t[0xA2] = cpu => { |
| // mov |
| cpu.safe_write8(cpu.read_moffs(), cpu.reg8[reg_al]); |
| }; |
| t16[0xA3] = cpu => { |
| // mov |
| cpu.safe_write16(cpu.read_moffs(), cpu.reg16[reg_ax]); |
| }; |
| t32[0xA3] = cpu => { |
| cpu.safe_write32(cpu.read_moffs(), cpu.reg32s[reg_eax]); |
| }; |
| |
| t[0xA4] = cpu => { cpu.movsb(); }; |
| t16[0xA5] = cpu => { cpu.movsw(); }; |
| t32[0xA5] = cpu => { cpu.movsd(); }; |
| t[0xA6] = cpu => { cmpsb(cpu); }; |
| t16[0xA7] = cpu => { cmpsw(cpu); }; |
| t32[0xA7] = cpu => { cmpsd(cpu); }; |
| |
| t[0xA8] = cpu => { |
| cpu.test8(cpu.reg8[reg_al], cpu.read_op8()); |
| }; |
| t16[0xA9] = cpu => { |
| cpu.test16(cpu.reg16[reg_ax], cpu.read_op16()); |
| }; |
| t32[0xA9] = cpu => { |
| cpu.test32(cpu.reg32s[reg_eax], cpu.read_op32s()); |
| }; |
| |
| t[0xAA] = cpu => { stosb(cpu); }; |
| t16[0xAB] = cpu => { stosw(cpu); }; |
| t32[0xAB] = cpu => { stosd(cpu); }; |
| t[0xAC] = cpu => { lodsb(cpu); }; |
| t16[0xAD] = cpu => { lodsw(cpu); }; |
| t32[0xAD] = cpu => { lodsd(cpu); }; |
| t[0xAE] = cpu => { scasb(cpu); }; |
| t16[0xAF] = cpu => { scasw(cpu); }; |
| t32[0xAF] = cpu => { scasd(cpu); }; |
| |
| |
| t[0xB0] = cpu => { cpu.reg8[reg_al] = cpu.read_op8(); }; |
| t[0xB1] = cpu => { cpu.reg8[reg_cl] = cpu.read_op8(); }; |
| t[0xB2] = cpu => { cpu.reg8[reg_dl] = cpu.read_op8(); }; |
| t[0xB3] = cpu => { cpu.reg8[reg_bl] = cpu.read_op8(); }; |
| t[0xB4] = cpu => { cpu.reg8[reg_ah] = cpu.read_op8(); }; |
| t[0xB5] = cpu => { cpu.reg8[reg_ch] = cpu.read_op8(); }; |
| t[0xB6] = cpu => { cpu.reg8[reg_dh] = cpu.read_op8(); }; |
| t[0xB7] = cpu => { cpu.reg8[reg_bh] = cpu.read_op8(); }; |
| |
| t16[0xB8] = cpu => { cpu.reg16[reg_ax] = cpu.read_op16(); }; |
| t32[0xB8] = cpu => { cpu.reg32s[reg_eax] = cpu.read_op32s(); }; |
| t16[0xB9] = cpu => { cpu.reg16[reg_cx] = cpu.read_op16(); }; |
| t32[0xB9] = cpu => { cpu.reg32s[reg_ecx] = cpu.read_op32s(); }; |
| t16[0xBA] = cpu => { cpu.reg16[reg_dx] = cpu.read_op16(); }; |
| t32[0xBA] = cpu => { cpu.reg32s[reg_edx] = cpu.read_op32s(); }; |
| t16[0xBB] = cpu => { cpu.reg16[reg_bx] = cpu.read_op16(); }; |
| t32[0xBB] = cpu => { cpu.reg32s[reg_ebx] = cpu.read_op32s(); }; |
| t16[0xBC] = cpu => { cpu.reg16[reg_sp] = cpu.read_op16(); }; |
| t32[0xBC] = cpu => { cpu.reg32s[reg_esp] = cpu.read_op32s(); }; |
| t16[0xBD] = cpu => { cpu.reg16[reg_bp] = cpu.read_op16(); }; |
| t32[0xBD] = cpu => { cpu.reg32s[reg_ebp] = cpu.read_op32s(); }; |
| t16[0xBE] = cpu => { cpu.reg16[reg_si] = cpu.read_op16(); }; |
| t32[0xBE] = cpu => { cpu.reg32s[reg_esi] = cpu.read_op32s(); }; |
| t16[0xBF] = cpu => { cpu.reg16[reg_di] = cpu.read_op16(); }; |
| t32[0xBF] = cpu => { cpu.reg32s[reg_edi] = cpu.read_op32s(); }; |
| |
| |
| t[0xC0] = cpu => { cpu.read_modrm_byte(); |
| var op1 = cpu.read_write_e8(); |
| var op2 = cpu.read_op8() & 31; |
| var result = 0; |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: result = cpu.rol8(op1, op2); break; |
| case 1: result = cpu.ror8(op1, op2); break; |
| case 2: result = cpu.rcl8(op1, op2); break; |
| case 3: result = cpu.rcr8(op1, op2); break; |
| case 4: result = cpu.shl8(op1, op2); break; |
| case 5: result = cpu.shr8(op1, op2); break; |
| case 6: result = cpu.shl8(op1, op2); break; |
| case 7: result = cpu.sar8(op1, op2); break; |
| } |
| cpu.write_e8(result); |
| }; |
| t16[0xC1] = cpu => { cpu.read_modrm_byte(); |
| var op1 = cpu.read_write_e16(); |
| var op2 = cpu.read_op8() & 31; |
| var result = 0; |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: result = cpu.rol16(op1, op2); break; |
| case 1: result = cpu.ror16(op1, op2); break; |
| case 2: result = cpu.rcl16(op1, op2); break; |
| case 3: result = cpu.rcr16(op1, op2); break; |
| case 4: result = cpu.shl16(op1, op2); break; |
| case 5: result = cpu.shr16(op1, op2); break; |
| case 6: result = cpu.shl16(op1, op2); break; |
| case 7: result = cpu.sar16(op1, op2); break; |
| } |
| cpu.write_e16(result); |
| }; |
| t32[0xC1] = cpu => { cpu.read_modrm_byte(); |
| var op1 = cpu.read_write_e32(); |
| var op2 = cpu.read_op8() & 31; |
| var result = 0; |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: result = cpu.rol32(op1, op2); break; |
| case 1: result = cpu.ror32(op1, op2); break; |
| case 2: result = cpu.rcl32(op1, op2); break; |
| case 3: result = cpu.rcr32(op1, op2); break; |
| case 4: result = cpu.shl32(op1, op2); break; |
| case 5: result = cpu.shr32(op1, op2); break; |
| case 6: result = cpu.shl32(op1, op2); break; |
| case 7: result = cpu.sar32(op1, op2); break; |
| } |
| cpu.write_e32(result); |
| }; |
| |
| t16[0xC2] = cpu => { |
| // retn |
| var imm16 = cpu.read_op16(); |
| |
| cpu.instruction_pointer = cpu.get_seg(reg_cs) + cpu.pop16() | 0; |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.adjust_stack_reg(imm16); |
| cpu.diverged(); |
| }; |
| t32[0xC2] = cpu => { |
| // retn |
| var imm16 = cpu.read_op16(); |
| var ip = cpu.pop32s(); |
| |
| dbg_assert(cpu.is_asize_32() || ip < 0x10000); |
| cpu.instruction_pointer = cpu.get_seg(reg_cs) + ip | 0; |
| cpu.adjust_stack_reg(imm16); |
| cpu.diverged(); |
| }; |
| t16[0xC3] = cpu => { |
| // retn |
| cpu.instruction_pointer = cpu.get_seg(reg_cs) + cpu.pop16() | 0; |
| cpu.diverged(); |
| }; |
| t32[0xC3] = cpu => { |
| // retn |
| var ip = cpu.pop32s(); |
| dbg_assert(cpu.is_asize_32() || ip < 0x10000); |
| cpu.instruction_pointer = cpu.get_seg(reg_cs) + ip | 0; |
| cpu.diverged(); |
| }; |
| |
| t16[0xC4] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss16(reg_es); |
| }; |
| t32[0xC4] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss32(reg_es); |
| }; |
| t16[0xC5] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss16(reg_ds); |
| }; |
| t32[0xC5] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss32(reg_ds); |
| }; |
| |
| t[0xC6] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.safe_write8(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op8()); |
| } else { |
| cpu.reg8[cpu.modrm_byte << 2 & 0xC | cpu.modrm_byte >> 2 & 1] = cpu.read_op8(); |
| } |
| } |
| t16[0xC7] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.safe_write16(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op16()); |
| } else { |
| cpu.reg16[cpu.modrm_byte << 1 & 14] = cpu.read_op16(); |
| } |
| }; |
| t32[0xC7] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.safe_write32(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op32s()); |
| } else { |
| cpu.reg32s[cpu.modrm_byte & 7] = cpu.read_op32s(); |
| } |
| } |
| |
| t16[0xC8] = cpu => { cpu.enter16(cpu.read_op16(), cpu.read_disp8()); }; |
| t32[0xC8] = cpu => { cpu.enter32(cpu.read_op16(), cpu.read_disp8()); }; |
| t16[0xC9] = cpu => { |
| // leave |
| var old_vbp = cpu.stack_size_32 ? cpu.reg32s[reg_ebp] : cpu.reg16[reg_bp]; |
| var new_bp = cpu.safe_read16(cpu.get_seg(reg_ss) + old_vbp | 0); |
| cpu.set_stack_reg(old_vbp + 2 | 0); |
| cpu.reg16[reg_bp] = new_bp; |
| }; |
| t32[0xC9] = cpu => { |
| var old_vbp = cpu.stack_size_32 ? cpu.reg32s[reg_ebp] : cpu.reg16[reg_bp]; |
| var new_ebp = cpu.safe_read32s(cpu.get_seg(reg_ss) + old_vbp | 0); |
| cpu.set_stack_reg(old_vbp + 4 | 0); |
| cpu.reg32s[reg_ebp] = new_ebp; |
| }; |
| t16[0xCA] = cpu => { |
| // retf |
| var imm16 = cpu.read_op16(); |
| var ip = cpu.safe_read16(cpu.get_stack_pointer(0)); |
| var cs = cpu.safe_read16(cpu.get_stack_pointer(2)); |
| |
| cpu.far_return(ip, cs, imm16); |
| cpu.diverged(); |
| }; |
| t32[0xCA] = cpu => { |
| // retf |
| var imm16 = cpu.read_op16(); |
| var ip = cpu.safe_read32s(cpu.get_stack_pointer(0)); |
| var cs = cpu.safe_read32s(cpu.get_stack_pointer(4)) & 0xFFFF; |
| |
| cpu.far_return(ip, cs, imm16); |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| t16[0xCB] = cpu => { |
| // retf |
| var ip = cpu.safe_read16(cpu.get_stack_pointer(0)); |
| var cs = cpu.safe_read16(cpu.get_stack_pointer(2)); |
| |
| cpu.far_return(ip, cs, 0); |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| t32[0xCB] = cpu => { |
| // retf |
| var ip = cpu.safe_read32s(cpu.get_stack_pointer(0)); |
| var cs = cpu.safe_read32s(cpu.get_stack_pointer(4)) & 0xFFFF; |
| |
| cpu.far_return(ip, cs, 0); |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| |
| t[0xCC] = cpu => { |
| // INT3 |
| // TODO: inhibit iopl checks |
| dbg_log("INT3", LOG_CPU); |
| cpu.call_interrupt_vector(3, true, false); |
| cpu.diverged(); |
| }; |
| t[0xCD] = cpu => { |
| // INT |
| var imm8 = cpu.read_op8(); |
| cpu.call_interrupt_vector(imm8, true, false); |
| cpu.diverged(); |
| }; |
| t[0xCE] = cpu => { |
| // INTO |
| dbg_log("INTO", LOG_CPU); |
| if(cpu.getof()) |
| { |
| // TODO: inhibit iopl checks |
| cpu.call_interrupt_vector(4, true, false); |
| } |
| cpu.diverged(); |
| }; |
| |
| t16[0xCF] = cpu => { |
| // iret |
| cpu.iret16(); |
| cpu.diverged(); |
| }; |
| t32[0xCF] = cpu => { |
| cpu.iret32(); |
| cpu.diverged(); |
| }; |
| |
| t[0xD0] = cpu => { cpu.read_modrm_byte(); |
| var op1 = cpu.read_write_e8(); |
| var result = 0; |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: result = cpu.rol8(op1, 1); break; |
| case 1: result = cpu.ror8(op1, 1); break; |
| case 2: result = cpu.rcl8(op1, 1); break; |
| case 3: result = cpu.rcr8(op1, 1); break; |
| case 4: result = cpu.shl8(op1, 1); break; |
| case 5: result = cpu.shr8(op1, 1); break; |
| case 6: result = cpu.shl8(op1, 1); break; |
| case 7: result = cpu.sar8(op1, 1); break; |
| } |
| cpu.write_e8(result); |
| }; |
| t16[0xD1] = cpu => { cpu.read_modrm_byte(); |
| var op1 = cpu.read_write_e16(); |
| var result = 0; |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: result = cpu.rol16(op1, 1); break; |
| case 1: result = cpu.ror16(op1, 1); break; |
| case 2: result = cpu.rcl16(op1, 1); break; |
| case 3: result = cpu.rcr16(op1, 1); break; |
| case 4: result = cpu.shl16(op1, 1); break; |
| case 5: result = cpu.shr16(op1, 1); break; |
| case 6: result = cpu.shl16(op1, 1); break; |
| case 7: result = cpu.sar16(op1, 1); break; |
| } |
| cpu.write_e16(result); |
| }; |
| t32[0xD1] = cpu => { cpu.read_modrm_byte(); |
| var op1 = cpu.read_write_e32(); |
| var result = 0; |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: result = cpu.rol32(op1, 1); break; |
| case 1: result = cpu.ror32(op1, 1); break; |
| case 2: result = cpu.rcl32(op1, 1); break; |
| case 3: result = cpu.rcr32(op1, 1); break; |
| case 4: result = cpu.shl32(op1, 1); break; |
| case 5: result = cpu.shr32(op1, 1); break; |
| case 6: result = cpu.shl32(op1, 1); break; |
| case 7: result = cpu.sar32(op1, 1); break; |
| } |
| cpu.write_e32(result); |
| }; |
| |
| t[0xD2] = cpu => { cpu.read_modrm_byte(); |
| var op1 = cpu.read_write_e8(); |
| var op2 = cpu.reg8[reg_cl] & 31; |
| var result = 0; |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: result = cpu.rol8(op1, op2); break; |
| case 1: result = cpu.ror8(op1, op2); break; |
| case 2: result = cpu.rcl8(op1, op2); break; |
| case 3: result = cpu.rcr8(op1, op2); break; |
| case 4: result = cpu.shl8(op1, op2); break; |
| case 5: result = cpu.shr8(op1, op2); break; |
| case 6: result = cpu.shl8(op1, op2); break; |
| case 7: result = cpu.sar8(op1, op2); break; |
| } |
| cpu.write_e8(result); |
| }; |
| t16[0xD3] = cpu => { cpu.read_modrm_byte(); |
| var op1 = cpu.read_write_e16(); |
| var op2 = cpu.reg8[reg_cl] & 31; |
| var result = 0; |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: result = cpu.rol16(op1, op2); break; |
| case 1: result = cpu.ror16(op1, op2); break; |
| case 2: result = cpu.rcl16(op1, op2); break; |
| case 3: result = cpu.rcr16(op1, op2); break; |
| case 4: result = cpu.shl16(op1, op2); break; |
| case 5: result = cpu.shr16(op1, op2); break; |
| case 6: result = cpu.shl16(op1, op2); break; |
| case 7: result = cpu.sar16(op1, op2); break; |
| } |
| cpu.write_e16(result); |
| }; |
| t32[0xD3] = cpu => { cpu.read_modrm_byte(); |
| var op1 = cpu.read_write_e32(); |
| var op2 = cpu.reg8[reg_cl] & 31; |
| var result = 0; |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: result = cpu.rol32(op1, op2); break; |
| case 1: result = cpu.ror32(op1, op2); break; |
| case 2: result = cpu.rcl32(op1, op2); break; |
| case 3: result = cpu.rcr32(op1, op2); break; |
| case 4: result = cpu.shl32(op1, op2); break; |
| case 5: result = cpu.shr32(op1, op2); break; |
| case 6: result = cpu.shl32(op1, op2); break; |
| case 7: result = cpu.sar32(op1, op2); break; |
| } |
| cpu.write_e32(result); |
| }; |
| |
| t[0xD4] = cpu => { |
| cpu.bcd_aam(cpu.read_op8()); |
| }; |
| t[0xD5] = cpu => { |
| cpu.bcd_aad(cpu.read_op8()); |
| }; |
| |
| t[0xD6] = cpu => { |
| // salc |
| cpu.reg8[reg_al] = -cpu.getcf(); |
| }; |
| t[0xD7] = cpu => { |
| // xlat |
| if(cpu.is_asize_32()) |
| { |
| cpu.reg8[reg_al] = cpu.safe_read8(cpu.get_seg_prefix(reg_ds) + cpu.reg32s[reg_ebx] + cpu.reg8[reg_al] | 0); |
| } |
| else |
| { |
| cpu.reg8[reg_al] = cpu.safe_read8(cpu.get_seg_prefix(reg_ds) + (cpu.reg16[reg_bx] + cpu.reg8[reg_al] & 0xFFFF) | 0); |
| } |
| }; |
| |
| t[0xD8] = cpu => { cpu.read_modrm_byte(); |
| cpu.task_switch_test(); |
| if(cpu.modrm_byte < 0xC0) |
| cpu.fpu.op_D8_mem(cpu.modrm_byte, cpu.modrm_resolve(cpu.modrm_byte)); |
| else |
| cpu.fpu.op_D8_reg(cpu.modrm_byte); |
| }; |
| t[0xD9] = cpu => { cpu.read_modrm_byte(); |
| cpu.task_switch_test(); |
| if(cpu.modrm_byte < 0xC0) |
| cpu.fpu.op_D9_mem(cpu.modrm_byte, cpu.modrm_resolve(cpu.modrm_byte)); |
| else |
| cpu.fpu.op_D9_reg(cpu.modrm_byte); |
| }; |
| t[0xDA] = cpu => { cpu.read_modrm_byte(); |
| cpu.task_switch_test(); |
| if(cpu.modrm_byte < 0xC0) |
| cpu.fpu.op_DA_mem(cpu.modrm_byte, cpu.modrm_resolve(cpu.modrm_byte)); |
| else |
| cpu.fpu.op_DA_reg(cpu.modrm_byte); |
| }; |
| t[0xDB] = cpu => { cpu.read_modrm_byte(); |
| cpu.task_switch_test(); |
| if(cpu.modrm_byte < 0xC0) |
| cpu.fpu.op_DB_mem(cpu.modrm_byte, cpu.modrm_resolve(cpu.modrm_byte)); |
| else |
| cpu.fpu.op_DB_reg(cpu.modrm_byte); |
| }; |
| t[0xDC] = cpu => { cpu.read_modrm_byte(); |
| cpu.task_switch_test(); |
| if(cpu.modrm_byte < 0xC0) |
| cpu.fpu.op_DC_mem(cpu.modrm_byte, cpu.modrm_resolve(cpu.modrm_byte)); |
| else |
| cpu.fpu.op_DC_reg(cpu.modrm_byte); |
| }; |
| t[0xDD] = cpu => { cpu.read_modrm_byte(); |
| cpu.task_switch_test(); |
| if(cpu.modrm_byte < 0xC0) |
| cpu.fpu.op_DD_mem(cpu.modrm_byte, cpu.modrm_resolve(cpu.modrm_byte)); |
| else |
| cpu.fpu.op_DD_reg(cpu.modrm_byte); |
| }; |
| t[0xDE] = cpu => { cpu.read_modrm_byte(); |
| cpu.task_switch_test(); |
| if(cpu.modrm_byte < 0xC0) |
| cpu.fpu.op_DE_mem(cpu.modrm_byte, cpu.modrm_resolve(cpu.modrm_byte)); |
| else |
| cpu.fpu.op_DE_reg(cpu.modrm_byte); |
| }; |
| t[0xDF] = cpu => { cpu.read_modrm_byte(); |
| cpu.task_switch_test(); |
| if(cpu.modrm_byte < 0xC0) |
| cpu.fpu.op_DF_mem(cpu.modrm_byte, cpu.modrm_resolve(cpu.modrm_byte)); |
| else |
| cpu.fpu.op_DF_reg(cpu.modrm_byte); |
| }; |
| |
| t[0xE0] = cpu => { cpu.loopne(cpu.read_op8s()); }; |
| t[0xE1] = cpu => { cpu.loope(cpu.read_op8s()); }; |
| t[0xE2] = cpu => { cpu.loop(cpu.read_op8s()); }; |
| t[0xE3] = cpu => { cpu.jcxz(cpu.read_op8s()); }; |
| |
| t[0xE4] = cpu => { |
| var port = cpu.read_op8(); |
| cpu.test_privileges_for_io(port, 1); |
| cpu.reg8[reg_al] = cpu.io.port_read8(port); |
| cpu.diverged(); |
| }; |
| t16[0xE5] = cpu => { |
| var port = cpu.read_op8(); |
| cpu.test_privileges_for_io(port, 2); |
| cpu.reg16[reg_ax] = cpu.io.port_read16(port); |
| cpu.diverged(); |
| }; |
| t32[0xE5] = cpu => { |
| var port = cpu.read_op8(); |
| cpu.test_privileges_for_io(port, 4); |
| cpu.reg32s[reg_eax] = cpu.io.port_read32(port); |
| cpu.diverged(); |
| }; |
| t[0xE6] = cpu => { |
| var port = cpu.read_op8(); |
| cpu.test_privileges_for_io(port, 1); |
| cpu.io.port_write8(port, cpu.reg8[reg_al]); |
| cpu.diverged(); |
| }; |
| t16[0xE7] = cpu => { |
| var port = cpu.read_op8(); |
| cpu.test_privileges_for_io(port, 2); |
| cpu.io.port_write16(port, cpu.reg16[reg_ax]); |
| cpu.diverged(); |
| }; |
| t32[0xE7] = cpu => { |
| var port = cpu.read_op8(); |
| cpu.test_privileges_for_io(port, 4); |
| cpu.io.port_write32(port, cpu.reg32s[reg_eax]); |
| cpu.diverged(); |
| }; |
| |
| t16[0xE8] = cpu => { |
| // call |
| var imm16 = cpu.read_op16(); |
| cpu.push16(cpu.get_real_eip()); |
| |
| cpu.jmp_rel16(imm16); |
| cpu.diverged(); |
| }; |
| t32[0xE8] = cpu => { |
| // call |
| var imm32s = cpu.read_op32s(); |
| cpu.push32(cpu.get_real_eip()); |
| |
| cpu.instruction_pointer = cpu.instruction_pointer + imm32s | 0; |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| t16[0xE9] = cpu => { |
| // jmp |
| var imm16 = cpu.read_op16(); |
| cpu.jmp_rel16(imm16); |
| cpu.diverged(); |
| }; |
| t32[0xE9] = cpu => { |
| // jmp |
| var imm32s = cpu.read_op32s(); |
| cpu.instruction_pointer = cpu.instruction_pointer + imm32s | 0; |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| t16[0xEA] = cpu => { |
| // jmpf |
| var ip = cpu.read_op16(); |
| var cs = cpu.read_disp16(); |
| cpu.far_jump(ip, cs, false); |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| t32[0xEA] = cpu => { |
| // jmpf |
| var new_ip = cpu.read_op32s(); |
| var cs = cpu.read_disp16(); |
| cpu.far_jump(new_ip, cs, false); |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| t[0xEB] = cpu => { |
| // jmp near |
| var imm8 = cpu.read_op8s(); |
| cpu.instruction_pointer = cpu.instruction_pointer + imm8 | 0; |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| }; |
| |
| t[0xEC] = cpu => { |
| var port = cpu.reg16[reg_dx]; |
| cpu.test_privileges_for_io(port, 1); |
| cpu.reg8[reg_al] = cpu.io.port_read8(port); |
| cpu.diverged(); |
| }; |
| t16[0xED] = cpu => { |
| var port = cpu.reg16[reg_dx]; |
| cpu.test_privileges_for_io(port, 2); |
| cpu.reg16[reg_ax] = cpu.io.port_read16(port); |
| cpu.diverged(); |
| }; |
| t32[0xED] = cpu => { |
| var port = cpu.reg16[reg_dx]; |
| cpu.test_privileges_for_io(port, 4); |
| cpu.reg32s[reg_eax] = cpu.io.port_read32(port); |
| cpu.diverged(); |
| }; |
| t[0xEE] = cpu => { |
| var port = cpu.reg16[reg_dx]; |
| cpu.test_privileges_for_io(port, 1); |
| cpu.io.port_write8(port, cpu.reg8[reg_al]); |
| cpu.diverged(); |
| }; |
| t16[0xEF] = cpu => { |
| var port = cpu.reg16[reg_dx]; |
| cpu.test_privileges_for_io(port, 2); |
| cpu.io.port_write16(port, cpu.reg16[reg_ax]); |
| cpu.diverged(); |
| }; |
| t32[0xEF] = cpu => { |
| var port = cpu.reg16[reg_dx]; |
| cpu.test_privileges_for_io(port, 4); |
| cpu.io.port_write32(port, cpu.reg32s[reg_eax]); |
| cpu.diverged(); |
| }; |
| |
| t[0xF0] = cpu => { |
| // lock |
| //dbg_log("lock", LOG_CPU); |
| |
| // TODO |
| // This triggers UD when used with |
| // some instructions that don't write to memory |
| cpu.run_prefix_instruction(); |
| }; |
| t[0xF1] = cpu => { |
| // INT1 |
| // https://code.google.com/p/corkami/wiki/x86oddities#IceBP |
| throw cpu.debug.unimpl("int1 instruction"); |
| }; |
| |
| t[0xF2] = cpu => { |
| // repnz |
| dbg_assert((cpu.prefixes & PREFIX_MASK_REP) === 0); |
| cpu.prefixes |= PREFIX_REPNZ; |
| cpu.run_prefix_instruction(); |
| cpu.prefixes = 0; |
| }; |
| t[0xF3] = cpu => { |
| // repz |
| dbg_assert((cpu.prefixes & PREFIX_MASK_REP) === 0); |
| cpu.prefixes |= PREFIX_REPZ; |
| cpu.run_prefix_instruction(); |
| cpu.prefixes = 0; |
| }; |
| |
| t[0xF4] = cpu => { |
| cpu.hlt_op(); |
| }; |
| |
| t[0xF5] = cpu => { |
| // cmc |
| cpu.flags = (cpu.flags | 1) ^ cpu.getcf(); |
| cpu.flags_changed &= ~1; |
| }; |
| |
| t[0xF6] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: |
| var data = cpu.read_e8(); cpu.test8(data, cpu.read_op8()); |
| break; |
| case 1: |
| var data = cpu.read_e8(); cpu.test8(data, cpu.read_op8()); |
| break; |
| case 2: |
| var data = cpu.read_write_e8(); cpu.write_e8(~(data)); |
| break; |
| case 3: |
| var data = cpu.read_write_e8(); cpu.write_e8(cpu.neg8(data)); |
| break; |
| case 4: |
| var data = cpu.read_e8(); cpu.mul8(data); |
| break; |
| case 5: |
| var data = cpu.read_e8s(); cpu.imul8(data); |
| break; |
| case 6: |
| var data = cpu.read_e8(); cpu.div8(data); |
| break; |
| case 7: |
| var data = cpu.read_e8s(); cpu.idiv8(data); |
| break; |
| } |
| }; |
| |
| t16[0xF7] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: |
| var data = cpu.read_e16(); cpu.test16(data, cpu.read_op16()); |
| break; |
| case 1: |
| var data = cpu.read_e16(); cpu.test16(data, cpu.read_op16()); |
| break; |
| case 2: |
| var data = cpu.read_write_e16(); cpu.write_e16(~(data)); |
| break; |
| case 3: |
| var data = cpu.read_write_e16(); cpu.write_e16(cpu.neg16(data)); |
| break; |
| case 4: |
| var data = cpu.read_e16(); cpu.mul16(data); |
| break; |
| case 5: |
| var data = cpu.read_e16s(); cpu.imul16(data); |
| break; |
| case 6: |
| var data = cpu.read_e16(); cpu.div16(data); |
| break; |
| case 7: |
| var data = cpu.read_e16s(); cpu.idiv16(data); |
| break; |
| } |
| }; |
| t32[0xF7] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: |
| var data = cpu.read_e32s(); cpu.test32(data, cpu.read_op32s()); |
| break; |
| case 1: |
| var data = cpu.read_e32s(); cpu.test32(data, cpu.read_op32s()); |
| break; |
| case 2: |
| var data = cpu.read_write_e32(); cpu.write_e32(~(data)); |
| break; |
| case 3: |
| var data = cpu.read_write_e32(); cpu.write_e32(cpu.neg32(data)); |
| break; |
| case 4: |
| var data = cpu.read_e32(); cpu.mul32(data); |
| break; |
| case 5: |
| var data = cpu.read_e32s(); cpu.imul32(data); |
| break; |
| case 6: |
| var data = cpu.read_e32(); cpu.div32(data); |
| break; |
| case 7: |
| var data = cpu.read_e32s(); cpu.idiv32(data); |
| break; |
| } |
| }; |
| |
| t[0xF8] = cpu => { |
| // clc |
| cpu.flags &= ~flag_carry; |
| cpu.flags_changed &= ~1; |
| }; |
| t[0xF9] = cpu => { |
| // stc |
| cpu.flags |= flag_carry; |
| cpu.flags_changed &= ~1; |
| }; |
| |
| t[0xFA] = cpu => { |
| // cli |
| //dbg_log("interrupts off"); |
| |
| if(!cpu.protected_mode || ((cpu.flags & flag_vm) ? |
| cpu.getiopl() === 3 : cpu.getiopl() >= cpu.cpl)) |
| { |
| cpu.flags &= ~flag_interrupt; |
| } |
| else |
| { |
| //if(cpu.getiopl() < 3 && ((cpu.flags & flag_vm) ? |
| // (cpu.cr[4] & CR4_VME) : |
| // (cpu.cpl === 3 && (cpu.cr[4] & CR4_PVI)))) |
| //{ |
| // cpu.flags &= ~flag_vif; |
| //} |
| //else |
| { |
| dbg_log("cli #gp", LOG_CPU); |
| cpu.trigger_gp(0); |
| } |
| } |
| }; |
| t[0xFB] = cpu => { |
| // sti |
| //dbg_log("interrupts on"); |
| |
| if(!cpu.protected_mode || ((cpu.flags & flag_vm) ? |
| cpu.getiopl() === 3 : cpu.getiopl() >= cpu.cpl)) |
| { |
| cpu.flags |= flag_interrupt; |
| |
| cpu.clear_prefixes(); |
| cpu.cycle_internal(); |
| |
| cpu.handle_irqs(); |
| } |
| else |
| { |
| //if(cpu.getiopl() < 3 && (cpu.flags & flag_vip) === 0 && ((cpu.flags & flag_vm) ? |
| // (cpu.cr[4] & CR4_VME) : |
| // (cpu.cpl === 3 && (cpu.cr[4] & CR4_PVI)))) |
| //{ |
| // cpu.flags |= flag_vif; |
| //} |
| //else |
| { |
| dbg_log("sti #gp", LOG_CPU); |
| cpu.trigger_gp(0); |
| } |
| } |
| |
| }; |
| |
| t[0xFC] = cpu => { |
| // cld |
| cpu.flags &= ~flag_direction; |
| }; |
| t[0xFD] = cpu => { |
| // std |
| cpu.flags |= flag_direction; |
| }; |
| |
| t[0xFE] = cpu => { cpu.read_modrm_byte(); |
| var mod = cpu.modrm_byte & 56; |
| |
| if(mod === 0) |
| { |
| var data = cpu.read_write_e8(); cpu.write_e8(cpu.inc8(data)); |
| } |
| else if(mod === 8) |
| { |
| var data = cpu.read_write_e8(); cpu.write_e8(cpu.dec8(data)); |
| } |
| else |
| { |
| cpu.todo(); |
| } |
| }; |
| t16[0xFF] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: |
| var data = cpu.read_write_e16(); cpu.write_e16(cpu.inc16(data)); |
| break; |
| case 1: |
| var data = cpu.read_write_e16(); cpu.write_e16(cpu.dec16(data)); |
| break; |
| case 2: |
| // 2, call near |
| var data = cpu.read_e16(); |
| cpu.push16(cpu.get_real_eip()); |
| cpu.instruction_pointer = cpu.get_seg(reg_cs) + data | 0; |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| break; |
| case 3: |
| // 3, callf |
| if(cpu.modrm_byte >= 0xC0) |
| { |
| dbg_log("callf #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| dbg_assert(false, "unreachable"); |
| } |
| |
| var virt_addr = cpu.modrm_resolve(cpu.modrm_byte); |
| var new_ip = cpu.safe_read16(virt_addr); |
| var new_cs = cpu.safe_read16(virt_addr + 2 | 0); |
| |
| cpu.far_jump(new_ip, new_cs, true); |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| break; |
| case 4: |
| // 4, jmp near |
| var data = cpu.read_e16(); |
| cpu.instruction_pointer = cpu.get_seg(reg_cs) + data | 0; |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| break; |
| case 5: |
| // 5, jmpf |
| if(cpu.modrm_byte >= 0xC0) |
| { |
| dbg_log("jmpf #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| dbg_assert(false, "unreachable"); |
| } |
| |
| var virt_addr = cpu.modrm_resolve(cpu.modrm_byte); |
| var new_ip = cpu.safe_read16(virt_addr); |
| var new_cs = cpu.safe_read16(virt_addr + 2 | 0); |
| |
| cpu.far_jump(new_ip, new_cs, false); |
| dbg_assert(cpu.is_asize_32() || cpu.get_real_eip() < 0x10000); |
| cpu.diverged(); |
| break; |
| case 6: |
| // 6, push |
| var data = cpu.read_e16(); |
| cpu.push16(data); |
| break; |
| case 7: |
| cpu.todo(); |
| } |
| }; |
| t32[0xFF] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: |
| var data = cpu.read_write_e32(); cpu.write_e32(cpu.inc32(data)); |
| break; |
| case 1: |
| var data = cpu.read_write_e32(); cpu.write_e32(cpu.dec32(data)); |
| break; |
| case 2: |
| // 2, call near |
| var data = cpu.read_e32s(); |
| cpu.push32(cpu.get_real_eip()); |
| |
| dbg_assert(cpu.is_asize_32() || data < 0x10000); |
| cpu.instruction_pointer = cpu.get_seg(reg_cs) + data | 0; |
| cpu.diverged(); |
| break; |
| case 3: |
| // 3, callf |
| if(cpu.modrm_byte >= 0xC0) |
| { |
| dbg_log("callf #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| dbg_assert(false, "unreachable"); |
| } |
| |
| var virt_addr = cpu.modrm_resolve(cpu.modrm_byte); |
| var new_ip = cpu.safe_read32s(virt_addr); |
| var new_cs = cpu.safe_read16(virt_addr + 4 | 0); |
| |
| if(!cpu.protected_mode || cpu.vm86_mode()) |
| { |
| if(new_ip & 0xFFFF0000) |
| { |
| throw cpu.debug.unimpl("#GP handler"); |
| } |
| } |
| |
| cpu.far_jump(new_ip, new_cs, true); |
| dbg_assert(cpu.is_asize_32() || new_ip < 0x10000); |
| cpu.diverged(); |
| break; |
| case 4: |
| // 4, jmp near |
| var data = cpu.read_e32s(); |
| dbg_assert(cpu.is_asize_32() || data < 0x10000); |
| cpu.instruction_pointer = cpu.get_seg(reg_cs) + data | 0; |
| cpu.diverged(); |
| break; |
| case 5: |
| // 5, jmpf |
| if(cpu.modrm_byte >= 0xC0) |
| { |
| dbg_log("jmpf #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| dbg_assert(false, "unreachable"); |
| } |
| |
| var virt_addr = cpu.modrm_resolve(cpu.modrm_byte); |
| var new_ip = cpu.safe_read32s(virt_addr); |
| var new_cs = cpu.safe_read16(virt_addr + 4 | 0); |
| |
| if(!cpu.protected_mode || cpu.vm86_mode()) |
| { |
| if(new_ip & 0xFFFF0000) |
| { |
| throw cpu.debug.unimpl("#GP handler"); |
| } |
| } |
| |
| cpu.far_jump(new_ip, new_cs, false); |
| dbg_assert(cpu.is_asize_32() || new_ip < 0x10000); |
| cpu.diverged(); |
| break; |
| case 6: |
| // push |
| var data = cpu.read_e32s(); |
| cpu.push32(data); |
| break; |
| case 7: |
| cpu.todo(); |
| } |
| }; |
| |
| var table16 = []; |
| var table32 = []; |
| CPU.prototype.table16 = table16; |
| CPU.prototype.table32 = table32; |
| |
| for(var i = 0; i < 256; i++) |
| { |
| if(t[i]) |
| { |
| //dbg_assert(!t16[i]); |
| //dbg_assert(!t32[i]); |
| table16[i] = table32[i] = t[i]; |
| } |
| else if(t16[i]) |
| { |
| //dbg_assert(!t[i]); |
| //dbg_assert(t32[i]); |
| table16[i] = t16[i]; |
| table32[i] = t32[i]; |
| } |
| } |
| |
| t = []; |
| t16 = []; |
| t32 = []; |
| |
| // 0F ops start here |
| |
| t[0x00] = cpu => { cpu.read_modrm_byte(); |
| if(!cpu.protected_mode || cpu.vm86_mode()) |
| { |
| // No GP, UD is correct here |
| dbg_log("0f 00 #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: |
| // sldt |
| cpu.set_e16(cpu.sreg[reg_ldtr]); |
| if(cpu.is_osize_32() && cpu.modrm_byte >= 0xC0) |
| { |
| cpu.reg32s[cpu.modrm_byte & 7] &= 0xFFFF; |
| } |
| break; |
| case 1: |
| // str |
| cpu.set_e16(cpu.sreg[reg_tr]); |
| if(cpu.is_osize_32() && cpu.modrm_byte >= 0xC0) |
| { |
| cpu.reg32s[cpu.modrm_byte & 7] &= 0xFFFF; |
| } |
| break; |
| case 2: |
| // lldt |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| var data = cpu.read_e16(); |
| cpu.load_ldt(data); |
| break; |
| case 3: |
| // ltr |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| var data = cpu.read_e16(); |
| cpu.load_tr(data); |
| break; |
| case 4: |
| cpu.verr(cpu.read_e16()); |
| break; |
| case 5: |
| cpu.verw(cpu.read_e16()); |
| break; |
| |
| default: |
| dbg_log(cpu.modrm_byte >> 3 & 7, LOG_CPU); |
| cpu.todo(); |
| } |
| }; |
| |
| t[0x01] = cpu => { cpu.read_modrm_byte(); |
| var mod = cpu.modrm_byte >> 3 & 7; |
| |
| if(mod === 4) |
| { |
| // smsw |
| if(cpu.modrm_byte >= 0xC0 && cpu.is_osize_32()) |
| { |
| cpu.set_e32(cpu.cr[0]); |
| } |
| else |
| { |
| cpu.set_e16(cpu.cr[0]); |
| } |
| return; |
| } |
| else if(mod === 6) |
| { |
| // lmsw |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| var cr0 = cpu.read_e16(); |
| |
| cr0 = (cpu.cr[0] & ~0xF) | (cr0 & 0xF); |
| |
| if(cpu.protected_mode) |
| { |
| // lmsw cannot be used to switch back |
| cr0 |= CR0_PE; |
| } |
| |
| cpu.set_cr0(cr0); |
| return; |
| } |
| |
| if(cpu.modrm_byte >= 0xC0) |
| { |
| // only memory |
| dbg_log("0f 01 #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| |
| switch(mod) |
| { |
| case 0: |
| // sgdt |
| cpu.writable_or_pagefault(addr, 6); |
| cpu.safe_write16(addr, cpu.gdtr_size); |
| var mask = cpu.is_osize_32() ? -1 : 0x00FFFFFF; |
| cpu.safe_write32(addr + 2, cpu.gdtr_offset & mask); |
| break; |
| case 1: |
| // sidt |
| cpu.writable_or_pagefault(addr, 6); |
| cpu.safe_write16(addr, cpu.idtr_size); |
| var mask = cpu.is_osize_32() ? -1 : 0x00FFFFFF; |
| cpu.safe_write32(addr + 2, cpu.idtr_offset & mask); |
| break; |
| case 2: |
| // lgdt |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| var size = cpu.safe_read16(addr); |
| var offset = cpu.safe_read32s(addr + 2); |
| |
| cpu.gdtr_size = size; |
| cpu.gdtr_offset = offset; |
| |
| if(!cpu.is_osize_32()) |
| { |
| cpu.gdtr_offset &= 0xFFFFFF; |
| } |
| |
| //dbg_log("gdt at " + h(cpu.gdtr_offset) + ", " + cpu.gdtr_size + " bytes", LOG_CPU); |
| //cpu.debug.dump_state(); |
| //cpu.debug.dump_regs_short(); |
| //cpu.debug.dump_gdt_ldt(); |
| break; |
| case 3: |
| // lidt |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| var size = cpu.safe_read16(addr); |
| var offset = cpu.safe_read32s(addr + 2); |
| |
| cpu.idtr_size = size; |
| cpu.idtr_offset = offset; |
| |
| if(!cpu.is_osize_32()) |
| { |
| cpu.idtr_offset &= 0xFFFFFF; |
| } |
| |
| //dbg_log("[" + h(cpu.instruction_pointer) + "] idt at " + |
| // h(idtr_offset) + ", " + cpu.idtr_size + " bytes " + h(addr), LOG_CPU); |
| break; |
| case 7: |
| // flush translation lookaside buffer |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| cpu.invlpg(addr); |
| break; |
| default: |
| dbg_log(mod); |
| cpu.todo(); |
| } |
| }; |
| |
| t16[0x02] = cpu => { cpu.read_modrm_byte(); |
| // lar |
| if(!cpu.protected_mode || cpu.vm86_mode()) |
| { |
| dbg_log("lar #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| var data = cpu.read_e16(); |
| cpu.write_g16(cpu.lar(data, cpu.read_g16())); |
| }; |
| t32[0x02] = cpu => { cpu.read_modrm_byte(); |
| if(!cpu.protected_mode || cpu.vm86_mode()) |
| { |
| dbg_log("lar #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| var data = cpu.read_e16(); |
| cpu.write_g32(cpu.lar(data, cpu.read_g32s())); |
| }; |
| |
| t16[0x03] = cpu => { cpu.read_modrm_byte(); |
| // lsl |
| if(!cpu.protected_mode || cpu.vm86_mode()) |
| { |
| dbg_log("lsl #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| var data = cpu.read_e16(); |
| cpu.write_g16(cpu.lsl(data, cpu.read_g16())); |
| }; |
| t32[0x03] = cpu => { cpu.read_modrm_byte(); |
| if(!cpu.protected_mode || cpu.vm86_mode()) |
| { |
| dbg_log("lsl #ud", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| var data = cpu.read_e16(); |
| cpu.write_g32(cpu.lsl(data, cpu.read_g32s())); |
| }; |
| |
| t[0x04] = cpu => { cpu.undefined_instruction(); }; |
| t[0x05] = cpu => { cpu.undefined_instruction(); }; |
| |
| t[0x06] = cpu => { |
| // clts |
| if(cpu.cpl) |
| { |
| dbg_log("clts #gp", LOG_CPU); |
| cpu.trigger_gp(0); |
| } |
| else |
| { |
| //dbg_log("clts", LOG_CPU); |
| cpu.cr[0] &= ~CR0_TS; |
| } |
| }; |
| |
| t[0x07] = cpu => { cpu.undefined_instruction(); }; |
| t[0x08] = cpu => { |
| // invd |
| cpu.todo(); |
| }; |
| |
| t[0x09] = cpu => { |
| if(cpu.cpl) |
| { |
| dbg_log("wbinvd #gp", LOG_CPU); |
| cpu.trigger_gp(0); |
| } |
| // wbinvd |
| }; |
| |
| |
| t[0x0A] = cpu => { cpu.undefined_instruction(); }; |
| t[0x0B] = cpu => { |
| // UD2 |
| cpu.trigger_ud(); |
| }; |
| t[0x0C] = cpu => { cpu.undefined_instruction(); }; |
| |
| t[0x0D] = cpu => { |
| // nop |
| cpu.todo(); |
| }; |
| |
| t[0x0E] = cpu => { cpu.undefined_instruction(); }; |
| t[0x0F] = cpu => { cpu.undefined_instruction(); }; |
| |
| t[0x10] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x11] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x12] = cpu => { |
| // movlpd xmm, xmm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| let data = cpu.read_xmm_mem64s(); |
| cpu.write_xmm64(data[0], data[1]); |
| }; |
| t[0x13] = cpu => { |
| // movlpd xmm/m64, xmm |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| let data = cpu.read_xmm64s(); |
| dbg_assert(cpu.modrm_byte < 0xC0); |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write64(addr, data[0], data[1]); |
| }; |
| t[0x14] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x15] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x16] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x17] = cpu => { cpu.unimplemented_sse(); }; |
| |
| t[0x18] = cpu => { cpu.read_modrm_byte(); |
| // prefetch |
| // nop for us |
| if(cpu.modrm_byte < 0xC0) |
| cpu.modrm_resolve(cpu.modrm_byte); |
| }; |
| |
| t[0x19] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x1A] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x1B] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x1C] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x1D] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x1E] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x1F] = cpu => { cpu.read_modrm_byte() |
| // multi-byte nop |
| if(cpu.modrm_byte < 0xC0) |
| cpu.modrm_resolve(cpu.modrm_byte); |
| }; |
| |
| |
| t[0x20] = cpu => { cpu.read_modrm_byte(); |
| |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| //dbg_log("cr" + (cpu.modrm_byte >> 3 & 7) + " read", LOG_CPU); |
| |
| // mov addr, cr |
| // mod = which control register |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: |
| cpu.write_reg_e32(cpu.cr[0]); |
| break; |
| case 2: |
| //dbg_log("read cr2 at " + h(cpu.instruction_pointer >>> 0, 8)); |
| cpu.write_reg_e32(cpu.cr[2]); |
| break; |
| case 3: |
| //dbg_log("read cr3 (" + h(cpu.cr[3], 8) + ")", LOG_CPU); |
| cpu.write_reg_e32(cpu.cr[3]); |
| break; |
| case 4: |
| cpu.write_reg_e32(cpu.cr[4]); |
| break; |
| default: |
| dbg_log(cpu.modrm_byte >> 3 & 7); |
| dbg_assert(false); |
| cpu.trigger_ud(); |
| } |
| }; |
| |
| t[0x21] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| var dreg = cpu.modrm_byte >> 3 & 7; |
| if((cpu.cr[4] & CR4_DE) && (dreg === 4 || dreg === 5)) |
| { |
| dbg_log("#ud mov dreg 4/5 with cr4.DE set", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| |
| // high two bits of modrm are ignored |
| cpu.reg32s[cpu.modrm_byte & 7] = cpu.dreg[dreg]; |
| |
| //dbg_log("read dr" + dreg + ": " + h(cpu.dreg[dreg] >>> 0), LOG_CPU); |
| }; |
| |
| t[0x22] = cpu => { cpu.read_modrm_byte(); |
| |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| var data = cpu.read_reg_e32s(); |
| //dbg_log("cr" + (cpu.modrm_byte >> 3 & 7) + " written: " + h(data >>> 0, 8), LOG_CPU); |
| |
| // mov cr, addr |
| // mod = which control register |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: |
| cpu.set_cr0(data); |
| //dbg_log("cr0=" + h(data >>> 0), LOG_CPU); |
| break; |
| |
| case 2: |
| cpu.cr[2] = data; |
| //dbg_log("cr2=" + h(data >>> 0), LOG_CPU); |
| break; |
| |
| case 3: |
| //dbg_log("cr3=" + h(data >>> 0), LOG_CPU); |
| data &= ~0b111111100111; |
| dbg_assert((data & 0xFFF) === 0, "TODO"); |
| cpu.cr[3] = data; |
| cpu.clear_tlb(); |
| |
| //dump_page_directory(); |
| //dbg_log("page directory loaded at " + h(cpu.cr[3] >>> 0, 8), LOG_CPU); |
| break; |
| |
| case 4: |
| cpu.set_cr4(data); |
| break; |
| |
| default: |
| dbg_log(cpu.modrm_byte >> 3 & 7); |
| dbg_assert(false); |
| cpu.trigger_ud(); |
| } |
| }; |
| t[0x23] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| var dreg = cpu.modrm_byte >> 3 & 7; |
| if((cpu.cr[4] & CR4_DE) && (dreg === 4 || dreg === 5)) |
| { |
| dbg_log("#ud mov dreg 4/5 with cr4.DE set", LOG_CPU); |
| cpu.trigger_ud(); |
| } |
| |
| // high two bits of modrm are ignored |
| cpu.dreg[dreg] = cpu.read_reg_e32s(); |
| |
| //dbg_log("write dr" + dreg + ": " + h(cpu.dreg[dreg] >>> 0), LOG_CPU); |
| }; |
| |
| t[0x24] = cpu => { cpu.undefined_instruction(); }; |
| t[0x25] = cpu => { cpu.undefined_instruction(); }; |
| t[0x26] = cpu => { cpu.undefined_instruction(); }; |
| t[0x27] = cpu => { cpu.undefined_instruction(); }; |
| |
| t[0x28] = cpu => { |
| // movaps xmm, xmm/m128 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| let data = cpu.read_xmm_mem128s(); |
| cpu.write_xmm128s(data[0], data[1], data[2], data[3]); |
| }; |
| t[0x29] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66) |
| { |
| // movapd xmm/m128, xmm |
| // (note: same as below, see google.com/?q=MOVAPD+vs+MOVAPS) |
| let data = cpu.read_xmm128s(); |
| dbg_assert(cpu.modrm_byte < 0xC0); |
| let addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write128(addr, data[0], data[1], data[2], data[3]); |
| } |
| else |
| { |
| // movaps xmm/m128, xmm |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === 0); |
| let data = cpu.read_xmm128s(); |
| dbg_assert(cpu.modrm_byte < 0xC0); |
| let addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write128(addr, data[0], data[1], data[2], data[3]); |
| } |
| }; |
| t[0x2A] = cpu => { |
| // cvtpi2ps xmm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| let data = cpu.read_mmx_mem64s(); |
| let float32 = new Float32Array(2); |
| let res32 = new Uint32Array(float32.buffer); |
| float32[0] = data[0]; |
| float32[1] = data[1]; |
| cpu.write_xmm64(res32[0], res32[1]); |
| }; |
| t[0x2B] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66) |
| { |
| // movntpd m128, xmm |
| let data = cpu.read_xmm128s(); |
| dbg_assert(cpu.modrm_byte < 0xC0); |
| let addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write128(addr, data[0], data[1], data[2], data[3]); |
| } |
| else |
| { |
| // movntps m128, xmm |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === 0); |
| let data = cpu.read_xmm128s(); |
| dbg_assert(cpu.modrm_byte < 0xC0); |
| let addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write128(addr, data[0], data[1], data[2], data[3]); |
| } |
| }; |
| t[0x2C] = cpu => { |
| // cvttps2pi mm, xmm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let data = cpu.read_xmm_mem64s(); |
| let float32 = new Float32Array(data.buffer); |
| let low = 0; |
| let high = 0; |
| |
| var res0 = Math.trunc(float32[0]); |
| if(res0 <= 0x7FFFFFFF && res0 >= -0x80000000) |
| { |
| low = res0; |
| } |
| else |
| { |
| low = 0x80000000|0; |
| } |
| |
| var res1 = Math.trunc(float32[1]); |
| if(res1 <= 0x7FFFFFFF && res1 >= -0x80000000) |
| { |
| high = res1; |
| } |
| else |
| { |
| high = 0x80000000|0; |
| } |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x2D] = cpu => { |
| // cvtps2pi mm, xmm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let data = cpu.read_xmm_mem64s(); |
| let float32 = new Float32Array(data.buffer); |
| let low = 0; |
| let high = 0; |
| var rc = cpu.mxcsr >> 13 & 3; |
| |
| var res0 = cpu.integer_round(float32[0], rc); |
| if(res0 <= 0x7FFFFFFF && res0 >= -0x80000000) |
| { |
| low = res0; |
| } |
| else |
| { |
| low = 0x80000000|0; |
| } |
| |
| var res1 = cpu.integer_round(float32[1], rc); |
| if(res1 <= 0x7FFFFFFF && res1 >= -0x80000000) |
| { |
| high = res1; |
| } |
| else |
| { |
| high = 0x80000000|0; |
| } |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x2E] = cpu => { |
| // ucomiss xmm1, xmm2/m32 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_xmm128s(); |
| let dest = cpu.read_xmm_mem128s(); |
| let source1 = new Float32Array(source.buffer); |
| let source2 = new Float32Array(dest.buffer); |
| |
| let x = source1[0]; |
| let y = source2[0]; |
| |
| cpu.flags_changed &= ~(1 | flag_parity | flag_zero); |
| cpu.flags &= ~(1 | flag_parity | flag_zero); |
| |
| if(x > y) |
| { |
| } |
| else if(y > x) |
| { |
| cpu.flags |= 1; |
| } |
| else if(x === y) |
| { |
| cpu.flags |= flag_zero; |
| } |
| else |
| { |
| cpu.flags |= 1 | flag_parity | flag_zero; |
| } |
| }; |
| t[0x2F] = cpu => { cpu.unimplemented_sse(); }; |
| |
| // wrmsr |
| t[0x30] = cpu => { |
| // wrmsr - write maschine specific register |
| |
| if(cpu.cpl) |
| { |
| // cpl > 0 or vm86 mode (vm86 mode is always runs with cpl=3) |
| cpu.trigger_gp(0); |
| } |
| |
| var index = cpu.reg32s[reg_ecx]; |
| var low = cpu.reg32s[reg_eax]; |
| var high = cpu.reg32s[reg_edx]; |
| |
| if(index !== IA32_SYSENTER_ESP) |
| { |
| dbg_log("wrmsr ecx=" + h(index >>> 0, 8) + |
| " data=" + h(high >>> 0, 8) + ":" + h(low >>> 0, 8), LOG_CPU); |
| } |
| |
| switch(index) |
| { |
| case IA32_SYSENTER_CS: |
| cpu.sysenter_cs = low & 0xFFFF; |
| break; |
| |
| case IA32_SYSENTER_EIP: |
| cpu.sysenter_eip = low; |
| break; |
| |
| case IA32_SYSENTER_ESP: |
| cpu.sysenter_esp = low; |
| break; |
| |
| case IA32_APIC_BASE_MSR: |
| dbg_assert(high === 0, "Changing APIC address (high 32 bits) not supported"); |
| let address = low & ~(IA32_APIC_BASE_BSP | IA32_APIC_BASE_EXTD | IA32_APIC_BASE_EN); |
| dbg_assert((address >>> 0) === APIC_ADDRESS, "Changing APIC address not supported"); |
| dbg_assert((low & IA32_APIC_BASE_EXTD) === 0, "x2apic not supported"); |
| cpu.apic_enabled = (low & IA32_APIC_BASE_EN) === IA32_APIC_BASE_EN; |
| break; |
| |
| case IA32_TIME_STAMP_COUNTER: |
| var new_tick = (low >>> 0) + 0x100000000 * (high >>> 0); |
| cpu.tsc_offset = v86.microtick() - new_tick / TSC_RATE; |
| break; |
| |
| case IA32_BIOS_SIGN_ID: |
| break; |
| |
| case IA32_MISC_ENABLE: // Enable Misc. Processor Features |
| dbg_log("IA32_MISC_ENABLE=" + h(low >>> 0, 8), LOG_CPU); |
| break; |
| |
| case IA32_MCG_CAP: |
| // netbsd |
| break; |
| |
| case IA32_KERNEL_GS_BASE: |
| // Only used in 64 bit mode (by SWAPGS), but set by kvm-unit-test |
| dbg_log("GS Base written", LOG_CPU); |
| break; |
| |
| default: |
| dbg_assert(false, "Unknown msr: " + h(index >>> 0, 8)); |
| } |
| }; |
| |
| t[0x31] = cpu => { |
| // rdtsc - read timestamp counter |
| |
| if(!cpu.cpl || !(cpu.cr[4] & CR4_TSD)) |
| { |
| var n = v86.microtick() - cpu.tsc_offset; |
| dbg_assert(isFinite(n), "non-finite tsc: " + n); |
| |
| cpu.reg32s[reg_eax] = n * TSC_RATE; |
| cpu.reg32s[reg_edx] = n * (TSC_RATE / 0x100000000); |
| |
| //dbg_log("rdtsc edx:eax=" + h(cpu.reg32[reg_edx], 8) + ":" + h(cpu.reg32[reg_eax], 8), LOG_CPU); |
| } |
| else |
| { |
| cpu.trigger_gp(0); |
| } |
| }; |
| |
| t[0x32] = cpu => { |
| // rdmsr - read maschine specific register |
| if(cpu.cpl) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| var index = cpu.reg32s[reg_ecx]; |
| |
| dbg_log("rdmsr ecx=" + h(index >>> 0, 8), LOG_CPU); |
| |
| var low = 0; |
| var high = 0; |
| |
| switch(index) |
| { |
| case IA32_SYSENTER_CS: |
| low = cpu.sysenter_cs; |
| break; |
| |
| case IA32_SYSENTER_EIP: |
| low = cpu.sysenter_eip; |
| break; |
| |
| case IA32_SYSENTER_ESP: |
| low = cpu.sysenter_esp; |
| break; |
| |
| case IA32_TIME_STAMP_COUNTER: |
| var n = v86.microtick() - cpu.tsc_offset; |
| low = n * TSC_RATE; |
| high = n * (TSC_RATE / 0x100000000); |
| break; |
| |
| case IA32_PLATFORM_ID: |
| break; |
| |
| case IA32_APIC_BASE_MSR: |
| if(ENABLE_ACPI) |
| { |
| low = APIC_ADDRESS; |
| |
| if(cpu.apic_enabled) |
| { |
| low |= IA32_APIC_BASE_EN; |
| } |
| } |
| break; |
| |
| case IA32_BIOS_SIGN_ID: |
| break; |
| |
| case IA32_MISC_ENABLE: // Enable Misc. Processor Features |
| break; |
| |
| case IA32_RTIT_CTL: |
| // linux4 |
| break; |
| |
| case MSR_SMI_COUNT: |
| break; |
| |
| case IA32_MCG_CAP: |
| // netbsd |
| break; |
| |
| case MSR_PKG_C2_RESIDENCY: |
| break; |
| |
| case MSR_EBC_FREQUENCY_ID: |
| low = 1 << 24; |
| break; |
| |
| default: |
| dbg_assert(false, "Unknown msr: " + h(index >>> 0, 8)); |
| } |
| |
| cpu.reg32s[reg_eax] = low; |
| cpu.reg32s[reg_edx] = high; |
| }; |
| |
| t[0x33] = cpu => { |
| // rdpmc |
| cpu.todo(); |
| }; |
| |
| t[0x34] = cpu => { |
| // sysenter |
| var seg = cpu.sysenter_cs & 0xFFFC; |
| |
| if(!cpu.protected_mode || seg === 0) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| //dbg_log("sysenter cs:eip=" + h(seg , 4) + ":" + h(cpu.sysenter_eip >>> 0, 8) + |
| // " ss:esp=" + h(seg + 8, 4) + ":" + h(cpu.sysenter_esp >>> 0, 8), LOG_CPU); |
| |
| cpu.flags &= ~flag_vm & ~flag_interrupt; |
| |
| cpu.instruction_pointer = cpu.sysenter_eip; |
| cpu.reg32s[reg_esp] = cpu.sysenter_esp; |
| |
| cpu.sreg[reg_cs] = seg; |
| cpu.segment_is_null[reg_cs] = 0; |
| cpu.segment_limits[reg_cs] = -1; |
| cpu.segment_offsets[reg_cs] = 0; |
| |
| cpu.update_cs_size(true); |
| |
| cpu.cpl = 0; |
| cpu.cpl_changed(); |
| |
| cpu.sreg[reg_ss] = seg + 8; |
| cpu.segment_is_null[reg_ss] = 0; |
| cpu.segment_limits[reg_ss] = -1; |
| cpu.segment_offsets[reg_ss] = 0; |
| |
| cpu.stack_size_32 = true; |
| cpu.diverged(); |
| }; |
| |
| t[0x35] = cpu => { |
| // sysexit |
| var seg = cpu.sysenter_cs & 0xFFFC; |
| |
| if(!cpu.protected_mode || cpu.cpl || seg === 0) |
| { |
| cpu.trigger_gp(0); |
| } |
| |
| //dbg_log("sysexit cs:eip=" + h(seg + 16, 4) + ":" + h(cpu.reg32s[reg_edx] >>> 0, 8) + |
| // " ss:esp=" + h(seg + 24, 4) + ":" + h(cpu.reg32s[reg_ecx] >>> 0, 8), LOG_CPU); |
| |
| cpu.instruction_pointer = cpu.reg32s[reg_edx]; |
| cpu.reg32s[reg_esp] = cpu.reg32s[reg_ecx]; |
| |
| cpu.sreg[reg_cs] = seg + 16 | 3; |
| |
| cpu.segment_is_null[reg_cs] = 0; |
| cpu.segment_limits[reg_cs] = -1; |
| cpu.segment_offsets[reg_cs] = 0; |
| |
| cpu.update_cs_size(true); |
| |
| cpu.cpl = 3; |
| cpu.cpl_changed(); |
| |
| cpu.sreg[reg_ss] = seg + 24 | 3; |
| cpu.segment_is_null[reg_ss] = 0; |
| cpu.segment_limits[reg_ss] = -1; |
| cpu.segment_offsets[reg_ss] = 0; |
| |
| cpu.stack_size_32 = true; |
| cpu.diverged(); |
| }; |
| |
| t[0x36] = cpu => { cpu.undefined_instruction(); }; |
| |
| t[0x37] = cpu => { |
| // getsec |
| cpu.todo(); |
| }; |
| |
| // sse3+ |
| t[0x38] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x39] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x3A] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x3B] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x3C] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x3D] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x3E] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x3F] = cpu => { cpu.unimplemented_sse(); }; |
| |
| // cmov |
| t16[0x40] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16( cpu.test_o()); }; |
| t32[0x40] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32( cpu.test_o()); }; |
| t16[0x41] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16(!cpu.test_o()); }; |
| t32[0x41] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32(!cpu.test_o()); }; |
| t16[0x42] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16( cpu.test_b()); }; |
| t32[0x42] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32( cpu.test_b()); }; |
| t16[0x43] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16(!cpu.test_b()); }; |
| t32[0x43] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32(!cpu.test_b()); }; |
| t16[0x44] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16( cpu.test_z()); }; |
| t32[0x44] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32( cpu.test_z()); }; |
| t16[0x45] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16(!cpu.test_z()); }; |
| t32[0x45] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32(!cpu.test_z()); }; |
| t16[0x46] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16( cpu.test_be()); }; |
| t32[0x46] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32( cpu.test_be()); }; |
| t16[0x47] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16(!cpu.test_be()); }; |
| t32[0x47] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32(!cpu.test_be()); }; |
| t16[0x48] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16( cpu.test_s()); }; |
| t32[0x48] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32( cpu.test_s()); }; |
| t16[0x49] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16(!cpu.test_s()); }; |
| t32[0x49] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32(!cpu.test_s()); }; |
| t16[0x4A] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16( cpu.test_p()); }; |
| t32[0x4A] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32( cpu.test_p()); }; |
| t16[0x4B] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16(!cpu.test_p()); }; |
| t32[0x4B] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32(!cpu.test_p()); }; |
| t16[0x4C] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16( cpu.test_l()); }; |
| t32[0x4C] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32( cpu.test_l()); }; |
| t16[0x4D] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16(!cpu.test_l()); }; |
| t32[0x4D] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32(!cpu.test_l()); }; |
| t16[0x4E] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16( cpu.test_le()); }; |
| t32[0x4E] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32( cpu.test_le()); }; |
| t16[0x4F] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc16(!cpu.test_le()); }; |
| t32[0x4F] = cpu => { cpu.read_modrm_byte(); cpu.cmovcc32(!cpu.test_le()); }; |
| |
| |
| t[0x50] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x51] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x52] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x53] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x54] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x55] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x56] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x57] = cpu => { |
| // xorps xmm, xmm/mem128 |
| // xorpd xmm, xmm/mem128 |
| // Note: Same code as pxor |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_xmm_mem128s(); |
| let destination = cpu.read_xmm128s(); |
| |
| cpu.write_xmm128s( |
| source[0] ^ destination[0], |
| source[1] ^ destination[1], |
| source[2] ^ destination[2], |
| source[3] ^ destination[3] |
| ); |
| }; |
| |
| t[0x58] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x59] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x5A] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x5B] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x5C] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x5D] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x5E] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x5F] = cpu => { cpu.unimplemented_sse(); }; |
| |
| t[0x60] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // punpcklbw xmm, xmm/m128 |
| let source = cpu.read_xmm_mem64s(); |
| let source8 = new Uint8Array(source.buffer); |
| |
| let destination = cpu.read_xmm64s(); |
| let destination8 = new Uint8Array(destination.buffer); |
| |
| cpu.write_xmm128s( |
| destination8[0] | source8[0] << 8 | destination8[1] << 16 | source8[1] << 24, |
| destination8[2] | source8[2] << 8 | destination8[3] << 16 | source8[3] << 24, |
| destination8[4] | source8[4] << 8 | destination8[5] << 16 | source8[5] << 24, |
| destination8[6] | source8[6] << 8 | destination8[7] << 16 | source8[7] << 24 |
| ); |
| } |
| else |
| { |
| // punpcklbw mm, mm/m32 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| |
| let source = cpu.read_mmx_mem32s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| |
| let byte0 = destination_low & 0xFF; |
| let byte1 = source & 0xFF; |
| let byte2 = (destination_low >> 8) & 0xFF; |
| let byte3 = (source >> 8) & 0xFF; |
| let byte4 = (destination_low >> 16) & 0xFF; |
| let byte5 = (source >> 16) & 0xFF; |
| let byte6 = destination_low >>> 24; |
| let byte7 = source >>> 24; |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0x61] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // punpcklwd xmm, xmm/m128 |
| let source = cpu.read_xmm_mem64s(); |
| let source16 = new Uint16Array(source.buffer); |
| |
| let destination = cpu.read_xmm64s(); |
| let destination16 = new Uint16Array(destination.buffer); |
| |
| cpu.write_xmm128s( |
| destination16[0] | source16[0] << 16, |
| destination16[1] | source16[1] << 16, |
| destination16[2] | source16[2] << 16, |
| destination16[3] | source16[3] << 16 |
| ); |
| } |
| else |
| { |
| // punpcklwd mm, mm/m32 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| let source = cpu.read_mmx_mem32s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| |
| let word0 = destination_low & 0xFFFF; |
| let word1 = source & 0xFFFF; |
| let word2 = destination_low >>> 16; |
| let word3 = source >>> 16; |
| |
| let low = word0 | word1 << 16; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0x62] = cpu => { |
| // punpckldq mm, mm/m32 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem32s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| |
| let low = destination_low; |
| let high = source; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x63] = cpu => { |
| // packsswb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = 0; |
| low |= (cpu.saturate_sw_to_sb((destination_low) & 0xFFFF)); |
| low |= (cpu.saturate_sw_to_sb(destination_low >>> 16)) << 8; |
| low |= (cpu.saturate_sw_to_sb((destination_high) & 0xFFFF)) << 16; |
| low |= (cpu.saturate_sw_to_sb(destination_high >>> 16)) << 24; |
| |
| let high = 0; |
| high |= (cpu.saturate_sw_to_sb((source[0]) & 0xFFFF)); |
| high |= (cpu.saturate_sw_to_sb(source[0] >>> 16)) << 8; |
| high |= (cpu.saturate_sw_to_sb((source[1]) & 0xFFFF)) << 16; |
| high |= (cpu.saturate_sw_to_sb(source[1] >>> 16)) << 24; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x64] = cpu => { |
| // pcmpgtb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source64s = cpu.read_mmx_mem64s(); |
| let source8s = new Int8Array(source64s.buffer); |
| |
| let reg_offset = 8 * (cpu.modrm_byte >> 3 & 7); |
| let destination8s = cpu.reg_mmx8s; |
| |
| let byte0 = destination8s[reg_offset] > source8s[0] ? 0xFF : 0; |
| let byte1 = destination8s[reg_offset + 1] > source8s[1] ? 0xFF : 0; |
| let byte2 = destination8s[reg_offset + 2] > source8s[2] ? 0xFF : 0; |
| let byte3 = destination8s[reg_offset + 3] > source8s[3] ? 0xFF : 0; |
| let byte4 = destination8s[reg_offset + 4] > source8s[4] ? 0xFF : 0; |
| let byte5 = destination8s[reg_offset + 5] > source8s[5] ? 0xFF : 0; |
| let byte6 = destination8s[reg_offset + 6] > source8s[6] ? 0xFF : 0; |
| let byte7 = destination8s[reg_offset + 7] > source8s[7] ? 0xFF : 0; |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x65] = cpu => { |
| // pcmpgtw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = (destination_low << 16 >> 16) > (source[0] << 16 >> 16) ? 0xFFFF : 0; |
| let word1 = (destination_low >> 16) > (source[0] >> 16) ? 0xFFFF : 0; |
| let word2 = (destination_high << 16 >> 16) > (source[1] << 16 >> 16) ? 0xFFFF : 0; |
| let word3 = (destination_high >> 16) > (source[1] >> 16) ? 0xFFFF : 0; |
| |
| let low = word0 | word1 << 16; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x66] = cpu => { |
| // pcmpgtd mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = destination_low > source[0] ? -1 : 0; |
| let high = destination_high > source[1] ? -1 : 0; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x67] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| let source = cpu.read_xmm_mem128s(); |
| let source16s = new Int16Array(source.buffer); |
| |
| let destination = cpu.read_xmm128s(); |
| let destination16s = new Int16Array(destination.buffer); |
| |
| let result = cpu.create_atom128s(0, 0, 0, 0); |
| let result8 = new Uint8Array(result.buffer); |
| |
| for(let i = 0; i < 8; i++) |
| { |
| result8[i] = cpu.saturate_sw_to_ub(destination16s[i]); |
| result8[i | 8] = cpu.saturate_sw_to_ub(source16s[i]); |
| } |
| |
| cpu.write_xmm128s(result[0], result[1], result[2], result[3]); |
| } |
| else |
| { |
| // packuswb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = 0; |
| low |= (cpu.saturate_sw_to_ub((destination_low) & 0xFFFF)); |
| low |= (cpu.saturate_sw_to_ub(destination_low >>> 16)) << 8; |
| low |= (cpu.saturate_sw_to_ub((destination_high) & 0xFFFF)) << 16; |
| low |= (cpu.saturate_sw_to_ub(destination_high >>> 16)) << 24; |
| |
| let high = 0; |
| high |= (cpu.saturate_sw_to_ub((source[0]) & 0xFFFF)); |
| high |= (cpu.saturate_sw_to_ub(source[0] >>> 16)) << 8; |
| high |= (cpu.saturate_sw_to_ub((source[1]) & 0xFFFF)) << 16; |
| high |= (cpu.saturate_sw_to_ub(source[1] >>> 16)) << 24; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0x68] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // punpckhbw xmm, xmm/m128 |
| let source = cpu.read_xmm_mem128s(); |
| let source8 = new Uint8Array(source.buffer); |
| |
| let destination = cpu.read_xmm128s(); |
| let destination8 = new Uint8Array(destination.buffer); |
| |
| cpu.write_xmm128s( |
| destination8[ 8] | source8[ 8] << 8 | destination8[ 9] << 16 | source8[ 9] << 24, |
| destination8[10] | source8[10] << 8 | destination8[11] << 16 | source8[11] << 24, |
| destination8[12] | source8[12] << 8 | destination8[13] << 16 | source8[13] << 24, |
| destination8[14] | source8[14] << 8 | destination8[15] << 16 | source8[15] << 24 |
| ); |
| } |
| else |
| { |
| // punpckhbw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let byte0 = destination_high & 0xFF; |
| let byte1 = source[1] & 0xFF; |
| let byte2 = (destination_high >> 8) & 0xFF; |
| let byte3 = (source[1] >> 8) & 0xFF; |
| let byte4 = (destination_high >> 16) & 0xFF; |
| let byte5 = (source[1] >> 16) & 0xFF; |
| let byte6 = destination_high >>> 24; |
| let byte7 = source[1] >>> 24; |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0x69] = cpu => { |
| // punpckhwd mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = destination_high & 0xFFFF; |
| let word1 = source[1] & 0xFFFF; |
| let word2 = destination_high >>> 16; |
| let word3 = source[1] >>> 16; |
| |
| let low = word0 | word1 << 16; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x6A] = cpu => { |
| // punpckhdq mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = destination_high; |
| let high = source[1]; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x6B] = cpu => { |
| // packssdw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = 0; |
| low |= cpu.saturate_sd_to_sw(destination_low); |
| low |= cpu.saturate_sd_to_sw(destination_high) << 16; |
| |
| let high = 0; |
| high |= cpu.saturate_sd_to_sw(source[0]); |
| high |= cpu.saturate_sd_to_sw(source[1]) << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x6C] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x6D] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x6E] = cpu => { |
| // movd mm, r/m32 |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66) |
| { |
| let data = cpu.read_e32s(); |
| cpu.write_xmm128s(data, 0, 0, 0); |
| } |
| else |
| { |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| let data = cpu.read_e32s(); |
| cpu.write_mmx64s(data, 0); |
| } |
| }; |
| t[0x6F] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // movdqa xmm, xmm/mem128 |
| let data = cpu.read_xmm_mem128s(); |
| cpu.write_xmm128s(data[0], data[1], data[2], data[3]); |
| } |
| else if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_F3) |
| { |
| // movdqu xmm, xmm/m128 |
| let data = cpu.read_xmm_mem128s_unaligned(); |
| cpu.write_xmm128s(data[0], data[1], data[2], data[3]); |
| } |
| else |
| { |
| // movq mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| let data = cpu.read_mmx_mem64s(); |
| cpu.write_mmx64s(data[0], data[1]); |
| } |
| }; |
| |
| t[0x70] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66) |
| { |
| // pshufd xmm, xmm/mem128 |
| let source = cpu.read_xmm_mem128s(); |
| let order = cpu.read_op8(); |
| |
| cpu.write_xmm128s( |
| source[order & 3], |
| source[order >> 2 & 3], |
| source[order >> 4 & 3], |
| source[order >> 6 & 3] |
| ); |
| } |
| else if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_F2) |
| { |
| // pshuflw xmm, xmm/m128, imm8 |
| let source = cpu.read_xmm_mem128s(); |
| let source16 = new Uint16Array(source.buffer); |
| let order = cpu.read_op8(); |
| |
| cpu.write_xmm128s( |
| source16[order & 3] | source16[order >> 2 & 3] << 16, |
| source16[order >> 4 & 3] | source16[order >> 6 & 3] << 16, |
| source[2], |
| source[3] |
| ); |
| } |
| else if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_F3) |
| { |
| // pshufhw xmm, xmm/m128, imm8 |
| let source = cpu.read_xmm_mem128s(); |
| let source16 = new Uint16Array(source.buffer); |
| let order = cpu.read_op8(); |
| |
| cpu.write_xmm128s( |
| source[0], |
| source[1], |
| source16[order & 3 | 4] | source16[order >> 2 & 3 | 4] << 16, |
| source16[order >> 4 & 3 | 4] | source16[order >> 6 & 3 | 4] << 16 |
| ); |
| } |
| else |
| { |
| // pshufw mm1, mm2/m64, imm8 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| let source = cpu.read_mmx_mem64s(); |
| let order = cpu.read_op8(); |
| |
| let word0_shift = order & 0b11; |
| let word0 = source[word0_shift >> 1] >>> ((word0_shift & 1) * 16) & 0xFFFF; |
| let word1_shift = (order >> 2) & 0b11; |
| let word1 = source[word1_shift >> 1] >>> ((word1_shift & 1) * 16); |
| let low = word0 | word1 << 16; |
| |
| let word2_shift = (order >> 4) & 0b11; |
| let word2 = source[word2_shift >> 1] >>> ((word2_shift & 1) * 16) & 0xFFFF; |
| let word3_shift = (order >>> 6); |
| let word3 = source[word3_shift >> 1] >>> ((word3_shift & 1) * 16); |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| t[0x71] = cpu => { |
| cpu.read_modrm_byte(); |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| |
| if(cpu.modrm_byte < 0xC0) |
| { |
| cpu.trigger_ud(); |
| } |
| |
| // psrlw, psraw, psllw |
| // 2, 4, 6 |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 2: |
| // psrlw mm, imm8 |
| var source = cpu.read_op8(); |
| var destination = cpu.modrm_byte & 7; |
| |
| var destination_low = cpu.reg_mmxs[2 * destination]; |
| var destination_high = cpu.reg_mmxs[2 * destination + 1]; |
| |
| var shift = source; |
| var low = 0; |
| var high = 0; |
| |
| if (shift <= 15) { |
| var word0 = (destination_low & 0xFFFF) >>> shift; |
| var word1 = (destination_low >>> 16) >>> shift; |
| low = word0 | word1 << 16; |
| |
| var word2 = (destination_high & 0xFFFF) >>> shift; |
| var word3 = (destination_high >>> 16) >>> shift; |
| high = word2 | word3 << 16; |
| } |
| |
| cpu.reg_mmxs[2 * destination] = low; |
| cpu.reg_mmxs[2 * destination + 1] = high; |
| |
| break; |
| case 4: |
| // psraw mm, imm8 |
| var source = cpu.read_op8(); |
| var destination = cpu.modrm_byte & 7; |
| |
| var destination_low = cpu.reg_mmxs[2 * destination]; |
| var destination_high = cpu.reg_mmxs[2 * destination + 1]; |
| |
| var shift = source; |
| if (shift > 15) { |
| shift = 16; |
| } |
| |
| var word0 = ((destination_low << 16 >> 16) >> shift) & 0xFFFF; |
| var word1 = ((destination_low >> 16) >> shift) & 0xFFFF; |
| var low = word0 | word1 << 16; |
| |
| var word2 = ((destination_high << 16 >> 16) >> shift) & 0xFFFF; |
| var word3 = ((destination_high >> 16) >> shift) & 0xFFFF; |
| var high = word2 | word3 << 16; |
| |
| cpu.reg_mmxs[2 * destination] = low; |
| cpu.reg_mmxs[2 * destination + 1] = high; |
| |
| break; |
| case 6: |
| // psllw mm, imm8 |
| var source = cpu.read_op8(); |
| var destination = cpu.modrm_byte & 7; |
| |
| var destination_low = cpu.reg_mmxs[2 * destination]; |
| var destination_high = cpu.reg_mmxs[2 * destination + 1]; |
| |
| var shift = source; |
| var low = 0; |
| var high = 0; |
| |
| if (shift <= 15) { |
| var word0 = ((destination_low & 0xFFFF) << shift) & 0xFFFF; |
| var word1 = (destination_low >>> 16) << shift; |
| low = word0 | word1 << 16; |
| |
| var word2 = ((destination_high & 0xFFFF) << shift) & 0xFFFF; |
| var word3 = (destination_high >>> 16) << shift; |
| high = word2 | word3 << 16; |
| } |
| |
| cpu.reg_mmxs[2 * destination] = low; |
| cpu.reg_mmxs[2 * destination + 1] = high; |
| |
| break; |
| default: |
| cpu.unimplemented_sse(); |
| break; |
| } |
| }; |
| |
| t[0x72] = cpu => { |
| cpu.read_modrm_byte(); |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| |
| if(cpu.modrm_byte < 0xC0) |
| { |
| cpu.trigger_ud(); |
| } |
| |
| // psrld, psrad, pslld |
| // 2, 4, 6 |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 2: |
| // psrld mm, imm8 |
| var source = cpu.read_op8(); |
| var destination = cpu.modrm_byte & 7; |
| |
| var destination_low = cpu.reg_mmxs[2 * destination]; |
| var destination_high = cpu.reg_mmxs[2 * destination + 1]; |
| |
| var shift = source; |
| var low = 0; |
| var high = 0; |
| |
| if (shift <= 31) { |
| low = destination_low >>> shift; |
| high = destination_high >>> shift; |
| } |
| |
| cpu.reg_mmxs[2 * destination] = low; |
| cpu.reg_mmxs[2 * destination + 1] = high; |
| |
| break; |
| case 4: |
| // psrad mm, imm8 |
| var source = cpu.read_op8(); |
| var destination = cpu.modrm_byte & 7; |
| |
| var destination_low = cpu.reg_mmxs[2 * destination]; |
| var destination_high = cpu.reg_mmxs[2 * destination + 1]; |
| |
| var shift = source; |
| if (shift > 31) { |
| shift = 31; |
| } |
| |
| var low = destination_low >> shift; |
| var high = destination_high >> shift; |
| |
| cpu.reg_mmxs[2 * destination] = low; |
| cpu.reg_mmxs[2 * destination + 1] = high; |
| |
| break; |
| case 6: |
| // pslld mm, imm8 |
| var source = cpu.read_op8(); |
| var destination = cpu.modrm_byte & 7; |
| |
| var destination_low = cpu.reg_mmxs[2 * destination]; |
| var destination_high = cpu.reg_mmxs[2 * destination + 1]; |
| |
| var shift = source; |
| var low = 0; |
| var high = 0; |
| |
| if (shift <= 31) { |
| low = destination_low << shift; |
| high = destination_high << shift; |
| } |
| |
| cpu.reg_mmxs[2 * destination] = low; |
| cpu.reg_mmxs[2 * destination + 1] = high; |
| |
| break; |
| default: |
| cpu.unimplemented_sse(); |
| break; |
| } |
| }; |
| |
| t[0x73] = cpu => { |
| cpu.read_modrm_byte(); |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| |
| if(cpu.modrm_byte < 0xC0) |
| { |
| cpu.trigger_ud(); |
| } |
| |
| // psrlq, psllq |
| // 2, 6 |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 2: |
| // psrlq mm, imm8 |
| var source = cpu.read_op8(); |
| var destination = cpu.modrm_byte & 7; |
| |
| var destination_low = cpu.reg_mmxs[2 * destination]; |
| var destination_high = cpu.reg_mmxs[2 * destination + 1]; |
| |
| var shift = source; |
| var low = 0; |
| var high = 0; |
| |
| if (shift <= 31) { |
| low = destination_low >>> shift | (destination_high << (32 - shift)); |
| high = destination_high >>> shift; |
| } |
| else if (shift <= 63) { |
| low = destination_high >>> (shift & 0x1F); |
| high = 0; |
| } |
| |
| cpu.reg_mmxs[2 * destination] = low; |
| cpu.reg_mmxs[2 * destination + 1] = high; |
| |
| break; |
| case 6: |
| // psllq mm, imm8 |
| var source = cpu.read_op8(); |
| var destination = cpu.modrm_byte & 7; |
| |
| var destination_low = cpu.reg_mmxs[2 * destination]; |
| var destination_high = cpu.reg_mmxs[2 * destination + 1]; |
| |
| var shift = source; |
| var low = 0; |
| var high = 0; |
| |
| if (shift <= 31) { |
| low = destination_low << shift; |
| high = destination_high << shift | (destination_low >>> (32 - shift)); |
| } |
| else if (shift <= 63) { |
| high = destination_low << (shift & 0x1F); |
| low = 0; |
| } |
| |
| cpu.reg_mmxs[2 * destination] = low; |
| cpu.reg_mmxs[2 * destination + 1] = high; |
| |
| break; |
| default: |
| cpu.unimplemented_sse(); |
| break; |
| } |
| }; |
| |
| t[0x74] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // pcmpeqb xmm, xmm/m128 |
| let source64s = cpu.read_xmm_mem128s(); |
| let source8 = new Uint8Array(source64s.buffer); |
| |
| let destination128 = cpu.read_xmm128s(); |
| let destination8 = new Uint8Array(destination128.buffer); |
| |
| let result = cpu.create_atom128s(0, 0, 0, 0); |
| let result8 = new Uint8Array(result.buffer); |
| |
| for(let i = 0; i < 16; i++) |
| { |
| result8[i] = source8[i] === destination8[i] ? 0xFF : 0; |
| } |
| |
| cpu.write_xmm128s(result[0], result[1], result[2], result[3]) |
| } |
| else |
| { |
| // pcmpeqb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| let source64s = cpu.read_mmx_mem64s(); |
| let source8s = new Int8Array(source64s.buffer); |
| |
| let reg_offset = 8 * (cpu.modrm_byte >> 3 & 7); |
| let destination8s = cpu.reg_mmx8s; |
| |
| let byte0 = destination8s[reg_offset] === source8s[0] ? 0xFF : 0; |
| let byte1 = destination8s[reg_offset + 1] === source8s[1] ? 0xFF : 0; |
| let byte2 = destination8s[reg_offset + 2] === source8s[2] ? 0xFF : 0; |
| let byte3 = destination8s[reg_offset + 3] === source8s[3] ? 0xFF : 0; |
| let byte4 = destination8s[reg_offset + 4] === source8s[4] ? 0xFF : 0; |
| let byte5 = destination8s[reg_offset + 5] === source8s[5] ? 0xFF : 0; |
| let byte6 = destination8s[reg_offset + 6] === source8s[6] ? 0xFF : 0; |
| let byte7 = destination8s[reg_offset + 7] === source8s[7] ? 0xFF : 0; |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0x75] = cpu => { |
| // pcmpeqw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = (destination_low & 0xFFFF) === (source[0] & 0xFFFF) ? 0xFFFF : 0; |
| let word1 = (destination_low & 0xFFFF0000) === (source[0] & 0xFFFF0000) ? 0xFFFF : 0; |
| let word2 = (destination_high & 0xFFFF) === (source[1] & 0xFFFF) ? 0xFFFF : 0; |
| let word3 = (destination_high & 0xFFFF0000) === (source[1] & 0xFFFF0000) ? 0xFFFF : 0; |
| |
| let low = word0 | word1 << 16; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0x76] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // pcmpeqd xmm, xmm/m128 |
| let source = cpu.read_xmm_mem128s(); |
| let destination = cpu.read_xmm128s(); |
| |
| cpu.write_xmm128s( |
| source[0] === destination[0] ? -1 : 0, |
| source[1] === destination[1] ? -1 : 0, |
| source[2] === destination[2] ? -1 : 0, |
| source[3] === destination[3] ? -1 : 0 |
| ); |
| } |
| else |
| { |
| // pcmpeqd mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = destination_low === source[0] ? -1 : 0; |
| let high = destination_high === source[1] ? -1 : 0; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0x77] = cpu => { |
| // emms |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.fpu.stack_empty = 0xFF; |
| }; |
| |
| t[0x78] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x79] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x7A] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x7B] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x7C] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x7D] = cpu => { cpu.unimplemented_sse(); }; |
| t[0x7E] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_F3) |
| { |
| // movq xmm, xmm/mem64 |
| let data = cpu.read_xmm_mem64s(); |
| cpu.write_xmm128s(data[0], data[1], 0, 0); |
| } |
| else if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // movd r/m32, xmm |
| let data = cpu.read_xmm64s(); |
| cpu.set_e32(data[0]); |
| } |
| else |
| { |
| // movd r/m32, mm |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| let data = cpu.read_mmx64s(); |
| cpu.set_e32(data[0]); |
| } |
| }; |
| t[0x7F] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_F3) |
| { |
| // movdqu xmm/m128, xmm |
| let data = cpu.read_xmm128s(); |
| dbg_assert(cpu.modrm_byte < 0xC0); |
| let addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write128(addr, data[0], data[1], data[2], data[3]); |
| } |
| else if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // movdqa xmm/m128, xmm |
| let data = cpu.read_xmm128s(); |
| dbg_assert(cpu.modrm_byte < 0xC0); |
| let addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write128(addr, data[0], data[1], data[2], data[3]); |
| } |
| else |
| { |
| // movq mm/m64, mm |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| |
| let data = cpu.read_mmx64s(); |
| cpu.set_mmx_mem64s(data[0], data[1]); |
| } |
| }; |
| |
| // jmpcc |
| t16[0x80] = cpu => { cpu.jmpcc16( cpu.test_o()); }; |
| t32[0x80] = cpu => { cpu.jmpcc32( cpu.test_o()); }; |
| t16[0x81] = cpu => { cpu.jmpcc16(!cpu.test_o()); }; |
| t32[0x81] = cpu => { cpu.jmpcc32(!cpu.test_o()); }; |
| t16[0x82] = cpu => { cpu.jmpcc16( cpu.test_b()); }; |
| t32[0x82] = cpu => { cpu.jmpcc32( cpu.test_b()); }; |
| t16[0x83] = cpu => { cpu.jmpcc16(!cpu.test_b()); }; |
| t32[0x83] = cpu => { cpu.jmpcc32(!cpu.test_b()); }; |
| t16[0x84] = cpu => { cpu.jmpcc16( cpu.test_z()); }; |
| t32[0x84] = cpu => { cpu.jmpcc32( cpu.test_z()); }; |
| t16[0x85] = cpu => { cpu.jmpcc16(!cpu.test_z()); }; |
| t32[0x85] = cpu => { cpu.jmpcc32(!cpu.test_z()); }; |
| t16[0x86] = cpu => { cpu.jmpcc16( cpu.test_be()); }; |
| t32[0x86] = cpu => { cpu.jmpcc32( cpu.test_be()); }; |
| t16[0x87] = cpu => { cpu.jmpcc16(!cpu.test_be()); }; |
| t32[0x87] = cpu => { cpu.jmpcc32(!cpu.test_be()); }; |
| t16[0x88] = cpu => { cpu.jmpcc16( cpu.test_s()); }; |
| t32[0x88] = cpu => { cpu.jmpcc32( cpu.test_s()); }; |
| t16[0x89] = cpu => { cpu.jmpcc16(!cpu.test_s()); }; |
| t32[0x89] = cpu => { cpu.jmpcc32(!cpu.test_s()); }; |
| t16[0x8A] = cpu => { cpu.jmpcc16( cpu.test_p()); }; |
| t32[0x8A] = cpu => { cpu.jmpcc32( cpu.test_p()); }; |
| t16[0x8B] = cpu => { cpu.jmpcc16(!cpu.test_p()); }; |
| t32[0x8B] = cpu => { cpu.jmpcc32(!cpu.test_p()); }; |
| t16[0x8C] = cpu => { cpu.jmpcc16( cpu.test_l()); }; |
| t32[0x8C] = cpu => { cpu.jmpcc32( cpu.test_l()); }; |
| t16[0x8D] = cpu => { cpu.jmpcc16(!cpu.test_l()); }; |
| t32[0x8D] = cpu => { cpu.jmpcc32(!cpu.test_l()); }; |
| t16[0x8E] = cpu => { cpu.jmpcc16( cpu.test_le()); }; |
| t32[0x8E] = cpu => { cpu.jmpcc32( cpu.test_le()); }; |
| t16[0x8F] = cpu => { cpu.jmpcc16(!cpu.test_le()); }; |
| t32[0x8F] = cpu => { cpu.jmpcc32(!cpu.test_le()); }; |
| |
| // setcc |
| t[0x90] = cpu => { cpu.read_modrm_byte(); cpu.setcc( cpu.test_o()); }; |
| t[0x91] = cpu => { cpu.read_modrm_byte(); cpu.setcc(!cpu.test_o()); }; |
| t[0x92] = cpu => { cpu.read_modrm_byte(); cpu.setcc( cpu.test_b()); }; |
| t[0x93] = cpu => { cpu.read_modrm_byte(); cpu.setcc(!cpu.test_b()); }; |
| t[0x94] = cpu => { cpu.read_modrm_byte(); cpu.setcc( cpu.test_z()); }; |
| t[0x95] = cpu => { cpu.read_modrm_byte(); cpu.setcc(!cpu.test_z()); }; |
| t[0x96] = cpu => { cpu.read_modrm_byte(); cpu.setcc( cpu.test_be()); }; |
| t[0x97] = cpu => { cpu.read_modrm_byte(); cpu.setcc(!cpu.test_be()); }; |
| t[0x98] = cpu => { cpu.read_modrm_byte(); cpu.setcc( cpu.test_s()); }; |
| t[0x99] = cpu => { cpu.read_modrm_byte(); cpu.setcc(!cpu.test_s()); }; |
| t[0x9A] = cpu => { cpu.read_modrm_byte(); cpu.setcc( cpu.test_p()); }; |
| t[0x9B] = cpu => { cpu.read_modrm_byte(); cpu.setcc(!cpu.test_p()); }; |
| t[0x9C] = cpu => { cpu.read_modrm_byte(); cpu.setcc( cpu.test_l()); }; |
| t[0x9D] = cpu => { cpu.read_modrm_byte(); cpu.setcc(!cpu.test_l()); }; |
| t[0x9E] = cpu => { cpu.read_modrm_byte(); cpu.setcc( cpu.test_le()); }; |
| t[0x9F] = cpu => { cpu.read_modrm_byte(); cpu.setcc(!cpu.test_le()); }; |
| |
| t16[0xA0] = cpu => { cpu.push16(cpu.sreg[reg_fs]); }; |
| t32[0xA0] = cpu => { cpu.push32(cpu.sreg[reg_fs]); }; |
| t16[0xA1] = cpu => { |
| cpu.switch_seg(reg_fs, cpu.safe_read16(cpu.get_stack_pointer(0))); |
| cpu.adjust_stack_reg(2); |
| }; |
| t32[0xA1] = cpu => { |
| cpu.switch_seg(reg_fs, cpu.safe_read32s(cpu.get_stack_pointer(0)) & 0xFFFF); |
| cpu.adjust_stack_reg(4); |
| }; |
| |
| t[0xA2] = cpu => { cpu.cpuid(); }; |
| |
| t16[0xA3] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) |
| { |
| cpu.bt_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_g16s()); |
| } |
| else |
| { |
| cpu.bt_reg(cpu.read_reg_e16(), cpu.read_g16() & 15); |
| } |
| }; |
| t32[0xA3] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) |
| { |
| cpu.bt_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_g32s()); |
| } |
| else |
| { |
| cpu.bt_reg(cpu.read_reg_e32s(), cpu.read_g32s() & 31); |
| } |
| }; |
| |
| t16[0xA4] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e16(); cpu.write_e16(cpu.shld16(data, cpu.read_g16(), cpu.read_op8() & 31)); |
| }; |
| t32[0xA4] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e32(); cpu.write_e32(cpu.shld32(data, cpu.read_g32s(), cpu.read_op8() & 31)); |
| }; |
| t16[0xA5] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e16(); cpu.write_e16(cpu.shld16(data, cpu.read_g16(), cpu.reg8[reg_cl] & 31)); |
| }; |
| t32[0xA5] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e32(); cpu.write_e32(cpu.shld32(data, cpu.read_g32s(), cpu.reg8[reg_cl] & 31)); |
| }; |
| |
| t[0xA6] = cpu => { |
| // obsolete cmpxchg (os/2) |
| cpu.trigger_ud(); |
| }; |
| t[0xA7] = cpu => { cpu.undefined_instruction(); }; |
| |
| t16[0xA8] = cpu => { cpu.push16(cpu.sreg[reg_gs]); }; |
| t32[0xA8] = cpu => { cpu.push32(cpu.sreg[reg_gs]); }; |
| t16[0xA9] = cpu => { |
| cpu.switch_seg(reg_gs, cpu.safe_read16(cpu.get_stack_pointer(0))); |
| cpu.adjust_stack_reg(2); |
| }; |
| t32[0xA9] = cpu => { |
| cpu.switch_seg(reg_gs, cpu.safe_read32s(cpu.get_stack_pointer(0)) & 0xFFFF); |
| cpu.adjust_stack_reg(4); |
| }; |
| |
| |
| t[0xAA] = cpu => { |
| // rsm |
| cpu.todo(); |
| }; |
| |
| t16[0xAB] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.bts_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_g16s()); |
| } else { |
| cpu.write_reg_e16(cpu.bts_reg(cpu.read_reg_e16(), cpu.read_g16s() & 15)); |
| } |
| }; |
| t32[0xAB] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.bts_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_g32s()); |
| } else { |
| cpu.write_reg_e32(cpu.bts_reg(cpu.read_reg_e32s(), cpu.read_g32s() & 31)); |
| } |
| }; |
| |
| |
| t16[0xAC] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e16(); cpu.write_e16(cpu.shrd16(data, cpu.read_g16(), cpu.read_op8() & 31)); |
| }; |
| t32[0xAC] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e32(); cpu.write_e32(cpu.shrd32(data, cpu.read_g32s(), cpu.read_op8() & 31)); |
| }; |
| t16[0xAD] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e16(); cpu.write_e16(cpu.shrd16(data, cpu.read_g16(), cpu.reg8[reg_cl] & 31)); |
| }; |
| t32[0xAD] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e32(); cpu.write_e32(cpu.shrd32(data, cpu.read_g32s(), cpu.reg8[reg_cl] & 31)); |
| }; |
| |
| t[0xAE] = cpu => { cpu.read_modrm_byte(); |
| // xsave, xrstor, ... |
| if(cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) cpu.todo(); |
| |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 0: // fxsave |
| if(cpu.modrm_byte >= 0xC0) cpu.trigger_ud(); |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.fxsave(addr); |
| break; |
| |
| case 1: // fxrstor |
| if(cpu.modrm_byte >= 0xC0) cpu.trigger_ud(); |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.fxrstor(addr); |
| break; |
| |
| case 2: // ldmxcsr |
| if(cpu.modrm_byte >= 0xC0) cpu.trigger_ud(); |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| let new_mxcsr = cpu.safe_read32s(addr); |
| if(new_mxcsr & ~MXCSR_MASK) |
| { |
| dbg_log("Invalid mxcsr bits: " + h((new_mxcsr & ~MXCSR_MASK) >>> 0, 8)); |
| cpu.trigger_gp(0); |
| } |
| cpu.mxcsr = new_mxcsr; |
| break; |
| |
| case 3: // stmxcsr |
| if(cpu.modrm_byte >= 0xC0) cpu.trigger_ud(); |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write32(addr, cpu.mxcsr); |
| break; |
| |
| case 5: |
| // lfence |
| dbg_assert(cpu.modrm_byte >= 0xC0, "Unexpected lfence encoding"); |
| if(cpu.modrm_byte < 0xC0) cpu.trigger_ud(); |
| break; |
| case 6: |
| // mfence |
| dbg_assert(cpu.modrm_byte >= 0xC0, "Unexpected mfence encoding"); |
| if(cpu.modrm_byte < 0xC0) cpu.trigger_ud(); |
| break; |
| case 7: |
| // sfence or clflush |
| dbg_assert(cpu.modrm_byte >= 0xC0, "Unexpected sfence encoding"); |
| if(cpu.modrm_byte < 0xC0) cpu.trigger_ud(); |
| break; |
| default: |
| dbg_log("missing " + (cpu.modrm_byte >> 3 & 7), LOG_CPU); |
| cpu.todo(); |
| } |
| }; |
| |
| t16[0xAF] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e16s(); |
| cpu.write_g16(cpu.imul_reg16(cpu.read_g16s(), data)); |
| }; |
| t32[0xAF] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e32s(); |
| cpu.write_g32(cpu.imul_reg32(cpu.read_g32s(), data)); |
| }; |
| |
| t[0xB0] = cpu => { cpu.read_modrm_byte(); |
| // cmpxchg8 |
| if(cpu.modrm_byte < 0xC0) |
| { |
| var virt_addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.writable_or_pagefault(virt_addr, 1); |
| |
| var data = cpu.safe_read8(virt_addr); |
| } |
| else |
| data = cpu.reg8[cpu.modrm_byte << 2 & 0xC | cpu.modrm_byte >> 2 & 1]; |
| |
| |
| cpu.cmp8(cpu.reg8[reg_al], data); |
| |
| if(cpu.getzf()) |
| { |
| if(cpu.modrm_byte < 0xC0) |
| cpu.safe_write8(virt_addr, cpu.read_g8()); |
| else |
| cpu.reg8[cpu.modrm_byte << 2 & 0xC | cpu.modrm_byte >> 2 & 1] = cpu.read_g8(); |
| } |
| else |
| { |
| if(cpu.modrm_byte < 0xC0) |
| cpu.safe_write8(virt_addr, data); |
| |
| cpu.reg8[reg_al] = data; |
| } |
| }; |
| t16[0xB1] = cpu => { cpu.read_modrm_byte(); |
| // cmpxchg16/32 |
| if(cpu.modrm_byte < 0xC0) |
| { |
| var virt_addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.writable_or_pagefault(virt_addr, 2); |
| |
| var data = cpu.safe_read16(virt_addr); |
| } |
| else |
| data = cpu.read_reg_e16(); |
| |
| cpu.cmp16(cpu.reg16[reg_ax], data); |
| |
| if(cpu.getzf()) |
| { |
| if(cpu.modrm_byte < 0xC0) |
| cpu.safe_write16(virt_addr, cpu.read_g16()); |
| else |
| cpu.write_reg_e16(cpu.read_g16()); |
| } |
| else |
| { |
| if(cpu.modrm_byte < 0xC0) |
| cpu.safe_write16(virt_addr, data); |
| |
| cpu.reg16[reg_ax] = data; |
| } |
| }; |
| t32[0xB1] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) |
| { |
| var virt_addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.writable_or_pagefault(virt_addr, 4); |
| |
| var data = cpu.safe_read32s(virt_addr); |
| } |
| else |
| { |
| data = cpu.read_reg_e32s(); |
| } |
| |
| cpu.cmp32(cpu.reg32s[reg_eax], data); |
| |
| if(cpu.getzf()) |
| { |
| if(cpu.modrm_byte < 0xC0) |
| cpu.safe_write32(virt_addr, cpu.read_g32s()); |
| else |
| cpu.write_reg_e32(cpu.read_g32s()); |
| } |
| else |
| { |
| if(cpu.modrm_byte < 0xC0) |
| cpu.safe_write32(virt_addr, data); |
| |
| cpu.reg32s[reg_eax] = data; |
| } |
| }; |
| |
| // lss |
| t16[0xB2] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss16(reg_ss); |
| }; |
| t32[0xB2] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss32(reg_ss); |
| }; |
| |
| t16[0xB3] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.btr_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_g16s()); |
| } else { |
| cpu.write_reg_e16(cpu.btr_reg(cpu.read_reg_e16(), cpu.read_g16s() & 15)); |
| } |
| }; |
| t32[0xB3] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.btr_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_g32s()); |
| } else { |
| cpu.write_reg_e32(cpu.btr_reg(cpu.read_reg_e32s(), cpu.read_g32s() & 31)); |
| } |
| }; |
| |
| // lfs, lgs |
| t16[0xB4] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss16(reg_fs); |
| }; |
| t32[0xB4] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss32(reg_fs); |
| }; |
| t16[0xB5] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss16(reg_gs); |
| }; |
| t32[0xB5] = cpu => { cpu.read_modrm_byte(); |
| cpu.lss32(reg_gs); |
| }; |
| |
| t16[0xB6] = cpu => { cpu.read_modrm_byte(); |
| // movzx |
| var data = cpu.read_e8(); |
| cpu.write_g16(data); |
| }; |
| t32[0xB6] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e8(); |
| cpu.write_g32(data); |
| }; |
| |
| t16[0xB7] = cpu => { cpu.read_modrm_byte(); |
| // movzx |
| dbg_assert(false, "Possibly invalid encoding"); |
| var data = cpu.read_e16(); |
| cpu.write_g16(data); |
| }; |
| t32[0xB7] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e16(); |
| cpu.write_g32(data); |
| }; |
| |
| t16[0xB8] = cpu => { cpu.read_modrm_byte(); |
| // popcnt |
| if((cpu.prefixes & PREFIX_F3) === 0) |
| { |
| cpu.trigger_ud(); |
| } |
| var data = cpu.read_e16(); |
| cpu.write_g16(cpu.popcnt(data)); |
| }; |
| t32[0xB8] = cpu => { cpu.read_modrm_byte(); |
| if((cpu.prefixes & PREFIX_F3) === 0) |
| { |
| cpu.trigger_ud(); |
| } |
| var data = cpu.read_e32s(); |
| cpu.write_g32(cpu.popcnt(data)); |
| }; |
| |
| t[0xB9] = cpu => { |
| // UD |
| cpu.todo(); |
| }; |
| |
| t16[0xBA] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 4: |
| if(cpu.modrm_byte < 0xC0) |
| { |
| cpu.bt_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op8() & 15); |
| } |
| else |
| { |
| cpu.bt_reg(cpu.read_reg_e16(), cpu.read_op8() & 15); |
| } |
| break; |
| case 5: |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.bts_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op8() & 15); |
| } else { |
| cpu.write_reg_e16(cpu.bts_reg(cpu.read_reg_e16(), cpu.read_op8() & 15)); |
| } |
| break; |
| case 6: |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.btr_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op8() & 15); |
| } else { |
| cpu.write_reg_e16(cpu.btr_reg(cpu.read_reg_e16(), cpu.read_op8() & 15)); |
| } |
| break; |
| case 7: |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.btc_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op8() & 15); |
| } else { |
| cpu.write_reg_e16(cpu.btc_reg(cpu.read_reg_e16(), cpu.read_op8() & 15)); |
| } |
| break; |
| default: |
| dbg_log(cpu.modrm_byte >> 3 & 7); |
| cpu.todo(); |
| } |
| }; |
| t32[0xBA] = cpu => { cpu.read_modrm_byte(); |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 4: |
| if(cpu.modrm_byte < 0xC0) |
| { |
| cpu.bt_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op8() & 31); |
| } |
| else |
| { |
| cpu.bt_reg(cpu.read_reg_e32s(), cpu.read_op8() & 31); |
| } |
| break; |
| case 5: |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.bts_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op8() & 31); |
| } else { |
| cpu.write_reg_e32(cpu.bts_reg(cpu.read_reg_e32s(), cpu.read_op8() & 31)); |
| } |
| break; |
| case 6: |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.btr_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op8() & 31); |
| } else { |
| cpu.write_reg_e32(cpu.btr_reg(cpu.read_reg_e32s(), cpu.read_op8() & 31)); |
| } |
| break; |
| case 7: |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.btc_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_op8() & 31); |
| } else { |
| cpu.write_reg_e32(cpu.btc_reg(cpu.read_reg_e32s(), cpu.read_op8() & 31)); |
| } |
| break; |
| default: |
| dbg_log(cpu.modrm_byte >> 3 & 7); |
| cpu.todo(); |
| } |
| }; |
| |
| t16[0xBB] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.btc_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_g16s()); |
| } else { |
| cpu.write_reg_e16(cpu.btc_reg(cpu.read_reg_e16(), cpu.read_g16s() & 15)); |
| } |
| }; |
| t32[0xBB] = cpu => { cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) { |
| cpu.btc_mem(cpu.modrm_resolve(cpu.modrm_byte), cpu.read_g32s()); |
| } else { |
| cpu.write_reg_e32(cpu.btc_reg(cpu.read_reg_e32s(), cpu.read_g32s() & 31)); |
| } |
| }; |
| |
| t16[0xBC] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e16(); |
| cpu.write_g16(cpu.bsf16(cpu.read_g16(), data)); |
| }; |
| t32[0xBC] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e32s(); |
| cpu.write_g32(cpu.bsf32(cpu.read_g32s(), data)); |
| }; |
| |
| t16[0xBD] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e16(); |
| cpu.write_g16(cpu.bsr16(cpu.read_g16(), data)); |
| }; |
| t32[0xBD] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e32s(); |
| cpu.write_g32(cpu.bsr32(cpu.read_g32s(), data)); |
| }; |
| |
| t16[0xBE] = cpu => { cpu.read_modrm_byte(); |
| // movsx |
| var data = cpu.read_e8s(); |
| cpu.write_g16(data); |
| }; |
| t32[0xBE] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e8s(); |
| cpu.write_g32(data); |
| }; |
| |
| t16[0xBF] = cpu => { cpu.read_modrm_byte(); |
| // movsx |
| dbg_assert(false, "Possibly invalid encoding"); |
| var data = cpu.read_e16(); |
| cpu.write_g16(data); |
| }; |
| |
| t32[0xBF] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_e16s(); |
| cpu.write_g32(data); |
| }; |
| |
| t[0xC0] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e8(); cpu.write_e8(cpu.xadd8(data, cpu.modrm_byte >> 1 & 0xC | cpu.modrm_byte >> 5 & 1)); |
| }; |
| |
| t16[0xC1] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e16(); |
| cpu.write_e16(cpu.xadd16(data, cpu.modrm_byte >> 2 & 14)); |
| }; |
| t32[0xC1] = cpu => { cpu.read_modrm_byte(); |
| var data = cpu.read_write_e32(); |
| cpu.write_e32(cpu.xadd32(data, cpu.modrm_byte >> 3 & 7)); |
| }; |
| |
| |
| t[0xC2] = cpu => { cpu.unimplemented_sse(); }; |
| t[0xC3] = cpu => { |
| // movnti |
| cpu.read_modrm_byte(); |
| if(cpu.modrm_byte >= 0xC0) cpu.trigger_ud(); |
| cpu.set_e32(cpu.read_g32s()); |
| }; |
| t[0xC4] = cpu => { cpu.unimplemented_sse(); }; |
| t[0xC5] = cpu => { cpu.unimplemented_sse(); }; |
| t[0xC6] = cpu => { cpu.unimplemented_sse(); }; |
| |
| t[0xC7] = cpu => { |
| cpu.read_modrm_byte(); |
| |
| switch(cpu.modrm_byte >> 3 & 7) |
| { |
| case 1: |
| // cmpxchg8b |
| if(cpu.modrm_byte >= 0xC0) |
| { |
| cpu.trigger_ud(); |
| } |
| |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.writable_or_pagefault(addr, 8); |
| |
| var m64_low = cpu.safe_read32s(addr); |
| var m64_high = cpu.safe_read32s(addr + 4 | 0); |
| |
| if(cpu.reg32s[reg_eax] === m64_low && |
| cpu.reg32s[reg_edx] === m64_high) |
| { |
| cpu.flags |= flag_zero; |
| |
| cpu.safe_write32(addr, cpu.reg32s[reg_ebx]); |
| cpu.safe_write32(addr + 4 | 0, cpu.reg32s[reg_ecx]); |
| } |
| else |
| { |
| cpu.flags &= ~flag_zero; |
| |
| cpu.reg32s[reg_eax] = m64_low; |
| cpu.reg32s[reg_edx] = m64_high; |
| |
| cpu.safe_write32(addr, m64_low); |
| cpu.safe_write32(addr + 4 | 0, m64_high); |
| } |
| |
| cpu.flags_changed &= ~flag_zero; |
| break; |
| |
| case 6: |
| var has_rand = v86util.has_rand_int(); |
| |
| if(has_rand) |
| { |
| var rand = v86util.get_rand_int(); |
| } |
| else |
| { |
| var rand = 0; |
| } |
| //dbg_log("rdrand -> " + h(rand >>> 0, 8), LOG_CPU); |
| |
| if(cpu.is_osize_32()) |
| { |
| cpu.set_e32(rand); |
| } |
| else |
| { |
| cpu.set_e16(rand); |
| } |
| |
| cpu.flags &= ~flags_all; |
| cpu.flags |= has_rand; |
| cpu.flags_changed = 0; |
| break; |
| |
| default: |
| dbg_log(cpu.modrm_byte >> 3 & 7, LOG_CPU); |
| cpu.todo(); |
| } |
| }; |
| |
| t[0xC8] = cpu => { cpu.bswap(reg_eax); }; |
| t[0xC9] = cpu => { cpu.bswap(reg_ecx); }; |
| t[0xCA] = cpu => { cpu.bswap(reg_edx); }; |
| t[0xCB] = cpu => { cpu.bswap(reg_ebx); }; |
| t[0xCC] = cpu => { cpu.bswap(reg_esp); }; |
| t[0xCD] = cpu => { cpu.bswap(reg_ebp); }; |
| t[0xCE] = cpu => { cpu.bswap(reg_esi); }; |
| t[0xCF] = cpu => { cpu.bswap(reg_edi); }; |
| |
| t[0xD0] = cpu => { cpu.unimplemented_sse(); }; |
| |
| t[0xD1] = cpu => { |
| // psrlw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let shift = source[0] >>> 0; |
| let low = 0; |
| let high = 0; |
| |
| if (shift <= 15) { |
| let word0 = (destination_low & 0xFFFF) >>> shift; |
| let word1 = (destination_low >>> 16) >>> shift; |
| low = word0 | word1 << 16; |
| |
| let word2 = (destination_high & 0xFFFF) >>> shift; |
| let word3 = (destination_high >>> 16) >>> shift; |
| high = word2 | word3 << 16; |
| } |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xD2] = cpu => { |
| // psrld mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let shift = source[0] >>> 0; |
| let low = 0; |
| let high = 0; |
| |
| if (shift <= 31) { |
| low = destination_low >>> shift; |
| high = destination_high >>> shift; |
| } |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xD3] = cpu => { |
| // psrlq mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let shift = source[0] >>> 0; |
| |
| if(shift === 0) |
| { |
| return; |
| } |
| |
| let low = 0; |
| let high = 0; |
| |
| if (shift <= 31) { |
| low = destination_low >>> shift | (destination_high << (32 - shift)); |
| high = destination_high >>> shift; |
| } |
| else if (shift <= 63) { |
| low = destination_high >>> (shift & 0x1F); |
| high = 0; |
| } |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xD4] = cpu => { cpu.unimplemented_sse(); }; |
| t[0xD5] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // pmullw xmm, xmm/m128 |
| let source = cpu.read_xmm_mem128s(); |
| let source16s = new Int16Array(source.buffer); |
| |
| let destination = cpu.read_xmm128s(); |
| let destination16s = new Int16Array(destination.buffer); |
| |
| cpu.write_xmm128s( |
| source16s[0] * destination16s[0] & 0xFFFF | source16s[1] * destination16s[1] << 16, |
| source16s[2] * destination16s[2] & 0xFFFF | source16s[3] * destination16s[3] << 16, |
| source16s[4] * destination16s[4] & 0xFFFF | source16s[5] * destination16s[5] << 16, |
| source16s[6] * destination16s[6] & 0xFFFF | source16s[7] * destination16s[7] << 16 |
| ); |
| } |
| else |
| { |
| // pmullw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = ((destination_low & 0xFFFF) * (source[0] & 0xFFFF)) & 0xFFFF; |
| let word1 = ((destination_low >>> 16) * (source[0] >>> 16)) & 0xFFFF; |
| let low = word0 | word1 << 16; |
| |
| let word2 = ((destination_high & 0xFFFF) * (source[1] & 0xFFFF)) & 0xFFFF; |
| let word3 = ((destination_high >>> 16) * (source[1] >>> 16)) & 0xFFFF; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| t[0xD6] = cpu => { |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| // movq xmm/m64, xmm |
| var data = cpu.read_xmm64s(); |
| dbg_assert(cpu.modrm_byte < 0xC0); |
| var addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write64(addr, data[0], data[1]); |
| }; |
| t[0xD7] = cpu => { |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| if(cpu.modrm_byte < 0xC0) cpu.trigger_ud(); |
| |
| // pmovmskb reg, xmm |
| let data = cpu.read_xmm_mem128s(); |
| let data8 = new Uint8Array(data.buffer); |
| let result = |
| data8[0] >> 7 << 0 | data8[1] >> 7 << 1 | data8[2] >> 7 << 2 | data8[3] >> 7 << 3 | |
| data8[4] >> 7 << 4 | data8[5] >> 7 << 5 | data8[6] >> 7 << 6 | data8[7] >> 7 << 7 | |
| data8[8] >> 7 << 8 | data8[9] >> 7 << 9 | data8[10] >> 7 << 10 | data8[11] >> 7 << 11 | |
| data8[12] >> 7 << 12 | data8[13] >> 7 << 13 | data8[14] >> 7 << 14 | data8[15] >> 7 << 15; |
| cpu.write_g32(result); |
| }; |
| |
| t[0xD8] = cpu => { |
| // psubusb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source64s = cpu.read_mmx_mem64s(); |
| let source8 = new Uint8Array(source64s.buffer); |
| |
| let reg_offset = 8 * (cpu.modrm_byte >> 3 & 7); |
| let destination8 = cpu.reg_mmx8; |
| |
| let byte0 = cpu.saturate_sd_to_ub(destination8[reg_offset] - source8[0]); |
| let byte1 = cpu.saturate_sd_to_ub(destination8[reg_offset + 1] - source8[1]); |
| let byte2 = cpu.saturate_sd_to_ub(destination8[reg_offset + 2] - source8[2]); |
| let byte3 = cpu.saturate_sd_to_ub(destination8[reg_offset + 3] - source8[3]); |
| let byte4 = cpu.saturate_sd_to_ub(destination8[reg_offset + 4] - source8[4]); |
| let byte5 = cpu.saturate_sd_to_ub(destination8[reg_offset + 5] - source8[5]); |
| let byte6 = cpu.saturate_sd_to_ub(destination8[reg_offset + 6] - source8[6]); |
| let byte7 = cpu.saturate_sd_to_ub(destination8[reg_offset + 7] - source8[7]); |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xD9] = cpu => { |
| // psubusw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = (destination_low & 0xFFFF) - (source[0] & 0xFFFF); |
| let word1 = (destination_low >>> 16) - (source[0] >>> 16); |
| if (word0 < 0) { |
| word0 = 0; |
| } |
| if (word1 < 0) { |
| word1 = 0; |
| } |
| |
| let word2 = (destination_high & 0xFFFF) - (source[1] & 0xFFFF); |
| let word3 = (destination_high >>> 16) - (source[1] >>> 16); |
| if (word2 < 0) { |
| word2 = 0; |
| } |
| if (word3 < 0) { |
| word3 = 0; |
| } |
| |
| let low = word0 | word1 << 16; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xDA] = cpu => { |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| // pminub xmm, xmm/m128 |
| let source = cpu.read_xmm_mem128s(); |
| let source8 = new Uint8Array(source.buffer); |
| |
| let destination = cpu.read_xmm128s(); |
| let destination8 = new Uint8Array(destination.buffer); |
| |
| let result = cpu.create_atom128s(0, 0, 0, 0); |
| let result8 = new Uint8Array(result.buffer); |
| |
| for(let i = 0; i < 16; i++) |
| { |
| result8[i] = source8[i] < destination8[i] ? source8[i] : destination8[i]; |
| } |
| |
| cpu.write_xmm128s(result[0], result[1], result[2], result[3]) |
| }; |
| t[0xDB] = cpu => { |
| // pand mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = source[0] & destination_low; |
| let high = source[1] & destination_high; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xDC] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // paddusb xmm, xmm/m128 |
| let source = cpu.read_xmm_mem128s(); |
| let source8 = new Uint8Array(source.buffer); |
| |
| let destination = cpu.read_xmm128s(); |
| let destination8 = new Uint8Array(destination.buffer); |
| |
| let result = cpu.create_atom128s(0, 0, 0, 0); |
| let result8 = new Uint8Array(result.buffer); |
| |
| for(let i = 0; i < 16; i++) |
| { |
| result8[i] = cpu.saturate_ud_to_ub(source8[i] + destination8[i]); |
| } |
| |
| cpu.write_xmm128s(result[0], result[1], result[2], result[3]) |
| } |
| else |
| { |
| // paddusb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| |
| let source64s = cpu.read_mmx_mem64s(); |
| let source8 = new Uint8Array(source64s.buffer); |
| |
| let reg_offset = 8 * (cpu.modrm_byte >> 3 & 7); |
| let destination8 = cpu.reg_mmx8; |
| |
| let byte0 = cpu.saturate_ud_to_ub(destination8[reg_offset] + source8[0]); |
| let byte1 = cpu.saturate_ud_to_ub(destination8[reg_offset + 1] + source8[1]); |
| let byte2 = cpu.saturate_ud_to_ub(destination8[reg_offset + 2] + source8[2]); |
| let byte3 = cpu.saturate_ud_to_ub(destination8[reg_offset + 3] + source8[3]); |
| let byte4 = cpu.saturate_ud_to_ub(destination8[reg_offset + 4] + source8[4]); |
| let byte5 = cpu.saturate_ud_to_ub(destination8[reg_offset + 5] + source8[5]); |
| let byte6 = cpu.saturate_ud_to_ub(destination8[reg_offset + 6] + source8[6]); |
| let byte7 = cpu.saturate_ud_to_ub(destination8[reg_offset + 7] + source8[7]); |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0xDD] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // paddusw mm, mm/m64 |
| let source = cpu.read_xmm_mem128s(); |
| let source16 = new Uint16Array(source.buffer); |
| |
| let destination = cpu.read_xmm128s(); |
| let destination16 = new Uint16Array(destination.buffer); |
| |
| cpu.write_xmm128s( |
| cpu.saturate_uw(source16[0] + destination16[0]) | cpu.saturate_uw(source16[1] + destination16[1]) << 16, |
| cpu.saturate_uw(source16[2] + destination16[2]) | cpu.saturate_uw(source16[3] + destination16[3]) << 16, |
| cpu.saturate_uw(source16[4] + destination16[4]) | cpu.saturate_uw(source16[5] + destination16[5]) << 16, |
| cpu.saturate_uw(source16[6] + destination16[6]) | cpu.saturate_uw(source16[7] + destination16[7]) << 16 |
| ); |
| } |
| else |
| { |
| // paddusw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = cpu.saturate_uw((destination_low & 0xFFFF) + (source[0] & 0xFFFF)); |
| let word1 = cpu.saturate_uw((destination_low >>> 16) + (source[0] >>> 16)); |
| let word2 = cpu.saturate_uw((destination_high & 0xFFFF) + (source[1] & 0xFFFF)); |
| let word3 = cpu.saturate_uw((destination_high >>> 16) + (source[1] >>> 16)); |
| |
| let low = word0 | word1 << 16; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0xDE] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // pmaxub xmm, xmm/m128 |
| let source = cpu.read_xmm_mem128s(); |
| let source8 = new Uint8Array(source.buffer); |
| |
| let destination = cpu.read_xmm128s(); |
| let destination8 = new Uint8Array(destination.buffer); |
| |
| let result = cpu.create_atom128s(0, 0, 0, 0); |
| let result8 = new Uint8Array(result.buffer); |
| |
| for(let i = 0; i < 16; i++) |
| { |
| result8[i] = source8[i] > destination8[i] ? source8[i] : destination8[i]; |
| } |
| |
| cpu.write_xmm128s(result[0], result[1], result[2], result[3]) |
| } |
| else |
| { |
| dbg_assert(false); |
| } |
| }; |
| t[0xDF] = cpu => { |
| // pandn mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = source[0] & ~destination_low; |
| let high = source[1] & ~destination_high; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xE0] = cpu => { cpu.unimplemented_sse(); }; |
| |
| t[0xE1] = cpu => { |
| // psraw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let shift = source[0] >>> 0; |
| if (shift > 15) { |
| shift = 16; |
| } |
| |
| let word0 = ((destination_low << 16 >> 16) >> shift) & 0xFFFF; |
| let word1 = ((destination_low >> 16) >> shift) & 0xFFFF; |
| let low = word0 | word1 << 16; |
| |
| let word2 = ((destination_high << 16 >> 16) >> shift) & 0xFFFF; |
| let word3 = ((destination_high >> 16) >> shift) & 0xFFFF; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xE2] = cpu => { |
| // psrad mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let shift = source[0] >>> 0; |
| if (shift > 31) { |
| shift = 31; |
| } |
| |
| let low = destination_low >> shift; |
| let high = destination_high >> shift; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xE3] = cpu => { cpu.unimplemented_sse(); }; |
| t[0xE4] = cpu => { |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| // pmulhuw xmm, xmm/m128 |
| let source = cpu.read_xmm_mem128s(); |
| let source16 = new Uint16Array(source.buffer); |
| |
| let destination = cpu.read_xmm128s(); |
| let destination16 = new Uint16Array(destination.buffer); |
| |
| cpu.write_xmm128s( |
| source16[0] * destination16[0] >>> 16 | source16[1] * destination16[1] & 0xFFFF0000, |
| source16[2] * destination16[2] >>> 16 | source16[3] * destination16[3] & 0xFFFF0000, |
| source16[4] * destination16[4] >>> 16 | source16[5] * destination16[5] & 0xFFFF0000, |
| source16[6] * destination16[6] >>> 16 | source16[7] * destination16[7] & 0xFFFF0000 |
| ); |
| }; |
| |
| t[0xE5] = cpu => { |
| // pmulhw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = ((destination_low << 16 >> 16) * (source[0] << 16 >> 16)) >>> 16; |
| let word1 = ((destination_low >> 16) * (source[0] >> 16)) >>> 16; |
| let word2 = ((destination_high << 16 >> 16) * (source[1] << 16 >> 16)) >>> 16; |
| let word3 = ((destination_high >> 16) * (source[1] >> 16)) >>> 16; |
| |
| let low = word0 | word1 << 16; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xE6] = cpu => { cpu.unimplemented_sse(); }; |
| t[0xE7] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if(cpu.modrm_byte >= 0xC0) |
| { |
| cpu.trigger_ud(); |
| } |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| let data = cpu.read_xmm128s(); |
| let addr = cpu.modrm_resolve(cpu.modrm_byte); |
| cpu.safe_write128(addr, data[0], data[1], data[2], data[3]); |
| } |
| else |
| { |
| dbg_assert(false); |
| } |
| }; |
| |
| t[0xE8] = cpu => { |
| // psubsb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source64s = cpu.read_mmx_mem64s(); |
| let source8s = new Int8Array(source64s.buffer); |
| |
| let reg_offset = 8 * (cpu.modrm_byte >> 3 & 7); |
| let destination8s = cpu.reg_mmx8s; |
| |
| let byte0 = cpu.saturate_sd_to_sb(destination8s[reg_offset] - source8s[0]); |
| let byte1 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 1] - source8s[1]); |
| let byte2 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 2] - source8s[2]); |
| let byte3 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 3] - source8s[3]); |
| let byte4 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 4] - source8s[4]); |
| let byte5 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 5] - source8s[5]); |
| let byte6 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 6] - source8s[6]); |
| let byte7 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 7] - source8s[7]); |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xE9] = cpu => { |
| // psubsw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = cpu.saturate_sd_to_sw((destination_low << 16 >> 16) - (source[0] << 16 >> 16)); |
| let word1 = cpu.saturate_sd_to_sw((destination_low >> 16) - (source[0] >> 16)); |
| let word2 = cpu.saturate_sd_to_sw((destination_high << 16 >> 16) - (source[1] << 16 >> 16)); |
| let word3 = cpu.saturate_sd_to_sw((destination_high >> 16) - (source[1] >> 16)); |
| |
| let low = word0 | word1 << 16; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xEA] = cpu => { cpu.unimplemented_sse(); }; |
| |
| t[0xEB] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) === PREFIX_66) |
| { |
| // por xmm, xmm/m128 |
| let source = cpu.read_xmm_mem128s(); |
| let destination = cpu.read_xmm128s(); |
| |
| cpu.write_xmm128s( |
| source[0] | destination[0], |
| source[1] | destination[1], |
| source[2] | destination[2], |
| source[3] | destination[3] |
| ); |
| } |
| else |
| { |
| // por mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = source[0] | destination_low; |
| let high = source[1] | destination_high; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0xEC] = cpu => { |
| // paddsb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source64s = cpu.read_mmx_mem64s(); |
| let source8s = new Int8Array(source64s.buffer); |
| |
| let reg_offset = 8 * (cpu.modrm_byte >> 3 & 7); |
| let destination8s = cpu.reg_mmx8s; |
| |
| let byte0 = cpu.saturate_sd_to_sb(destination8s[reg_offset] + source8s[0]); |
| let byte1 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 1] + source8s[1]); |
| let byte2 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 2] + source8s[2]); |
| let byte3 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 3] + source8s[3]); |
| let byte4 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 4] + source8s[4]); |
| let byte5 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 5] + source8s[5]); |
| let byte6 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 6] + source8s[6]); |
| let byte7 = cpu.saturate_sd_to_sb(destination8s[reg_offset + 7] + source8s[7]); |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xED] = cpu => { |
| // paddsw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = cpu.saturate_sd_to_sw((destination_low << 16 >> 16) + (source[0] << 16 >> 16)); |
| let word1 = cpu.saturate_sd_to_sw((destination_low >> 16) + (source[0] >> 16)); |
| let word2 = cpu.saturate_sd_to_sw((destination_high << 16 >> 16) + (source[1] << 16 >> 16)); |
| let word3 = cpu.saturate_sd_to_sw((destination_high >> 16) + (source[1] >> 16)); |
| |
| let low = word0 | word1 << 16; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xEE] = cpu => { cpu.unimplemented_sse(); }; |
| t[0xEF] = cpu => { |
| // pxor mm, mm/m64 |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| let source = cpu.read_xmm_mem128s(); |
| let destination = cpu.read_xmm128s(); |
| |
| cpu.write_xmm128s( |
| source[0] ^ destination[0], |
| source[1] ^ destination[1], |
| source[2] ^ destination[2], |
| source[3] ^ destination[3] |
| ); |
| } |
| else |
| { |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = source[0] ^ destination_low; |
| let high = source[1] ^ destination_high; |
| |
| cpu.write_mmx64s(low, high); |
| } |
| }; |
| |
| t[0xF0] = cpu => { cpu.unimplemented_sse(); }; |
| |
| t[0xF1] = cpu => { |
| // psllw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let shift = source[0] >>> 0; |
| let low = 0; |
| let high = 0; |
| |
| if (shift <= 15) { |
| let word0 = ((destination_low & 0xFFFF) << shift) & 0xFFFF; |
| let word1 = (destination_low >>> 16) << shift; |
| low = word0 | word1 << 16; |
| |
| let word2 = ((destination_high & 0xFFFF) << shift) & 0xFFFF; |
| let word3 = (destination_high >>> 16) << shift; |
| high = word2 | word3 << 16; |
| } |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xF2] = cpu => { |
| // pslld mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let shift = source[0] >>> 0; |
| let low = 0; |
| let high = 0; |
| |
| if (shift <= 31) { |
| low = destination_low << shift; |
| high = destination_high << shift; |
| } |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xF3] = cpu => { |
| // psllq mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let shift = source[0] >>> 0; |
| |
| if(shift === 0) |
| { |
| return; |
| } |
| |
| let low = 0; |
| let high = 0; |
| |
| if (shift <= 31) { |
| low = destination_low << shift; |
| high = destination_high << shift | (destination_low >>> (32 - shift)); |
| } |
| else if (shift <= 63) { |
| high = destination_low << (shift & 0x1F); |
| low = 0; |
| } |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xF4] = cpu => { |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| if((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == PREFIX_66) |
| { |
| // pmuludq xmm1, xmm2/m128 |
| let source = cpu.read_xmm_mem128s(); |
| let destination = cpu.read_xmm128s(); |
| |
| let i = (cpu.modrm_byte >> 3 & 7) << 2; |
| |
| let result = cpu.do_mul32(destination[0] , source[0]); |
| cpu.reg_xmm32s[i] = result[0]; |
| cpu.reg_xmm32s[i + 1] = result[1]; |
| |
| result = cpu.do_mul32(destination[2] , source[2]); |
| cpu.reg_xmm32s[i + 2] = result[0]; |
| cpu.reg_xmm32s[i + 3] = result[1]; |
| } |
| else |
| { |
| // pmuludq mm1, mm2/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| let source64s = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| |
| let result = cpu.do_mul32(destination_low,source64s[0]) |
| |
| cpu.write_mmx64s(result[0], result[1]); |
| } |
| }; |
| |
| t[0xF5] = cpu => { |
| // pmaddwd mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let mul0 = ((destination_low << 16 >> 16) * (source[0] << 16 >> 16)); |
| let mul1 = ((destination_low >> 16) * (source[0] >> 16)); |
| let mul2 = ((destination_high << 16 >> 16) * (source[1] << 16 >> 16)); |
| let mul3 = ((destination_high >> 16) * (source[1] >> 16)); |
| |
| let low = mul0 + mul1 | 0; |
| let high = mul2 + mul3 | 0; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xF6] = cpu => { cpu.unimplemented_sse(); }; |
| t[0xF7] = cpu => { cpu.unimplemented_sse(); }; |
| |
| t[0xF8] = cpu => { |
| // psubb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source64s = cpu.read_mmx_mem64s(); |
| let source8s = new Int8Array(source64s.buffer); |
| |
| let reg_offset = 8 * (cpu.modrm_byte >> 3 & 7); |
| let destination8s = cpu.reg_mmx8s; |
| |
| let byte0 = (destination8s[reg_offset] - source8s[0]) & 0xFF; |
| let byte1 = (destination8s[reg_offset + 1] - source8s[1]) & 0xFF; |
| let byte2 = (destination8s[reg_offset + 2] - source8s[2]) & 0xFF; |
| let byte3 = (destination8s[reg_offset + 3] - source8s[3]) & 0xFF; |
| let byte4 = (destination8s[reg_offset + 4] - source8s[4]) & 0xFF; |
| let byte5 = (destination8s[reg_offset + 5] - source8s[5]) & 0xFF; |
| let byte6 = (destination8s[reg_offset + 6] - source8s[6]) & 0xFF; |
| let byte7 = (destination8s[reg_offset + 7] - source8s[7]) & 0xFF; |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xF9] = cpu => { |
| // psubw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = (destination_low - source[0]) & 0xFFFF; |
| let word1 = ((destination_low >>> 16) - (source[0] >>> 16)) & 0xFFFF; |
| let low = word0 | word1 << 16; |
| |
| let word2 = (destination_high - source[1]) & 0xFFFF; |
| let word3 = ((destination_high >>> 16) - (source[1] >>> 16)) & 0xFFFF; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xFA] = cpu => { |
| // psubd mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = destination_low - source[0]; |
| let high = destination_high - source[1]; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xFB] = cpu => { cpu.unimplemented_sse(); }; |
| |
| t[0xFC] = cpu => { |
| // paddb mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source64s = cpu.read_mmx_mem64s(); |
| let source8s = new Int8Array(source64s.buffer); |
| |
| let reg_offset = 8 * (cpu.modrm_byte >> 3 & 7); |
| let destination8s = cpu.reg_mmx8s; |
| |
| let byte0 = (destination8s[reg_offset] + source8s[0]) & 0xFF; |
| let byte1 = (destination8s[reg_offset + 1] + source8s[1]) & 0xFF; |
| let byte2 = (destination8s[reg_offset + 2] + source8s[2]) & 0xFF; |
| let byte3 = (destination8s[reg_offset + 3] + source8s[3]) & 0xFF; |
| let byte4 = (destination8s[reg_offset + 4] + source8s[4]) & 0xFF; |
| let byte5 = (destination8s[reg_offset + 5] + source8s[5]) & 0xFF; |
| let byte6 = (destination8s[reg_offset + 6] + source8s[6]) & 0xFF; |
| let byte7 = (destination8s[reg_offset + 7] + source8s[7]) & 0xFF; |
| |
| let low = byte0 | byte1 << 8 | byte2 << 16 | byte3 << 24; |
| let high = byte4 | byte5 << 8 | byte6 << 16 | byte7 << 24; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xFD] = cpu => { |
| // paddw mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let word0 = (destination_low + source[0]) & 0xFFFF; |
| let word1 = ((destination_low >>> 16) + (source[0] >>> 16)) & 0xFFFF; |
| let low = word0 | word1 << 16; |
| |
| let word2 = (destination_high + source[1]) & 0xFFFF; |
| let word3 = ((destination_high >>> 16) + (source[1] >>> 16)) & 0xFFFF; |
| let high = word2 | word3 << 16; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xFE] = cpu => { |
| // paddd mm, mm/m64 |
| dbg_assert((cpu.prefixes & (PREFIX_MASK_REP | PREFIX_MASK_OPSIZE)) == 0); |
| cpu.task_switch_test_mmx(); |
| cpu.read_modrm_byte(); |
| |
| let source = cpu.read_mmx_mem64s(); |
| let destination_low = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7)]; |
| let destination_high = cpu.reg_mmxs[2 * (cpu.modrm_byte >> 3 & 7) + 1]; |
| |
| let low = destination_low + source[0] | 0; |
| let high = destination_high + source[1] | 0; |
| |
| cpu.write_mmx64s(low, high); |
| }; |
| |
| t[0xFF] = cpu => { |
| // Windows 98 |
| dbg_log("#ud: 0F FF"); |
| cpu.trigger_ud(); |
| }; |
| |
| |
| var table0F_16 = []; |
| var table0F_32 = []; |
| CPU.prototype.table0F_16 = table0F_16; |
| CPU.prototype.table0F_32 = table0F_32; |
| |
| for(i = 0; i < 256; i++) |
| { |
| if(t[i]) |
| { |
| //dbg_assert(!t16[i]); |
| //dbg_assert(!t32[i]); |
| table0F_16[i] = table0F_32[i] = t[i]; |
| } |
| else if(t16[i]) |
| { |
| //dbg_assert(!t[i]); |
| //dbg_assert(t32[i]); |
| table0F_16[i] = t16[i]; |
| table0F_32[i] = t32[i]; |
| } |
| } |