mirror of https://github.com/bsnes-emu/bsnes.git
634 lines
16 KiB
C++
Executable File
634 lines
16 KiB
C++
Executable File
#ifdef PROCESSOR_ARM_HPP
|
|
|
|
void ARM::arm_step() {
|
|
if(pipeline.reload) {
|
|
pipeline.reload = false;
|
|
r(15).data &= ~3;
|
|
|
|
sequential() = false;
|
|
pipeline.fetch.address = r(15) & ~3;
|
|
pipeline.fetch.instruction = read(pipeline.fetch.address, Word);
|
|
|
|
pipeline_step();
|
|
}
|
|
|
|
pipeline_step();
|
|
|
|
if(processor.irqline && cpsr().i == 0) {
|
|
vector(0x00000018, Processor::Mode::IRQ);
|
|
return;
|
|
}
|
|
|
|
instructions++;
|
|
if(trace) {
|
|
print(disassemble_registers(), "\n");
|
|
print(disassemble_arm_instruction(pipeline.execute.address), "\n");
|
|
usleep(100000);
|
|
}
|
|
|
|
if(condition(instruction() >> 28) == false) return;
|
|
|
|
#define decode(pattern, execute) if( \
|
|
(instruction() & std::integral_constant<uint32, bit::mask(pattern)>::value) \
|
|
== std::integral_constant<uint32, bit::test(pattern)>::value \
|
|
) return arm_op_ ## execute()
|
|
|
|
decode("???? 0001 0010 ++++ ++++ ++++ 0001 ????", branch_exchange_register);
|
|
decode("???? 0000 00?? ???? ???? ???? 1001 ????", multiply);
|
|
decode("???? 0000 1??? ???? ???? ???? 1001 ????", multiply_long);
|
|
decode("???? 0001 0?00 ++++ ???? ---- 0000 ----", move_to_register_from_status);
|
|
decode("???? 0001 0?00 ???? ???? ---- 1001 ????", memory_swap);
|
|
decode("???? 0001 0?10 ???? ++++ ---- 0000 ????", move_to_status_from_register);
|
|
decode("???? 0011 0?10 ???? ++++ ???? ???? ????", move_to_status_from_immediate);
|
|
decode("???? 000? ?0?1 ???? ???? ---- 11?1 ????", load_register);
|
|
decode("???? 000? ?1?1 ???? ???? ???? 11?1 ????", load_immediate);
|
|
decode("???? 000? ?0?? ???? ???? ---- 1011 ????", move_half_register);
|
|
decode("???? 000? ?1?? ???? ???? ???? 1011 ????", move_half_immediate);
|
|
decode("???? 000? ???? ???? ???? ???? ???0 ????", data_immediate_shift);
|
|
decode("???? 000? ???? ???? ???? ???? 0??1 ????", data_register_shift);
|
|
decode("???? 001? ???? ???? ???? ???? ???? ????", data_immediate);
|
|
decode("???? 010? ???? ???? ???? ???? ???? ????", move_immediate_offset);
|
|
decode("???? 011? ???? ???? ???? ???? ???0 ????", move_register_offset);
|
|
decode("???? 100? ???? ???? ???? ???? ???? ????", move_multiple);
|
|
decode("???? 101? ???? ???? ???? ???? ???? ????", branch);
|
|
decode("???? 1111 ???? ???? ???? ???? ???? ????", software_interrupt);
|
|
|
|
#undef decode
|
|
|
|
crash = true;
|
|
}
|
|
|
|
void ARM::arm_opcode(uint32 rm) {
|
|
uint4 opcode = instruction() >> 21;
|
|
uint1 save = instruction() >> 20;
|
|
uint4 n = instruction() >> 16;
|
|
uint4 d = instruction() >> 12;
|
|
|
|
uint32 rn = r(n);
|
|
|
|
switch(opcode) {
|
|
case 0: r(d) = bit(rn & rm); break; //AND
|
|
case 1: r(d) = bit(rn ^ rm); break; //EOR
|
|
case 2: r(d) = sub(rn, rm, 1); break; //SUB
|
|
case 3: r(d) = sub(rm, rn, 1); break; //RSB
|
|
case 4: r(d) = add(rn, rm, 0); break; //ADD
|
|
case 5: r(d) = add(rn, rm, cpsr().c); break; //ADC
|
|
case 6: r(d) = sub(rn, rm, cpsr().c); break; //SBC
|
|
case 7: r(d) = sub(rm, rn, cpsr().c); break; //RSC
|
|
case 8: bit(rn & rm); break; //TST
|
|
case 9: bit(rn ^ rm); break; //TEQ
|
|
case 10: sub(rn, rm, 1); break; //CMP
|
|
case 11: add(rn, rm, 0); break; //CMN
|
|
case 12: r(d) = bit(rn | rm); break; //ORR
|
|
case 13: r(d) = bit(rm); break; //MOV
|
|
case 14: r(d) = bit(rn & ~rm); break; //BIC
|
|
case 15: r(d) = bit(~rm); break; //MVN
|
|
}
|
|
|
|
if(exceptionmode() && d == 15 && save) {
|
|
cpsr() = spsr();
|
|
processor.setMode((Processor::Mode)cpsr().m);
|
|
}
|
|
}
|
|
|
|
void ARM::arm_move_to_status(uint32 rm) {
|
|
uint1 source = instruction() >> 22;
|
|
uint4 field = instruction() >> 16;
|
|
|
|
if(source == 1) {
|
|
if(mode() == Processor::Mode::USR) return;
|
|
if(mode() == Processor::Mode::SYS) return;
|
|
}
|
|
|
|
PSR &psr = source ? spsr() : cpsr();
|
|
|
|
if(field & 1) {
|
|
if(source == 1 || privilegedmode()) {
|
|
psr.i = rm & 0x00000080;
|
|
psr.f = rm & 0x00000040;
|
|
psr.t = rm & 0x00000020;
|
|
psr.m = rm & 0x0000001f;
|
|
if(source == 0) processor.setMode((Processor::Mode)psr.m);
|
|
}
|
|
}
|
|
|
|
if(field & 8) {
|
|
psr.n = rm & 0x80000000;
|
|
psr.z = rm & 0x40000000;
|
|
psr.c = rm & 0x20000000;
|
|
psr.v = rm & 0x10000000;
|
|
}
|
|
}
|
|
|
|
//mul{condition}{s} rd,rm,rs
|
|
//mla{condition}{s} rd,rm,rs,rn
|
|
//cccc 0000 00as dddd nnnn ssss 1001 mmmm
|
|
//c = condition
|
|
//a = accumulate
|
|
//s = save flags
|
|
//d = rd
|
|
//n = rn
|
|
//s = rs
|
|
//m = rm
|
|
void ARM::arm_op_multiply() {
|
|
uint1 accumulate = instruction() >> 21;
|
|
uint1 save = instruction() >> 20;
|
|
uint4 d = instruction() >> 16;
|
|
uint4 n = instruction() >> 12;
|
|
uint4 s = instruction() >> 8;
|
|
uint4 m = instruction();
|
|
|
|
step(1);
|
|
r(d) = mul(accumulate ? r(n) : 0u, r(m), r(s));
|
|
}
|
|
|
|
//(u,s)mull{condition}{s} rdlo,rdhi,rm,rs
|
|
//(u,s)mlal{condition}{s} rdlo,rdhi,rm,rs
|
|
//cccc 0000 1sas hhhh llll ssss 1001 mmmm
|
|
//c = condition
|
|
//s = sign-extend
|
|
//a = accumulate
|
|
//s = save flags
|
|
//h = rdhi
|
|
//l = rdlo
|
|
//s = rs
|
|
//m = rm
|
|
void ARM::arm_op_multiply_long() {
|
|
uint1 signextend = instruction() >> 22;
|
|
uint1 accumulate = instruction() >> 21;
|
|
uint1 save = instruction() >> 20;
|
|
uint4 dhi = instruction() >> 16;
|
|
uint4 dlo = instruction() >> 12;
|
|
uint4 s = instruction() >> 8;
|
|
uint4 m = instruction();
|
|
|
|
uint64 rm = r(m);
|
|
uint64 rs = r(s);
|
|
if(signextend) {
|
|
rm = (int32)rm;
|
|
rs = (int32)rs;
|
|
}
|
|
|
|
uint64 rd = rm * rs;
|
|
if(accumulate) rd += ((uint64)r(dhi) << 32) + ((uint64)r(dlo) << 0);
|
|
|
|
r(dhi) = rd >> 32;
|
|
r(dlo) = rd >> 0;
|
|
|
|
if(save) {
|
|
cpsr().n = r(dhi) >> 31;
|
|
cpsr().z = r(dhi) == 0 && r(dlo) == 0;
|
|
//cpsr().c = 0; //(undefined)
|
|
//cpsr().v = 0; //(undefined)
|
|
}
|
|
}
|
|
|
|
//swp{condition}{b} rd,rm,[rn]
|
|
//cccc 0001 0b00 nnnn dddd ---- 1001 mmmm
|
|
//c = condition
|
|
//b = byte (0 = word)
|
|
//n = rn
|
|
//d = rd
|
|
//m = rm
|
|
void ARM::arm_op_memory_swap() {
|
|
uint1 byte = instruction() >> 22;
|
|
uint4 n = instruction() >> 16;
|
|
uint4 d = instruction() >> 12;
|
|
uint4 m = instruction();
|
|
|
|
uint32 word = load(r(n), byte ? Byte : Word);
|
|
store(r(n), byte ? Byte : Word, r(m));
|
|
r(d) = word;
|
|
}
|
|
|
|
//(ldr,str){condition}h rd,[rn,rm]{!}
|
|
//(ldr,str){condition}h rd,[rn],rm
|
|
//cccc 000p u0wl nnnn dddd ---- 1011 mmmm
|
|
//c = condition
|
|
//p = pre (0 = post)
|
|
//u = up
|
|
//w = writeback
|
|
//l = load (0 = save)
|
|
//n = rn
|
|
//d = rd
|
|
//m = rm
|
|
void ARM::arm_op_move_half_register() {
|
|
uint1 pre = instruction() >> 24;
|
|
uint1 up = instruction() >> 23;
|
|
uint1 writeback = instruction() >> 21;
|
|
uint1 l = instruction() >> 20;
|
|
uint4 n = instruction() >> 16;
|
|
uint4 d = instruction() >> 12;
|
|
uint4 m = instruction();
|
|
|
|
uint32 rn = r(n);
|
|
uint32 rm = r(m);
|
|
|
|
if(pre == 1) rn = up ? rn + rm : rn - rm;
|
|
if(l == 1) r(d) = load(rn, Half);
|
|
if(l == 0) store(rn, Half, r(d));
|
|
if(pre == 0) rn = up ? rn + rm : rn - rm;
|
|
|
|
if(pre == 0 || writeback == 1) r(n) = rn;
|
|
}
|
|
|
|
//(ldr,str){condition}h rd,[rn{,+/-offset}]{!}
|
|
//(ldr,str){condition}h rd,[rn]{,+/-offset}
|
|
//cccc 000p u1wl nnnn dddd iiii 1011 iiii
|
|
//c = condition
|
|
//p = pre (0 = post)
|
|
//u = up
|
|
//w = writeback
|
|
//l = load (0 = save)
|
|
//n = rn
|
|
//d = rd
|
|
//i = immediate hi
|
|
//i = immediate lo
|
|
void ARM::arm_op_move_half_immediate() {
|
|
uint1 pre = instruction() >> 24;
|
|
uint1 up = instruction() >> 23;
|
|
uint1 writeback = instruction() >> 21;
|
|
uint1 l = instruction() >> 20;
|
|
uint4 n = instruction() >> 16;
|
|
uint4 d = instruction() >> 12;
|
|
uint4 ih = instruction() >> 8;
|
|
uint4 il = instruction();
|
|
|
|
uint32 rn = r(n);
|
|
uint8 immediate = (ih << 4) + (il << 0);
|
|
|
|
if(pre == 1) rn = up ? rn + immediate : rn - immediate;
|
|
if(l == 1) r(d) = load(rn, Half);
|
|
if(l == 0) store(rn, Half, r(d));
|
|
if(pre == 0) rn = up ? rn + immediate : rn - immediate;
|
|
|
|
if(pre == 0 || writeback == 1) r(n) = rn;
|
|
}
|
|
|
|
//ldr{condition}s(h,b) rd,[rn,rm]{!}
|
|
//ldr{condition}s(h,b) rd,[rn],rm
|
|
//cccc 000p u0w1 nnnn dddd ---- 11h1 mmmm
|
|
//c = condition
|
|
//p = pre (0 = post)
|
|
//u = up
|
|
//w = writeback
|
|
//n = rn
|
|
//d = rd
|
|
//h = half (0 = byte)
|
|
//m = rm
|
|
void ARM::arm_op_load_register() {
|
|
uint1 pre = instruction() >> 24;
|
|
uint1 up = instruction() >> 23;
|
|
uint1 writeback = instruction() >> 21;
|
|
uint4 n = instruction() >> 16;
|
|
uint4 d = instruction() >> 12;
|
|
uint1 half = instruction() >> 5;
|
|
uint4 m = instruction();
|
|
|
|
uint32 rn = r(n);
|
|
uint32 rm = r(m);
|
|
|
|
if(pre == 1) rn = up ? rn + rm : rn - rm;
|
|
uint32 word = load(rn, half ? Half : Byte);
|
|
r(d) = half ? (int16)word : (int8)word;
|
|
if(pre == 0) rn = up ? rn + rm : rn - rm;
|
|
|
|
if(pre == 0 || writeback == 1) r(n) = rn;
|
|
}
|
|
|
|
//ldr{condition}s(h,b) rd,[rn{,+/-offset}]{!}
|
|
//ldr{condition}s(h,b) rd,[rn]{,+/-offset}
|
|
//cccc 000p u1w1 nnnn dddd iiii 11h1 iiii
|
|
//c = condition
|
|
//p = pre (0 = post)
|
|
//u = up
|
|
//w = writeback
|
|
//n = rn
|
|
//d = rd
|
|
//i = immediate hi
|
|
//h = half (0 = byte)
|
|
//i = immediate lo
|
|
void ARM::arm_op_load_immediate() {
|
|
uint1 pre = instruction() >> 24;
|
|
uint1 up = instruction() >> 23;
|
|
uint1 writeback = instruction() >> 21;
|
|
uint4 n = instruction() >> 16;
|
|
uint4 d = instruction() >> 12;
|
|
uint4 ih = instruction() >> 8;
|
|
uint1 half = instruction() >> 5;
|
|
uint4 il = instruction();
|
|
|
|
uint32 rn = r(n);
|
|
uint8 immediate = (ih << 4) + (il << 0);
|
|
|
|
if(pre == 1) rn = up ? rn + immediate : rn - immediate;
|
|
uint32 word = load(rn, half ? Half : Byte);
|
|
r(d) = half ? (int16)word : (int8)word;
|
|
if(pre == 0) rn = up ? rn + immediate : rn - immediate;
|
|
|
|
if(pre == 0 || writeback == 1) r(n) = rn;
|
|
}
|
|
|
|
//mrs{condition} rd,(c,s)psr
|
|
//cccc 0001 0r00 ++++ dddd ---- 0000 ----
|
|
//c = condition
|
|
//r = SPSR (0 = CPSR)
|
|
//d = rd
|
|
void ARM::arm_op_move_to_register_from_status() {
|
|
uint1 source = instruction() >> 22;
|
|
uint4 d = instruction() >> 12;
|
|
|
|
if(source) {
|
|
if(mode() == Processor::Mode::USR) return;
|
|
if(mode() == Processor::Mode::SYS) return;
|
|
}
|
|
|
|
r(d) = source ? spsr() : cpsr();
|
|
}
|
|
|
|
//msr{condition} (c,s)psr:{fields},rm
|
|
//cccc 0001 0r10 ffff ++++ ---- 0000 mmmm
|
|
//c = condition
|
|
//r = SPSR (0 = CPSR)
|
|
//f = field mask
|
|
//m = rm
|
|
void ARM::arm_op_move_to_status_from_register() {
|
|
uint4 m = instruction();
|
|
|
|
arm_move_to_status(r(m));
|
|
}
|
|
|
|
//bx{condition} rm
|
|
//cccc 0001 0010 ++++ ++++ ++++ 0001 mmmm
|
|
//c = condition
|
|
//m = rm
|
|
void ARM::arm_op_branch_exchange_register() {
|
|
uint4 m = instruction();
|
|
|
|
cpsr().t = r(m) & 1;
|
|
r(15) = r(m);
|
|
}
|
|
|
|
//msr{condition} (c,s)psr:{fields},#immediate
|
|
//cccc 0011 0r10 ffff ++++ rrrr iiii iiii
|
|
//c = condition
|
|
//r = SPSR (0 = CPSR)
|
|
//f = field mask
|
|
//r = rotate
|
|
//i = immediate
|
|
void ARM::arm_op_move_to_status_from_immediate() {
|
|
uint4 rotate = instruction() >> 8;
|
|
uint8 immediate = instruction();
|
|
|
|
uint32 rm = immediate;
|
|
if(rotate) rm = ror(rm, 2 * rotate);
|
|
|
|
arm_move_to_status(rm);
|
|
}
|
|
|
|
//{opcode}{condition}{s} rd,rm {shift} #immediate
|
|
//{opcode}{condition} rn,rm {shift} #immediate
|
|
//{opcode}{condition}{s} rd,rn,rm {shift} #immediate
|
|
//cccc 000o ooos nnnn dddd llll lss0 mmmm
|
|
//c = condition
|
|
//o = opcode
|
|
//s = save flags
|
|
//n = rn
|
|
//d = rd
|
|
//l = shift immediate
|
|
//s = shift
|
|
//m = rm
|
|
void ARM::arm_op_data_immediate_shift() {
|
|
uint1 save = instruction() >> 20;
|
|
uint5 shift = instruction() >> 7;
|
|
uint2 mode = instruction() >> 5;
|
|
uint4 m = instruction();
|
|
|
|
uint32 rs = shift;
|
|
uint32 rm = r(m);
|
|
carryout() = cpsr().c;
|
|
|
|
if(mode == 0) rm = lsl(rm, rs);
|
|
if(mode == 1) rm = lsr(rm, rs ? rs : 32);
|
|
if(mode == 2) rm = asr(rm, rs ? rs : 32);
|
|
if(mode == 3) rm = rs ? ror(rm, rs) : rrx(rm);
|
|
|
|
arm_opcode(rm);
|
|
}
|
|
|
|
//{opcode}{condition}{s} rd,rm {shift} rs
|
|
//{opcode}{condition} rn,rm {shift} rs
|
|
//{opcode}{condition}{s} rd,rn,rm {shift} rs
|
|
//cccc 000o ooos nnnn dddd ssss 0ss1 mmmm
|
|
//c = condition
|
|
//o = opcode
|
|
//s = save flags
|
|
//n = rn
|
|
//d = rd
|
|
//s = rs
|
|
//s = shift
|
|
//m = rm
|
|
void ARM::arm_op_data_register_shift() {
|
|
uint1 save = instruction() >> 20;
|
|
uint4 s = instruction() >> 8;
|
|
uint2 mode = instruction() >> 5;
|
|
uint4 m = instruction();
|
|
|
|
uint8 rs = r(s);
|
|
uint32 rm = r(m);
|
|
carryout() = cpsr().c;
|
|
|
|
if(mode == 0 ) rm = lsl(rm, rs < 33 ? rs : 33);
|
|
if(mode == 1 ) rm = lsr(rm, rs < 33 ? rs : 33);
|
|
if(mode == 2 ) rm = asr(rm, rs < 32 ? rs : 32);
|
|
if(mode == 3 && rs) rm = ror(rm, rs & 31 == 0 ? 32 : rs & 31);
|
|
|
|
arm_opcode(rm);
|
|
}
|
|
|
|
//{opcode}{condition}{s} rd,#immediate
|
|
//{opcode}{condition} rn,#immediate
|
|
//{opcode}{condition}{s} rd,rn,#immediate
|
|
//cccc 001o ooos nnnn dddd ssss iiii iiii
|
|
//c = condition
|
|
//o = opcode
|
|
//s = save flags
|
|
//n = rn
|
|
//d = rd
|
|
//s = shift immediate
|
|
//i = immediate
|
|
void ARM::arm_op_data_immediate() {
|
|
uint1 save = instruction() >> 20;
|
|
uint4 shift = instruction() >> 8;
|
|
uint8 immediate = instruction();
|
|
|
|
uint32 rm = immediate;
|
|
|
|
carryout() = cpsr().c;
|
|
if(shift) rm = ror(immediate, 2 * shift);
|
|
|
|
arm_opcode(rm);
|
|
}
|
|
|
|
//(ldr,str){condition}{b} rd,[rn{,+/-offset}]{!}
|
|
//(ldr,str){condition}{b} rd,[rn]{,+/-offset}
|
|
//cccc 010p ubwl nnnn dddd iiii iiii iiii
|
|
//c = condition
|
|
//p = pre (0 = post-indexed addressing)
|
|
//u = up (add/sub offset to base)
|
|
//b = byte (1 = word)
|
|
//w = writeback
|
|
//l = load (0 = save)
|
|
//n = rn
|
|
//d = rd
|
|
//i = immediate
|
|
void ARM::arm_op_move_immediate_offset() {
|
|
uint1 pre = instruction() >> 24;
|
|
uint1 up = instruction() >> 23;
|
|
uint1 byte = instruction() >> 22;
|
|
uint1 writeback = instruction() >> 21;
|
|
uint1 l = instruction() >> 20;
|
|
uint4 n = instruction() >> 16;
|
|
uint4 d = instruction() >> 12;
|
|
uint12 rm = instruction();
|
|
|
|
uint32 rn = r(n);
|
|
auto& rd = r(d);
|
|
|
|
if(pre == 1) rn = up ? rn + rm : rn - rm;
|
|
if(l == 1) rd = load(rn, byte ? Byte : Word);
|
|
if(l == 0) store(rn, byte ? Byte : Word, rd);
|
|
if(pre == 0) rn = up ? rn + rm : rn - rm;
|
|
|
|
if(pre == 0 || writeback == 1) r(n) = rn;
|
|
}
|
|
|
|
//(ldr,str){condition}{b} rd,[rn,rm {mode} #immediate]{!}
|
|
//(ldr,str){condition}{b} rd,[rn],rm {mode} #immediate
|
|
//cccc 011p ubwl nnnn dddd llll lss0 mmmm
|
|
//c = condition
|
|
//p = pre (0 = post-indexed addressing)
|
|
//u = up
|
|
//b = byte (1 = word)
|
|
//w = writeback
|
|
//l = load (0 = save)
|
|
//n = rn
|
|
//d = rd
|
|
//l = shift immediate
|
|
//s = shift mode
|
|
//m = rm
|
|
void ARM::arm_op_move_register_offset() {
|
|
uint1 pre = instruction() >> 24;
|
|
uint1 up = instruction() >> 23;
|
|
uint1 byte = instruction() >> 22;
|
|
uint1 writeback = instruction() >> 21;
|
|
uint1 l = instruction() >> 20;
|
|
uint4 n = instruction() >> 16;
|
|
uint4 d = instruction() >> 12;
|
|
uint5 immediate = instruction() >> 7;
|
|
uint2 mode = instruction() >> 5;
|
|
uint4 m = instruction();
|
|
|
|
uint32 rn = r(n);
|
|
auto& rd = r(d);
|
|
uint32 rs = immediate;
|
|
uint32 rm = r(m);
|
|
bool c = cpsr().c;
|
|
|
|
if(mode == 0) rm = lsl(rm, rs);
|
|
if(mode == 1) rm = lsr(rm, rs ? rs : 32);
|
|
if(mode == 2) rm = asr(rm, rs ? rs : 32);
|
|
if(mode == 3) rm = rs ? ror(rm, rs) : rrx(rm);
|
|
|
|
if(pre == 1) rn = up ? rn + rm : rn - rm;
|
|
if(l == 1) rd = load(rn, byte ? Byte : Word);
|
|
if(l == 0) store(rn, byte ? Byte : Word, rd);
|
|
if(pre == 0) rn = up ? rn + rm : rn - rm;
|
|
|
|
if(pre == 0 || writeback == 1) r(n) = rn;
|
|
}
|
|
|
|
//(ldm,stm){condition}{mode} rn{!},{r...}
|
|
//cccc 100p uswl nnnn llll llll llll llll
|
|
//c = condition
|
|
//p = pre (0 = post-indexed addressing)
|
|
//u = up (add/sub offset to base)
|
|
//s = spsr copy -or- usr register copy
|
|
//w = writeback
|
|
//l = load (0 = save)
|
|
//n = rn
|
|
//l = register list
|
|
void ARM::arm_op_move_multiple() {
|
|
uint1 pre = instruction() >> 24;
|
|
uint1 up = instruction() >> 23;
|
|
uint1 s = instruction() >> 22;
|
|
uint1 writeback = instruction() >> 21;
|
|
uint1 l = instruction() >> 20;
|
|
uint4 n = instruction() >> 16;
|
|
uint16 list = instruction();
|
|
|
|
uint32 rn = r(n);
|
|
if(pre == 0 && up == 1) rn = rn + 0; //IA
|
|
if(pre == 1 && up == 1) rn = rn + 4; //IB
|
|
if(pre == 1 && up == 0) rn = rn - bit::count(list) * 4 + 0; //DB
|
|
if(pre == 0 && up == 0) rn = rn - bit::count(list) * 4 + 4; //DA
|
|
|
|
Processor::Mode pmode = mode();
|
|
bool usr = false;
|
|
if(s && l == 1 && (list & 0x8000) == 0) usr = true;
|
|
if(s && l == 0) usr = true;
|
|
|
|
if(usr) processor.setMode(Processor::Mode::USR);
|
|
|
|
sequential() = false;
|
|
for(unsigned m = 0; m < 16; m++) {
|
|
if(list & (1 << m)) {
|
|
if(l == 1) r(m) = read(rn, Word);
|
|
if(l == 0) write(rn, Word, r(m));
|
|
rn += 4;
|
|
}
|
|
}
|
|
|
|
if(usr) processor.setMode(pmode);
|
|
|
|
if(l == 1) {
|
|
idle();
|
|
if(s && (list & 0x8000)) {
|
|
if(mode() != Processor::Mode::USR && mode() != Processor::Mode::SYS) {
|
|
cpsr() = spsr();
|
|
processor.setMode((Processor::Mode)cpsr().m);
|
|
}
|
|
}
|
|
}
|
|
|
|
if(writeback) {
|
|
if(up == 1) r(n) = r(n) + bit::count(list) * 4; //IA, IB
|
|
if(up == 0) r(n) = r(n) - bit::count(list) * 4; //DA, DB
|
|
}
|
|
}
|
|
|
|
//b{l}{condition} address
|
|
//cccc 101l dddd dddd dddd dddd dddd dddd
|
|
//c = condition
|
|
//l = link
|
|
//d = displacement (24-bit signed)
|
|
void ARM::arm_op_branch() {
|
|
uint1 link = instruction() >> 24;
|
|
int24 displacement = instruction();
|
|
|
|
if(link) r(14) = r(15) - 4;
|
|
r(15) += displacement * 4;
|
|
}
|
|
|
|
//swi #immediate
|
|
//cccc 1111 iiii iiii iiii iiii iiii iiii
|
|
//c = condition
|
|
//i = immediate
|
|
void ARM::arm_op_software_interrupt() {
|
|
uint24 immediate = instruction();
|
|
|
|
vector(0x00000008, Processor::Mode::SVC);
|
|
}
|
|
|
|
#endif
|