701 lines
23 KiB
C
701 lines
23 KiB
C
/* This file is included into vbu.c and cannot be compiled on its own. */
|
|
#ifdef VBUAPI
|
|
|
|
|
|
|
|
/********************************* Constants *********************************/
|
|
|
|
/* Operand types */
|
|
#define DASM_BCOND 0
|
|
#define DASM_DISP9 1
|
|
#define DASM_DISP26 2
|
|
#define DASM_IMM5S 3
|
|
#define DASM_IMM5U 4
|
|
#define DASM_IMM16S 5
|
|
#define DASM_IMM16U 6
|
|
#define DASM_JMP 7
|
|
#define DASM_MEMORY 8
|
|
#define DASM_REG1 9
|
|
#define DASM_REG2 10
|
|
#define DASM_SETF 11
|
|
#define DASM_SYSTEM 12
|
|
|
|
|
|
|
|
/******************************** Lookup Data ********************************/
|
|
|
|
/* Instruction descriptor */
|
|
typedef struct {
|
|
char mnemonic[8]; /* Mnemonic, uppercase */
|
|
uint8_t operandsLength; /* Number of operands */
|
|
uint8_t operands[3]; /* Operand types */
|
|
} OpDef;
|
|
|
|
/* Instruction descriptors */
|
|
static const OpDef OPDEFS[] = {
|
|
{ "MOV" , 2, { DASM_REG1, DASM_REG2 } }, /* 000000 */
|
|
{ "ADD" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "SUB" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "CMP" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "SHL" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "SHR" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "JMP" , 1, { DASM_JMP } },
|
|
{ "SAR" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "MUL" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "DIV" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "MULU" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "DIVU" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "OR" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "AND" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "XOR" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "NOT" , 1, { DASM_REG1 } },
|
|
|
|
{ "MOV" , 2, { DASM_IMM5S, DASM_REG2 } }, /* 010000 */
|
|
{ "ADD" , 2, { DASM_IMM5S, DASM_REG2 } },
|
|
{ "" , 0, { 0 } }, /* SETF */
|
|
{ "CMP" , 2, { DASM_IMM5S, DASM_REG2 } },
|
|
{ "SHL" , 2, { DASM_IMM5U, DASM_REG2 } },
|
|
{ "SHR" , 2, { DASM_IMM5U, DASM_REG2 } },
|
|
{ "CLI" , 0, { 0 } },
|
|
{ "SAR" , 2, { DASM_IMM5U, DASM_REG2 } },
|
|
{ "TRAP" , 1, { DASM_IMM5U } },
|
|
{ "RETI" , 0, { 0 } },
|
|
{ "HALT" , 0, { 0 } },
|
|
{ "---" , 0, { 0 } },
|
|
{ "LDSR" , 2, { DASM_REG2, DASM_SYSTEM } },
|
|
{ "STSR" , 2, { DASM_SYSTEM, DASM_REG2 } },
|
|
{ "SEI" , 0, { 0 } },
|
|
{ "" , 0, { 0 } }, /* Bit string */
|
|
|
|
{ "" , 0, { 0 } }, /* BCOND */ /* 100000 */
|
|
{ "" , 0, { 0 } }, /* BCOND */
|
|
{ "" , 0, { 0 } }, /* BCOND */
|
|
{ "" , 0, { 0 } }, /* BCOND */
|
|
{ "" , 0, { 0 } }, /* BCOND */
|
|
{ "" , 0, { 0 } }, /* BCOND */
|
|
{ "" , 0, { 0 } }, /* BCOND */
|
|
{ "" , 0, { 0 } }, /* BCOND */
|
|
{ "MOVEA", 3, { DASM_IMM16U, DASM_REG1, DASM_REG2 } },
|
|
{ "ADDI" , 3, { DASM_IMM16S, DASM_REG1, DASM_REG2 } },
|
|
{ "JR" , 1, { DASM_DISP26 } },
|
|
{ "JAL" , 1, { DASM_DISP26 } },
|
|
{ "ORI" , 3, { DASM_IMM16U, DASM_REG1, DASM_REG2 } },
|
|
{ "ANDI" , 3, { DASM_IMM16U, DASM_REG1, DASM_REG2 } },
|
|
{ "XORI" , 3, { DASM_IMM16U, DASM_REG1, DASM_REG2 } },
|
|
{ "MOVHI", 3, { DASM_IMM16U, DASM_REG1, DASM_REG2 } },
|
|
|
|
{ "LD.B" , 2, { DASM_MEMORY, DASM_REG2 } }, /* 110000 */
|
|
{ "LD.H" , 2, { DASM_MEMORY, DASM_REG2 } },
|
|
{ "---" , 0, { 0 } },
|
|
{ "LD.W" , 2, { DASM_MEMORY, DASM_REG2 } },
|
|
{ "ST.B" , 2, { DASM_REG2, DASM_MEMORY } },
|
|
{ "ST.H" , 2, { DASM_REG2, DASM_MEMORY } },
|
|
{ "---" , 0, { 0 } },
|
|
{ "ST.W" , 2, { DASM_REG2, DASM_MEMORY } },
|
|
{ "IN.B" , 2, { DASM_MEMORY, DASM_REG2 } },
|
|
{ "IN.H" , 2, { DASM_MEMORY, DASM_REG2 } },
|
|
{ "CAXI" , 0, { DASM_MEMORY, DASM_REG2 } },
|
|
{ "IN.W" , 2, { DASM_MEMORY, DASM_REG2 } },
|
|
{ "OUT.B", 2, { DASM_REG2, DASM_MEMORY } },
|
|
{ "OUT.H", 2, { DASM_REG2, DASM_MEMORY } },
|
|
{ "" , 0, { 0 } }, /* Floating-point/Nintendo */
|
|
{ "OUT.W", 2, { DASM_REG2, DASM_MEMORY } }
|
|
};
|
|
|
|
/* Bit string instruction descriptors */
|
|
static const OpDef OPDEFS_BITSTRING[] = {
|
|
{ "SCH0BSU", 0, { 0 } }, /* 00000 */
|
|
{ "SCH0BSD", 0, { 0 } },
|
|
{ "SCH1BSU", 0, { 0 } },
|
|
{ "SCH1BSD", 0, { 0 } },
|
|
{ "---" , 0, { 0 } },
|
|
{ "---" , 0, { 0 } },
|
|
{ "---" , 0, { 0 } },
|
|
{ "---" , 0, { 0 } },
|
|
{ "ORBSU" , 0, { 0 } },
|
|
{ "ANDBSU" , 0, { 0 } },
|
|
{ "XORBSU" , 0, { 0 } },
|
|
{ "MOVBSU" , 0, { 0 } },
|
|
{ "ORNBSU" , 0, { 0 } },
|
|
{ "ANDNBSU", 0, { 0 } },
|
|
{ "XORNBSU", 0, { 0 } },
|
|
{ "NOTBSU" , 0, { 0 } }
|
|
/* +16 invalid opcodes */
|
|
};
|
|
|
|
/* Floating-point/Nintendo instruction descriptors */
|
|
static const OpDef OPDEFS_FLOATENDO[] = {
|
|
{ "CMPF.S" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "---" , 0, { 0 } },
|
|
{ "CVT.WS" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "CVT.SW" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "ADDF.S" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "SUBF.S" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "MULF.S" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "DIVF.S" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "XB" , 1, { DASM_REG2 } },
|
|
{ "XH" , 1, { DASM_REG2 } },
|
|
{ "REV" , 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "TRNC.SW", 2, { DASM_REG1, DASM_REG2 } },
|
|
{ "MPYHW" , 2, { DASM_REG1, DASM_REG2 } },
|
|
/* +51 invalid opcodes */
|
|
};
|
|
|
|
/* Invalid instruction descriptors */
|
|
static const OpDef OPDEF_ILLEGAL = { "---", 0, { 0 } };
|
|
|
|
|
|
|
|
/***************************** Module Functions ******************************/
|
|
|
|
/* Select the display text for a condition */
|
|
static char* dasmCondition(VBU_DasmConfig *config, int cond) {
|
|
switch (cond) {
|
|
case 0: return "V" ;
|
|
case 1: return config->conditionCL == VBU_C ? "C" : "L";
|
|
case 2: return config->conditionEZ == VBU_E ? "E" : "Z";
|
|
case 3: return "NH";
|
|
case 4: return "N" ;
|
|
case 5: return "T" ;
|
|
case 6: return "LT";
|
|
case 7: return "LE";
|
|
case 8: return "NV";
|
|
case 9: return config->conditionCL == VBU_C ? "NC" : "NL";
|
|
case 10: return config->conditionEZ == VBU_E ? "NE" : "NZ";
|
|
case 11: return "H" ;
|
|
case 12: return "P" ;
|
|
case 13: return "F" ;
|
|
case 14: return "GE";
|
|
}
|
|
return "GT";
|
|
}
|
|
|
|
/* Format a number as hexadecimal */
|
|
static void dasmToHex(char *dest, VBU_DasmConfig *config,
|
|
int minDigits, int32_t value) {
|
|
char format[9];
|
|
char *prefix = "";
|
|
char *suffix = "";
|
|
switch (config->hexNotation) {
|
|
case VBU_0X : prefix = "0x"; break;
|
|
case VBU_DOLLAR: prefix = "$" ; break;
|
|
case VBU_H : suffix = "h" ; break;
|
|
}
|
|
sprintf(format, "%%s%%0%d%s%%s", minDigits,
|
|
config->hexCase == VBU_LOWER ? "x" : "X");
|
|
sprintf(dest, format, prefix, value, suffix);
|
|
if (config->hexNotation != VBU_H || *dest <= '9')
|
|
return;
|
|
*dest++ = '0';
|
|
sprintf(dest, format, prefix, value, suffix);
|
|
}
|
|
|
|
/* Convert a string to lowercase */
|
|
static void dasmToLower(char *str) {
|
|
for (; *str; str++) {
|
|
if (*str >= 'A' && *str <= 'Z')
|
|
*str += 'a' - 'A';
|
|
}
|
|
}
|
|
|
|
/* Format a program register */
|
|
static void dasmProgram(char *dest, VBU_DasmConfig *config, int id) {
|
|
char *text = NULL;
|
|
if (config->programNotation != VBU_NUMBERS) {
|
|
switch (id) {
|
|
case 2: text = "HP"; break;
|
|
case 3: text = "SP"; break;
|
|
case 4: text = "GP"; break;
|
|
case 5: text = "TP"; break;
|
|
case 31: text = "LP"; break;
|
|
}
|
|
}
|
|
if (text == NULL)
|
|
sprintf(dest, "R%d", id);
|
|
else strcpy(dest, text);
|
|
if (config->programCase == VBU_LOWER)
|
|
dasmToLower(dest);
|
|
}
|
|
|
|
|
|
|
|
/**************************** Operand Formatters *****************************/
|
|
|
|
/* Format the condition operand of a split BCOND instruction */
|
|
static void dasmOpBCOND(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
int cond = line->code[1] >> 2 & 15;
|
|
if (config->conditionNotation == VBU_NUMBERS)
|
|
sprintf(dest, "%d", cond);
|
|
else {
|
|
strcpy(dest, dasmCondition(config, cond));
|
|
if (config->conditionCase == VBU_LOWER)
|
|
dasmToLower(dest);
|
|
}
|
|
}
|
|
|
|
/* Format a 9-bit displacement operand */
|
|
static void dasmOpDisp9(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
int32_t disp = SignExtend((int32_t) line->code[1] << 8 | line->code[0], 9);
|
|
sprintf(dest, config->hexCase == VBU_LOWER ?
|
|
"%08x" : "%08X", line->address + disp);
|
|
}
|
|
|
|
/* Format a 26-bit displacement operand */
|
|
static void dasmOpDisp26(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
int32_t disp = SignExtend(
|
|
(int32_t) line->code[1] << 24 |
|
|
(int32_t) line->code[0] << 16 |
|
|
(int32_t) line->code[3] << 8 |
|
|
line->code[2]
|
|
, 26);
|
|
sprintf(dest, config->hexCase == VBU_LOWER ?
|
|
"%08x" : "%08X", line->address + disp);
|
|
}
|
|
|
|
/* Format a 5-bit sign-extended immediate operand */
|
|
static void dasmOpImm5S(char *dest, VBU_DasmConfig*config, VBU_DasmLine *line){
|
|
if (config->immediateNotation == VBU_NUMBER)
|
|
*dest++ = '#';
|
|
sprintf(dest, "%d", SignExtend(line->code[0], 5));
|
|
}
|
|
|
|
/* Format a 5-bit zero-filled immediate operand */
|
|
static void dasmOpImm5U(char *dest, VBU_DasmConfig*config, VBU_DasmLine *line){
|
|
if (config->immediateNotation == VBU_NUMBER)
|
|
*dest++ = '#';
|
|
sprintf(dest, "%d", line->code[0] & 31);
|
|
}
|
|
|
|
/* Format a 16-bit sign-extended immediate operand */
|
|
static void dasmOpImm16S(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
int32_t imm = (int16_t) ((int16_t) line->code[3] << 8 | line->code[2]);
|
|
if (config->immediateNotation == VBU_NUMBER)
|
|
*dest++ = '#';
|
|
if (imm >= -256 && imm <= 256) {
|
|
sprintf(dest, "%d", imm);
|
|
return;
|
|
}
|
|
if (imm < 0) {
|
|
*dest++ = '-';
|
|
imm = -imm;
|
|
}
|
|
dasmToHex(dest, config, 0, imm);
|
|
}
|
|
|
|
/* Format a 16-bit zero-filled immediate operand */
|
|
static void dasmOpImm16U(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
uint16_t imm = (uint16_t) line->code[3] << 8 | line->code[2];
|
|
if (config->immediateNotation == VBU_NUMBER)
|
|
*dest++ = '#';
|
|
dasmToHex(dest, config, 4, imm);
|
|
}
|
|
|
|
/* Format the operand of a JMP instruction */
|
|
static void dasmOpJMP(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
*dest++ = '[';
|
|
dasmProgram(dest, config, line->code[0] & 31);
|
|
strcat(dest, "]");
|
|
}
|
|
|
|
/* Format a memory operand */
|
|
static void dasmOpMemory(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
int reg1 = line->code[0] & 31;
|
|
int32_t disp = (int16_t) ((int16_t) line->code[3] << 8 | line->code[2]);
|
|
|
|
/* Displacement of zero */
|
|
if (disp == 0) {
|
|
*dest++ = '[';
|
|
dasmProgram(dest, config, reg1);
|
|
strcat(dest, "]");
|
|
return;
|
|
}
|
|
|
|
/* Inside */
|
|
if (config->memoryNotation == VBU_INSIDE) {
|
|
*dest++ = '[';
|
|
dasmProgram(dest, config, reg1);
|
|
dest += strlen(dest);
|
|
if (disp < 0) {
|
|
strcpy(dest, " - ");
|
|
disp = -disp;
|
|
} else strcpy(dest, " + ");
|
|
dest += 3;
|
|
if (disp <= 256)
|
|
sprintf(dest, "%d", disp);
|
|
else dasmToHex(dest, config, 0, disp);
|
|
strcat(dest, "]");
|
|
}
|
|
|
|
/* Outside */
|
|
else {
|
|
if (disp < 0) {
|
|
*dest++ = '-';
|
|
disp = -disp;
|
|
}
|
|
if (disp <= 256)
|
|
sprintf(dest, "%d", disp);
|
|
else dasmToHex(dest, config, 0, disp);
|
|
dest += strlen(dest);
|
|
*dest++ = '[';
|
|
dasmProgram(dest, config, reg1);
|
|
strcat(dest, "]");
|
|
}
|
|
|
|
}
|
|
|
|
/* Format a source register operand */
|
|
static void dasmOpReg1(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
dasmProgram(dest, config, line->code[0] & 31);
|
|
}
|
|
|
|
/* Format a destination register operand */
|
|
static void dasmOpReg2(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
dasmProgram(dest, config,
|
|
((uint16_t) line->code[1] << 8 | line->code[0]) >> 5 & 31);
|
|
}
|
|
|
|
/* Format the condition operand of a split SETF instruction */
|
|
static void dasmOpSETF(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
int cond = line->code[0] & 15;
|
|
if (config->conditionNotation == VBU_NUMBERS)
|
|
sprintf(dest, "%d", cond);
|
|
else {
|
|
strcpy(dest, dasmCondition(config, cond));
|
|
if (config->conditionCase == VBU_LOWER)
|
|
dasmToLower(dest);
|
|
}
|
|
}
|
|
|
|
/* Format a system register operand */
|
|
static void dasmOpSystem(char*dest, VBU_DasmConfig*config, VBU_DasmLine*line) {
|
|
int id = line->code[0] & 31;
|
|
char *text = NULL;
|
|
if (config->systemNotation != VBU_NUMBERS) {
|
|
switch (id) {
|
|
case 0: text = "EIPC" ; break;
|
|
case 1: text = "EIPSW"; break;
|
|
case 2: text = "FEPC" ; break;
|
|
case 3: text = "FEPSW"; break;
|
|
case 4: text = "ECR" ; break;
|
|
case 5: text = "PSW" ; break;
|
|
case 6: text = "PIR" ; break;
|
|
case 7: text = "TKCW" ; break;
|
|
case 24: text = "CHCW" ; break;
|
|
case 25: text = "ADTRE"; break;
|
|
}
|
|
}
|
|
if (text == NULL) {
|
|
sprintf(dest, "%d", id);
|
|
return;
|
|
}
|
|
strcpy(dest, text);
|
|
if (config->systemCase == VBU_LOWER)
|
|
dasmToLower(dest);
|
|
}
|
|
|
|
|
|
|
|
/***************************** Module Functions ******************************/
|
|
|
|
/* Prepare an OpDef for a BCOND instruction */
|
|
static OpDef* dasmBCOND(VBU_DasmConfig *config,
|
|
VBU_DasmLine *line, OpDef *opdef) {
|
|
int cond; /* Condition code */
|
|
char *text; /* Mnemonic text */
|
|
|
|
/* Split notation */
|
|
if (config->bcondNotation == VBU_SPLIT) {
|
|
strcpy(opdef->mnemonic, "BCOND");
|
|
opdef->operandsLength = 2;
|
|
opdef->operands[0] = DASM_BCOND;
|
|
opdef->operands[1] = DASM_DISP9;
|
|
return opdef;
|
|
}
|
|
|
|
/* Joined notation */
|
|
cond = line->code[1] >> 1 & 15;
|
|
switch (cond) {
|
|
case 0: text = "BV" ; break;
|
|
case 1: text = config->conditionCL == VBU_C ? "BC" : "BL"; break;
|
|
case 2: text = config->conditionEZ == VBU_E ? "BE" : "BZ"; break;
|
|
case 3: text = "BNH"; break;
|
|
case 4: text = "BN" ; break;
|
|
case 5: text = "BR" ; break;
|
|
case 6: text = "BLT"; break;
|
|
case 7: text = "BLE"; break;
|
|
case 8: text = "BNV"; break;
|
|
case 9: text = config->conditionCL == VBU_C ? "BNC" : "BNL"; break;
|
|
case 10: text = config->conditionEZ == VBU_E ? "BNE" : "BNZ"; break;
|
|
case 11: text = "BH" ; break;
|
|
case 12: text = "BP" ; break;
|
|
case 13: text = "NOP"; break;
|
|
case 14: text = "BGE"; break;
|
|
default: text = "BGT";
|
|
}
|
|
strcpy(opdef->mnemonic, text);
|
|
opdef->operandsLength = cond != 13;
|
|
opdef->operands[0] = DASM_DISP9;
|
|
return opdef;
|
|
}
|
|
|
|
/* Ensure an output buffer matches the required size */
|
|
static VBU_DasmLine* dasmGrow(
|
|
VBU_DasmLine **lines, size_t *size, size_t target, unsigned index) {
|
|
if (*size < target) {
|
|
while (*size < target)
|
|
*size *= 2;
|
|
*lines = VBU_REALLOC(*lines, *size);
|
|
}
|
|
return *lines == NULL ? NULL : &(*lines)[index];
|
|
}
|
|
|
|
/* Format an operand */
|
|
static void dasmOperand(char *dest, VBU_DasmConfig *config,
|
|
VBU_DasmLine *line, uint8_t type) {
|
|
switch (type) {
|
|
case DASM_BCOND : dasmOpBCOND (dest, config, line); break;
|
|
case DASM_DISP9 : dasmOpDisp9 (dest, config, line); break;
|
|
case DASM_DISP26: dasmOpDisp26(dest, config, line); break;
|
|
case DASM_IMM5S : dasmOpImm5S (dest, config, line); break;
|
|
case DASM_IMM5U : dasmOpImm5U (dest, config, line); break;
|
|
case DASM_IMM16S: dasmOpImm16S(dest, config, line); break;
|
|
case DASM_IMM16U: dasmOpImm16U(dest, config, line); break;
|
|
case DASM_JMP : dasmOpJMP (dest, config, line); break;
|
|
case DASM_MEMORY: dasmOpMemory(dest, config, line); break;
|
|
case DASM_REG1 : dasmOpReg1 (dest, config, line); break;
|
|
case DASM_REG2 : dasmOpReg2 (dest, config, line); break;
|
|
case DASM_SETF : dasmOpSETF (dest, config, line); break;
|
|
case DASM_SYSTEM: dasmOpSystem(dest, config, line); break;
|
|
}
|
|
}
|
|
|
|
/* Prepare an OpDef for a SETF instruction */
|
|
static OpDef* dasmSETF(VBU_DasmConfig *config,
|
|
VBU_DasmLine *line, OpDef *opdef) {
|
|
|
|
/* Split notation */
|
|
if (config->bcondNotation == VBU_SPLIT) {
|
|
strcpy(opdef->mnemonic, "SETF");
|
|
opdef->operandsLength = 2;
|
|
opdef->operands[0] = DASM_SETF;
|
|
opdef->operands[1] = DASM_REG2;
|
|
return opdef;
|
|
}
|
|
|
|
/* Joined notation */
|
|
sprintf(opdef->mnemonic, "SETF%s",
|
|
dasmCondition(config, line->code[1] >> 1 & 15));
|
|
opdef->operandsLength = 1;
|
|
opdef->operands[0] = DASM_REG2;
|
|
return opdef;
|
|
}
|
|
|
|
/* Disassemble one instruction */
|
|
static int dasmLine(VB *sim, uint32_t *address, uint32_t pc,
|
|
VBU_DasmConfig *config, VBU_DasmLine **lines, size_t *size, size_t *offset,
|
|
unsigned index) {
|
|
char *dest; /* Text output pointer */
|
|
char *format; /* Hexadecimal format string */
|
|
VBU_DasmLine *line; /* Output line */
|
|
size_t more; /* Number of bytes to output */
|
|
OpDef *opdef; /* Instruction descriptor */
|
|
OpDef special; /* Special-case descriptor */
|
|
char tAddress[9]; /* Address display text */
|
|
char tCode[4][3]; /* Code display text */
|
|
char tMnemonic[8]; /* Mnemonic display text */
|
|
char tOperands[3][15]; /* Operands display text */
|
|
uint32_t x; /* Scratch and iterator */
|
|
|
|
/* Process non-text members */
|
|
line = &(*lines)[index];
|
|
line->address = *address;
|
|
line->codeLength = vbuCodeSize(sim, *address);
|
|
line->isPC = *address == pc;
|
|
for (x = 0; x < line->codeLength; x++)
|
|
line->code[x] = vbRead(sim, *address + x, VB_U8);
|
|
|
|
/* Advance to the next instruction or PC, whichever is sooner */
|
|
*address += pc != *address && pc - *address < line->codeLength ?
|
|
pc - *address : line->codeLength;
|
|
|
|
/* Do not process text members */
|
|
if (config == NULL) {
|
|
line->text.address = NULL;
|
|
line->text.mnemonic = NULL;
|
|
line->text.operandsLength = 0;
|
|
for (x = 0; x < line->codeLength; x++)
|
|
line->text.code[x] = NULL;
|
|
return 0;
|
|
}
|
|
|
|
/* Select instruction descriptor */
|
|
x = line->code[1] >> 2 & 63;
|
|
switch (x) {
|
|
case 0x12: /* SETF */
|
|
opdef = dasmSETF(config, line, &special);
|
|
break;
|
|
case 0x1F: /* Bit string */
|
|
x = line->code[0] & 31;
|
|
opdef = (OpDef *) (x < 16 ? &OPDEFS_BITSTRING[x] : &OPDEF_ILLEGAL);
|
|
break;
|
|
case 0x20: case 0x21: case 0x22: case 0x23: /* BCOND */
|
|
case 0x24: case 0x25: case 0x26: case 0x27:
|
|
opdef = dasmBCOND(config, line, &special);
|
|
break;
|
|
case 0x3E: /* Floating-point/Nintendo */
|
|
x = line->code[3] >> 2 & 63;
|
|
opdef = (OpDef *) (x < 13 ? &OPDEFS_FLOATENDO[x] : &OPDEF_ILLEGAL);
|
|
break;
|
|
default: opdef = (OpDef *) &OPDEFS[x]; /* Other */
|
|
}
|
|
|
|
/* Address */
|
|
format = config->hexCase == VBU_LOWER ? "%08x" : "%08X";
|
|
sprintf(tAddress, format, line->address);
|
|
more = 9;
|
|
|
|
/* Code */
|
|
format = config->hexCase == VBU_LOWER ? "%02x" : "%02X";
|
|
for (x = 0; x < line->codeLength; x++) {
|
|
sprintf(tCode[x], format, (int) line->code[x]);
|
|
more += 3;
|
|
}
|
|
|
|
/* Mnemonic */
|
|
strcpy(tMnemonic, opdef->mnemonic);
|
|
if (config->mnemonicCase == VBU_LOWER)
|
|
dasmToLower(tMnemonic);
|
|
more += strlen(tMnemonic) + 1;
|
|
|
|
/* Operands */
|
|
line->text.operandsLength = opdef->operandsLength;
|
|
for (x = 0; x < opdef->operandsLength; x++) {
|
|
dasmOperand(tOperands[x], config, line, opdef->operands[x]);
|
|
more += strlen(tOperands[x]) + 1;
|
|
}
|
|
|
|
/* Grow the text buffer */
|
|
line = dasmGrow(lines, size, *offset + more, index);
|
|
if (line == NULL)
|
|
return 1;
|
|
|
|
/* Working variables */
|
|
dest = &((char *) *lines)[*offset];
|
|
*offset += more;
|
|
|
|
/* Address */
|
|
strcpy(dest, tAddress);
|
|
line->text.address = dest;
|
|
dest += 9;
|
|
|
|
/* Code */
|
|
for (x = 0; x < line->codeLength; x++) {
|
|
strcpy(dest, tCode[x]);
|
|
line->text.code[x] = dest;
|
|
dest += 3;
|
|
}
|
|
|
|
/* Mnemonic */
|
|
strcpy(dest, tMnemonic);
|
|
line->text.mnemonic = dest;
|
|
dest += strlen(dest) + 1;
|
|
|
|
/* Operands */
|
|
for (x = 0; x < opdef->operandsLength; x++) {
|
|
strcpy(dest, tOperands[
|
|
config->operandOrder == VBU_DEST_FIRST ?
|
|
opdef->operandsLength - 1 - x :
|
|
x
|
|
]);
|
|
line->text.operands[x] = dest;
|
|
dest += strlen(dest) + 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
|
|
|
|
/***************************** Library Functions *****************************/
|
|
|
|
/* Disassemble from a simulation */
|
|
static VBU_DasmLine* dasmDisassemble(VB *sim, uint32_t address,
|
|
VBU_DasmConfig *config, unsigned length, int line) {
|
|
uint32_t addr; /* Working address */
|
|
uint32_t *circle = NULL; /* Circular address buffer */
|
|
VBU_DasmLine *lines = NULL; /* Output line buffer */
|
|
size_t offset; /* Offset into lines buffer */
|
|
uint32_t pc; /* Program counter */
|
|
size_t size; /* Number of bytes in lines */
|
|
unsigned x; /* Iterator */
|
|
|
|
/* Nothing to disassemble */
|
|
if (length == 0)
|
|
return NULL;
|
|
|
|
/* Establish a circular buffer */
|
|
if (line > 0) {
|
|
circle = VBU_REALLOC(NULL, (size_t) (line + 1) * sizeof (uint32_t));
|
|
if (circle == NULL)
|
|
goto catch;
|
|
}
|
|
|
|
/* Begin decoding from at least 10 lines before first/reference */
|
|
addr = (address & 0xFFFFFFFE) -
|
|
(int32_t) (line <= 0 ? 10 : line + 10) * 4;
|
|
|
|
/* Locate the address of the line containing the reference address */
|
|
pc = vbGetProgramCounter(sim);
|
|
for (x = 0;;) {
|
|
|
|
/* Track the address in the circular buffer */
|
|
if (line > 0) {
|
|
circle[x] = addr;
|
|
if (++x > (size_t) line)
|
|
x = 0;
|
|
}
|
|
|
|
/* Check if the instruction contains the reference address */
|
|
size = vbuCodeSize(sim, addr);
|
|
if (address - addr < size)
|
|
break;
|
|
|
|
/* Advance to the next instruction or PC, whichever is sooner */
|
|
addr += addr != pc && pc - addr < size ? pc - addr : size;
|
|
}
|
|
|
|
/* Address of first line is in the circular buffer */
|
|
if (line > 0) {
|
|
addr = circle[x];
|
|
circle = VBU_REALLOC(circle, 0);
|
|
}
|
|
|
|
/* Keep decoding until the first line of output */
|
|
else for (; line < 0; line++)
|
|
addr += vbuCodeSize(sim, addr);
|
|
|
|
/* Working variables */
|
|
size = length * sizeof (VBU_DasmLine);
|
|
lines = VBU_REALLOC(NULL, size);
|
|
offset = size;
|
|
if (lines == NULL)
|
|
goto catch;
|
|
|
|
/* Process output lines */
|
|
for (x = 0; x < length; x++) {
|
|
if (dasmLine(sim, &addr, pc, config, &lines, &size, &offset, x))
|
|
goto catch;
|
|
}
|
|
return VBU_REALLOC(lines, offset);
|
|
|
|
/* Exception handler */
|
|
catch:
|
|
circle = VBU_REALLOC(circle, 0);
|
|
lines = VBU_REALLOC(lines , 0);
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
#endif /* VBUAPI */
|