-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy patharm_instruction.c
183 lines (163 loc) · 4.98 KB
/
arm_instruction.c
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
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
/*
Armator - simulateur de jeu d'instruction ARMv5T � but p�dagogique
Copyright (C) 2011 Guillaume Huard
Ce programme est libre, vous pouvez le redistribuer et/ou le modifier selon les
termes de la Licence Publique G�n�rale GNU publi�e par la Free Software
Foundation (version 2 ou bien toute autre version ult�rieure choisie par vous).
Ce programme est distribu� car potentiellement utile, mais SANS AUCUNE
GARANTIE, ni explicite ni implicite, y compris les garanties de
commercialisation ou d'adaptation dans un but sp�cifique. Reportez-vous � la
Licence Publique G�n�rale GNU pour plus de d�tails.
Vous devez avoir re�u une copie de la Licence Publique G�n�rale GNU en m�me
temps que ce programme ; si ce n'est pas le cas, �crivez � la Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307,
�tats-Unis.
Contact: [email protected]
B�timent IMAG
700 avenue centrale, domaine universitaire
38401 Saint Martin d'H�res
*/
#include "arm_instruction.h"
#include "arm_exception.h"
#include "arm_data_processing.h"
#include "arm_load_store.h"
#include "arm_branch_other.h"
#include "arm_constants.h"
#include "util.h"
/*
La fonction arm_execute_instruction est la toute première partie de la simulation
d’un cycle d’exécution d’une instruction dans le processeur.
cette fonction devra contenir la partie commune à l’exécution de
toute instruction ARM :
DONE la lecture de la prochaine instruction,
DONE l’incrémentation du compteur ordinal
DONE l’interprétation du champ d’exécution conditionnelle.
DONE Ensuite, cette fonction devra sélectionner la classe d’instructions
concernée par l’instruction courante et
appeler la fonction appropriée, chargée de la suite du décodage.
*/
int condition_passed(arm_core p, uint32_t ins) {
uint8_t cond = get_bits(ins, 31, 28);
uint32_t cpsr = arm_read_cpsr(p);
int n = get_bit(cpsr, 31);
int z = get_bit(cpsr, 30);
int c = get_bit(cpsr, 29);
int v = get_bit(cpsr, 28);
switch (cond) {
case 0x0:
return z == 1;
case 0x1:
return z == 0;
case 0x2:
return c == 1;
case 0x3:
return c == 0;
case 0x4:
return n == 1;
case 0x5:
return n == 0;
case 0x6:
return v == 1;
case 0x7:
return v == 0;
case 0x8:
return c == 1 && z == 0;
case 0x9:
return c == 0 || z == 1;
case 0xA:
return n == v;
case 0xB:
return n != v;
case 0xC:
return z == 0 && n == v;
case 0xD:
return z == 1 || n != v;
case 0xF:
return 0xF;
default:
return 1;
}
}
static int arm_execute_instruction(arm_core p) {
uint32_t ins;
int exception = arm_fetch(p, & ins);
if (exception != 0)
return 1;
if (!condition_passed(p, ins))
return 1;
uint8_t opcode = get_bits(ins, 27, 25);
switch (opcode) {
case 0x0:
if (get_bit(ins, 4) == 0) {
if (get_bits(ins, 24, 23) == 2 && get_bit(ins, 20) == 0)
return arm_miscellaneous(p, ins);
else{
return arm_data_processing(p, ins);
}
} else {
if (get_bit(ins, 7) == 0) {
if (get_bits(ins, 24, 23) == 2 && get_bit(ins, 20) == 1)
return arm_miscellaneous(p, ins);
else{
return arm_data_processing(p, ins);
}
} else {
return arm_load_store(p, ins);
}
}
if (get_bits(ins, 24, 23) == 2 && get_bit(ins, 20) == 0){ // Instruction MRS
if (get_bit(ins, 20) == 1){
arm_write_register(p, get_bits(ins, 15, 12), arm_read_spsr(p));
return 0;
}
else{
arm_write_register(p, get_bits(ins, 15, 12), arm_read_cpsr(p));
return 0;
}
}
break;
case 0x1:
if ((get_bits(ins, 24, 23) == 2) && (get_bits(ins, 21, 20) == 2))
return arm_data_processing_immediate_msr(p, ins);
else {
return arm_data_processing(p, ins);
}
case 0x2:
return arm_load_store(p, ins);
case 0x3:
if (get_bit(ins, 4) == 0) {
return arm_load_store(p, ins);
} else {
if (get_bits(ins, 24, 20) == 0x1F && get_bits(ins, 7, 5) == 0x07) {
return UNDEFINED_INSTRUCTION;
} else {
return UNDEFINED_INSTRUCTION;
}
}
case 0x4:
return arm_load_store_multiple(p, ins);
case 0x5:
return arm_branch(p, ins);
case 0x6:
return arm_coprocessor_load_store(p, ins);
case 0x7:
if (get_bit(ins, 24) == 0) {
if (get_bit(ins, 4) == 0) {
return arm_data_processing(p, ins);
} else {
return UNDEFINED_INSTRUCTION;
}
} else {
return arm_coprocessor_others_swi(p, ins);
}
default:
return UNDEFINED_INSTRUCTION;
}
}
int arm_step(arm_core p) {
int result;
result = arm_execute_instruction(p);
if (result)
arm_exception(p, result);
return result;
}