OSDN Git Service

[VM][I386] TRY: Import from DOSBOX(J3100) v.2019-05-20.
[csp-qt/common_source_project-fm7.git] / source / src / vm / libcpu_newdev / dosbox-i386 / core_full / string.h
diff --git a/source/src/vm/libcpu_newdev/dosbox-i386/core_full/string.h b/source/src/vm/libcpu_newdev/dosbox-i386/core_full/string.h
new file mode 100644 (file)
index 0000000..8cb6ec0
--- /dev/null
@@ -0,0 +1,252 @@
+/*
+ *  Copyright (C) 2002-2015  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.
+ */
+
+{
+       EAPoint si_base,di_base;
+       Bitu    si_index,di_index;
+       Bitu    add_mask;
+       Bitu    count,count_left;
+       Bits    add_index;
+       
+       if (inst.prefix & PREFIX_SEG) si_base=inst.seg.base;
+       else si_base=SegBase(ds);
+       di_base=SegBase(es);
+       if (inst.prefix & PREFIX_ADDR) {
+               add_mask=0xFFFFFFFF;
+               si_index=reg_esi;
+               di_index=reg_edi;
+               count=reg_ecx;
+       } else {
+               add_mask=0xFFFF;
+               si_index=reg_si;
+               di_index=reg_di;
+               count=reg_cx;
+       }
+       if (!(inst.prefix & PREFIX_REP)) {
+               count=1;
+       } else {
+               /* Calculate amount of ops to do before cycles run out */
+               CPU_Cycles++;
+               if ((count>(Bitu)CPU_Cycles) && (inst.code.op<R_SCASB)) {
+                       count_left=count-CPU_Cycles;
+                       count=CPU_Cycles;
+                       CPU_Cycles=0;
+                       LoadIP();
+               } else {
+                       /* Won't interrupt scas and cmps instruction since they can interrupt themselves */
+                       if (inst.code.op<R_SCASB) CPU_Cycles-=count;
+                       count_left=0;
+               }
+       }
+       add_index=cpu.direction;
+       if (count) switch (inst.code.op) {
+       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_INSD:
+               add_index<<=2;
+               for (;count>0;count--) {
+                       SaveMd(di_base+di_index,IO_ReadD(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)!=inst.repz) 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)!=inst.repz) 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)!=inst.repz) 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)!=inst.repz) 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)!=inst.repz) 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)!=inst.repz) break;
+                       }
+                       CMPD(val1,val2,LoadD,0);
+               }
+               break;
+       default:
+               LOG(LOG_CPU,LOG_ERROR)("Unhandled string %d entry %X",inst.code.op,inst.entry);
+       }
+       /* 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 (inst.prefix & PREFIX_REP) {
+               count+=count_left;
+               reg_ecx&=(~add_mask);
+               reg_ecx|=(count & add_mask);
+       }
+}