blob: 570fb0732a89812a391d89337bd2aac8406bbd23 [file] [log] [blame]
jkummerow@chromium.org05ed9dd2012-01-23 14:42:48 +00001// Copyright 2012 the V8 project authors. All rights reserved.
ager@chromium.org5c838252010-02-19 08:53:10 +00002// Redistribution and use in source and binary forms, with or without
3// modification, are permitted provided that the following conditions are
4// met:
5//
6// * Redistributions of source code must retain the above copyright
7// notice, this list of conditions and the following disclaimer.
8// * Redistributions in binary form must reproduce the above
9// copyright notice, this list of conditions and the following
10// disclaimer in the documentation and/or other materials provided
11// with the distribution.
12// * Neither the name of Google Inc. nor the names of its
13// contributors may be used to endorse or promote products derived
14// from this software without specific prior written permission.
15//
16// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
lrn@chromium.org7516f052011-03-30 08:52:27 +000028#ifndef V8_MIPS_LITHIUM_CODEGEN_MIPS_H_
29#define V8_MIPS_LITHIUM_CODEGEN_MIPS_H_
ager@chromium.org5c838252010-02-19 08:53:10 +000030
lrn@chromium.org7516f052011-03-30 08:52:27 +000031#include "mips/lithium-mips.h"
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +000032#include "mips/lithium-gap-resolver-mips.h"
lrn@chromium.org7516f052011-03-30 08:52:27 +000033#include "deoptimizer.h"
34#include "safepoint-table.h"
35#include "scopes.h"
36
ager@chromium.org5c838252010-02-19 08:53:10 +000037namespace v8 {
38namespace internal {
39
lrn@chromium.org7516f052011-03-30 08:52:27 +000040// Forward declarations.
41class LDeferredCode;
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +000042class SafepointGenerator;
ager@chromium.org5c838252010-02-19 08:53:10 +000043
lrn@chromium.org7516f052011-03-30 08:52:27 +000044class LCodeGen BASE_EMBEDDED {
45 public:
yangguo@chromium.org5a11aaf2012-06-20 11:29:00 +000046 LCodeGen(LChunk* chunk, MacroAssembler* assembler, CompilationInfo* info)
47 : zone_(info->zone()),
jkummerow@chromium.org28583c92012-07-16 11:31:55 +000048 chunk_(static_cast<LPlatformChunk*>(chunk)),
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +000049 masm_(assembler),
50 info_(info),
51 current_block_(-1),
52 current_instruction_(-1),
53 instructions_(chunk->instructions()),
yangguo@chromium.org5a11aaf2012-06-20 11:29:00 +000054 deoptimizations_(4, info->zone()),
55 deopt_jump_table_(4, info->zone()),
56 deoptimization_literals_(8, info->zone()),
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +000057 inlined_function_count_(0),
58 scope_(info->scope()),
59 status_(UNUSED),
yangguo@chromium.org5a11aaf2012-06-20 11:29:00 +000060 translations_(info->zone()),
61 deferred_(8, info->zone()),
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +000062 osr_pc_offset_(-1),
erikcorry0ad885c2011-11-21 13:51:57 +000063 last_lazy_deopt_pc_(0),
yangguo@chromium.org5a11aaf2012-06-20 11:29:00 +000064 safepoints_(info->zone()),
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +000065 resolver_(this),
66 expected_safepoint_kind_(Safepoint::kSimple) {
67 PopulateDeoptimizationLiteralsWithInlinedFunctions();
68 }
69
70
71 // Simple accessors.
72 MacroAssembler* masm() const { return masm_; }
73 CompilationInfo* info() const { return info_; }
74 Isolate* isolate() const { return info_->isolate(); }
75 Factory* factory() const { return isolate()->factory(); }
76 Heap* heap() const { return isolate()->heap(); }
mmassi@chromium.org7028c052012-06-13 11:51:58 +000077 Zone* zone() const { return zone_; }
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +000078
79 // Support for converting LOperands to assembler types.
80 // LOperand must be a register.
81 Register ToRegister(LOperand* op) const;
82
83 // LOperand is loaded into scratch, unless already a register.
84 Register EmitLoadRegister(LOperand* op, Register scratch);
85
86 // LOperand must be a double register.
87 DoubleRegister ToDoubleRegister(LOperand* op) const;
88
89 // LOperand is loaded into dbl_scratch, unless already a double register.
90 DoubleRegister EmitLoadDoubleRegister(LOperand* op,
91 FloatRegister flt_scratch,
92 DoubleRegister dbl_scratch);
93 int ToInteger32(LConstantOperand* op) const;
94 double ToDouble(LConstantOperand* op) const;
95 Operand ToOperand(LOperand* op);
96 MemOperand ToMemOperand(LOperand* op) const;
97 // Returns a MemOperand pointing to the high word of a DoubleStackSlot.
98 MemOperand ToHighMemOperand(LOperand* op) const;
ager@chromium.orgce5e87b2010-03-10 10:24:18 +000099
rossberg@chromium.orgfab14982012-01-05 15:02:15 +0000100 bool IsInteger32(LConstantOperand* op) const;
101 Handle<Object> ToHandle(LConstantOperand* op) const;
102
lrn@chromium.org7516f052011-03-30 08:52:27 +0000103 // Try to generate code for the entire chunk, but it may fail if the
104 // chunk contains constructs we cannot handle. Returns true if the
105 // code generation attempt succeeded.
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000106 bool GenerateCode();
ager@chromium.orgce5e87b2010-03-10 10:24:18 +0000107
lrn@chromium.org7516f052011-03-30 08:52:27 +0000108 // Finish the code by setting stack height, safepoint, and bailout
109 // information on it.
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000110 void FinishCode(Handle<Code> code);
111
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000112 void DoDeferredNumberTagD(LNumberTagD* instr);
113 void DoDeferredNumberTagI(LNumberTagI* instr);
114 void DoDeferredTaggedToI(LTaggedToI* instr);
115 void DoDeferredMathAbsTaggedHeapNumber(LUnaryMathOperation* instr);
116 void DoDeferredStackCheck(LStackCheck* instr);
erik.corry@gmail.combbceb572012-03-09 10:52:05 +0000117 void DoDeferredRandom(LRandom* instr);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000118 void DoDeferredStringCharCodeAt(LStringCharCodeAt* instr);
119 void DoDeferredStringCharFromCode(LStringCharFromCode* instr);
ulan@chromium.org967e2702012-02-28 09:49:15 +0000120 void DoDeferredAllocateObject(LAllocateObject* instr);
erikcorry0ad885c2011-11-21 13:51:57 +0000121 void DoDeferredInstanceOfKnownGlobal(LInstanceOfKnownGlobal* instr,
122 Label* map_check);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000123
jkummerow@chromium.org05ed9dd2012-01-23 14:42:48 +0000124 void DoCheckMapCommon(Register reg, Register scratch, Handle<Map> map,
125 CompareMapMode mode, LEnvironment* env);
126
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000127 // Parallel move support.
128 void DoParallelMove(LParallelMove* move);
129 void DoGap(LGap* instr);
130
danno@chromium.org129d3982012-07-25 15:01:47 +0000131 MemOperand PrepareKeyedOperand(Register key,
132 Register base,
133 bool key_is_constant,
134 int constant_key,
135 int element_size,
136 int shift_size,
137 int additional_index,
138 int additional_offset);
139
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000140 // Emit frame translation commands for an environment.
141 void WriteTranslation(LEnvironment* environment, Translation* translation);
142
143 // Declare methods that deal with the individual node types.
144#define DECLARE_DO(type) void Do##type(L##type* node);
145 LITHIUM_CONCRETE_INSTRUCTION_LIST(DECLARE_DO)
146#undef DECLARE_DO
147
148 private:
149 enum Status {
150 UNUSED,
151 GENERATING,
152 DONE,
153 ABORTED
154 };
155
156 bool is_unused() const { return status_ == UNUSED; }
157 bool is_generating() const { return status_ == GENERATING; }
158 bool is_done() const { return status_ == DONE; }
159 bool is_aborted() const { return status_ == ABORTED; }
160
161 StrictModeFlag strict_mode_flag() const {
mstarzinger@chromium.org1b3afd12011-11-29 14:28:56 +0000162 return info()->is_classic_mode() ? kNonStrictMode : kStrictMode;
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000163 }
164
jkummerow@chromium.org28583c92012-07-16 11:31:55 +0000165 LPlatformChunk* chunk() const { return chunk_; }
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000166 Scope* scope() const { return scope_; }
167 HGraph* graph() const { return chunk_->graph(); }
168
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +0000169 Register scratch0() { return kLithiumScratchReg; }
170 Register scratch1() { return kLithiumScratchReg2; }
171 DoubleRegister double_scratch0() { return kLithiumScratchDouble; }
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000172
173 int GetNextEmittedBlock(int block);
174 LInstruction* GetNextInstruction();
175
176 void EmitClassOfTest(Label* if_true,
177 Label* if_false,
178 Handle<String> class_name,
179 Register input,
180 Register temporary,
181 Register temporary2);
182
183 int GetStackSlotCount() const { return chunk()->spill_slot_count(); }
184 int GetParameterCount() const { return scope()->num_parameters(); }
185
186 void Abort(const char* format, ...);
187 void Comment(const char* format, ...);
188
mmassi@chromium.org7028c052012-06-13 11:51:58 +0000189 void AddDeferredCode(LDeferredCode* code) { deferred_.Add(code, zone()); }
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000190
191 // Code generation passes. Returns true if code generation should
192 // continue.
193 bool GeneratePrologue();
194 bool GenerateBody();
195 bool GenerateDeferredCode();
196 bool GenerateDeoptJumpTable();
197 bool GenerateSafepointTable();
198
199 enum SafepointMode {
200 RECORD_SIMPLE_SAFEPOINT,
201 RECORD_SAFEPOINT_WITH_REGISTERS_AND_NO_ARGUMENTS
202 };
203
204 void CallCode(Handle<Code> code,
205 RelocInfo::Mode mode,
206 LInstruction* instr);
207
208 void CallCodeGeneric(Handle<Code> code,
209 RelocInfo::Mode mode,
210 LInstruction* instr,
211 SafepointMode safepoint_mode);
212
213 void CallRuntime(const Runtime::Function* function,
214 int num_arguments,
215 LInstruction* instr);
216
217 void CallRuntime(Runtime::FunctionId id,
218 int num_arguments,
219 LInstruction* instr) {
220 const Runtime::Function* function = Runtime::FunctionForId(id);
221 CallRuntime(function, num_arguments, instr);
222 }
223
224 void CallRuntimeFromDeferred(Runtime::FunctionId id,
225 int argc,
226 LInstruction* instr);
227
svenpanne@chromium.orgfb046332012-04-19 12:02:44 +0000228 enum A1State {
229 A1_UNINITIALIZED,
230 A1_CONTAINS_TARGET
231 };
232
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000233 // Generate a direct call to a known function. Expects the function
234 // to be in a1.
235 void CallKnownFunction(Handle<JSFunction> function,
236 int arity,
237 LInstruction* instr,
svenpanne@chromium.orgfb046332012-04-19 12:02:44 +0000238 CallKind call_kind,
239 A1State a1_state);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000240
241 void LoadHeapObject(Register result, Handle<HeapObject> object);
242
erikcorry0ad885c2011-11-21 13:51:57 +0000243 void RecordSafepointWithLazyDeopt(LInstruction* instr,
244 SafepointMode safepoint_mode);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000245
erikcorry0ad885c2011-11-21 13:51:57 +0000246 void RegisterEnvironmentForDeoptimization(LEnvironment* environment,
247 Safepoint::DeoptMode mode);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000248 void DeoptimizeIf(Condition cc,
249 LEnvironment* environment,
jkummerow@chromium.org05ed9dd2012-01-23 14:42:48 +0000250 Register src1 = zero_reg,
251 const Operand& src2 = Operand(zero_reg));
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000252
253 void AddToTranslation(Translation* translation,
254 LOperand* op,
255 bool is_tagged);
256 void PopulateDeoptimizationData(Handle<Code> code);
257 int DefineDeoptimizationLiteral(Handle<Object> literal);
258
259 void PopulateDeoptimizationLiteralsWithInlinedFunctions();
260
261 Register ToRegister(int index) const;
262 DoubleRegister ToDoubleRegister(int index) const;
263
264 // Specific math operations - used from DoUnaryMathOperation.
265 void EmitIntegerMathAbs(LUnaryMathOperation* instr);
266 void DoMathAbs(LUnaryMathOperation* instr);
267 void DoMathFloor(LUnaryMathOperation* instr);
268 void DoMathRound(LUnaryMathOperation* instr);
269 void DoMathSqrt(LUnaryMathOperation* instr);
270 void DoMathPowHalf(LUnaryMathOperation* instr);
271 void DoMathLog(LUnaryMathOperation* instr);
svenpanne@chromium.orgecb9dd62011-12-01 08:22:35 +0000272 void DoMathTan(LUnaryMathOperation* instr);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000273 void DoMathCos(LUnaryMathOperation* instr);
274 void DoMathSin(LUnaryMathOperation* instr);
275
276 // Support for recording safepoint and position information.
277 void RecordSafepoint(LPointerMap* pointers,
278 Safepoint::Kind kind,
279 int arguments,
erikcorry0ad885c2011-11-21 13:51:57 +0000280 Safepoint::DeoptMode mode);
281 void RecordSafepoint(LPointerMap* pointers, Safepoint::DeoptMode mode);
282 void RecordSafepoint(Safepoint::DeoptMode mode);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000283 void RecordSafepointWithRegisters(LPointerMap* pointers,
284 int arguments,
erikcorry0ad885c2011-11-21 13:51:57 +0000285 Safepoint::DeoptMode mode);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000286 void RecordSafepointWithRegistersAndDoubles(LPointerMap* pointers,
287 int arguments,
erikcorry0ad885c2011-11-21 13:51:57 +0000288 Safepoint::DeoptMode mode);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000289 void RecordPosition(int position);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000290
291 static Condition TokenToCondition(Token::Value op, bool is_unsigned);
292 void EmitGoto(int block);
293 void EmitBranch(int left_block,
294 int right_block,
295 Condition cc,
296 Register src1,
297 const Operand& src2);
298 void EmitBranchF(int left_block,
299 int right_block,
300 Condition cc,
301 FPURegister src1,
302 FPURegister src2);
303 void EmitCmpI(LOperand* left, LOperand* right);
304 void EmitNumberUntagD(Register input,
305 DoubleRegister result,
306 bool deoptimize_on_undefined,
ulan@chromium.org2efb9002012-01-19 15:36:35 +0000307 bool deoptimize_on_minus_zero,
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000308 LEnvironment* env);
309
310 // Emits optimized code for typeof x == "y". Modifies input register.
311 // Returns the condition on which a final split to
312 // true and false label should be made, to optimize fallthrough.
313 // Returns two registers in cmp1 and cmp2 that can be used in the
314 // Branch instruction after EmitTypeofIs.
315 Condition EmitTypeofIs(Label* true_label,
316 Label* false_label,
317 Register input,
318 Handle<String> type_name,
319 Register& cmp1,
320 Operand& cmp2);
321
322 // Emits optimized code for %_IsObject(x). Preserves input register.
323 // Returns the condition on which a final split to
324 // true and false label should be made, to optimize fallthrough.
325 Condition EmitIsObject(Register input,
326 Register temp1,
erikcorry0ad885c2011-11-21 13:51:57 +0000327 Register temp2,
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000328 Label* is_not_object,
329 Label* is_object);
330
mstarzinger@chromium.orgf8c6bd52011-11-23 12:13:52 +0000331 // Emits optimized code for %_IsString(x). Preserves input register.
332 // Returns the condition on which a final split to
333 // true and false label should be made, to optimize fallthrough.
334 Condition EmitIsString(Register input,
335 Register temp1,
336 Label* is_not_string);
337
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000338 // Emits optimized code for %_IsConstructCall().
339 // Caller should branch on equal condition.
340 void EmitIsConstructCall(Register temp1, Register temp2);
341
342 void EmitLoadFieldOrConstantFunction(Register result,
343 Register object,
344 Handle<Map> type,
mmassi@chromium.org7028c052012-06-13 11:51:58 +0000345 Handle<String> name,
346 LEnvironment* env);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000347
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000348 // Emits optimized code to deep-copy the contents of statically known
349 // object graphs (e.g. object literal boilerplate).
350 void EmitDeepCopy(Handle<JSObject> object,
351 Register result,
352 Register source,
353 int* offset);
354
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000355 struct JumpTableEntry {
356 explicit inline JumpTableEntry(Address entry)
357 : label(),
358 address(entry) { }
359 Label label;
360 Address address;
361 };
362
erikcorry0ad885c2011-11-21 13:51:57 +0000363 void EnsureSpaceForLazyDeopt();
364
yangguo@chromium.org5a11aaf2012-06-20 11:29:00 +0000365 Zone* zone_;
jkummerow@chromium.org28583c92012-07-16 11:31:55 +0000366 LPlatformChunk* const chunk_;
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000367 MacroAssembler* const masm_;
368 CompilationInfo* const info_;
369
370 int current_block_;
371 int current_instruction_;
372 const ZoneList<LInstruction*>* instructions_;
373 ZoneList<LEnvironment*> deoptimizations_;
374 ZoneList<JumpTableEntry> deopt_jump_table_;
375 ZoneList<Handle<Object> > deoptimization_literals_;
376 int inlined_function_count_;
377 Scope* const scope_;
378 Status status_;
379 TranslationBuffer translations_;
380 ZoneList<LDeferredCode*> deferred_;
381 int osr_pc_offset_;
erikcorry0ad885c2011-11-21 13:51:57 +0000382 int last_lazy_deopt_pc_;
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000383
384 // Builder that keeps track of safepoints in the code. The table
385 // itself is emitted at the end of the generated code.
386 SafepointTableBuilder safepoints_;
387
388 // Compiler from a set of parallel moves to a sequential list of moves.
389 LGapResolver resolver_;
390
391 Safepoint::Kind expected_safepoint_kind_;
392
393 class PushSafepointRegistersScope BASE_EMBEDDED {
394 public:
395 PushSafepointRegistersScope(LCodeGen* codegen,
396 Safepoint::Kind kind)
397 : codegen_(codegen) {
398 ASSERT(codegen_->expected_safepoint_kind_ == Safepoint::kSimple);
399 codegen_->expected_safepoint_kind_ = kind;
400
401 switch (codegen_->expected_safepoint_kind_) {
402 case Safepoint::kWithRegisters:
403 codegen_->masm_->PushSafepointRegisters();
404 break;
405 case Safepoint::kWithRegistersAndDoubles:
406 codegen_->masm_->PushSafepointRegistersAndDoubles();
407 break;
408 default:
409 UNREACHABLE();
410 }
411 }
412
413 ~PushSafepointRegistersScope() {
414 Safepoint::Kind kind = codegen_->expected_safepoint_kind_;
415 ASSERT((kind & Safepoint::kWithRegisters) != 0);
416 switch (kind) {
417 case Safepoint::kWithRegisters:
418 codegen_->masm_->PopSafepointRegisters();
419 break;
420 case Safepoint::kWithRegistersAndDoubles:
421 codegen_->masm_->PopSafepointRegistersAndDoubles();
422 break;
423 default:
424 UNREACHABLE();
425 }
426 codegen_->expected_safepoint_kind_ = Safepoint::kSimple;
427 }
428
429 private:
430 LCodeGen* codegen_;
431 };
432
433 friend class LDeferredCode;
434 friend class LEnvironment;
435 friend class SafepointGenerator;
436 DISALLOW_COPY_AND_ASSIGN(LCodeGen);
437};
438
439
440class LDeferredCode: public ZoneObject {
441 public:
442 explicit LDeferredCode(LCodeGen* codegen)
443 : codegen_(codegen),
444 external_exit_(NULL),
445 instruction_index_(codegen->current_instruction_) {
446 codegen->AddDeferredCode(this);
447 }
448
449 virtual ~LDeferredCode() { }
450 virtual void Generate() = 0;
451 virtual LInstruction* instr() = 0;
452
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +0000453 void SetExit(Label* exit) { external_exit_ = exit; }
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000454 Label* entry() { return &entry_; }
455 Label* exit() { return external_exit_ != NULL ? external_exit_ : &exit_; }
456 int instruction_index() const { return instruction_index_; }
457
458 protected:
459 LCodeGen* codegen() const { return codegen_; }
460 MacroAssembler* masm() const { return codegen_->masm(); }
461
462 private:
463 LCodeGen* codegen_;
464 Label entry_;
465 Label exit_;
466 Label* external_exit_;
467 int instruction_index_;
lrn@chromium.org7516f052011-03-30 08:52:27 +0000468};
ager@chromium.org5c838252010-02-19 08:53:10 +0000469
470} } // namespace v8::internal
471
lrn@chromium.org7516f052011-03-30 08:52:27 +0000472#endif // V8_MIPS_LITHIUM_CODEGEN_MIPS_H_