Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(22)

Side by Side Diff: runtime/vm/simulator_mips.cc

Issue 12541003: - Add a skeleton MIPS assembler, disassembler and simulator. (Closed) Base URL: http://dart.googlecode.com/svn/branches/bleeding_edge/dart/
Patch Set: Created 7 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file 1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file
2 // for details. All rights reserved. Use of this source code is governed by a 2 // for details. All rights reserved. Use of this source code is governed by a
3 // BSD-style license that can be found in the LICENSE file. 3 // BSD-style license that can be found in the LICENSE file.
4 4
5 #include "vm/globals.h" 5 #include "vm/globals.h"
6 #if defined(TARGET_ARCH_MIPS) 6 #if defined(TARGET_ARCH_MIPS)
7 7
8 // Only build the simulator if not compiling for real MIPS hardware. 8 // Only build the simulator if not compiling for real MIPS hardware.
9 #if !defined(HOST_ARCH_MIPS) 9 #if !defined(HOST_ARCH_MIPS)
10 10
11 #include "vm/simulator.h" 11 #include "vm/simulator.h"
12 12
13 #include "vm/assembler.h" 13 #include "vm/assembler.h"
14 #include "vm/constants_mips.h" 14 #include "vm/constants_mips.h"
15 #include "vm/disassembler.h" 15 #include "vm/disassembler.h"
16 16
17 namespace dart { 17 namespace dart {
18 18
19 DEFINE_FLAG(int, stop_sim_at, 0, "Address to stop simulator at.");
20
21 void Simulator::InitOnce() {
22 }
23
24
19 Simulator::Simulator() { 25 Simulator::Simulator() {
20 // Setup simulator support first. Some of this information is needed to 26 // Setup simulator support first. Some of this information is needed to
21 // setup the architecture state. 27 // setup the architecture state.
22 // We allocate the stack here, the size is computed as the sum of 28 // We allocate the stack here, the size is computed as the sum of
23 // the size specified by the user and the buffer space needed for 29 // the size specified by the user and the buffer space needed for
24 // handling stack overflow exceptions. To be safe in potential 30 // handling stack overflow exceptions. To be safe in potential
25 // stack underflows we also add some underflow buffer space. 31 // stack underflows we also add some underflow buffer space.
26 stack_ = new char[(Isolate::GetSpecifiedStackSize() + 32 stack_ = new char[(Isolate::GetSpecifiedStackSize() +
27 Isolate::kStackSizeBuffer + 33 Isolate::kStackSizeBuffer +
28 kSimulatorStackUnderflowSize)]; 34 kSimulatorStackUnderflowSize)];
35 icount_ = 0;
36 delay_slot_ = false;
37
38 // Setup architecture state.
39 // All registers are initialized to zero to start with.
40 for (int i = 0; i < kNumberOfCpuRegisters; i++) {
41 registers_[i] = 0;
42 }
43 pc_ = 0;
44 // The sp is initialized to point to the bottom (high address) of the
45 // allocated stack area.
46 registers_[SP] = StackTop();
29 } 47 }
30 48
31 49
32 Simulator::~Simulator() { 50 Simulator::~Simulator() {
33 Isolate* isolate = Isolate::Current(); 51 Isolate* isolate = Isolate::Current();
34 if (isolate != NULL) { 52 if (isolate != NULL) {
35 isolate->set_simulator(NULL); 53 isolate->set_simulator(NULL);
36 } 54 }
37 } 55 }
38 56
39 57
40 // Get the active Simulator for the current isolate. 58 // Get the active Simulator for the current isolate.
41 Simulator* Simulator::Current() { 59 Simulator* Simulator::Current() {
42 Simulator* simulator = Isolate::Current()->simulator(); 60 Simulator* simulator = Isolate::Current()->simulator();
43 if (simulator == NULL) { 61 if (simulator == NULL) {
44 simulator = new Simulator(); 62 simulator = new Simulator();
45 Isolate::Current()->set_simulator(simulator); 63 Isolate::Current()->set_simulator(simulator);
46 } 64 }
47 return simulator; 65 return simulator;
48 } 66 }
49 67
50 68
51 // Sets the register in the architecture state. It will also deal with updating 69 // Sets the register in the architecture state. It will also deal with updating
52 // Simulator internal state for special registers such as PC. 70 // Simulator internal state for special registers such as PC.
53 void Simulator::set_register(Register reg, int32_t value) { 71 void Simulator::set_register(Register reg, int32_t value) {
54 UNIMPLEMENTED(); 72 if (reg != R0) {
73 registers_[reg] = value;
74 }
55 } 75 }
56 76
57 77
58 // Get the register from the architecture state. This function does handle 78 // Get the register from the architecture state. This function does handle
59 // the special case of accessing the PC register. 79 // the special case of accessing the PC register.
60 int32_t Simulator::get_register(Register reg) const { 80 int32_t Simulator::get_register(Register reg) const {
61 UNIMPLEMENTED(); 81 if (reg == R0) {
62 return 0; 82 return 0;
83 }
84 return registers_[reg];
63 } 85 }
64 86
65 87
66 // Returns the top of the stack area to enable checking for stack pointer 88 // Returns the top of the stack area to enable checking for stack pointer
67 // validity. 89 // validity.
68 uword Simulator::StackTop() const { 90 uword Simulator::StackTop() const {
69 // To be safe in potential stack underflows we leave some buffer above and 91 // To be safe in potential stack underflows we leave some buffer above and
70 // set the stack top. 92 // set the stack top.
71 return reinterpret_cast<uword>(stack_) + 93 return reinterpret_cast<uword>(stack_) +
72 (Isolate::GetSpecifiedStackSize() + Isolate::kStackSizeBuffer); 94 (Isolate::GetSpecifiedStackSize() + Isolate::kStackSizeBuffer);
73 } 95 }
74 96
75 97
76 void Simulator::InitOnce() { 98 void Simulator::Format(Instr* instr, const char* format) {
99 OS::PrintErr("Simulator - unknown instruction: %s\n", format);
100 UNIMPLEMENTED();
101 }
102
103 void Simulator::DecodeSpecial(Instr* instr) {
104 switch (instr->FunctionField()) {
105 case SLL: {
106 if ((instr->RdField() == R0) &&
107 (instr->RtField() == R0) &&
108 (instr->SaField() == 0)) {
109 // Format(instr, "nop");
110 // Nothing to be done for NOP.
111 } else {
112 Format(instr, "sll 'rd, 'rt, 'sa");
113 }
114 break;
115 }
116 case JR: {
117 ASSERT(instr->RtField() == R0);
118 ASSERT(instr->RdField() == R0);
119 ASSERT(!delay_slot_);
120 // Format(instr, "jr'hint 'rs");
121 uword next_pc = get_register(instr->RsField());
122 ExecuteDelaySlot();
123 pc_ = next_pc - Instr::kInstrSize; // Account for regular PC increment.
124 break;
125 }
126 default: {
127 OS::PrintErr("DecodeSpecial: 0x%x\n", instr->InstructionBits());
128 UNREACHABLE();
129 break;
130 }
131 }
132 }
133
134
135 void Simulator::InstructionDecode(Instr* instr) {
136 switch (instr->OpcodeField()) {
137 case SPECIAL: {
138 DecodeSpecial(instr);
139 break;
140 }
141 case ORI: {
142 // Format(instr, "ori 'rt, 'rs, 'imm");
143 int32_t rs_val = get_register(instr->RsField());
144 set_register(instr->RtField(), rs_val | instr->ImmField());
regis 2013/03/06 21:18:42 The immediate field should be zero extended, i.e.
Ivan Posva 2013/03/07 11:03:22 The result of ImmField() is already unsigned.
145 break;
146 }
147 default: {
148 OS::PrintErr("Undecoded instruction: 0x%x\n", instr->InstructionBits());
149 UNREACHABLE();
150 break;
151 }
152 }
153 pc_ += Instr::kInstrSize;
154 }
155
156
157 void Simulator::ExecuteDelaySlot() {
158 ASSERT(pc_ != kEndSimulatingPC);
159 delay_slot_ = true;
160 icount_++;
161 if (icount_ == FLAG_stop_sim_at) {
162 UNIMPLEMENTED();
163 }
164 Instr* instr = Instr::At(pc_ + Instr::kInstrSize);
165 InstructionDecode(instr);
166 delay_slot_ = false;
167 }
168
169
170 void Simulator::Execute() {
171 if (FLAG_stop_sim_at == 0) {
172 // Fast version of the dispatch loop without checking whether the simulator
173 // should be stopping at a particular executed instruction.
174 while (pc_ != kEndSimulatingPC) {
175 icount_++;
176 Instr* instr = Instr::At(pc_);
177 InstructionDecode(instr);
178 }
179 } else {
180 // FLAG_stop_sim_at is at the non-default value. Stop in the debugger when
181 // we reach the particular instruction count.
182 while (pc_ != kEndSimulatingPC) {
183 icount_++;
184 if (icount_ == FLAG_stop_sim_at) {
185 UNIMPLEMENTED();
186 } else {
187 Instr* instr = Instr::At(pc_);
188 InstructionDecode(instr);
189 }
190 }
191 }
77 } 192 }
78 193
79 194
80 int64_t Simulator::Call(int32_t entry, 195 int64_t Simulator::Call(int32_t entry,
81 int32_t parameter0, 196 int32_t parameter0,
82 int32_t parameter1, 197 int32_t parameter1,
83 int32_t parameter2, 198 int32_t parameter2,
84 int32_t parameter3, 199 int32_t parameter3) {
85 int32_t parameter4) { 200 // Save the SP register before the call so we can restore it.
86 UNIMPLEMENTED(); 201 int32_t sp_before_call = get_register(SP);
87 return 0LL; 202
203 // Setup parameters.
204 set_register(A0, parameter0);
205 set_register(A1, parameter1);
206 set_register(A2, parameter2);
207 set_register(A3, parameter3);
208
209 // Make sure the activation frames are properly aligned.
210 int32_t stack_pointer = sp_before_call;
211 static const int kFrameAlignment = OS::ActivationFrameAlignment();
212 if (kFrameAlignment > 0) {
213 stack_pointer = Utils::RoundDown(stack_pointer, kFrameAlignment);
214 }
215 set_register(SP, stack_pointer);
216
217 // Prepare to execute the code at entry.
218 set_pc(entry);
219 // Put down marker for end of simulation. The simulator will stop simulation
220 // when the PC reaches this value. By saving the "end simulation" value into
221 // RA the simulation stops when returning to this call point.
222 set_register(RA, kEndSimulatingPC);
223
224 // Remember the values of callee-saved registers.
225 // The code below assumes that r9 is not used as sb (static base) in
226 // simulator code and therefore is regarded as a callee-saved register.
227 int32_t r16_val = get_register(R16);
228 int32_t r17_val = get_register(R17);
229 int32_t r18_val = get_register(R18);
230 int32_t r19_val = get_register(R19);
231 int32_t r20_val = get_register(R20);
232 int32_t r21_val = get_register(R21);
233 int32_t r22_val = get_register(R22);
234 int32_t r23_val = get_register(R23);
235
236 // Setup the callee-saved registers with a known value. To be able to check
237 // that they are preserved properly across dart execution.
238 int32_t callee_saved_value = icount_;
239 set_register(R16, callee_saved_value);
240 set_register(R17, callee_saved_value);
241 set_register(R18, callee_saved_value);
242 set_register(R19, callee_saved_value);
243 set_register(R20, callee_saved_value);
244 set_register(R21, callee_saved_value);
245 set_register(R22, callee_saved_value);
246 set_register(R23, callee_saved_value);
247
248 // Start the simulation
249 Execute();
250
251 // Check that the callee-saved registers have been preserved.
252 ASSERT(callee_saved_value == get_register(R16));
253 ASSERT(callee_saved_value == get_register(R17));
254 ASSERT(callee_saved_value == get_register(R18));
255 ASSERT(callee_saved_value == get_register(R19));
256 ASSERT(callee_saved_value == get_register(R20));
257 ASSERT(callee_saved_value == get_register(R21));
258 ASSERT(callee_saved_value == get_register(R22));
259 ASSERT(callee_saved_value == get_register(R23));
260
261 // Restore callee-saved registers with the original value.
262 set_register(R16, r16_val);
263 set_register(R17, r17_val);
264 set_register(R18, r18_val);
265 set_register(R19, r19_val);
266 set_register(R20, r20_val);
267 set_register(R21, r21_val);
268 set_register(R22, r22_val);
269 set_register(R23, r23_val);
270
271 // Restore the SP register and return R1:R0.
272 set_register(SP, sp_before_call);
273 return Utils::LowHighTo64Bits(get_register(V0), get_register(V1));
88 } 274 }
89 275
90 } // namespace dart 276 } // namespace dart
91 277
92 #endif // !defined(HOST_ARCH_MIPS) 278 #endif // !defined(HOST_ARCH_MIPS)
93 279
94 #endif // defined TARGET_ARCH_MIPS 280 #endif // defined TARGET_ARCH_MIPS
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698