-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsnes_instructions.cpp
115 lines (102 loc) · 3.46 KB
/
snes_instructions.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
#include "p4lib/traits.h"
#include "snes_instructions.h"
#include "p4lib/print.h"
snes_line get_directive(const string &directive)
{
snes_line line;
line.type = DIRECTIVE;
line.directive() = directive;
return line;
}
snes_line get_width_directive(asm_register_width width)
{
snes_line line;
line.type = DIRECTIVE;
line.directive() = ".width";
line.create_operand().width = width;
return line;
}
snes_line get_instruction(const string &instruction)
{
snes_line line;
line.type = INSTRUCTION;
line.mnemonic() = instruction;
return line;
}
template <typename T>
snes_line get_instruction(const string &instruction, asm_operand_type type, T value)
{
snes_line line = get_instruction(instruction);
add_operand(line, type, value);
return line;
}
template snes_line get_instruction<int>(const string &instruction, asm_operand_type type, int value);
template snes_line get_instruction<unsigned int>(const string &instruction, asm_operand_type type, unsigned int value);
template snes_line get_instruction<string>(const string &instruction, asm_operand_type type, string value);
template <typename T>
snes_operand &add_operand(snes_line &line, asm_operand_type type, T value)
{
int operand = line.operands.length();
line.create_operand().type = type;
if constexpr(is_integral_v<T>){
line.operand(operand).value = value;
}else if constexpr(is_same_v<string, T>){
line.operand(operand).label = value;
}
return line.operand(operand);
}
template snes_operand &add_operand<unsigned int>(snes_line &line, asm_operand_type type, unsigned int value);
template snes_operand &add_operand<string>(snes_line &line, asm_operand_type type, string value);
template <typename T>
snes_line get_memory_instruction(const string &instruction, snes_address_mode mode, T address)
{
snes_line line = get_instruction(instruction);
snes_operand &operand = add_operand(line, MEMORY, address);
operand.mode = mode;
//if we know the address is dp we should optimize it
//todo handle long addresses
if constexpr(is_integral_v<T>){
if(address < 0x100){
operand.width = BITS_8;
}else if(address < 0x10000){
operand.width = BITS_16;
}else{
operand.width = BITS_24;
}
}else{
//todo don't assume word length
operand.width = BITS_16;
}
return line;
}
template snes_line get_memory_instruction(const string &instruction, snes_address_mode mode, int address);
template snes_line get_memory_instruction(const string &instruction, snes_address_mode mode, unsigned int address);
template snes_line get_memory_instruction(const string &instruction, snes_address_mode mode, string address);
template <typename T>
snes_line get_pc_instruction(const string &instruction, snes_address_mode mode, T address)
{
snes_line line = get_instruction(instruction);
snes_operand &operand = add_operand(line, CONSTANT, address);
operand.mode = mode;
if constexpr(is_integral_v<T>){
if(mode != RELATIVE && address < 0x10000){
operand.width = BITS_16;
}else if(mode != RELATIVE){
operand.width = BITS_24;
}else{
operand.width = BITS_8;
}
}else{
//assumes no brl for now
if(instruction[0] == 'b'){
operand.width = BITS_8;
}else if(instruction[2] == 'r' || instruction[2] == 'p'){ //jsr/jmp
operand.width = BITS_16;
}else{ //jsl
operand.width = BITS_24;
}
}
return line;
}
template snes_line get_pc_instruction(const string &instruction, snes_address_mode mode, unsigned int address);
template snes_line get_pc_instruction(const string &instruction, snes_address_mode mode, string address);