/* * Copyright (C) 2002-2011 The DOSBox 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ enum STRING_OP { R_OUTSB,R_OUTSW,R_OUTSD, R_INSB,R_INSW,R_INSD, R_MOVSB,R_MOVSW,R_MOVSD, R_LODSB,R_LODSW,R_LODSD, R_STOSB,R_STOSW,R_STOSD, R_SCASB,R_SCASW,R_SCASD, R_CMPSB,R_CMPSW,R_CMPSD }; #define LoadD(_BLAH) _BLAH static void DoString(STRING_OP type) { PhysPt si_base,di_base; Bitu si_index,di_index; Bitu add_mask; Bitu count,count_left = 0; Bits add_index; si_base=BaseDS; di_base=SegBase(es); add_mask=AddrMaskTable[core.prefixes & PREFIX_ADDR]; si_index=reg_esi & add_mask; di_index=reg_edi & add_mask; count=reg_ecx & add_mask; if (!TEST_PREFIX_REP) { count=1; } else { CPU_Cycles++; /* Calculate amount of ops to do before cycles run out */ if ((count>(Bitu)CPU_Cycles) && (type0;count--) { IO_WriteB(reg_dx,LoadMb(si_base+si_index)); si_index=(si_index+add_index) & add_mask; } break; case R_OUTSW: add_index<<=1; for (;count>0;count--) { IO_WriteW(reg_dx,LoadMw(si_base+si_index)); si_index=(si_index+add_index) & add_mask; } break; case R_OUTSD: add_index<<=2; for (;count>0;count--) { IO_WriteD(reg_dx,LoadMd(si_base+si_index)); si_index=(si_index+add_index) & add_mask; } break; case R_INSB: for (;count>0;count--) { SaveMb(di_base+di_index,IO_ReadB(reg_dx)); di_index=(di_index+add_index) & add_mask; } break; case R_INSW: add_index<<=1; for (;count>0;count--) { SaveMw(di_base+di_index,IO_ReadW(reg_dx)); di_index=(di_index+add_index) & add_mask; } break; case R_STOSB: for (;count>0;count--) { SaveMb(di_base+di_index,reg_al); di_index=(di_index+add_index) & add_mask; } break; case R_STOSW: add_index<<=1; for (;count>0;count--) { SaveMw(di_base+di_index,reg_ax); di_index=(di_index+add_index) & add_mask; } break; case R_STOSD: add_index<<=2; for (;count>0;count--) { SaveMd(di_base+di_index,reg_eax); di_index=(di_index+add_index) & add_mask; } break; case R_MOVSB: for (;count>0;count--) { SaveMb(di_base+di_index,LoadMb(si_base+si_index)); di_index=(di_index+add_index) & add_mask; si_index=(si_index+add_index) & add_mask; } break; case R_MOVSW: add_index<<=1; for (;count>0;count--) { SaveMw(di_base+di_index,LoadMw(si_base+si_index)); di_index=(di_index+add_index) & add_mask; si_index=(si_index+add_index) & add_mask; } break; case R_MOVSD: add_index<<=2; for (;count>0;count--) { SaveMd(di_base+di_index,LoadMd(si_base+si_index)); di_index=(di_index+add_index) & add_mask; si_index=(si_index+add_index) & add_mask; } break; case R_LODSB: for (;count>0;count--) { reg_al=LoadMb(si_base+si_index); si_index=(si_index+add_index) & add_mask; } break; case R_LODSW: add_index<<=1; for (;count>0;count--) { reg_ax=LoadMw(si_base+si_index); si_index=(si_index+add_index) & add_mask; } break; case R_LODSD: add_index<<=2; for (;count>0;count--) { reg_eax=LoadMd(si_base+si_index); si_index=(si_index+add_index) & add_mask; } break; case R_SCASB: { Bit8u val2; for (;count>0;) { count--;CPU_Cycles--; val2=LoadMb(di_base+di_index); di_index=(di_index+add_index) & add_mask; if ((reg_al==val2)!=core.rep_zero) break; } CMPB(reg_al,val2,LoadD,0); } break; case R_SCASW: { add_index<<=1;Bit16u val2; for (;count>0;) { count--;CPU_Cycles--; val2=LoadMw(di_base+di_index); di_index=(di_index+add_index) & add_mask; if ((reg_ax==val2)!=core.rep_zero) break; } CMPW(reg_ax,val2,LoadD,0); } break; case R_SCASD: { add_index<<=2;Bit32u val2; for (;count>0;) { count--;CPU_Cycles--; val2=LoadMd(di_base+di_index); di_index=(di_index+add_index) & add_mask; if ((reg_eax==val2)!=core.rep_zero) break; } CMPD(reg_eax,val2,LoadD,0); } break; case R_CMPSB: { Bit8u val1,val2; for (;count>0;) { count--;CPU_Cycles--; val1=LoadMb(si_base+si_index); val2=LoadMb(di_base+di_index); si_index=(si_index+add_index) & add_mask; di_index=(di_index+add_index) & add_mask; if ((val1==val2)!=core.rep_zero) break; } CMPB(val1,val2,LoadD,0); } break; case R_CMPSW: { add_index<<=1;Bit16u val1,val2; for (;count>0;) { count--;CPU_Cycles--; val1=LoadMw(si_base+si_index); val2=LoadMw(di_base+di_index); si_index=(si_index+add_index) & add_mask; di_index=(di_index+add_index) & add_mask; if ((val1==val2)!=core.rep_zero) break; } CMPW(val1,val2,LoadD,0); } break; case R_CMPSD: { add_index<<=2;Bit32u val1,val2; for (;count>0;) { count--;CPU_Cycles--; val1=LoadMd(si_base+si_index); val2=LoadMd(di_base+di_index); si_index=(si_index+add_index) & add_mask; di_index=(di_index+add_index) & add_mask; if ((val1==val2)!=core.rep_zero) break; } CMPD(val1,val2,LoadD,0); } break; default: LOG(LOG_CPU,LOG_ERROR)("Unhandled string op %d",type); } /* Clean up after certain amount of instructions */ reg_esi&=(~add_mask); reg_esi|=(si_index & add_mask); reg_edi&=(~add_mask); reg_edi|=(di_index & add_mask); if (TEST_PREFIX_REP) { count+=count_left; reg_ecx&=(~add_mask); reg_ecx|=(count & add_mask); } }