//============================================================================ // // SSSS tt lll lll // SS SS tt ll ll // SS tttttt eeee ll ll aaaa // SSSS tt ee ee ll ll aa // SS tt eeeeee ll ll aaaaa -- "An Atari 2600 VCS Emulator" // SS SS tt ee ll ll aa aa // SSSS ttt eeeee llll llll aaaaa // // Copyright (c) 1995-2014 by Bradford W. Mott, Stephen Anthony // and the Stella Team // // See the file "License.txt" for information on usage and redistribution of // this file, and for a DISCLAIMER OF ALL WARRANTIES. // // $Id$ //============================================================================ //============================================================================ // This class provides Thumb emulation code ("Thumbulator") // by David Welch (dwelch@dwelch.com) // Modified by Fred Quimby // Code is public domain and used with the author's consent //============================================================================ #ifdef THUMB_SUPPORT #include "bspf.hxx" #include "Base.hxx" #include "Thumbulator.hxx" using namespace Common; // Uncomment the following to enable specific functionality // WARNING!!! This slows the runtime to a crawl //#define THUMB_DISS //#define THUMB_DBUG #if defined(THUMB_DISS) #define DO_DISS(statement) statement #else #define DO_DISS(statement) #endif #if defined(THUMB_DBUG) #define DO_DBUG(statement) statement #else #define DO_DBUG(statement) #endif // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Thumbulator::Thumbulator(const uInt16* rom_ptr, uInt16* ram_ptr, bool traponfatal) : rom(rom_ptr), ram(ram_ptr) { trapFatalErrors(traponfatal); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - Thumbulator::~Thumbulator() { } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - string Thumbulator::run( void ) { reset(); for(;;) { if (execute()) break; if (instructions > 500000) // way more than would otherwise be possible throw "instructions > 500000"; } #if defined(THUMB_DISS) || defined(THUMB_DBUG) dump_counters(); cout << statusMsg.str() << endl; #endif return statusMsg.str(); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - inline int Thumbulator::fatalError(const char* opcode, uInt32 v1, const char* msg) { statusMsg << "Thumb ARM emulation fatal error: " << endl << opcode << "(" << Base::HEX8 << v1 << "), " << msg << endl; dump_regs(); if(trapOnFatal) throw statusMsg.str(); return 0; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - inline int Thumbulator::fatalError(const char* opcode, uInt32 v1, uInt32 v2, const char* msg) { statusMsg << "Thumb ARM emulation fatal error: " << endl << opcode << "(" << Base::HEX8 << v1 << "," << v2 << "), " << msg << endl; dump_regs(); if(trapOnFatal) throw statusMsg.str(); return 0; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::dump_counters ( void ) { cout << endl << endl << "instructions " << instructions << endl << "fetches " << fetches << endl << "reads " << reads << endl << "writes " << writes << endl << "memcycles " << (fetches+reads+writes) << endl; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::dump_regs( void ) { for (int cnt = 1; cnt < 14; cnt++) { statusMsg << "R" << cnt << " = " << Base::HEX8 << reg_sys[cnt-1] << " "; if(cnt % 4 == 0) statusMsg << endl; } statusMsg << endl << "SP = " << Base::HEX8 << reg_svc[13] << " " << "LR = " << Base::HEX8 << reg_svc[14] << " " << "PC = " << Base::HEX8 << reg_sys[15] << " " << endl; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - uInt32 Thumbulator::fetch16 ( uInt32 addr ) { fetches++; uInt32 data; switch(addr&0xF0000000) { case 0x00000000: //ROM addr &= ROMADDMASK; if(addr<0x50) fatalError("fetch16", addr, "abort"); addr>>=1; #ifdef __BIG_ENDIAN__ data=((rom[addr]>>8)|(rom[addr]<<8))&0xffff; #else data=rom[addr]; #endif DO_DBUG(statusMsg << "fetch16(" << Base::HEX8 << addr << ")=" << Base::HEX4 << data << endl); return(data); case 0x40000000: //RAM addr &= RAMADDMASK; addr>>=1; #ifdef __BIG_ENDIAN__ data=((ram[addr]>>8)|(ram[addr]<<8))&0xffff; #else data=ram[addr]; #endif DO_DBUG(statusMsg << "fetch16(" << Base::HEX8 << addr << ")=" << Base::HEX4 << data << endl); return(data); } return fatalError("fetch16", addr, "abort"); } #if 0 // Currently not used anywhere in this class // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - uInt32 Thumbulator::fetch32 ( uInt32 addr ) { uInt32 data; switch(addr&0xF0000000) { case 0x00000000: //ROM if(addr<0x50) { data=read32(addr); DO_DBUG(statusMsg << "fetch32(" << Base::HEX8 << addr << ")=" << Base::HEX8 << data << endl); if(addr==0x00000000) return(data); if(addr==0x00000004) return(data); fatalError("fetch32", addr, "abort"); } case 0x40000000: //RAM data =fetch16(addr+2); data<<=16; data|=fetch16(addr+0); DO_DBUG(statusMsg << "fetch32(" << Base::HEX8 << addr << ")=" << Base::HEX8 << data << endl); return(data); } return fatalError("fetch32", addr, "abort"); } #endif // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::write16 ( uInt32 addr, uInt32 data ) { if((addr>0x40001fff)&&(addr<0x50000000)) fatalError("write16", addr, "abort - out of range"); else if((addr>0x40000028)&&(addr<0x40000c00)) fatalError("write16", addr, "to bankswitch code area"); if(addr&1) fatalError("write16", addr, "abort - misaligned"); writes++; DO_DBUG(statusMsg << "write16(" << Base::HEX8 << addr << "," << Base::HEX8 << data << ")" << endl); switch(addr&0xF0000000) { case 0x40000000: //RAM addr&=RAMADDMASK; addr>>=1; #ifdef __BIG_ENDIAN__ ram[addr]=(((data&0xFFFF)>>8)|((data&0xffff)<<8))&0xffff; #else ram[addr]=data&0xFFFF; #endif return; case 0xE0000000: //MAMCR if(addr == 0xE01FC000) { DO_DBUG(statusMsg << "write16(" << Base::HEX8 << "MAMCR" << "," << Base::HEX8 << data << ") *" << endl); mamcr = data; return; } } fatalError("write16", addr, data, "abort"); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::write32 ( uInt32 addr, uInt32 data ) { if(addr&3) fatalError("write32", addr, "abort - misaligned"); DO_DBUG(statusMsg << "write32(" << Base::HEX8 << addr << "," << Base::HEX8 << data << ")" << endl); switch(addr&0xF0000000) { case 0xF0000000: //halt dump_counters(); throw "HALT";// exit(0); case 0xE0000000: //periph switch(addr) { case 0xE0000000: DO_DISS(statusMsg << "uart: [" << char(data&0xFF) << "]" << endl); break; } return; case 0xD0000000: //debug statusMsg << "[" << Base::HEX8 << read_register(14) << "][" << addr << "] " << data << endl; return; case 0x40000000: //RAM write16(addr+0,(data>> 0)&0xFFFF); write16(addr+2,(data>>16)&0xFFFF); return; } fatalError("write32", addr, data, "abort"); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - uInt32 Thumbulator::read16 ( uInt32 addr ) { uInt32 data; if((addr>0x40001fff)&&(addr<0x50000000)) fatalError("read16", addr, "abort - out of range"); else if((addr>0x7fff)&&(addr<0x10000000)) fatalError("read16", addr, "abort - out of range"); if(addr&1) fatalError("read16", addr, "abort - misaligned"); reads++; switch(addr&0xF0000000) { case 0x00000000: //ROM addr&=ROMADDMASK; addr>>=1; #ifdef __BIG_ENDIAN__ data=((rom[addr]>>8)|(rom[addr]<<8))&0xffff; #else data=rom[addr]; #endif DO_DBUG(statusMsg << "read16(" << Base::HEX8 << addr << ")=" << Base::HEX4 << data << endl); return(data); case 0x40000000: //RAM addr&=RAMADDMASK; addr>>=1; #ifdef __BIG_ENDIAN__ data=((ram[addr]>>8)|(ram[addr]<<8))&0xffff; #else data=ram[addr]; #endif DO_DBUG(statusMsg << "read16(" << Base::HEX8 << addr << ")=" << Base::HEX4 << data << endl); return(data); case 0xE0000000: //MAMCR if(addr == 0xE01FC000) { DO_DBUG(statusMsg << "read16(" << "MAMCR" << addr << ")=" << mamcr << " *"); return mamcr; } } return fatalError("read16", addr, "abort"); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - uInt32 Thumbulator::read32 ( uInt32 addr ) { if(addr&3) fatalError("read32", addr, "abort - misaligned"); uInt32 data; switch(addr&0xF0000000) { case 0x00000000: //ROM case 0x40000000: //RAM data =read16(addr+2); data<<=16; data|=read16(addr+0); DO_DBUG(statusMsg << "read32(" << Base::HEX8 << addr << ")=" << Base::HEX8 << data << endl); return(data); } return fatalError("read32", addr, "abort"); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - uInt32 Thumbulator::read_register ( uInt32 reg ) { reg&=0xF; uInt32 data; switch(cpsr&0x1F) { case MODE_SVC: switch(reg) // TODO (SA) - does this do anything other than default? { default: data=reg_sys[reg]; break; case 13: case 14: data=reg_svc[reg]; break; } DO_DBUG(statusMsg << "read_register(" << dec << reg << ")=" << Base::HEX8 << data << endl); return(data); } return fatalError("read_register", cpsr, "invalid cpsr mode"); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - uInt32 Thumbulator::write_register ( uInt32 reg, uInt32 data ) { reg&=0xF; DO_DBUG(statusMsg << "write_register(" << dec << reg << "," << Base::HEX8 << data << ")" << endl); switch(cpsr&0x1F) { case MODE_SVC: switch(reg) // TODO (SA) - does this do anything other than default? { default: reg_sys[reg]=data; break; case 13: case 14: reg_svc[reg]=data; break; } return(data); } return fatalError("write_register", cpsr, "invalid cpsr mode"); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::do_zflag ( uInt32 x ) { if(x==0) cpsr|=CPSR_Z; else cpsr&=~CPSR_Z; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::do_nflag ( uInt32 x ) { if(x&0x80000000) cpsr|=CPSR_N; else cpsr&=~CPSR_N; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::do_cflag ( uInt32 a, uInt32 b, uInt32 c ) { uInt32 rc; cpsr&=~CPSR_C; rc=(a&0x7FFFFFFF)+(b&0x7FFFFFFF)+c; //carry in rc = (rc>>31)+(a>>31)+(b>>31); //carry out if(rc&2) cpsr|=CPSR_C; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::do_sub_vflag ( uInt32 a, uInt32 b, uInt32 c ) { cpsr&=~CPSR_V; //if the sign bits are different if((a&0x80000000)^(b&0x80000000)) { //and result matches b if((b&0x80000000)==(c&0x80000000)) cpsr|=CPSR_V; } } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::do_add_vflag ( uInt32 a, uInt32 b, uInt32 c ) { cpsr&=~CPSR_V; //if sign bits are the same if((a&0x80000000)==(b&0x80000000)) { //and the result is different if((b&0x80000000)!=(c&0x80000000)) cpsr|=CPSR_V; } } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::do_cflag_bit ( uInt32 x ) { if(x) cpsr|=CPSR_C; else cpsr&=~CPSR_C; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - void Thumbulator::do_vflag_bit ( uInt32 x ) { if(x) cpsr|=CPSR_V; else cpsr&=~CPSR_V; } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - int Thumbulator::execute ( void ) { uInt32 pc, sp, inst, ra,rb,rc, rm,rd,rn,rs, op; pc=read_register(15); inst=fetch16(pc-2); pc+=2; write_register(15,pc); DO_DISS(statusMsg << Base::HEX8 << (pc-5) << ": " << Base::HEX4 << inst << " "); instructions++; //ADC if((inst&0xFFC0)==0x4140) { rd=(inst>>0)&0x07; rm=(inst>>3)&0x07; DO_DISS(statusMsg << "adc r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rd); rb=read_register(rm); rc=ra+rb; if(cpsr&CPSR_C) rc++; write_register(rd,rc); do_nflag(rc); do_zflag(rc); if(cpsr&CPSR_C) do_cflag(ra,rb,1); else do_cflag(ra,rb,0); do_add_vflag(ra,rb,rc); return(0); } //ADD(1) small immediate two registers if((inst&0xFE00)==0x1C00) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rb=(inst>>6)&0x7; if(rb) { DO_DISS(statusMsg << "adds r" << dec << rd << ",r" << dec << rn << "," << "#0x" << Base::HEX2 << rb << endl); ra=read_register(rn); rc=ra+rb; //fprintf(stderr,"0x%08X = 0x%08X + 0x%08X\n",rc,ra,rb); write_register(rd,rc); do_nflag(rc); do_zflag(rc); do_cflag(ra,rb,0); do_add_vflag(ra,rb,rc); return(0); } else { //this is a mov } } //ADD(2) big immediate one register if((inst&0xF800)==0x3000) { rb=(inst>>0)&0xFF; rd=(inst>>8)&0x7; DO_DISS(statusMsg << "adds r" << dec << rd << ",#0x" << Base::HEX2 << rb << endl); ra=read_register(rd); rc=ra+rb; write_register(rd,rc); do_nflag(rc); do_zflag(rc); do_cflag(ra,rb,0); do_add_vflag(ra,-rb,rc); return(0); } //ADD(3) three registers if((inst&0xFE00)==0x1800) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "adds r" << dec << rd << ",r" << dec << rn << ",r" << rm << endl); ra=read_register(rn); rb=read_register(rm); rc=ra+rb; write_register(rd,rc); do_nflag(rc); do_zflag(rc); do_cflag(ra,rb,0); do_add_vflag(ra,rb,rc); return(0); } //ADD(4) two registers one or both high no flags if((inst&0xFF00)==0x4400) { if((inst>>6)&3) { //UNPREDICTABLE } rd=(inst>>0)&0x7; rd|=(inst>>4)&0x8; rm=(inst>>3)&0xF; DO_DISS(statusMsg << "add r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rd); rb=read_register(rm); rc=ra+rb; //fprintf(stderr,"0x%08X = 0x%08X + 0x%08X\n",rc,ra,rb); write_register(rd,rc); return(0); } //ADD(5) rd = pc plus immediate if((inst&0xF800)==0xA000) { rb=(inst>>0)&0xFF; rd=(inst>>8)&0x7; rb<<=2; DO_DISS(statusMsg << "add r" << dec << rd << ",PC,#0x" << Base::HEX2 << rb << endl); ra=read_register(15); rc=(ra&(~3))+rb; write_register(rd,rc); return(0); } //ADD(6) rd = sp plus immediate if((inst&0xF800)==0xA800) { rb=(inst>>0)&0xFF; rd=(inst>>8)&0x7; rb<<=2; DO_DISS(statusMsg << "add r" << dec << rd << ",SP,#0x" << Base::HEX2 << rb << endl); ra=read_register(13); rc=ra+rb; write_register(rd,rc); return(0); } //ADD(7) sp plus immediate if((inst&0xFF80)==0xB000) { rb=(inst>>0)&0x7F; rb<<=2; DO_DISS(statusMsg << "add SP,#0x" << Base::HEX2 << rb << endl); ra=read_register(13); rc=ra+rb; write_register(13,rc); return(0); } //AND if((inst&0xFFC0)==0x4000) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "ands r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rd); rb=read_register(rm); rc=ra&rb; write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //ASR(1) two register immediate if((inst&0xF800)==0x1000) { rd=(inst>>0)&0x07; rm=(inst>>3)&0x07; rb=(inst>>6)&0x1F; DO_DISS(statusMsg << "asrs r" << dec << rd << ",r" << dec << rm << ",#0x" << Base::HEX2 << rb << endl); rc=read_register(rm); if(rb==0) { if(rc&0x80000000) { do_cflag_bit(1); rc=~0; } else { do_cflag_bit(0); rc=0; } } else { do_cflag_bit(rc&(1<<(rb-1))); ra=rc&0x80000000; rc>>=rb; if(ra) //asr, sign is shifted in { rc|=(~0)<<(32-rb); } } write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //ASR(2) two register if((inst&0xFFC0)==0x4100) { rd=(inst>>0)&0x07; rs=(inst>>3)&0x07; DO_DISS(statusMsg << "asrs r" << dec << rd << ",r" << dec << rs << endl); rc=read_register(rd); rb=read_register(rs); rb&=0xFF; if(rb==0) { } else if(rb<32) { do_cflag_bit(rc&(1<<(rb-1))); ra=rc&0x80000000; rc>>=rb; if(ra) //asr, sign is shifted in { rc|=(~0)<<(32-rb); } } else { if(rc&0x80000000) { do_cflag_bit(1); rc=(~0); } else { do_cflag_bit(0); rc=0; } } write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //B(1) conditional branch if((inst&0xF000)==0xD000) { rb=(inst>>0)&0xFF; if(rb&0x80) rb|=(~0)<<8; op=(inst>>8)&0xF; rb<<=1; rb+=pc; rb+=2; switch(op) { case 0x0: //b eq z set DO_DISS(statusMsg << "beq 0x" << Base::HEX8 << (rb-3) << endl); if(cpsr&CPSR_Z) { write_register(15,rb); } return(0); case 0x1: //b ne z clear DO_DISS(statusMsg << "bne 0x" << Base::HEX8 << (rb-3) << endl); if(!(cpsr&CPSR_Z)) { write_register(15,rb); } return(0); case 0x2: //b cs c set DO_DISS(statusMsg << "bcs 0x" << Base::HEX8 << (rb-3) << endl); if(cpsr&CPSR_C) { write_register(15,rb); } return(0); case 0x3: //b cc c clear DO_DISS(statusMsg << "bcc 0x" << Base::HEX8 << (rb-3) << endl); if(!(cpsr&CPSR_C)) { write_register(15,rb); } return(0); case 0x4: //b mi n set DO_DISS(statusMsg << "bmi 0x" << Base::HEX8 << (rb-3) << endl); if(cpsr&CPSR_N) { write_register(15,rb); } return(0); case 0x5: //b pl n clear DO_DISS(statusMsg << "bpl 0x" << Base::HEX8 << (rb-3) << endl); if(!(cpsr&CPSR_N)) { write_register(15,rb); } return(0); case 0x6: //b vs v set DO_DISS(statusMsg << "bvs 0x" << Base::HEX8 << (rb-3) << endl); if(cpsr&CPSR_V) { write_register(15,rb); } return(0); case 0x7: //b vc v clear DO_DISS(statusMsg << "bvc 0x" << Base::HEX8 << (rb-3) << endl); if(!(cpsr&CPSR_V)) { write_register(15,rb); } return(0); case 0x8: //b hi c set z clear DO_DISS(statusMsg << "bhi 0x" << Base::HEX8 << (rb-3) << endl); if((cpsr&CPSR_C)&&(!(cpsr&CPSR_Z))) { write_register(15,rb); } return(0); case 0x9: //b ls c clear or z set DO_DISS(statusMsg << "bls 0x" << Base::HEX8 << (rb-3) << endl); if((cpsr&CPSR_Z)||(!(cpsr&CPSR_C))) { write_register(15,rb); } return(0); case 0xA: //b ge N == V DO_DISS(statusMsg << "bge 0x" << Base::HEX8 << (rb-3) << endl); ra=0; if( (cpsr&CPSR_N) && (cpsr&CPSR_V) ) ra++; if((!(cpsr&CPSR_N))&&(!(cpsr&CPSR_V))) ra++; if(ra) { write_register(15,rb); } return(0); case 0xB: //b lt N != V DO_DISS(statusMsg << "blt 0x" << Base::HEX8 << (rb-3) << endl); ra=0; if((!(cpsr&CPSR_N))&&(cpsr&CPSR_V)) ra++; if((!(cpsr&CPSR_V))&&(cpsr&CPSR_N)) ra++; if(ra) { write_register(15,rb); } return(0); case 0xC: //b gt Z==0 and N == V DO_DISS(statusMsg << "bgt 0x" << Base::HEX8 << (rb-3) << endl); ra=0; if( (cpsr&CPSR_N) && (cpsr&CPSR_V) ) ra++; if((!(cpsr&CPSR_N))&&(!(cpsr&CPSR_V))) ra++; if(cpsr&CPSR_Z) ra=0; if(ra) { write_register(15,rb); } return(0); case 0xD: //b le Z==1 or N != V DO_DISS(statusMsg << "ble 0x" << Base::HEX8 << (rb-3) << endl); ra=0; if((!(cpsr&CPSR_N))&&(cpsr&CPSR_V)) ra++; if((!(cpsr&CPSR_V))&&(cpsr&CPSR_N)) ra++; if(cpsr&CPSR_Z) ra++; if(ra) { write_register(15,rb); } return(0); case 0xE: //undefined instruction break; case 0xF: //swi break; } } //B(2) unconditional branch if((inst&0xF800)==0xE000) { rb=(inst>>0)&0x7FF; if(rb&(1<<10)) rb|=(~0)<<11; rb<<=1; rb+=pc; rb+=2; DO_DISS(statusMsg << "B 0x" << Base::HEX8 << (rb-3) << endl); write_register(15,rb); return(0); } //BIC if((inst&0xFFC0)==0x4380) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "bics r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rd); rb=read_register(rm); rc=ra&(~rb); write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //BKPT if((inst&0xFF00)==0xBE00) { rb=(inst>>0)&0xFF; statusMsg << "bkpt 0x" << Base::HEX2 << rb << endl; return(1); } //BL/BLX(1) if((inst&0xE000)==0xE000) //BL,BLX { if((inst&0x1800)==0x1000) //H=b10 { DO_DISS(statusMsg << endl); halfadd=inst; return(0); } else if((inst&0x1800)==0x1800) //H=b11 { //branch to thumb rb=halfadd&((1<<11)-1); if(rb&1<<10) rb|=(~((1<<11)-1)); //sign extend rb<<=11; rb|=inst&((1<<11)-1); rb<<=1; rb+=pc; DO_DISS(statusMsg << "bl 0x" << Base::HEX8 << (rb-3) << endl); write_register(14,pc-2); write_register(15,rb); return(0); } else if((inst&0x1800)==0x0800) //H=b01 { //fprintf(stderr,"cannot branch to arm 0x%08X 0x%04X\n",pc,inst); // fxq: this should exit the code without having to detect it return(1); } } //BLX(2) if((inst&0xFF87)==0x4780) { rm=(inst>>3)&0xF; DO_DISS(statusMsg << "blx r" << dec << rm << endl); rc=read_register(rm); //fprintf(stderr,"blx r%u 0x%X 0x%X\n",rm,rc,pc); rc+=2; if(rc&1) { write_register(14,pc-2); write_register(15,rc); return(0); } else { //fprintf(stderr,"cannot branch to arm 0x%08X 0x%04X\n",pc,inst); // fxq: this could serve as exit code return(1); } } //BX if((inst&0xFF87)==0x4700) { rm=(inst>>3)&0xF; DO_DISS(statusMsg << "bx r" << dec << rm << endl); rc=read_register(rm); rc+=2; //fprintf(stderr,"bx r%u 0x%X 0x%X\n",rm,rc,pc); if(rc&1) { write_register(15,rc); return(0); } else { //fprintf(stderr,"cannot branch to arm 0x%08X 0x%04X\n",pc,inst); // fxq: or maybe this one?? return(1); } } //CMN if((inst&0xFFC0)==0x42C0) { rn=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "cmns r" << dec << rn << ",r" << dec << rm << endl); ra=read_register(rn); rb=read_register(rm); rc=ra+rb; do_nflag(rc); do_zflag(rc); do_cflag(ra,rb,0); do_add_vflag(ra,rb,rc); return(0); } //CMP(1) compare immediate if((inst&0xF800)==0x2800) { rb=(inst>>0)&0xFF; rn=(inst>>8)&0x07; DO_DISS(statusMsg << "cmp r" << dec << rn << ",#0x" << Base::HEX2 << rb << endl); ra=read_register(rn); rc=ra-rb; //fprintf(stderr,"0x%08X 0x%08X\n",ra,rb); do_nflag(rc); do_zflag(rc); do_cflag(ra,~rb,1); do_sub_vflag(ra,rb,rc); return(0); } //CMP(2) compare register if((inst&0xFFC0)==0x4280) { rn=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "cmps r" << dec << rn << ",r" << dec << rm << endl); ra=read_register(rn); rb=read_register(rm); rc=ra-rb; //fprintf(stderr,"0x%08X 0x%08X\n",ra,rb); do_nflag(rc); do_zflag(rc); do_cflag(ra,~rb,1); do_sub_vflag(ra,rb,rc); return(0); } //CMP(3) compare high register if((inst&0xFF00)==0x4500) { if(((inst>>6)&3)==0x0) { //UNPREDICTABLE } rn=(inst>>0)&0x7; rn|=(inst>>4)&0x8; if(rn==0xF) { //UNPREDICTABLE } rm=(inst>>3)&0xF; DO_DISS(statusMsg << "cmps r" << dec << rn << ",r" << dec << rm << endl); ra=read_register(rn); rb=read_register(rm); rc=ra-rb; do_nflag(rc); do_zflag(rc); do_cflag(ra,~rb,1); do_sub_vflag(ra,rb,rc); #if 0 if(cpsr&CPSR_N) statusMsg << "N"; else statusMsg << "n"; if(cpsr&CPSR_Z) statusMsg << "Z"; else statusMsg << "z"; if(cpsr&CPSR_C) statusMsg << "C"; else statusMsg << "c"; if(cpsr&CPSR_V) statusMsg << "V"; else statusMsg << "v"; statusMsg << " -- 0x" << Base::HEX8 << ra << " 0x" << Base::HEX8 << rb << endl; #endif return(0); } //CPS if((inst&0xFFE8)==0xB660) { DO_DISS(statusMsg << "cps TODO" << endl); return(1); } //CPY copy high register if((inst&0xFFC0)==0x4600) { //same as mov except you can use both low registers //going to let mov handle high registers rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "cpy r" << dec << rd << ",r" << dec << rm << endl); rc=read_register(rm); write_register(rd,rc); return(0); } //EOR if((inst&0xFFC0)==0x4040) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "eors r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rd); rb=read_register(rm); rc=ra^rb; write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //LDMIA if((inst&0xF800)==0xC800) { rn=(inst>>8)&0x7; #if defined(THUMB_DISS) statusMsg << "ldmia r" << dec << rn << "!,{"; for(ra=0,rb=0x01,rc=0;rb;rb=(rb<<1)&0xFF,ra++) { if(inst&rb) { if(rc) statusMsg << ","; statusMsg << "r" << dec << ra; rc++; } } statusMsg << "}" << endl; #endif sp=read_register(rn); for(ra=0,rb=0x01;rb;rb=(rb<<1)&0xFF,ra++) { if(inst&rb) { write_register(ra,read32(sp)); sp+=4; } } write_register(rn,sp); return(0); } //LDR(1) two register immediate if((inst&0xF800)==0x6800) { rd=(inst>>0)&0x07; rn=(inst>>3)&0x07; rb=(inst>>6)&0x1F; rb<<=2; DO_DISS(statusMsg << "ldr r" << dec << rd << ",[r" << dec << rn << ",#0x" << Base::HEX2 << rb << "]" << endl); rb=read_register(rn)+rb; rc=read32(rb); write_register(rd,rc); return(0); } //LDR(2) three register if((inst&0xFE00)==0x5800) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "ldr r" << dec << rd << ",[r" << dec << rn << ",r" << dec << "]" << endl); rb=read_register(rn)+read_register(rm); rc=read32(rb); write_register(rd,rc); return(0); } //LDR(3) if((inst&0xF800)==0x4800) { rb=(inst>>0)&0xFF; rd=(inst>>8)&0x07; rb<<=2; DO_DISS(statusMsg << "ldr r" << dec << rd << ",[PC+#0x" << Base::HEX2 << rb << "] "); ra=read_register(15); ra&=~3; rb+=ra; DO_DISS(statusMsg << ";@ 0x" << Base::HEX2 << rb << endl); rc=read32(rb); write_register(rd,rc); return(0); } //LDR(4) if((inst&0xF800)==0x9800) { rb=(inst>>0)&0xFF; rd=(inst>>8)&0x07; rb<<=2; DO_DISS(statusMsg << "ldr r" << dec << rd << ",[SP+#0x" << Base::HEX2 << rb << "]" << endl); ra=read_register(13); //ra&=~3; rb+=ra; rc=read32(rb); write_register(rd,rc); return(0); } //LDRB(1) if((inst&0xF800)==0x7800) { rd=(inst>>0)&0x07; rn=(inst>>3)&0x07; rb=(inst>>6)&0x1F; DO_DISS(statusMsg << "ldrb r" << dec << rd << ",[r" << dec << rn << ",#0x" << Base::HEX2 << rb << "]" << endl); rb=read_register(rn)+rb; rc=read16(rb&(~1)); if(rb&1) { rc>>=8; } else { } write_register(rd,rc&0xFF); return(0); } //LDRB(2) if((inst&0xFE00)==0x5C00) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "ldrb r" << dec << rd << ",[r" << dec << rn << ",r" << dec << rm << "]" << endl); rb=read_register(rn)+read_register(rm); rc=read16(rb&(~1)); if(rb&1) { rc>>=8; } else { } write_register(rd,rc&0xFF); return(0); } //LDRH(1) if((inst&0xF800)==0x8800) { rd=(inst>>0)&0x07; rn=(inst>>3)&0x07; rb=(inst>>6)&0x1F; rb<<=1; DO_DISS(statusMsg << "ldrh r" << dec << rd << ",[r" << dec << rn << ",#0x" << Base::HEX2 << rb << "]" << endl); rb=read_register(rn)+rb; rc=read16(rb); write_register(rd,rc&0xFFFF); return(0); } //LDRH(2) if((inst&0xFE00)==0x5A00) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "ldrh r" << dec << rd << ",[r" << dec << rn << ",r" << dec << rm << "]" << endl); rb=read_register(rn)+read_register(rm); rc=read16(rb); write_register(rd,rc&0xFFFF); return(0); } //LDRSB if((inst&0xFE00)==0x5600) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "ldrsb r" << dec << rd << ",[r" << dec << rn << ",r" << dec << rm << "]" << endl); rb=read_register(rn)+read_register(rm); rc=read16(rb&(~1)); if(rb&1) { rc>>=8; } else { } rc&=0xFF; if(rc&0x80) rc|=((~0)<<8); write_register(rd,rc); return(0); } //LDRSH if((inst&0xFE00)==0x5E00) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "ldrsh r" << dec << rd << ",[r" << dec << rn << ",r" << dec << rm << "]" << endl); rb=read_register(rn)+read_register(rm); rc=read16(rb); rc&=0xFFFF; if(rc&0x8000) rc|=((~0)<<16); write_register(rd,rc); return(0); } //LSL(1) if((inst&0xF800)==0x0000) { rd=(inst>>0)&0x07; rm=(inst>>3)&0x07; rb=(inst>>6)&0x1F; DO_DISS(statusMsg << "lsls r" << dec << rd << ",r" << dec << rm << ",#0x" << Base::HEX2 << rb << endl); rc=read_register(rm); if(rb==0) { //if immed_5 == 0 //C unnaffected //result not shifted } else { //else immed_5 > 0 do_cflag_bit(rc&(1<<(32-rb))); rc<<=rb; } write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //LSL(2) two register if((inst&0xFFC0)==0x4080) { rd=(inst>>0)&0x07; rs=(inst>>3)&0x07; DO_DISS(statusMsg << "lsls r" << dec << rd << ",r" << dec << rs << endl); rc=read_register(rd); rb=read_register(rs); rb&=0xFF; if(rb==0) { } else if(rb<32) { do_cflag_bit(rc&(1<<(32-rb))); rc<<=rb; } else if(rb==32) { do_cflag_bit(rc&1); rc=0; } else { do_cflag_bit(0); rc=0; } write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //LSR(1) two register immediate if((inst&0xF800)==0x0800) { rd=(inst>>0)&0x07; rm=(inst>>3)&0x07; rb=(inst>>6)&0x1F; DO_DISS(statusMsg << "lsrs r" << dec << rd << ",r" << dec << rm << ",#0x" << Base::HEX2 << rb << endl); rc=read_register(rm); if(rb==0) { do_cflag_bit(rc&0x80000000); rc=0; } else { do_cflag_bit(rc&(1<<(rb-1))); rc>>=rb; } write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //LSR(2) two register if((inst&0xFFC0)==0x40C0) { rd=(inst>>0)&0x07; rs=(inst>>3)&0x07; DO_DISS(statusMsg << "lsrs r" << dec << rd << ",r" << dec << rs << endl); rc=read_register(rd); rb=read_register(rs); rb&=0xFF; if(rb==0) { } else if(rb<32) { do_cflag_bit(rc&(1<<(32-rb))); rc>>=rb; } else if(rb==32) { do_cflag_bit(rc&0x80000000); rc=0; } else { do_cflag_bit(0); rc=0; } write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //MOV(1) immediate if((inst&0xF800)==0x2000) { rb=(inst>>0)&0xFF; rd=(inst>>8)&0x07; DO_DISS(statusMsg << "movs r" << dec << rd << ",#0x" << Base::HEX2 << rb << endl); write_register(rd,rb); do_nflag(rb); do_zflag(rb); return(0); } //MOV(2) two low registers if((inst&0xFFC0)==0x1C00) { rd=(inst>>0)&7; rn=(inst>>3)&7; DO_DISS(statusMsg << "movs r" << dec << rd << ",r" << dec << rn << endl); rc=read_register(rn); //fprintf(stderr,"0x%08X\n",rc); write_register(rd,rc); do_nflag(rc); do_zflag(rc); do_cflag_bit(0); do_vflag_bit(0); return(0); } //MOV(3) if((inst&0xFF00)==0x4600) { rd=(inst>>0)&0x7; rd|=(inst>>4)&0x8; rm=(inst>>3)&0xF; DO_DISS(statusMsg << "mov r" << dec << rd << ",r" << dec << rm << endl); rc=read_register(rm); if (rd==15) rc+=2; // fxq fix for MOV R15 write_register(rd,rc); return(0); } //MUL if((inst&0xFFC0)==0x4340) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "muls r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rd); rb=read_register(rm); rc=ra*rb; write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //MVN if((inst&0xFFC0)==0x43C0) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "mvns r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rm); rc=(~ra); write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //NEG if((inst&0xFFC0)==0x4240) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "negs r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rm); rc=0-ra; write_register(rd,rc); do_nflag(rc); do_zflag(rc); do_cflag(0,~ra,1); do_sub_vflag(0,ra,rc); return(0); } //ORR if((inst&0xFFC0)==0x4300) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "orrs r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rd); rb=read_register(rm); rc=ra|rb; write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //POP if((inst&0xFE00)==0xBC00) { #if defined(THUMB_DISS) statusMsg << "pop {"; for(ra=0,rb=0x01,rc=0;rb;rb=(rb<<1)&0xFF,ra++) { if(inst&rb) { if(rc) statusMsg << ","; statusMsg << "r" << dec << ra; rc++; } } if(inst&0x100) { if(rc) statusMsg << ","; statusMsg << "pc"; } statusMsg << "}" << endl; #endif sp=read_register(13); for(ra=0,rb=0x01;rb;rb=(rb<<1)&0xFF,ra++) { if(inst&rb) { write_register(ra,read32(sp)); sp+=4; } } if(inst&0x100) { rc=read32(sp); rc+=2; write_register(15,rc); sp+=4; } write_register(13,sp); return(0); } //PUSH if((inst&0xFE00)==0xB400) { #if defined(THUMB_DISS) statusMsg << "push {"; for(ra=0,rb=0x01,rc=0;rb;rb=(rb<<1)&0xFF,ra++) { if(inst&rb) { if(rc) statusMsg << ","; statusMsg << "r" << dec << ra; rc++; } } if(inst&0x100) { if(rc) statusMsg << ","; statusMsg << "lr"; } statusMsg << "}" << endl; #endif sp=read_register(13); //fprintf(stderr,"sp 0x%08X\n",sp); for(ra=0,rb=0x01,rc=0;rb;rb=(rb<<1)&0xFF,ra++) { if(inst&rb) { rc++; } } if(inst&0x100) rc++; rc<<=2; sp-=rc; rd=sp; for(ra=0,rb=0x01;rb;rb=(rb<<1)&0xFF,ra++) { if(inst&rb) { write32(rd,read_register(ra)); rd+=4; } } if(inst&0x100) { write32(rd,read_register(14)); } write_register(13,sp); return(0); } //REV if((inst&0xFFC0)==0xBA00) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; DO_DISS(statusMsg << "rev r" << dec << rd << ",r" << dec << rn << endl); ra=read_register(rn); rc =((ra>> 0)&0xFF)<<24; rc|=((ra>> 8)&0xFF)<<16; rc|=((ra>>16)&0xFF)<< 8; rc|=((ra>>24)&0xFF)<< 0; write_register(rd,rc); return(0); } //REV16 if((inst&0xFFC0)==0xBA40) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; DO_DISS(statusMsg << "rev16 r" << dec << rd << ",r" << dec << rn << endl); ra=read_register(rn); rc =((ra>> 0)&0xFF)<< 8; rc|=((ra>> 8)&0xFF)<< 0; rc|=((ra>>16)&0xFF)<<24; rc|=((ra>>24)&0xFF)<<16; write_register(rd,rc); return(0); } //REVSH if((inst&0xFFC0)==0xBAC0) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; DO_DISS(statusMsg << "revsh r" << dec << rd << ",r" << dec << rn << endl); ra=read_register(rn); rc =((ra>> 0)&0xFF)<< 8; rc|=((ra>> 8)&0xFF)<< 0; if(rc&0x8000) rc|=0xFFFF0000; else rc&=0x0000FFFF; write_register(rd,rc); return(0); } //ROR if((inst&0xFFC0)==0x41C0) { rd=(inst>>0)&0x7; rs=(inst>>3)&0x7; DO_DISS(statusMsg << "rors r" << dec << rd << ",r" << dec << rs << endl); rc=read_register(rd); ra=read_register(rs); ra&=0xFF; if(ra==0) { } else { ra&=0x1F; if(ra==0) { do_cflag_bit(rc&0x80000000); } else { do_cflag_bit(rc&(1<<(ra-1))); rb=rc<<(32-ra); rc>>=ra; rc|=rb; } } write_register(rd,rc); do_nflag(rc); do_zflag(rc); return(0); } //SBC if((inst&0xFFC0)==0x4180) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "sbc r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rd); rb=read_register(rm); rc=ra-rb; if(!(cpsr&CPSR_C)) rc--; write_register(rd,rc); do_nflag(rc); do_zflag(rc); do_cflag(ra,rb,0); do_sub_vflag(ra,rb,rc); return(0); } //SETEND if((inst&0xFFF7)==0xB650) { statusMsg << "setend not implemented" << endl; return(1); } //STMIA if((inst&0xF800)==0xC000) { rn=(inst>>8)&0x7; #if defined(THUMB_DISS) statusMsg << "stmia r" << dec << rn << "!,{"; for(ra=0,rb=0x01,rc=0;rb;rb=(rb<<1)&0xFF,ra++) { if(inst&rb) { if(rc) statusMsg << ","; statusMsg << "r" << dec << ra; rc++; } } statusMsg << "}" << endl; #endif sp=read_register(rn); for(ra=0,rb=0x01;rb;rb=(rb<<1)&0xFF,ra++) { if(inst&rb) { write32(sp,read_register(ra)); sp+=4; } } write_register(rn,sp); return(0); } //STR(1) if((inst&0xF800)==0x6000) { rd=(inst>>0)&0x07; rn=(inst>>3)&0x07; rb=(inst>>6)&0x1F; rb<<=2; DO_DISS(statusMsg << "str r" << dec << rd << ",[r" << dec << rn << ",#0x" << Base::HEX2 << rb << "]" << endl); rb=read_register(rn)+rb; rc=read_register(rd); write32(rb,rc); return(0); } //STR(2) if((inst&0xFE00)==0x5000) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "str r" << dec << rd << ",[r" << dec << rn << ",r" << dec << rm << "]" << endl); rb=read_register(rn)+read_register(rm); rc=read_register(rd); write32(rb,rc); return(0); } //STR(3) if((inst&0xF800)==0x9000) { rb=(inst>>0)&0xFF; rd=(inst>>8)&0x07; rb<<=2; DO_DISS(statusMsg << "str r" << dec << rd << ",[SP,#0x" << Base::HEX2 << rb << "]" << endl); rb=read_register(13)+rb; //fprintf(stderr,"0x%08X\n",rb); rc=read_register(rd); write32(rb,rc); return(0); } //STRB(1) if((inst&0xF800)==0x7000) { rd=(inst>>0)&0x07; rn=(inst>>3)&0x07; rb=(inst>>6)&0x1F; DO_DISS(statusMsg << "strb r" << dec << rd << ",[r" << dec << rn << ",#0x" << Base::HEX8 << rb << "]" << endl); rb=read_register(rn)+rb; rc=read_register(rd); ra=read16(rb&(~1)); if(rb&1) { ra&=0x00FF; ra|=rc<<8; } else { ra&=0xFF00; ra|=rc&0x00FF; } write16(rb&(~1),ra&0xFFFF); return(0); } //STRB(2) if((inst&0xFE00)==0x5400) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "strb r" << dec << rd << ",[r" << dec << rn << ",r" << rm << "]" << endl); rb=read_register(rn)+read_register(rm); rc=read_register(rd); ra=read16(rb&(~1)); if(rb&1) { ra&=0x00FF; ra|=rc<<8; } else { ra&=0xFF00; ra|=rc&0x00FF; } write16(rb&(~1),ra&0xFFFF); return(0); } //STRH(1) if((inst&0xF800)==0x8000) { rd=(inst>>0)&0x07; rn=(inst>>3)&0x07; rb=(inst>>6)&0x1F; rb<<=1; DO_DISS(statusMsg << "strh r" << dec << rd << ",[r" << dec << rn << ",#0x" << Base::HEX2 << rb << "]" << endl); rb=read_register(rn)+rb; rc=read_register(rd); write16(rb,rc&0xFFFF); return(0); } //STRH(2) if((inst&0xFE00)==0x5200) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "strh r" << dec << rd << ",[r" << dec << rn << ",r" << dec << rm << "]" << endl); rb=read_register(rn)+read_register(rm); rc=read_register(rd); write16(rb,rc&0xFFFF); return(0); } //SUB(1) if((inst&0xFE00)==0x1E00) { rd=(inst>>0)&7; rn=(inst>>3)&7; rb=(inst>>6)&7; DO_DISS(statusMsg << "subs r" << dec << rd << ",r" << dec << rn << ",#0x" << Base::HEX2 << rb << endl); ra=read_register(rn); rc=ra-rb; write_register(rd,rc); do_nflag(rc); do_zflag(rc); do_cflag(ra,~rb,1); do_sub_vflag(ra,rb,rc); return(0); } //SUB(2) if((inst&0xF800)==0x3800) { rb=(inst>>0)&0xFF; rd=(inst>>8)&0x07; DO_DISS(statusMsg << "subs r" << dec << rd << ",#0x" << Base::HEX2 << rb << endl); ra=read_register(rd); rc=ra-rb; write_register(rd,rc); do_nflag(rc); do_zflag(rc); do_cflag(ra,~rb,1); do_sub_vflag(ra,rb,rc); return(0); } //SUB(3) if((inst&0xFE00)==0x1A00) { rd=(inst>>0)&0x7; rn=(inst>>3)&0x7; rm=(inst>>6)&0x7; DO_DISS(statusMsg << "subs r" << dec << rd << ",r" << dec << rn << ",r" << dec << rm << endl); ra=read_register(rn); rb=read_register(rm); rc=ra-rb; write_register(rd,rc); do_nflag(rc); do_zflag(rc); do_cflag(ra,~rb,1); do_sub_vflag(ra,rb,rc); return(0); } //SUB(4) if((inst&0xFF80)==0xB080) { rb=inst&0x7F; rb<<=2; DO_DISS(statusMsg << "sub SP,#0x" << Base::HEX2 << rb << endl); ra=read_register(13); ra-=rb; write_register(13,ra); return(0); } //SWI if((inst&0xFF00)==0xDF00) { rb=inst&0xFF; DO_DISS(statusMsg << "swi 0x" << Base::HEX2 << rb << endl); statusMsg << endl << endl << "swi 0x" << Base::HEX2 << rb << endl; return(1); } //SXTB if((inst&0xFFC0)==0xB240) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "sxtb r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rm); rc=ra&0xFF; if(rc&0x80) rc|=(~0)<<8; write_register(rd,rc); return(0); } //SXTH if((inst&0xFFC0)==0xB200) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "sxth r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rm); rc=ra&0xFFFF; if(rc&0x8000) rc|=(~0)<<16; write_register(rd,rc); return(0); } //TST if((inst&0xFFC0)==0x4200) { rn=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "tst r" << dec << rn << ",r" << dec << rm << endl); ra=read_register(rn); rb=read_register(rm); rc=ra&rb; do_nflag(rc); do_zflag(rc); return(0); } //UXTB if((inst&0xFFC0)==0xB2C0) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "uxtb r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rm); rc=ra&0xFF; write_register(rd,rc); return(0); } //UXTH if((inst&0xFFC0)==0xB280) { rd=(inst>>0)&0x7; rm=(inst>>3)&0x7; DO_DISS(statusMsg << "uxth r" << dec << rd << ",r" << dec << rm << endl); ra=read_register(rm); rc=ra&0xFFFF; write_register(rd,rc); return(0); } statusMsg << "invalid instruction " << Base::HEX8 << pc << " " << Base::HEX4 << inst << endl; return(1); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - int Thumbulator::reset ( void ) { //memset(ram,0xFF,sizeof(ram)); cpsr=CPSR_T|CPSR_I|CPSR_F|MODE_SVC; reg_svc[13]=0x40001fb4; //sp reg_svc[14]=0x00000c00; //lr (duz this use odd addrs) reg_sys[15]=0x00000c0b; // entry point of 0xc09+2 // reg_sys[15]+=2; mamcr = 0; // fxq: don't care about below so much (maybe to guess timing???) instructions=0; fetches=0; reads=0; writes=0; statusMsg.str(""); return(0); } // - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - bool Thumbulator::trapOnFatal = true; #endif