Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
#ifdef PROCESSOR_ARM_HPP
|
|
|
|
|
2012-03-21 11:08:16 +00:00
|
|
|
void ARM::arm_step() {
|
|
|
|
if(pipeline.reload) {
|
|
|
|
pipeline.reload = false;
|
2012-03-31 08:14:31 +00:00
|
|
|
r(15).data &= ~3;
|
2012-03-21 11:08:16 +00:00
|
|
|
|
2012-04-15 06:49:56 +00:00
|
|
|
sequential() = false;
|
2012-04-03 23:50:40 +00:00
|
|
|
pipeline.fetch.address = r(15) & ~3;
|
|
|
|
pipeline.fetch.instruction = read(pipeline.fetch.address, Word);
|
2012-03-21 11:08:16 +00:00
|
|
|
|
2012-03-29 11:58:10 +00:00
|
|
|
pipeline_step();
|
2012-03-21 11:08:16 +00:00
|
|
|
}
|
|
|
|
|
2012-03-29 11:58:10 +00:00
|
|
|
pipeline_step();
|
2012-03-21 11:08:16 +00:00
|
|
|
|
2012-04-07 08:17:49 +00:00
|
|
|
if(processor.irqline && cpsr().i == 0) {
|
|
|
|
vector(0x00000018, Processor::Mode::IRQ);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-03-27 11:02:57 +00:00
|
|
|
instructions++;
|
|
|
|
if(trace) {
|
|
|
|
print(disassemble_registers(), "\n");
|
|
|
|
print(disassemble_arm_instruction(pipeline.execute.address), "\n");
|
2012-03-31 08:14:31 +00:00
|
|
|
usleep(100000);
|
2012-03-27 11:02:57 +00:00
|
|
|
}
|
2012-03-21 11:08:16 +00:00
|
|
|
|
2012-03-27 11:02:57 +00:00
|
|
|
if(condition(instruction() >> 28) == false) return;
|
2012-03-21 11:08:16 +00:00
|
|
|
|
2012-03-27 11:02:57 +00:00
|
|
|
#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()
|
|
|
|
|
2012-03-29 11:58:10 +00:00
|
|
|
decode("???? 0001 0010 ++++ ++++ ++++ 0001 ????", branch_exchange_register);
|
2012-03-27 11:02:57 +00:00
|
|
|
decode("???? 0000 00?? ???? ???? ???? 1001 ????", multiply);
|
2012-03-31 08:14:31 +00:00
|
|
|
decode("???? 0000 1??? ???? ???? ???? 1001 ????", multiply_long);
|
2012-03-29 11:58:10 +00:00
|
|
|
decode("???? 0001 0?00 ++++ ???? ---- 0000 ----", move_to_register_from_status);
|
2012-03-27 11:02:57 +00:00
|
|
|
decode("???? 0001 0?00 ???? ???? ---- 1001 ????", memory_swap);
|
2012-03-29 11:58:10 +00:00
|
|
|
decode("???? 0001 0?10 ???? ++++ ---- 0000 ????", move_to_status_from_register);
|
|
|
|
decode("???? 0011 0?10 ???? ++++ ???? ???? ????", move_to_status_from_immediate);
|
2012-03-27 11:02:57 +00:00
|
|
|
decode("???? 000? ?0?1 ???? ???? ---- 11?1 ????", load_register);
|
|
|
|
decode("???? 000? ?1?1 ???? ???? ???? 11?1 ????", load_immediate);
|
2012-03-29 11:58:10 +00:00
|
|
|
decode("???? 000? ?0?? ???? ???? ---- 1011 ????", move_half_register);
|
|
|
|
decode("???? 000? ?1?? ???? ???? ???? 1011 ????", move_half_immediate);
|
2012-03-27 11:02:57 +00:00
|
|
|
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
|
|
|
|
|
2012-03-29 11:58:10 +00:00
|
|
|
crash = true;
|
2012-03-21 11:08:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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) {
|
2012-03-29 11:58:10 +00:00
|
|
|
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);
|
2012-03-21 11:08:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
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) {
|
2012-03-29 11:58:10 +00:00
|
|
|
if(source == 1 || privilegedmode()) {
|
Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-03-21 11:08:16 +00:00
|
|
|
//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
|
2012-03-31 08:14:31 +00:00
|
|
|
//m = rm
|
2012-03-21 11:08:16 +00:00
|
|
|
void ARM::arm_op_multiply() {
|
|
|
|
uint1 accumulate = instruction() >> 21;
|
2012-03-31 08:14:31 +00:00
|
|
|
uint1 save = instruction() >> 20;
|
2012-03-21 11:08:16 +00:00
|
|
|
uint4 d = instruction() >> 16;
|
|
|
|
uint4 n = instruction() >> 12;
|
|
|
|
uint4 s = instruction() >> 8;
|
2012-03-31 08:14:31 +00:00
|
|
|
uint4 m = instruction();
|
2012-03-21 11:08:16 +00:00
|
|
|
|
Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
step(1);
|
2012-04-01 01:41:15 +00:00
|
|
|
r(d) = mul(accumulate ? r(n) : 0u, r(m), r(s));
|
2012-03-21 11:08:16 +00:00
|
|
|
}
|
|
|
|
|
2012-03-31 08:14:31 +00:00
|
|
|
//(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();
|
|
|
|
|
2012-03-31 08:17:36 +00:00
|
|
|
uint64 rm = r(m);
|
|
|
|
uint64 rs = r(s);
|
|
|
|
if(signextend) {
|
|
|
|
rm = (int32)rm;
|
|
|
|
rs = (int32)rs;
|
|
|
|
}
|
2012-03-31 08:14:31 +00:00
|
|
|
|
|
|
|
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;
|
2012-04-01 01:41:15 +00:00
|
|
|
//cpsr().c = 0; //(undefined)
|
|
|
|
//cpsr().v = 0; //(undefined)
|
2012-03-31 08:14:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
//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();
|
|
|
|
|
2012-04-15 06:49:56 +00:00
|
|
|
uint32 word = load(r(n), byte ? Byte : Word);
|
|
|
|
store(r(n), byte ? Byte : Word, r(m));
|
Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
r(d) = word;
|
|
|
|
}
|
|
|
|
|
2012-03-27 11:02:57 +00:00
|
|
|
//(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;
|
2012-04-15 06:49:56 +00:00
|
|
|
uint1 l = instruction() >> 20;
|
2012-03-27 11:02:57 +00:00
|
|
|
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;
|
2012-04-15 06:49:56 +00:00
|
|
|
if(l == 1) r(d) = load(rn, Half);
|
|
|
|
if(l == 0) store(rn, Half, r(d));
|
2012-03-27 11:02:57 +00:00
|
|
|
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;
|
2012-04-15 06:49:56 +00:00
|
|
|
uint1 l = instruction() >> 20;
|
2012-03-27 11:02:57 +00:00
|
|
|
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;
|
2012-04-15 06:49:56 +00:00
|
|
|
if(l == 1) r(d) = load(rn, Half);
|
|
|
|
if(l == 0) store(rn, Half, r(d));
|
2012-03-27 11:02:57 +00:00
|
|
|
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;
|
2012-04-15 06:49:56 +00:00
|
|
|
uint32 word = load(rn, half ? Half : Byte);
|
2012-03-27 11:02:57 +00:00
|
|
|
r(d) = half ? (int16)word : (int8)word;
|
2012-03-29 11:58:10 +00:00
|
|
|
if(pre == 0) rn = up ? rn + rm : rn - rm;
|
2012-03-27 11:02:57 +00:00
|
|
|
|
|
|
|
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;
|
2012-04-15 06:49:56 +00:00
|
|
|
uint32 word = load(rn, half ? Half : Byte);
|
2012-03-27 11:02:57 +00:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2012-03-21 11:08:16 +00:00
|
|
|
//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();
|
|
|
|
|
Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
arm_move_to_status(r(m));
|
2012-03-21 11:08:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//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;
|
2012-03-26 10:13:02 +00:00
|
|
|
r(15) = r(m);
|
2012-03-21 11:08:16 +00:00
|
|
|
}
|
|
|
|
|
Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
//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;
|
2012-03-29 11:58:10 +00:00
|
|
|
if(rotate) rm = ror(rm, 2 * rotate);
|
Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
|
|
|
|
arm_move_to_status(rm);
|
|
|
|
}
|
|
|
|
|
2012-03-21 11:08:16 +00:00
|
|
|
//{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);
|
2012-03-31 08:17:36 +00:00
|
|
|
carryout() = cpsr().c;
|
2012-03-21 11:08:16 +00:00
|
|
|
|
2012-03-29 11:58:10 +00:00
|
|
|
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);
|
2012-03-21 11:08:16 +00:00
|
|
|
|
|
|
|
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);
|
2012-03-31 08:17:36 +00:00
|
|
|
carryout() = cpsr().c;
|
2012-03-21 11:08:16 +00:00
|
|
|
|
2012-03-29 11:58:10 +00:00
|
|
|
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);
|
2012-03-21 11:08:16 +00:00
|
|
|
|
|
|
|
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();
|
|
|
|
|
2012-03-31 08:17:36 +00:00
|
|
|
uint32 rm = immediate;
|
|
|
|
|
|
|
|
carryout() = cpsr().c;
|
|
|
|
if(shift) rm = ror(immediate, 2 * shift);
|
2012-03-21 11:08:16 +00:00
|
|
|
|
|
|
|
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;
|
2012-04-15 06:49:56 +00:00
|
|
|
uint1 l = instruction() >> 20;
|
2012-03-21 11:08:16 +00:00
|
|
|
uint4 n = instruction() >> 16;
|
|
|
|
uint4 d = instruction() >> 12;
|
|
|
|
uint12 rm = instruction();
|
|
|
|
|
|
|
|
uint32 rn = r(n);
|
2013-05-05 09:21:30 +00:00
|
|
|
auto& rd = r(d);
|
2012-03-21 11:08:16 +00:00
|
|
|
|
|
|
|
if(pre == 1) rn = up ? rn + rm : rn - rm;
|
2012-04-15 06:49:56 +00:00
|
|
|
if(l == 1) rd = load(rn, byte ? Byte : Word);
|
|
|
|
if(l == 0) store(rn, byte ? Byte : Word, rd);
|
2012-03-21 11:08:16 +00:00
|
|
|
if(pre == 0) rn = up ? rn + rm : rn - rm;
|
|
|
|
|
|
|
|
if(pre == 0 || writeback == 1) r(n) = rn;
|
|
|
|
}
|
|
|
|
|
2012-03-27 11:02:57 +00:00
|
|
|
//(ldr,str){condition}{b} rd,[rn,rm {mode} #immediate]{!}
|
2012-03-21 11:08:16 +00:00
|
|
|
//(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;
|
2012-04-15 06:49:56 +00:00
|
|
|
uint1 l = instruction() >> 20;
|
2012-03-21 11:08:16 +00:00
|
|
|
uint4 n = instruction() >> 16;
|
|
|
|
uint4 d = instruction() >> 12;
|
|
|
|
uint5 immediate = instruction() >> 7;
|
|
|
|
uint2 mode = instruction() >> 5;
|
|
|
|
uint4 m = instruction();
|
|
|
|
|
|
|
|
uint32 rn = r(n);
|
2013-05-05 09:21:30 +00:00
|
|
|
auto& rd = r(d);
|
2012-03-21 11:08:16 +00:00
|
|
|
uint32 rs = immediate;
|
|
|
|
uint32 rm = r(m);
|
|
|
|
bool c = cpsr().c;
|
|
|
|
|
2012-03-29 11:58:10 +00:00
|
|
|
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);
|
2012-03-21 11:08:16 +00:00
|
|
|
|
|
|
|
if(pre == 1) rn = up ? rn + rm : rn - rm;
|
2012-04-15 06:49:56 +00:00
|
|
|
if(l == 1) rd = load(rn, byte ? Byte : Word);
|
|
|
|
if(l == 0) store(rn, byte ? Byte : Word, rd);
|
2012-03-21 11:08:16 +00:00
|
|
|
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;
|
2012-04-15 06:49:56 +00:00
|
|
|
uint1 l = instruction() >> 20;
|
2012-03-21 11:08:16 +00:00
|
|
|
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;
|
2012-04-15 06:49:56 +00:00
|
|
|
if(s && l == 1 && (list & 0x8000) == 0) usr = true;
|
|
|
|
if(s && l == 0) usr = true;
|
2012-03-21 11:08:16 +00:00
|
|
|
|
|
|
|
if(usr) processor.setMode(Processor::Mode::USR);
|
|
|
|
|
2012-04-15 06:49:56 +00:00
|
|
|
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));
|
2012-03-21 11:08:16 +00:00
|
|
|
rn += 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if(usr) processor.setMode(pmode);
|
|
|
|
|
2012-04-15 06:49:56 +00:00
|
|
|
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);
|
|
|
|
}
|
2012-03-21 11:08:16 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
Update to v087r08 release.
byuu says:
Added some more ARM opcodes, hooked up MMIO. Bind it with mmio[(addr
000-3ff)] = this; inside CPU/PPU/APU, goes to read(), write().
Also moved the Hitachi HG51B core to processor/, and split it apart from
the snes/chip/hitachidsp implementation.
This one actually worked really well. Very clean split between MMIO/DMA
and the processor core. I may move a more generic DMA function inside
the core, not sure yet.
I still believe the HG51B169 to be a variant of the HG51BS family, but
given they're meant to be incredibly flexible microcontrollers, it's
possible that each variant gets its own instruction set.
So, who knows. We'll worry about it if we ever find another HG51B DSP,
I guess.
GBA BIOS is constantly reading from 04000300, but it never writes. If
I return prng()&1, I can get it to proceed until it hits a bad opcode
(stc opcode, which the GBA lacks a coprocessor so ... bad codepath.)
Without it, it just reads that register forever and keeps resetting the
system, or something ...
I guess we're going to have to try and get ARMwrestler working, because
the BIOS seems to need too much emulation code to do anything at all.
2012-03-24 07:52:36 +00:00
|
|
|
|
|
|
|
//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
|