2011-06-22 06:18:55 +02:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2010-11-05 05:55:33 +01:00
|
|
|
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;
|
|
|
|
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) && (type<R_SCASB)) {
|
|
|
|
count_left=count-CPU_Cycles;
|
|
|
|
count=CPU_Cycles;
|
|
|
|
CPU_Cycles=0;
|
|
|
|
LOADIP; //RESET IP to the start
|
|
|
|
} else {
|
|
|
|
/* Won't interrupt scas and cmps instruction since they can interrupt themselves */
|
|
|
|
if ((count<=1) && (CPU_Cycles<=1)) CPU_Cycles--;
|
|
|
|
else if (type<R_SCASB) CPU_Cycles-=count;
|
|
|
|
count_left=0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
add_index=cpu.direction;
|
|
|
|
if (count) switch (type) {
|
|
|
|
case R_OUTSB:
|
|
|
|
for (;count>0;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);
|
|
|
|
}
|
|
|
|
}
|