/* Pcsx2 - Pc Ps2 Emulator * Copyright (C) 2002-2009 Pcsx2 Team * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ #pragma once // Implementations found here: TEST + BTS/BT/BTC/BTR + BSF/BSR! (for lack of better location) // Note: This header is meant to be included from within the x86Emitter::Internal namespace. ////////////////////////////////////////////////////////////////////////////////////////// // TEST instruction Implementation // class xImpl_Test { public: // ------------------------------------------------------------------------ template< typename T > __forceinline void operator()( const xRegister& to, const xRegister& from ) const { prefix16(); xWrite8( Is8BitOp() ? 0x84 : 0x85 ); EmitSibMagic( from, to ); } // ------------------------------------------------------------------------ template< typename T > __forceinline void operator()( const ModSibStrict& dest, int imm ) const { prefix16(); xWrite8( Is8BitOp() ? 0xf6 : 0xf7 ); EmitSibMagic( 0, dest ); xWrite( imm ); } // ------------------------------------------------------------------------ template< typename T > __forceinline void operator()( const xRegister& to, int imm ) const { prefix16(); if( to.IsAccumulator() ) xWrite8( Is8BitOp() ? 0xa8 : 0xa9 ); else { xWrite8( Is8BitOp() ? 0xf6 : 0xf7 ); EmitSibMagic( 0, to ); } xWrite( imm ); } xImpl_Test() {} // Why does GCC need these? }; enum G8Type { G8Type_BT = 4, G8Type_BTS, G8Type_BTR, G8Type_BTC, }; ////////////////////////////////////////////////////////////////////////////////////////// // BSF / BSR -- 16/32 operands supported only. // // 0xbc [fwd] / 0xbd [rev] // template< u16 Opcode > class xImpl_BitScan { public: xImpl_BitScan() {} __forceinline void operator()( const xRegister32& to, const xRegister32& from ) const { xOpWrite0F( Opcode, to, from ); } __forceinline void operator()( const xRegister16& to, const xRegister16& from ) const { xOpWrite0F( 0x66, Opcode, to, from ); } __forceinline void operator()( const xRegister32& to, const ModSibBase& sibsrc ) const { xOpWrite0F( Opcode, to, sibsrc ); } __forceinline void operator()( const xRegister16& to, const ModSibBase& sibsrc ) const { xOpWrite0F( 0x66, Opcode, to, sibsrc ); } }; ////////////////////////////////////////////////////////////////////////////////////////// // Bit Test Instructions - Valid on 16/32 bit instructions only. // template< G8Type InstType > class xImpl_Group8 { static const uint RegFormOp = 0xa3 | (InstType << 3); public: __forceinline void operator()( const xRegister32& bitbase, const xRegister32& bitoffset ) const { xOpWrite0F( RegFormOp, bitbase, bitoffset ); } __forceinline void operator()( const xRegister16& bitbase, const xRegister16& bitoffset ) const { xOpWrite0F( 0x66, RegFormOp, bitbase, bitoffset ); } __forceinline void operator()( const ModSibBase& bitbase, const xRegister32& bitoffset ) const { xOpWrite0F( RegFormOp, bitoffset, bitbase ); } __forceinline void operator()( const ModSibBase& bitbase, const xRegister16& bitoffset ) const { xOpWrite0F( 0x66, RegFormOp, bitoffset, bitbase ); } __forceinline void operator()( const ModSibStrict& bitbase, u8 bitoffset ) const { xOpWrite0F( 0xba, InstType, bitbase, bitoffset ); } __forceinline void operator()( const ModSibStrict& bitbase, u8 bitoffset ) const { xOpWrite0F( 0x66, 0xba, InstType, bitbase, bitoffset ); } __forceinline void operator()( const xRegister& bitbase, u8 bitoffset ) const { xOpWrite0F( 0xba, InstType, bitbase, bitoffset ); } __forceinline void operator()( const xRegister& bitbase, u8 bitoffset ) const { xOpWrite0F( 0x66, 0xba, InstType, bitbase, bitoffset ); } xImpl_Group8() {} };