mirror of https://github.com/mgba-emu/mgba.git
Start cycle counting
This commit is contained in:
parent
9b2cd97505
commit
c8a2f595d4
|
@ -17,6 +17,15 @@
|
||||||
buffer += AMOUNT; \
|
buffer += AMOUNT; \
|
||||||
blen -= AMOUNT;
|
blen -= AMOUNT;
|
||||||
|
|
||||||
|
#define LOAD_CYCLES \
|
||||||
|
info->iCycles = 1; \
|
||||||
|
info->nDataCycles = 1;
|
||||||
|
|
||||||
|
#define STORE_CYCLES \
|
||||||
|
info->sInstructionCycles = 0; \
|
||||||
|
info->nInstructionCycles = 1; \
|
||||||
|
info->nDataCycles = 1;
|
||||||
|
|
||||||
static int _decodeRegister(int reg, char* buffer, int blen);
|
static int _decodeRegister(int reg, char* buffer, int blen);
|
||||||
static int _decodeRegisterList(int list, char* buffer, int blen);
|
static int _decodeRegisterList(int list, char* buffer, int blen);
|
||||||
static int _decodePCRelative(uint32_t address, uint32_t pc, char* buffer, int blen);
|
static int _decodePCRelative(uint32_t address, uint32_t pc, char* buffer, int blen);
|
||||||
|
|
|
@ -26,7 +26,7 @@
|
||||||
ARM_OPERAND_REGISTER_2 | \
|
ARM_OPERAND_REGISTER_2 | \
|
||||||
ARM_OPERAND_IMMEDIATE_3;)
|
ARM_OPERAND_IMMEDIATE_3;)
|
||||||
|
|
||||||
#define DEFINE_IMMEDIATE_5_DECODER_MEM_THUMB(NAME, IMMEDIATE, MNEMONIC) \
|
#define DEFINE_IMMEDIATE_5_DECODER_MEM_THUMB(NAME, IMMEDIATE, MNEMONIC, CYCLES) \
|
||||||
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
||||||
info->op1.reg = opcode & 0x0007; \
|
info->op1.reg = opcode & 0x0007; \
|
||||||
info->memory.baseReg = (opcode >> 3) & 0x0007; \
|
info->memory.baseReg = (opcode >> 3) & 0x0007; \
|
||||||
|
@ -35,7 +35,14 @@
|
||||||
ARM_OPERAND_AFFECTED_1 | \
|
ARM_OPERAND_AFFECTED_1 | \
|
||||||
ARM_OPERAND_MEMORY_2; \
|
ARM_OPERAND_MEMORY_2; \
|
||||||
info->memory.format = ARM_MEMORY_REGISTER_BASE | \
|
info->memory.format = ARM_MEMORY_REGISTER_BASE | \
|
||||||
ARM_MEMORY_IMMEDIATE_OFFSET;)
|
ARM_MEMORY_IMMEDIATE_OFFSET; \
|
||||||
|
CYCLES)
|
||||||
|
|
||||||
|
#define DEFINE_IMMEDIATE_5_DECODER_MEM_LOAD_THUMB(NAME, IMMEDIATE, MNEMONIC) \
|
||||||
|
DEFINE_IMMEDIATE_5_DECODER_MEM_THUMB(NAME, IMMEDIATE, MNEMONIC, LOAD_CYCLES)
|
||||||
|
|
||||||
|
#define DEFINE_IMMEDIATE_5_DECODER_MEM_STORE_THUMB(NAME, IMMEDIATE, MNEMONIC) \
|
||||||
|
DEFINE_IMMEDIATE_5_DECODER_MEM_THUMB(NAME, IMMEDIATE, MNEMONIC, STORE_CYCLES)
|
||||||
|
|
||||||
#define DEFINE_IMMEDIATE_5_DECODER_THUMB(NAME, MNEMONIC, TYPE) \
|
#define DEFINE_IMMEDIATE_5_DECODER_THUMB(NAME, MNEMONIC, TYPE) \
|
||||||
COUNT_5(DEFINE_IMMEDIATE_5_DECODER_ ## TYPE ## _THUMB, NAME ## _, MNEMONIC)
|
COUNT_5(DEFINE_IMMEDIATE_5_DECODER_ ## TYPE ## _THUMB, NAME ## _, MNEMONIC)
|
||||||
|
@ -43,12 +50,12 @@
|
||||||
DEFINE_IMMEDIATE_5_DECODER_THUMB(LSL1, LSL, DATA)
|
DEFINE_IMMEDIATE_5_DECODER_THUMB(LSL1, LSL, DATA)
|
||||||
DEFINE_IMMEDIATE_5_DECODER_THUMB(LSR1, LSR, DATA)
|
DEFINE_IMMEDIATE_5_DECODER_THUMB(LSR1, LSR, DATA)
|
||||||
DEFINE_IMMEDIATE_5_DECODER_THUMB(ASR1, ASR, DATA)
|
DEFINE_IMMEDIATE_5_DECODER_THUMB(ASR1, ASR, DATA)
|
||||||
DEFINE_IMMEDIATE_5_DECODER_THUMB(LDR1, LDR, MEM)
|
DEFINE_IMMEDIATE_5_DECODER_THUMB(LDR1, LDR, MEM_LOAD)
|
||||||
DEFINE_IMMEDIATE_5_DECODER_THUMB(LDRB1, LDRB, MEM)
|
DEFINE_IMMEDIATE_5_DECODER_THUMB(LDRB1, LDRB, MEM_LOAD)
|
||||||
DEFINE_IMMEDIATE_5_DECODER_THUMB(LDRH1, LDRH, MEM)
|
DEFINE_IMMEDIATE_5_DECODER_THUMB(LDRH1, LDRH, MEM_LOAD)
|
||||||
DEFINE_IMMEDIATE_5_DECODER_THUMB(STR1, STR, MEM)
|
DEFINE_IMMEDIATE_5_DECODER_THUMB(STR1, STR, MEM_STORE)
|
||||||
DEFINE_IMMEDIATE_5_DECODER_THUMB(STRB1, STRB, MEM)
|
DEFINE_IMMEDIATE_5_DECODER_THUMB(STRB1, STRB, MEM_STORE)
|
||||||
DEFINE_IMMEDIATE_5_DECODER_THUMB(STRH1, STRH, MEM)
|
DEFINE_IMMEDIATE_5_DECODER_THUMB(STRH1, STRH, MEM_STORE)
|
||||||
|
|
||||||
#define DEFINE_DATA_FORM_1_DECODER_EX_THUMB(NAME, RM, MNEMONIC) \
|
#define DEFINE_DATA_FORM_1_DECODER_EX_THUMB(NAME, RM, MNEMONIC) \
|
||||||
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
||||||
|
@ -159,7 +166,7 @@ DEFINE_DECODER_WITH_HIGH_THUMB(MOV3, MOV, ARM_OPERAND_AFFECTED_1, 0)
|
||||||
ARM_OPERAND_REGISTER_2 | \
|
ARM_OPERAND_REGISTER_2 | \
|
||||||
ARM_OPERAND_IMMEDIATE_3;)
|
ARM_OPERAND_IMMEDIATE_3;)
|
||||||
|
|
||||||
#define DEFINE_IMMEDIATE_WITH_REGISTER_MEM_THUMB(NAME, RD, MNEMONIC, REG) \
|
#define DEFINE_IMMEDIATE_WITH_REGISTER_MEM_THUMB(NAME, RD, MNEMONIC, REG, CYCLES) \
|
||||||
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
||||||
info->op1.reg = RD; \
|
info->op1.reg = RD; \
|
||||||
info->memory.baseReg = REG; \
|
info->memory.baseReg = REG; \
|
||||||
|
@ -169,19 +176,26 @@ DEFINE_DECODER_WITH_HIGH_THUMB(MOV3, MOV, ARM_OPERAND_AFFECTED_1, 0)
|
||||||
ARM_OPERAND_AFFECTED_1 | \
|
ARM_OPERAND_AFFECTED_1 | \
|
||||||
ARM_OPERAND_MEMORY_2; \
|
ARM_OPERAND_MEMORY_2; \
|
||||||
info->memory.format = ARM_MEMORY_REGISTER_BASE | \
|
info->memory.format = ARM_MEMORY_REGISTER_BASE | \
|
||||||
ARM_MEMORY_IMMEDIATE_OFFSET;)
|
ARM_MEMORY_IMMEDIATE_OFFSET; \
|
||||||
|
CYCLES;)
|
||||||
|
|
||||||
|
#define DEFINE_IMMEDIATE_WITH_REGISTER_MEM_LOAD_THUMB(NAME, RD, MNEMONIC, REG) \
|
||||||
|
DEFINE_IMMEDIATE_WITH_REGISTER_MEM_THUMB(NAME, RD, MNEMONIC, REG, LOAD_CYCLES)
|
||||||
|
|
||||||
|
#define DEFINE_IMMEDIATE_WITH_REGISTER_MEM_STORE_THUMB(NAME, RD, MNEMONIC, REG) \
|
||||||
|
DEFINE_IMMEDIATE_WITH_REGISTER_MEM_THUMB(NAME, RD, MNEMONIC, REG, STORE_CYCLES)
|
||||||
|
|
||||||
#define DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(NAME, MNEMONIC, TYPE, REG) \
|
#define DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(NAME, MNEMONIC, TYPE, REG) \
|
||||||
COUNT_3(DEFINE_IMMEDIATE_WITH_REGISTER_ ## TYPE ## _THUMB, NAME ## _R, MNEMONIC, REG)
|
COUNT_3(DEFINE_IMMEDIATE_WITH_REGISTER_ ## TYPE ## _THUMB, NAME ## _R, MNEMONIC, REG)
|
||||||
|
|
||||||
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(LDR3, LDR, MEM, ARM_PC)
|
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(LDR3, LDR, MEM_LOAD, ARM_PC)
|
||||||
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(LDR4, LDR, MEM, ARM_SP)
|
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(LDR4, LDR, MEM_LOAD, ARM_SP)
|
||||||
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(STR3, STR, MEM, ARM_SP)
|
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(STR3, STR, MEM_STORE, ARM_SP)
|
||||||
|
|
||||||
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(ADD5, ADD, DATA, ARM_PC)
|
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(ADD5, ADD, DATA, ARM_PC)
|
||||||
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(ADD6, ADD, DATA, ARM_SP)
|
DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(ADD6, ADD, DATA, ARM_SP)
|
||||||
|
|
||||||
#define DEFINE_LOAD_STORE_WITH_REGISTER_EX_THUMB(NAME, RM, MNEMONIC) \
|
#define DEFINE_LOAD_STORE_WITH_REGISTER_EX_THUMB(NAME, RM, MNEMONIC, CYCLES) \
|
||||||
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
||||||
info->memory.offset.reg = RM; \
|
info->memory.offset.reg = RM; \
|
||||||
info->op1.reg = opcode & 0x0007; \
|
info->op1.reg = opcode & 0x0007; \
|
||||||
|
@ -190,19 +204,20 @@ DEFINE_IMMEDIATE_WITH_REGISTER_THUMB(ADD6, ADD, DATA, ARM_SP)
|
||||||
ARM_OPERAND_AFFECTED_1 | \
|
ARM_OPERAND_AFFECTED_1 | \
|
||||||
ARM_OPERAND_MEMORY_2; \
|
ARM_OPERAND_MEMORY_2; \
|
||||||
info->memory.format = ARM_MEMORY_REGISTER_BASE | \
|
info->memory.format = ARM_MEMORY_REGISTER_BASE | \
|
||||||
ARM_MEMORY_REGISTER_OFFSET;)
|
ARM_MEMORY_REGISTER_OFFSET; \
|
||||||
|
CYCLES;)
|
||||||
|
|
||||||
#define DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(NAME, MNEMONIC) \
|
#define DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(NAME, MNEMONIC, CYCLES) \
|
||||||
COUNT_3(DEFINE_LOAD_STORE_WITH_REGISTER_EX_THUMB, NAME ## _R, MNEMONIC)
|
COUNT_3(DEFINE_LOAD_STORE_WITH_REGISTER_EX_THUMB, NAME ## _R, MNEMONIC, CYCLES)
|
||||||
|
|
||||||
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDR2, LDR)
|
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDR2, LDR, LOAD_CYCLES)
|
||||||
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDRB2, LDRB)
|
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDRB2, LDRB, LOAD_CYCLES)
|
||||||
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDRH2, LDRH)
|
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDRH2, LDRH, LOAD_CYCLES)
|
||||||
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDRSB, LDRSB)
|
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDRSB, LDRSB, LOAD_CYCLES)
|
||||||
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDRSH, LDRSH)
|
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(LDRSH, LDRSH, LOAD_CYCLES)
|
||||||
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(STR2, STR)
|
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(STR2, STR, STORE_CYCLES)
|
||||||
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(STRB2, STRB)
|
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(STRB2, STRB, STORE_CYCLES)
|
||||||
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(STRH2, STRH)
|
DEFINE_LOAD_STORE_WITH_REGISTER_THUMB(STRH2, STRH, STORE_CYCLES)
|
||||||
|
|
||||||
#define DEFINE_LOAD_STORE_MULTIPLE_EX_THUMB(NAME, RN, MNEMONIC, SPECIAL_REG, ADDITIONAL_REG) \
|
#define DEFINE_LOAD_STORE_MULTIPLE_EX_THUMB(NAME, RN, MNEMONIC, SPECIAL_REG, ADDITIONAL_REG) \
|
||||||
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
DEFINE_THUMB_DECODER(NAME, MNEMONIC, \
|
||||||
|
@ -303,6 +318,12 @@ void _decodeThumb(uint16_t opcode, struct ThumbInstructionInfo* info) {
|
||||||
info->accessesSpecialRegisters = 0;
|
info->accessesSpecialRegisters = 0;
|
||||||
info->affectsCPSR = 0;
|
info->affectsCPSR = 0;
|
||||||
info->condition = ARM_CONDITION_AL;
|
info->condition = ARM_CONDITION_AL;
|
||||||
|
info->sDataCycles = 0;
|
||||||
|
info->nDataCycles = 0;
|
||||||
|
info->sInstructionCycles = 1;
|
||||||
|
info->nInstructionCycles = 0;
|
||||||
|
info->iCycles = 0;
|
||||||
|
info->cCycles = 0;
|
||||||
ThumbDecoder decoder = _thumbDecoderTable[opcode >> 6];
|
ThumbDecoder decoder = _thumbDecoderTable[opcode >> 6];
|
||||||
decoder(opcode, info);
|
decoder(opcode, info);
|
||||||
}
|
}
|
||||||
|
|
|
@ -139,6 +139,12 @@ struct ThumbInstructionInfo {
|
||||||
int accessesSpecialRegisters;
|
int accessesSpecialRegisters;
|
||||||
int affectsCPSR;
|
int affectsCPSR;
|
||||||
int condition;
|
int condition;
|
||||||
|
int sDataCycles;
|
||||||
|
int nDataCycles;
|
||||||
|
int sInstructionCycles;
|
||||||
|
int nInstructionCycles;
|
||||||
|
int iCycles;
|
||||||
|
int cCycles;
|
||||||
};
|
};
|
||||||
|
|
||||||
void ARMDecodeThumb(uint16_t opcode, struct ThumbInstructionInfo* info);
|
void ARMDecodeThumb(uint16_t opcode, struct ThumbInstructionInfo* info);
|
||||||
|
|
Loading…
Reference in New Issue