blob: aa88b277b18f0ddaad74d030fc26d946c4c1fdc9 [file] [log] [blame]
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +00001// Copyright 2012 the V8 project authors. All rights reserved.
kmillikin@chromium.orgd2c22f02011-01-10 08:15:37 +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
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +000028#include "v8.h"
29
30#if defined(V8_TARGET_ARCH_X64)
31
ricow@chromium.org83aa5492011-02-07 12:42:56 +000032#include "lithium-allocator-inl.h"
kmillikin@chromium.orgd2c22f02011-01-10 08:15:37 +000033#include "x64/lithium-x64.h"
34#include "x64/lithium-codegen-x64.h"
35
36namespace v8 {
37namespace internal {
38
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +000039#define DEFINE_COMPILE(type) \
40 void L##type::CompileToNative(LCodeGen* generator) { \
41 generator->Do##type(this); \
42 }
43LITHIUM_CONCRETE_INSTRUCTION_LIST(DEFINE_COMPILE)
44#undef DEFINE_COMPILE
45
46LOsrEntry::LOsrEntry() {
47 for (int i = 0; i < Register::kNumAllocatableRegisters; ++i) {
48 register_spills_[i] = NULL;
49 }
50 for (int i = 0; i < DoubleRegister::kNumAllocatableRegisters; ++i) {
51 double_register_spills_[i] = NULL;
52 }
53}
54
55
56void LOsrEntry::MarkSpilledRegister(int allocation_index,
57 LOperand* spill_operand) {
58 ASSERT(spill_operand->IsStackSlot());
59 ASSERT(register_spills_[allocation_index] == NULL);
60 register_spills_[allocation_index] = spill_operand;
61}
62
63
64void LOsrEntry::MarkSpilledDoubleRegister(int allocation_index,
65 LOperand* spill_operand) {
66 ASSERT(spill_operand->IsDoubleStackSlot());
67 ASSERT(double_register_spills_[allocation_index] == NULL);
68 double_register_spills_[allocation_index] = spill_operand;
69}
70
71
ricow@chromium.org83aa5492011-02-07 12:42:56 +000072#ifdef DEBUG
73void LInstruction::VerifyCall() {
danno@chromium.org160a7b02011-04-18 15:51:38 +000074 // Call instructions can use only fixed registers as temporaries and
75 // outputs because all registers are blocked by the calling convention.
76 // Inputs operands must use a fixed register or use-at-start policy or
77 // a non-register policy.
ricow@chromium.org83aa5492011-02-07 12:42:56 +000078 ASSERT(Output() == NULL ||
79 LUnallocated::cast(Output())->HasFixedPolicy() ||
80 !LUnallocated::cast(Output())->HasRegisterPolicy());
jkummerow@chromium.orge297f592011-06-08 10:05:15 +000081 for (UseIterator it(this); !it.Done(); it.Advance()) {
82 LUnallocated* operand = LUnallocated::cast(it.Current());
danno@chromium.org160a7b02011-04-18 15:51:38 +000083 ASSERT(operand->HasFixedPolicy() ||
84 operand->IsUsedAtStart());
ricow@chromium.org83aa5492011-02-07 12:42:56 +000085 }
jkummerow@chromium.orge297f592011-06-08 10:05:15 +000086 for (TempIterator it(this); !it.Done(); it.Advance()) {
87 LUnallocated* operand = LUnallocated::cast(it.Current());
danno@chromium.org160a7b02011-04-18 15:51:38 +000088 ASSERT(operand->HasFixedPolicy() ||!operand->HasRegisterPolicy());
ricow@chromium.org83aa5492011-02-07 12:42:56 +000089 }
90}
91#endif
92
93
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +000094void LInstruction::PrintTo(StringStream* stream) {
95 stream->Add("%s ", this->Mnemonic());
ricow@chromium.org83aa5492011-02-07 12:42:56 +000096
97 PrintOutputOperandTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +000098
99 PrintDataTo(stream);
100
101 if (HasEnvironment()) {
102 stream->Add(" ");
103 environment()->PrintTo(stream);
104 }
105
106 if (HasPointerMap()) {
107 stream->Add(" ");
108 pointer_map()->PrintTo(stream);
109 }
110}
111
112
113template<int R, int I, int T>
114void LTemplateInstruction<R, I, T>::PrintDataTo(StringStream* stream) {
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000115 stream->Add("= ");
svenpanne@chromium.org6d786c92011-06-15 10:58:27 +0000116 for (int i = 0; i < inputs_.length(); i++) {
117 if (i > 0) stream->Add(" ");
118 inputs_[i]->PrintTo(stream);
119 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000120}
121
122
123template<int R, int I, int T>
124void LTemplateInstruction<R, I, T>::PrintOutputOperandTo(StringStream* stream) {
svenpanne@chromium.org6d786c92011-06-15 10:58:27 +0000125 for (int i = 0; i < results_.length(); i++) {
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000126 if (i > 0) stream->Add(" ");
svenpanne@chromium.org6d786c92011-06-15 10:58:27 +0000127 results_[i]->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000128 }
129}
130
131
132void LLabel::PrintDataTo(StringStream* stream) {
133 LGap::PrintDataTo(stream);
134 LLabel* rep = replacement();
135 if (rep != NULL) {
136 stream->Add(" Dead block replaced with B%d", rep->block_id());
137 }
138}
139
140
141bool LGap::IsRedundant() const {
142 for (int i = 0; i < 4; i++) {
143 if (parallel_moves_[i] != NULL && !parallel_moves_[i]->IsRedundant()) {
144 return false;
145 }
146 }
147
148 return true;
149}
150
151
152void LGap::PrintDataTo(StringStream* stream) {
153 for (int i = 0; i < 4; i++) {
154 stream->Add("(");
155 if (parallel_moves_[i] != NULL) {
156 parallel_moves_[i]->PrintDataTo(stream);
157 }
158 stream->Add(") ");
159 }
160}
161
162
163const char* LArithmeticD::Mnemonic() const {
164 switch (op()) {
165 case Token::ADD: return "add-d";
166 case Token::SUB: return "sub-d";
167 case Token::MUL: return "mul-d";
168 case Token::DIV: return "div-d";
169 case Token::MOD: return "mod-d";
170 default:
171 UNREACHABLE();
172 return NULL;
173 }
174}
175
176
177const char* LArithmeticT::Mnemonic() const {
178 switch (op()) {
179 case Token::ADD: return "add-t";
180 case Token::SUB: return "sub-t";
181 case Token::MUL: return "mul-t";
182 case Token::MOD: return "mod-t";
183 case Token::DIV: return "div-t";
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000184 case Token::BIT_AND: return "bit-and-t";
185 case Token::BIT_OR: return "bit-or-t";
186 case Token::BIT_XOR: return "bit-xor-t";
187 case Token::SHL: return "sal-t";
188 case Token::SAR: return "sar-t";
189 case Token::SHR: return "shr-t";
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000190 default:
191 UNREACHABLE();
192 return NULL;
193 }
194}
195
196
197void LGoto::PrintDataTo(StringStream* stream) {
198 stream->Add("B%d", block_id());
199}
200
201
202void LBranch::PrintDataTo(StringStream* stream) {
203 stream->Add("B%d | B%d on ", true_block_id(), false_block_id());
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000204 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000205}
206
207
208void LCmpIDAndBranch::PrintDataTo(StringStream* stream) {
209 stream->Add("if ");
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000210 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000211 stream->Add(" %s ", Token::String(op()));
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000212 InputAt(1)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000213 stream->Add(" then B%d else B%d", true_block_id(), false_block_id());
214}
215
216
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000217void LIsNilAndBranch::PrintDataTo(StringStream* stream) {
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000218 stream->Add("if ");
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000219 InputAt(0)->PrintTo(stream);
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000220 stream->Add(kind() == kStrictEquality ? " === " : " == ");
221 stream->Add(nil() == kNullValue ? "null" : "undefined");
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000222 stream->Add(" then B%d else B%d", true_block_id(), false_block_id());
223}
224
225
226void LIsObjectAndBranch::PrintDataTo(StringStream* stream) {
227 stream->Add("if is_object(");
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000228 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000229 stream->Add(") then B%d else B%d", true_block_id(), false_block_id());
230}
231
232
erikcorry0ad885c2011-11-21 13:51:57 +0000233void LIsStringAndBranch::PrintDataTo(StringStream* stream) {
234 stream->Add("if is_string(");
235 InputAt(0)->PrintTo(stream);
236 stream->Add(") then B%d else B%d", true_block_id(), false_block_id());
237}
238
239
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000240void LIsSmiAndBranch::PrintDataTo(StringStream* stream) {
241 stream->Add("if is_smi(");
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000242 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000243 stream->Add(") then B%d else B%d", true_block_id(), false_block_id());
244}
245
246
vegorov@chromium.org7304bca2011-05-16 12:14:13 +0000247void LIsUndetectableAndBranch::PrintDataTo(StringStream* stream) {
248 stream->Add("if is_undetectable(");
249 InputAt(0)->PrintTo(stream);
250 stream->Add(") then B%d else B%d", true_block_id(), false_block_id());
251}
252
253
erikcorry0ad885c2011-11-21 13:51:57 +0000254void LStringCompareAndBranch::PrintDataTo(StringStream* stream) {
255 stream->Add("if string_compare(");
256 InputAt(0)->PrintTo(stream);
257 InputAt(1)->PrintTo(stream);
258 stream->Add(") then B%d else B%d", true_block_id(), false_block_id());
259}
260
261
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000262void LHasInstanceTypeAndBranch::PrintDataTo(StringStream* stream) {
263 stream->Add("if has_instance_type(");
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000264 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000265 stream->Add(") then B%d else B%d", true_block_id(), false_block_id());
266}
267
268
269void LHasCachedArrayIndexAndBranch::PrintDataTo(StringStream* stream) {
270 stream->Add("if has_cached_array_index(");
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000271 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000272 stream->Add(") then B%d else B%d", true_block_id(), false_block_id());
273}
274
275
276void LClassOfTestAndBranch::PrintDataTo(StringStream* stream) {
277 stream->Add("if class_of_test(");
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000278 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000279 stream->Add(", \"%o\") then B%d else B%d",
280 *hydrogen()->class_name(),
281 true_block_id(),
282 false_block_id());
283}
284
285
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000286void LTypeofIsAndBranch::PrintDataTo(StringStream* stream) {
287 stream->Add("if typeof ");
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000288 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000289 stream->Add(" == \"%s\" then B%d else B%d",
290 *hydrogen()->type_literal()->ToCString(),
291 true_block_id(), false_block_id());
292}
293
294
295void LCallConstantFunction::PrintDataTo(StringStream* stream) {
296 stream->Add("#%d / ", arity());
297}
298
299
300void LUnaryMathOperation::PrintDataTo(StringStream* stream) {
301 stream->Add("/%s ", hydrogen()->OpName());
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000302 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000303}
304
305
306void LLoadContextSlot::PrintDataTo(StringStream* stream) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000307 InputAt(0)->PrintTo(stream);
308 stream->Add("[%d]", slot_index());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000309}
310
311
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000312void LStoreContextSlot::PrintDataTo(StringStream* stream) {
313 InputAt(0)->PrintTo(stream);
314 stream->Add("[%d] <- ", slot_index());
315 InputAt(1)->PrintTo(stream);
316}
317
318
danno@chromium.org160a7b02011-04-18 15:51:38 +0000319void LInvokeFunction::PrintDataTo(StringStream* stream) {
320 stream->Add("= ");
321 InputAt(0)->PrintTo(stream);
322 stream->Add(" #%d / ", arity());
323}
324
325
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000326void LCallKeyed::PrintDataTo(StringStream* stream) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000327 stream->Add("[rcx] #%d / ", arity());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000328}
329
330
331void LCallNamed::PrintDataTo(StringStream* stream) {
kmillikin@chromium.org83e16822011-09-13 08:21:47 +0000332 SmartArrayPointer<char> name_string = name()->ToCString();
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000333 stream->Add("%s #%d / ", *name_string, arity());
334}
335
336
337void LCallGlobal::PrintDataTo(StringStream* stream) {
kmillikin@chromium.org83e16822011-09-13 08:21:47 +0000338 SmartArrayPointer<char> name_string = name()->ToCString();
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000339 stream->Add("%s #%d / ", *name_string, arity());
340}
341
342
343void LCallKnownGlobal::PrintDataTo(StringStream* stream) {
344 stream->Add("#%d / ", arity());
345}
346
347
348void LCallNew::PrintDataTo(StringStream* stream) {
349 stream->Add("= ");
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000350 InputAt(0)->PrintTo(stream);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000351 stream->Add(" #%d / ", arity());
352}
353
354
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000355void LAccessArgumentsAt::PrintDataTo(StringStream* stream) {
356 arguments()->PrintTo(stream);
357
358 stream->Add(" length ");
359 length()->PrintTo(stream);
360
361 stream->Add(" index ");
362 index()->PrintTo(stream);
363}
364
365
366int LChunk::GetNextSpillIndex(bool is_double) {
367 return spill_slot_count_++;
368}
369
370
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000371LOperand* LChunk::GetNextSpillSlot(bool is_double) {
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000372 // All stack slots are Double stack slots on x64.
373 // Alternatively, at some point, start using half-size
374 // stack slots for int32 values.
375 int index = GetNextSpillIndex(is_double);
376 if (is_double) {
377 return LDoubleStackSlot::Create(index);
378 } else {
379 return LStackSlot::Create(index);
380 }
381}
382
383
384void LChunk::MarkEmptyBlocks() {
ulan@chromium.org9a21ec42012-03-06 08:42:24 +0000385 HPhase phase("L_Mark empty blocks", this);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000386 for (int i = 0; i < graph()->blocks()->length(); ++i) {
387 HBasicBlock* block = graph()->blocks()->at(i);
388 int first = block->first_instruction_index();
389 int last = block->last_instruction_index();
390 LInstruction* first_instr = instructions()->at(first);
391 LInstruction* last_instr = instructions()->at(last);
392
393 LLabel* label = LLabel::cast(first_instr);
394 if (last_instr->IsGoto()) {
395 LGoto* goto_instr = LGoto::cast(last_instr);
ager@chromium.org04921a82011-06-27 13:21:41 +0000396 if (label->IsRedundant() &&
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000397 !label->is_loop_header()) {
398 bool can_eliminate = true;
399 for (int i = first + 1; i < last && can_eliminate; ++i) {
400 LInstruction* cur = instructions()->at(i);
401 if (cur->IsGap()) {
402 LGap* gap = LGap::cast(cur);
403 if (!gap->IsRedundant()) {
404 can_eliminate = false;
405 }
406 } else {
407 can_eliminate = false;
408 }
409 }
410
411 if (can_eliminate) {
412 label->set_replacement(GetLabel(goto_instr->block_id()));
413 }
414 }
415 }
416 }
417}
418
419
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000420void LStoreNamedField::PrintDataTo(StringStream* stream) {
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000421 object()->PrintTo(stream);
422 stream->Add(".");
423 stream->Add(*String::cast(*name())->ToCString());
424 stream->Add(" <- ");
425 value()->PrintTo(stream);
426}
427
428
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000429void LStoreNamedGeneric::PrintDataTo(StringStream* stream) {
430 object()->PrintTo(stream);
431 stream->Add(".");
432 stream->Add(*String::cast(*name())->ToCString());
433 stream->Add(" <- ");
434 value()->PrintTo(stream);
435}
436
437
438void LStoreKeyedFastElement::PrintDataTo(StringStream* stream) {
439 object()->PrintTo(stream);
440 stream->Add("[");
441 key()->PrintTo(stream);
442 stream->Add("] <- ");
443 value()->PrintTo(stream);
444}
445
446
rossberg@chromium.org717967f2011-07-20 13:44:42 +0000447void LStoreKeyedFastDoubleElement::PrintDataTo(StringStream* stream) {
448 elements()->PrintTo(stream);
449 stream->Add("[");
450 key()->PrintTo(stream);
451 stream->Add("] <- ");
452 value()->PrintTo(stream);
453}
454
455
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +0000456void LStoreKeyedGeneric::PrintDataTo(StringStream* stream) {
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000457 object()->PrintTo(stream);
458 stream->Add("[");
459 key()->PrintTo(stream);
460 stream->Add("] <- ");
461 value()->PrintTo(stream);
462}
463
464
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000465void LTransitionElementsKind::PrintDataTo(StringStream* stream) {
466 object()->PrintTo(stream);
467 stream->Add(" %p -> %p", *original_map(), *transitioned_map());
468}
469
470
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000471void LChunk::AddInstruction(LInstruction* instr, HBasicBlock* block) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000472 LInstructionGap* gap = new(graph_->zone()) LInstructionGap(block);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000473 int index = -1;
474 if (instr->IsControl()) {
475 instructions_.Add(gap);
476 index = instructions_.length();
477 instructions_.Add(instr);
478 } else {
479 index = instructions_.length();
480 instructions_.Add(instr);
481 instructions_.Add(gap);
482 }
483 if (instr->HasPointerMap()) {
484 pointer_maps_.Add(instr->pointer_map());
485 instr->pointer_map()->set_lithium_position(index);
486 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000487}
488
489
490LConstantOperand* LChunk::DefineConstantOperand(HConstant* constant) {
491 return LConstantOperand::Create(constant->id());
492}
493
494
495int LChunk::GetParameterStackSlot(int index) const {
496 // The receiver is at index 0, the first parameter at index 1, so we
497 // shift all parameter indexes down by the number of parameters, and
498 // make sure they end up negative so they are distinguishable from
499 // spill slots.
karlklose@chromium.org8f806e82011-03-07 14:06:08 +0000500 int result = index - info()->scope()->num_parameters() - 1;
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000501 ASSERT(result < 0);
502 return result;
503}
504
505// A parameter relative to ebp in the arguments stub.
506int LChunk::ParameterAt(int index) {
507 ASSERT(-1 <= index); // -1 is the receiver.
karlklose@chromium.org8f806e82011-03-07 14:06:08 +0000508 return (1 + info()->scope()->num_parameters() - index) *
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000509 kPointerSize;
510}
511
512
513LGap* LChunk::GetGapAt(int index) const {
514 return LGap::cast(instructions_[index]);
515}
516
517
518bool LChunk::IsGapAt(int index) const {
519 return instructions_[index]->IsGap();
520}
521
522
523int LChunk::NearestGapPos(int index) const {
524 while (!IsGapAt(index)) index--;
525 return index;
526}
527
528
529void LChunk::AddGapMove(int index, LOperand* from, LOperand* to) {
530 GetGapAt(index)->GetOrCreateParallelMove(LGap::START)->AddMove(from, to);
531}
532
533
534Handle<Object> LChunk::LookupLiteral(LConstantOperand* operand) const {
535 return HConstant::cast(graph_->LookupValue(operand->index()))->handle();
536}
537
538
539Representation LChunk::LookupLiteralRepresentation(
540 LConstantOperand* operand) const {
541 return graph_->LookupValue(operand->index())->representation();
542}
543
544
kmillikin@chromium.orgd2c22f02011-01-10 08:15:37 +0000545LChunk* LChunkBuilder::Build() {
546 ASSERT(is_unused());
ulan@chromium.org812308e2012-02-29 15:58:45 +0000547 chunk_ = new(zone()) LChunk(info(), graph());
ulan@chromium.org9a21ec42012-03-06 08:42:24 +0000548 HPhase phase("L_Building chunk", chunk_);
kmillikin@chromium.orgd2c22f02011-01-10 08:15:37 +0000549 status_ = BUILDING;
550 const ZoneList<HBasicBlock*>* blocks = graph()->blocks();
551 for (int i = 0; i < blocks->length(); i++) {
552 HBasicBlock* next = NULL;
553 if (i < blocks->length() - 1) next = blocks->at(i + 1);
554 DoBasicBlock(blocks->at(i), next);
555 if (is_aborted()) return NULL;
556 }
557 status_ = DONE;
558 return chunk_;
559}
560
561
562void LChunkBuilder::Abort(const char* format, ...) {
563 if (FLAG_trace_bailout) {
kmillikin@chromium.org83e16822011-09-13 08:21:47 +0000564 SmartArrayPointer<char> name(
565 info()->shared_info()->DebugName()->ToCString());
karlklose@chromium.org8f806e82011-03-07 14:06:08 +0000566 PrintF("Aborting LChunk building in @\"%s\": ", *name);
kmillikin@chromium.orgd2c22f02011-01-10 08:15:37 +0000567 va_list arguments;
568 va_start(arguments, format);
569 OS::VPrint(format, arguments);
570 va_end(arguments);
571 PrintF("\n");
572 }
573 status_ = ABORTED;
574}
575
576
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000577LUnallocated* LChunkBuilder::ToUnallocated(Register reg) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000578 return new(zone()) LUnallocated(LUnallocated::FIXED_REGISTER,
579 Register::ToAllocationIndex(reg));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000580}
581
582
583LUnallocated* LChunkBuilder::ToUnallocated(XMMRegister reg) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000584 return new(zone()) LUnallocated(LUnallocated::FIXED_DOUBLE_REGISTER,
585 XMMRegister::ToAllocationIndex(reg));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000586}
587
588
589LOperand* LChunkBuilder::UseFixed(HValue* value, Register fixed_register) {
590 return Use(value, ToUnallocated(fixed_register));
591}
592
593
594LOperand* LChunkBuilder::UseFixedDouble(HValue* value, XMMRegister reg) {
595 return Use(value, ToUnallocated(reg));
596}
597
598
599LOperand* LChunkBuilder::UseRegister(HValue* value) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000600 return Use(value, new(zone()) LUnallocated(LUnallocated::MUST_HAVE_REGISTER));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000601}
602
603
604LOperand* LChunkBuilder::UseRegisterAtStart(HValue* value) {
605 return Use(value,
ulan@chromium.org812308e2012-02-29 15:58:45 +0000606 new(zone()) LUnallocated(LUnallocated::MUST_HAVE_REGISTER,
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000607 LUnallocated::USED_AT_START));
608}
609
610
611LOperand* LChunkBuilder::UseTempRegister(HValue* value) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000612 return Use(value, new(zone()) LUnallocated(LUnallocated::WRITABLE_REGISTER));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000613}
614
615
616LOperand* LChunkBuilder::Use(HValue* value) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000617 return Use(value, new(zone()) LUnallocated(LUnallocated::NONE));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000618}
619
620
621LOperand* LChunkBuilder::UseAtStart(HValue* value) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000622 return Use(value, new(zone()) LUnallocated(LUnallocated::NONE,
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000623 LUnallocated::USED_AT_START));
624}
625
626
627LOperand* LChunkBuilder::UseOrConstant(HValue* value) {
628 return value->IsConstant()
629 ? chunk_->DefineConstantOperand(HConstant::cast(value))
630 : Use(value);
631}
632
633
634LOperand* LChunkBuilder::UseOrConstantAtStart(HValue* value) {
635 return value->IsConstant()
636 ? chunk_->DefineConstantOperand(HConstant::cast(value))
637 : UseAtStart(value);
638}
639
640
641LOperand* LChunkBuilder::UseRegisterOrConstant(HValue* value) {
642 return value->IsConstant()
643 ? chunk_->DefineConstantOperand(HConstant::cast(value))
644 : UseRegister(value);
645}
646
647
648LOperand* LChunkBuilder::UseRegisterOrConstantAtStart(HValue* value) {
649 return value->IsConstant()
650 ? chunk_->DefineConstantOperand(HConstant::cast(value))
651 : UseRegisterAtStart(value);
652}
653
654
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000655LOperand* LChunkBuilder::UseAny(HValue* value) {
656 return value->IsConstant()
657 ? chunk_->DefineConstantOperand(HConstant::cast(value))
ulan@chromium.org812308e2012-02-29 15:58:45 +0000658 : Use(value, new(zone()) LUnallocated(LUnallocated::ANY));
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000659}
660
661
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000662LOperand* LChunkBuilder::Use(HValue* value, LUnallocated* operand) {
663 if (value->EmitAtUses()) {
664 HInstruction* instr = HInstruction::cast(value);
665 VisitInstruction(instr);
666 }
rossberg@chromium.org994edf62012-02-06 10:12:55 +0000667 operand->set_virtual_register(value->id());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000668 return operand;
669}
670
671
672template<int I, int T>
673LInstruction* LChunkBuilder::Define(LTemplateInstruction<1, I, T>* instr,
674 LUnallocated* result) {
rossberg@chromium.org994edf62012-02-06 10:12:55 +0000675 result->set_virtual_register(current_instruction_->id());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000676 instr->set_result(result);
677 return instr;
678}
679
680
681template<int I, int T>
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000682LInstruction* LChunkBuilder::DefineAsRegister(
683 LTemplateInstruction<1, I, T>* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000684 return Define(instr,
685 new(zone()) LUnallocated(LUnallocated::MUST_HAVE_REGISTER));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000686}
687
688
689template<int I, int T>
690LInstruction* LChunkBuilder::DefineAsSpilled(
691 LTemplateInstruction<1, I, T>* instr,
692 int index) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000693 return Define(instr,
694 new(zone()) LUnallocated(LUnallocated::FIXED_SLOT, index));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000695}
696
697
698template<int I, int T>
699LInstruction* LChunkBuilder::DefineSameAsFirst(
700 LTemplateInstruction<1, I, T>* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000701 return Define(instr,
702 new(zone()) LUnallocated(LUnallocated::SAME_AS_FIRST_INPUT));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000703}
704
705
706template<int I, int T>
707LInstruction* LChunkBuilder::DefineFixed(LTemplateInstruction<1, I, T>* instr,
708 Register reg) {
709 return Define(instr, ToUnallocated(reg));
710}
711
712
713template<int I, int T>
714LInstruction* LChunkBuilder::DefineFixedDouble(
715 LTemplateInstruction<1, I, T>* instr,
716 XMMRegister reg) {
717 return Define(instr, ToUnallocated(reg));
718}
719
720
721LInstruction* LChunkBuilder::AssignEnvironment(LInstruction* instr) {
722 HEnvironment* hydrogen_env = current_block_->last_environment();
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000723 int argument_index_accumulator = 0;
724 instr->set_environment(CreateEnvironment(hydrogen_env,
725 &argument_index_accumulator));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000726 return instr;
727}
728
729
fschneider@chromium.org1df6b472011-01-26 08:23:03 +0000730LInstruction* LChunkBuilder::SetInstructionPendingDeoptimizationEnvironment(
731 LInstruction* instr, int ast_id) {
kmillikin@chromium.org31b12772011-02-02 16:08:26 +0000732 ASSERT(instruction_pending_deoptimization_environment_ == NULL);
fschneider@chromium.org1df6b472011-01-26 08:23:03 +0000733 ASSERT(pending_deoptimization_ast_id_ == AstNode::kNoNumber);
kmillikin@chromium.org31b12772011-02-02 16:08:26 +0000734 instruction_pending_deoptimization_environment_ = instr;
fschneider@chromium.org1df6b472011-01-26 08:23:03 +0000735 pending_deoptimization_ast_id_ = ast_id;
736 return instr;
737}
738
739
740void LChunkBuilder::ClearInstructionPendingDeoptimizationEnvironment() {
kmillikin@chromium.org31b12772011-02-02 16:08:26 +0000741 instruction_pending_deoptimization_environment_ = NULL;
fschneider@chromium.org1df6b472011-01-26 08:23:03 +0000742 pending_deoptimization_ast_id_ = AstNode::kNoNumber;
743}
744
745
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000746LInstruction* LChunkBuilder::MarkAsCall(LInstruction* instr,
747 HInstruction* hinstr,
748 CanDeoptimize can_deoptimize) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000749#ifdef DEBUG
750 instr->VerifyCall();
751#endif
752 instr->MarkAsCall();
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000753 instr = AssignPointerMap(instr);
754
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000755 if (hinstr->HasObservableSideEffects()) {
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000756 ASSERT(hinstr->next()->IsSimulate());
757 HSimulate* sim = HSimulate::cast(hinstr->next());
fschneider@chromium.org1df6b472011-01-26 08:23:03 +0000758 instr = SetInstructionPendingDeoptimizationEnvironment(
759 instr, sim->ast_id());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000760 }
761
762 // If instruction does not have side-effects lazy deoptimization
763 // after the call will try to deoptimize to the point before the call.
764 // Thus we still need to attach environment to this call even if
765 // call sequence can not deoptimize eagerly.
766 bool needs_environment =
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +0000767 (can_deoptimize == CAN_DEOPTIMIZE_EAGERLY) ||
768 !hinstr->HasObservableSideEffects();
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000769 if (needs_environment && !instr->HasEnvironment()) {
770 instr = AssignEnvironment(instr);
771 }
772
773 return instr;
774}
775
776
777LInstruction* LChunkBuilder::MarkAsSaveDoubles(LInstruction* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000778 instr->MarkAsSaveDoubles();
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000779 return instr;
780}
781
782
783LInstruction* LChunkBuilder::AssignPointerMap(LInstruction* instr) {
784 ASSERT(!instr->HasPointerMap());
ulan@chromium.org812308e2012-02-29 15:58:45 +0000785 instr->set_pointer_map(new(zone()) LPointerMap(position_));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000786 return instr;
787}
788
789
790LUnallocated* LChunkBuilder::TempRegister() {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000791 LUnallocated* operand =
792 new(zone()) LUnallocated(LUnallocated::MUST_HAVE_REGISTER);
rossberg@chromium.org994edf62012-02-06 10:12:55 +0000793 operand->set_virtual_register(allocator_->GetVirtualRegister());
794 if (!allocator_->AllocationOk()) Abort("Not enough virtual registers.");
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000795 return operand;
796}
797
798
799LOperand* LChunkBuilder::FixedTemp(Register reg) {
800 LUnallocated* operand = ToUnallocated(reg);
rossberg@chromium.org994edf62012-02-06 10:12:55 +0000801 ASSERT(operand->HasFixedPolicy());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000802 return operand;
803}
804
805
806LOperand* LChunkBuilder::FixedTemp(XMMRegister reg) {
807 LUnallocated* operand = ToUnallocated(reg);
rossberg@chromium.org994edf62012-02-06 10:12:55 +0000808 ASSERT(operand->HasFixedPolicy());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000809 return operand;
810}
811
812
813LInstruction* LChunkBuilder::DoBlockEntry(HBlockEntry* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000814 return new(zone()) LLabel(instr->block());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000815}
816
817
jkummerow@chromium.orge297f592011-06-08 10:05:15 +0000818LInstruction* LChunkBuilder::DoSoftDeoptimize(HSoftDeoptimize* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000819 return AssignEnvironment(new(zone()) LDeoptimize);
jkummerow@chromium.orge297f592011-06-08 10:05:15 +0000820}
821
822
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000823LInstruction* LChunkBuilder::DoDeoptimize(HDeoptimize* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +0000824 return AssignEnvironment(new(zone()) LDeoptimize);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000825}
826
827
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000828LInstruction* LChunkBuilder::DoShift(Token::Value op,
829 HBitwiseBinaryOperation* instr) {
830 if (instr->representation().IsTagged()) {
831 ASSERT(instr->left()->representation().IsTagged());
832 ASSERT(instr->right()->representation().IsTagged());
833
834 LOperand* left = UseFixed(instr->left(), rdx);
835 LOperand* right = UseFixed(instr->right(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +0000836 LArithmeticT* result = new(zone()) LArithmeticT(op, left, right);
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000837 return MarkAsCall(DefineFixed(result, rax), instr);
838 }
839
840 ASSERT(instr->representation().IsInteger32());
jkummerow@chromium.orgddda9e82011-07-06 11:27:02 +0000841 ASSERT(instr->left()->representation().IsInteger32());
842 ASSERT(instr->right()->representation().IsInteger32());
843 LOperand* left = UseRegisterAtStart(instr->left());
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000844
jkummerow@chromium.orgddda9e82011-07-06 11:27:02 +0000845 HValue* right_value = instr->right();
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000846 LOperand* right = NULL;
847 int constant_value = 0;
848 if (right_value->IsConstant()) {
849 HConstant* constant = HConstant::cast(right_value);
850 right = chunk_->DefineConstantOperand(constant);
851 constant_value = constant->Integer32Value() & 0x1f;
852 } else {
853 right = UseFixed(right_value, rcx);
854 }
855
erik.corry@gmail.com3847bd52011-04-27 10:38:56 +0000856 // Shift operations can only deoptimize if we do a logical shift by 0 and
857 // the result cannot be truncated to int32.
858 bool may_deopt = (op == Token::SHR && constant_value == 0);
859 bool does_deopt = false;
860 if (may_deopt) {
861 for (HUseIterator it(instr->uses()); !it.Done(); it.Advance()) {
862 if (!it.value()->CheckFlag(HValue::kTruncatingToInt32)) {
863 does_deopt = true;
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000864 break;
865 }
866 }
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000867 }
868
erik.corry@gmail.com3847bd52011-04-27 10:38:56 +0000869 LInstruction* result =
ulan@chromium.org812308e2012-02-29 15:58:45 +0000870 DefineSameAsFirst(new(zone()) LShiftI(op, left, right, does_deopt));
erik.corry@gmail.com3847bd52011-04-27 10:38:56 +0000871 return does_deopt ? AssignEnvironment(result) : result;
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000872}
873
874
875LInstruction* LChunkBuilder::DoArithmeticD(Token::Value op,
876 HArithmeticBinaryOperation* instr) {
kmillikin@chromium.org49edbdf2011-02-16 12:32:18 +0000877 ASSERT(instr->representation().IsDouble());
878 ASSERT(instr->left()->representation().IsDouble());
879 ASSERT(instr->right()->representation().IsDouble());
karlklose@chromium.org8f806e82011-03-07 14:06:08 +0000880 ASSERT(op != Token::MOD);
kmillikin@chromium.org49edbdf2011-02-16 12:32:18 +0000881 LOperand* left = UseRegisterAtStart(instr->left());
882 LOperand* right = UseRegisterAtStart(instr->right());
ulan@chromium.org812308e2012-02-29 15:58:45 +0000883 LArithmeticD* result = new(zone()) LArithmeticD(op, left, right);
kmillikin@chromium.org49edbdf2011-02-16 12:32:18 +0000884 return DefineSameAsFirst(result);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000885}
886
887
888LInstruction* LChunkBuilder::DoArithmeticT(Token::Value op,
889 HArithmeticBinaryOperation* instr) {
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000890 ASSERT(op == Token::ADD ||
891 op == Token::DIV ||
892 op == Token::MOD ||
893 op == Token::MUL ||
894 op == Token::SUB);
895 HValue* left = instr->left();
896 HValue* right = instr->right();
897 ASSERT(left->representation().IsTagged());
898 ASSERT(right->representation().IsTagged());
899 LOperand* left_operand = UseFixed(left, rdx);
900 LOperand* right_operand = UseFixed(right, rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +0000901 LArithmeticT* result =
902 new(zone()) LArithmeticT(op, left_operand, right_operand);
erik.corry@gmail.com0511e242011-01-19 11:11:08 +0000903 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000904}
905
906
907void LChunkBuilder::DoBasicBlock(HBasicBlock* block, HBasicBlock* next_block) {
908 ASSERT(is_building());
909 current_block_ = block;
910 next_block_ = next_block;
911 if (block->IsStartBlock()) {
912 block->UpdateEnvironment(graph_->start_environment());
913 argument_count_ = 0;
914 } else if (block->predecessors()->length() == 1) {
915 // We have a single predecessor => copy environment and outgoing
916 // argument count from the predecessor.
917 ASSERT(block->phis()->length() == 0);
918 HBasicBlock* pred = block->predecessors()->at(0);
919 HEnvironment* last_environment = pred->last_environment();
920 ASSERT(last_environment != NULL);
921 // Only copy the environment, if it is later used again.
922 if (pred->end()->SecondSuccessor() == NULL) {
923 ASSERT(pred->end()->FirstSuccessor() == block);
924 } else {
925 if (pred->end()->FirstSuccessor()->block_id() > block->block_id() ||
926 pred->end()->SecondSuccessor()->block_id() > block->block_id()) {
927 last_environment = last_environment->Copy();
928 }
929 }
930 block->UpdateEnvironment(last_environment);
931 ASSERT(pred->argument_count() >= 0);
932 argument_count_ = pred->argument_count();
933 } else {
934 // We are at a state join => process phis.
935 HBasicBlock* pred = block->predecessors()->at(0);
936 // No need to copy the environment, it cannot be used later.
937 HEnvironment* last_environment = pred->last_environment();
938 for (int i = 0; i < block->phis()->length(); ++i) {
939 HPhi* phi = block->phis()->at(i);
940 last_environment->SetValueAt(phi->merged_index(), phi);
941 }
942 for (int i = 0; i < block->deleted_phis()->length(); ++i) {
943 last_environment->SetValueAt(block->deleted_phis()->at(i),
944 graph_->GetConstantUndefined());
945 }
946 block->UpdateEnvironment(last_environment);
947 // Pick up the outgoing argument count of one of the predecessors.
948 argument_count_ = pred->argument_count();
949 }
950 HInstruction* current = block->first();
951 int start = chunk_->instructions()->length();
952 while (current != NULL && !is_aborted()) {
953 // Code for constants in registers is generated lazily.
954 if (!current->EmitAtUses()) {
955 VisitInstruction(current);
956 }
957 current = current->next();
958 }
959 int end = chunk_->instructions()->length() - 1;
960 if (end >= start) {
961 block->set_first_instruction_index(start);
962 block->set_last_instruction_index(end);
963 }
964 block->set_argument_count(argument_count_);
965 next_block_ = NULL;
966 current_block_ = NULL;
967}
968
969
970void LChunkBuilder::VisitInstruction(HInstruction* current) {
971 HInstruction* old_current = current_instruction_;
972 current_instruction_ = current;
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000973 if (current->has_position()) position_ = current->position();
974 LInstruction* instr = current->CompileToLithium(this);
975
976 if (instr != NULL) {
977 if (FLAG_stress_pointer_maps && !instr->HasPointerMap()) {
978 instr = AssignPointerMap(instr);
979 }
980 if (FLAG_stress_environments && !instr->HasEnvironment()) {
981 instr = AssignEnvironment(instr);
982 }
ricow@chromium.org4f693d62011-07-04 14:01:31 +0000983 instr->set_hydrogen_value(current);
ricow@chromium.org83aa5492011-02-07 12:42:56 +0000984 chunk_->AddInstruction(instr, current_block_);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000985 }
986 current_instruction_ = old_current;
987}
988
989
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000990LEnvironment* LChunkBuilder::CreateEnvironment(
991 HEnvironment* hydrogen_env,
992 int* argument_index_accumulator) {
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000993 if (hydrogen_env == NULL) return NULL;
994
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000995 LEnvironment* outer =
996 CreateEnvironment(hydrogen_env->outer(), argument_index_accumulator);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +0000997 int ast_id = hydrogen_env->ast_id();
ulan@chromium.org967e2702012-02-28 09:49:15 +0000998 ASSERT(ast_id != AstNode::kNoNumber ||
999 hydrogen_env->frame_type() != JS_FUNCTION);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001000 int value_count = hydrogen_env->length();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001001 LEnvironment* result = new(zone()) LEnvironment(
1002 hydrogen_env->closure(),
1003 hydrogen_env->frame_type(),
1004 ast_id,
1005 hydrogen_env->parameter_count(),
1006 argument_count_,
1007 value_count,
1008 outer);
yangguo@chromium.org659ceec2012-01-26 07:37:54 +00001009 int argument_index = *argument_index_accumulator;
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001010 for (int i = 0; i < value_count; ++i) {
karlklose@chromium.org83a47282011-05-11 11:54:09 +00001011 if (hydrogen_env->is_special_index(i)) continue;
1012
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001013 HValue* value = hydrogen_env->values()->at(i);
1014 LOperand* op = NULL;
1015 if (value->IsArgumentsObject()) {
1016 op = NULL;
1017 } else if (value->IsPushArgument()) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001018 op = new(zone()) LArgument(argument_index++);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001019 } else {
erik.corry@gmail.com0511e242011-01-19 11:11:08 +00001020 op = UseAny(value);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001021 }
1022 result->AddValue(op, value->representation());
1023 }
1024
ulan@chromium.org967e2702012-02-28 09:49:15 +00001025 if (hydrogen_env->frame_type() == JS_FUNCTION) {
yangguo@chromium.org659ceec2012-01-26 07:37:54 +00001026 *argument_index_accumulator = argument_index;
1027 }
1028
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001029 return result;
1030}
1031
1032
1033LInstruction* LChunkBuilder::DoGoto(HGoto* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001034 return new(zone()) LGoto(instr->FirstSuccessor()->block_id());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001035}
1036
1037
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001038LInstruction* LChunkBuilder::DoBranch(HBranch* instr) {
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +00001039 HValue* value = instr->value();
1040 if (value->EmitAtUses()) {
1041 ASSERT(value->IsConstant());
1042 ASSERT(!value->representation().IsDouble());
1043 HBasicBlock* successor = HConstant::cast(value)->ToBoolean()
ager@chromium.orgea91cc52011-05-23 06:06:11 +00001044 ? instr->FirstSuccessor()
1045 : instr->SecondSuccessor();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001046 return new(zone()) LGoto(successor->block_id());
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00001047 }
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +00001048
ulan@chromium.org812308e2012-02-29 15:58:45 +00001049 LBranch* result = new(zone()) LBranch(UseRegister(value));
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +00001050 // Tagged values that are not known smis or booleans require a
1051 // deoptimization environment.
1052 Representation rep = value->representation();
1053 HType type = value->type();
1054 if (rep.IsTagged() && !type.IsSmi() && !type.IsBoolean()) {
1055 return AssignEnvironment(result);
1056 }
1057 return result;
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001058}
1059
1060
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00001061LInstruction* LChunkBuilder::DoCompareMap(HCompareMap* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001062 ASSERT(instr->value()->representation().IsTagged());
1063 LOperand* value = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001064 return new(zone()) LCmpMapAndBranch(value);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001065}
1066
1067
1068LInstruction* LChunkBuilder::DoArgumentsLength(HArgumentsLength* length) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001069 return DefineAsRegister(new(zone()) LArgumentsLength(Use(length->value())));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001070}
1071
1072
1073LInstruction* LChunkBuilder::DoArgumentsElements(HArgumentsElements* elems) {
danno@chromium.orgb2a1c072012-03-23 15:47:56 +00001074 return DefineAsRegister(new(zone()) LArgumentsElements);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001075}
1076
1077
1078LInstruction* LChunkBuilder::DoInstanceOf(HInstanceOf* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001079 LOperand* left = UseFixed(instr->left(), rax);
1080 LOperand* right = UseFixed(instr->right(), rdx);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001081 LInstanceOf* result = new(zone()) LInstanceOf(left, right);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001082 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001083}
1084
1085
1086LInstruction* LChunkBuilder::DoInstanceOfKnownGlobal(
1087 HInstanceOfKnownGlobal* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001088 LInstanceOfKnownGlobal* result =
ulan@chromium.org812308e2012-02-29 15:58:45 +00001089 new(zone()) LInstanceOfKnownGlobal(UseFixed(instr->left(), rax),
1090 FixedTemp(rdi));
ager@chromium.org9ee27ae2011-03-02 13:43:26 +00001091 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001092}
1093
1094
yangguo@chromium.org154ff992012-03-13 08:09:54 +00001095LInstruction* LChunkBuilder::DoWrapReceiver(HWrapReceiver* instr) {
1096 LOperand* receiver = UseRegister(instr->receiver());
1097 LOperand* function = UseRegisterAtStart(instr->function());
1098 LWrapReceiver* result = new(zone()) LWrapReceiver(receiver, function);
1099 return AssignEnvironment(DefineSameAsFirst(result));
1100}
1101
1102
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001103LInstruction* LChunkBuilder::DoApplyArguments(HApplyArguments* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001104 LOperand* function = UseFixed(instr->function(), rdi);
1105 LOperand* receiver = UseFixed(instr->receiver(), rax);
1106 LOperand* length = UseFixed(instr->length(), rbx);
1107 LOperand* elements = UseFixed(instr->elements(), rcx);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001108 LApplyArguments* result = new(zone()) LApplyArguments(function,
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001109 receiver,
1110 length,
1111 elements);
1112 return MarkAsCall(DefineFixed(result, rax), instr, CAN_DEOPTIMIZE_EAGERLY);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001113}
1114
1115
1116LInstruction* LChunkBuilder::DoPushArgument(HPushArgument* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00001117 ++argument_count_;
1118 LOperand* argument = UseOrConstant(instr->argument());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001119 return new(zone()) LPushArgument(argument);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001120}
1121
1122
ricow@chromium.orgd2be9012011-06-01 06:00:58 +00001123LInstruction* LChunkBuilder::DoThisFunction(HThisFunction* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001124 return instr->HasNoUses()
1125 ? NULL
1126 : DefineAsRegister(new(zone()) LThisFunction);
ricow@chromium.orgd2be9012011-06-01 06:00:58 +00001127}
1128
1129
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001130LInstruction* LChunkBuilder::DoContext(HContext* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001131 return instr->HasNoUses() ? NULL : DefineAsRegister(new(zone()) LContext);
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001132}
1133
1134
1135LInstruction* LChunkBuilder::DoOuterContext(HOuterContext* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001136 LOperand* context = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001137 return DefineAsRegister(new(zone()) LOuterContext(context));
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001138}
1139
1140
yangguo@chromium.org56454712012-02-16 15:33:53 +00001141LInstruction* LChunkBuilder::DoDeclareGlobals(HDeclareGlobals* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001142 return MarkAsCall(new(zone()) LDeclareGlobals, instr);
yangguo@chromium.org56454712012-02-16 15:33:53 +00001143}
1144
1145
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001146LInstruction* LChunkBuilder::DoGlobalObject(HGlobalObject* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001147 return DefineAsRegister(new(zone()) LGlobalObject);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001148}
1149
1150
1151LInstruction* LChunkBuilder::DoGlobalReceiver(HGlobalReceiver* instr) {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00001152 LOperand* global_object = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001153 return DefineAsRegister(new(zone()) LGlobalReceiver(global_object));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001154}
1155
1156
1157LInstruction* LChunkBuilder::DoCallConstantFunction(
1158 HCallConstantFunction* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001159 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001160 return MarkAsCall(DefineFixed(new(zone()) LCallConstantFunction, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001161}
1162
1163
danno@chromium.org160a7b02011-04-18 15:51:38 +00001164LInstruction* LChunkBuilder::DoInvokeFunction(HInvokeFunction* instr) {
1165 LOperand* function = UseFixed(instr->function(), rdi);
1166 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001167 LInvokeFunction* result = new(zone()) LInvokeFunction(function);
danno@chromium.org160a7b02011-04-18 15:51:38 +00001168 return MarkAsCall(DefineFixed(result, rax), instr, CANNOT_DEOPTIMIZE_EAGERLY);
1169}
1170
1171
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001172LInstruction* LChunkBuilder::DoUnaryMathOperation(HUnaryMathOperation* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001173 BuiltinFunctionId op = instr->op();
ulan@chromium.org9a21ec42012-03-06 08:42:24 +00001174 if (op == kMathLog || op == kMathSin || op == kMathCos || op == kMathTan) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001175 LOperand* input = UseFixedDouble(instr->value(), xmm1);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001176 LUnaryMathOperation* result = new(zone()) LUnaryMathOperation(input);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001177 return MarkAsCall(DefineFixedDouble(result, xmm1), instr);
1178 } else {
1179 LOperand* input = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001180 LUnaryMathOperation* result = new(zone()) LUnaryMathOperation(input);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001181 switch (op) {
1182 case kMathAbs:
1183 return AssignEnvironment(AssignPointerMap(DefineSameAsFirst(result)));
1184 case kMathFloor:
1185 return AssignEnvironment(DefineAsRegister(result));
1186 case kMathRound:
1187 return AssignEnvironment(DefineAsRegister(result));
1188 case kMathSqrt:
1189 return DefineSameAsFirst(result);
1190 case kMathPowHalf:
1191 return DefineSameAsFirst(result);
1192 default:
1193 UNREACHABLE();
1194 return NULL;
1195 }
1196 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001197}
1198
1199
1200LInstruction* LChunkBuilder::DoCallKeyed(HCallKeyed* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001201 ASSERT(instr->key()->representation().IsTagged());
1202 LOperand* key = UseFixed(instr->key(), rcx);
1203 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001204 LCallKeyed* result = new(zone()) LCallKeyed(key);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001205 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001206}
1207
1208
1209LInstruction* LChunkBuilder::DoCallNamed(HCallNamed* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001210 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001211 return MarkAsCall(DefineFixed(new(zone()) LCallNamed, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001212}
1213
1214
1215LInstruction* LChunkBuilder::DoCallGlobal(HCallGlobal* instr) {
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +00001216 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001217 return MarkAsCall(DefineFixed(new(zone()) LCallGlobal, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001218}
1219
1220
1221LInstruction* LChunkBuilder::DoCallKnownGlobal(HCallKnownGlobal* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001222 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001223 return MarkAsCall(DefineFixed(new(zone()) LCallKnownGlobal, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001224}
1225
1226
1227LInstruction* LChunkBuilder::DoCallNew(HCallNew* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00001228 LOperand* constructor = UseFixed(instr->constructor(), rdi);
1229 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001230 LCallNew* result = new(zone()) LCallNew(constructor);
ager@chromium.org378b34e2011-01-28 08:04:38 +00001231 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001232}
1233
1234
1235LInstruction* LChunkBuilder::DoCallFunction(HCallFunction* instr) {
danno@chromium.orgc612e022011-11-10 11:38:15 +00001236 LOperand* function = UseFixed(instr->function(), rdi);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001237 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001238 LCallFunction* result = new(zone()) LCallFunction(function);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001239 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001240}
1241
1242
1243LInstruction* LChunkBuilder::DoCallRuntime(HCallRuntime* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001244 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001245 return MarkAsCall(DefineFixed(new(zone()) LCallRuntime, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001246}
1247
1248
1249LInstruction* LChunkBuilder::DoShr(HShr* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001250 return DoShift(Token::SHR, instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001251}
1252
1253
1254LInstruction* LChunkBuilder::DoSar(HSar* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001255 return DoShift(Token::SAR, instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001256}
1257
1258
1259LInstruction* LChunkBuilder::DoShl(HShl* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001260 return DoShift(Token::SHL, instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001261}
1262
1263
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +00001264LInstruction* LChunkBuilder::DoBitwise(HBitwise* instr) {
1265 if (instr->representation().IsInteger32()) {
1266 ASSERT(instr->left()->representation().IsInteger32());
1267 ASSERT(instr->right()->representation().IsInteger32());
1268
1269 LOperand* left = UseRegisterAtStart(instr->LeastConstantOperand());
1270 LOperand* right = UseOrConstantAtStart(instr->MostConstantOperand());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001271 return DefineSameAsFirst(new(zone()) LBitI(left, right));
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +00001272 } else {
1273 ASSERT(instr->representation().IsTagged());
1274 ASSERT(instr->left()->representation().IsTagged());
1275 ASSERT(instr->right()->representation().IsTagged());
1276
1277 LOperand* left = UseFixed(instr->left(), rdx);
1278 LOperand* right = UseFixed(instr->right(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001279 LArithmeticT* result = new(zone()) LArithmeticT(instr->op(), left, right);
jkummerow@chromium.orgc3b37122011-11-07 10:14:12 +00001280 return MarkAsCall(DefineFixed(result, rax), instr);
1281 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001282}
1283
1284
1285LInstruction* LChunkBuilder::DoBitNot(HBitNot* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001286 ASSERT(instr->value()->representation().IsInteger32());
1287 ASSERT(instr->representation().IsInteger32());
1288 LOperand* input = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001289 LBitNotI* result = new(zone()) LBitNotI(input);
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001290 return DefineSameAsFirst(result);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001291}
1292
1293
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001294LInstruction* LChunkBuilder::DoDiv(HDiv* instr) {
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00001295 if (instr->representation().IsDouble()) {
1296 return DoArithmeticD(Token::DIV, instr);
1297 } else if (instr->representation().IsInteger32()) {
1298 // The temporary operand is necessary to ensure that right is not allocated
1299 // into rdx.
1300 LOperand* temp = FixedTemp(rdx);
1301 LOperand* dividend = UseFixed(instr->left(), rax);
1302 LOperand* divisor = UseRegister(instr->right());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001303 LDivI* result = new(zone()) LDivI(dividend, divisor, temp);
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00001304 return AssignEnvironment(DefineFixed(result, rax));
1305 } else {
1306 ASSERT(instr->representation().IsTagged());
1307 return DoArithmeticT(Token::DIV, instr);
1308 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001309}
1310
1311
1312LInstruction* LChunkBuilder::DoMod(HMod* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001313 if (instr->representation().IsInteger32()) {
1314 ASSERT(instr->left()->representation().IsInteger32());
1315 ASSERT(instr->right()->representation().IsInteger32());
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001316
1317 LInstruction* result;
1318 if (instr->HasPowerOf2Divisor()) {
1319 ASSERT(!instr->CheckFlag(HValue::kCanBeDivByZero));
1320 LOperand* value = UseRegisterAtStart(instr->left());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001321 LModI* mod =
1322 new(zone()) LModI(value, UseOrConstant(instr->right()), NULL);
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001323 result = DefineSameAsFirst(mod);
1324 } else {
1325 // The temporary operand is necessary to ensure that right is not
1326 // allocated into edx.
1327 LOperand* temp = FixedTemp(rdx);
1328 LOperand* value = UseFixed(instr->left(), rax);
1329 LOperand* divisor = UseRegister(instr->right());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001330 LModI* mod = new(zone()) LModI(value, divisor, temp);
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001331 result = DefineFixed(mod, rdx);
1332 }
1333
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001334 return (instr->CheckFlag(HValue::kBailoutOnMinusZero) ||
1335 instr->CheckFlag(HValue::kCanBeDivByZero))
1336 ? AssignEnvironment(result)
1337 : result;
1338 } else if (instr->representation().IsTagged()) {
1339 return DoArithmeticT(Token::MOD, instr);
1340 } else {
1341 ASSERT(instr->representation().IsDouble());
1342 // We call a C function for double modulo. It can't trigger a GC.
1343 // We need to use fixed result register for the call.
1344 // TODO(fschneider): Allow any register as input registers.
karlklose@chromium.org8f806e82011-03-07 14:06:08 +00001345 LOperand* left = UseFixedDouble(instr->left(), xmm2);
1346 LOperand* right = UseFixedDouble(instr->right(), xmm1);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001347 LArithmeticD* result = new(zone()) LArithmeticD(Token::MOD, left, right);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001348 return MarkAsCall(DefineFixedDouble(result, xmm1), instr);
1349 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001350}
1351
1352
1353LInstruction* LChunkBuilder::DoMul(HMul* instr) {
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00001354 if (instr->representation().IsInteger32()) {
1355 ASSERT(instr->left()->representation().IsInteger32());
1356 ASSERT(instr->right()->representation().IsInteger32());
1357 LOperand* left = UseRegisterAtStart(instr->LeastConstantOperand());
1358 LOperand* right = UseOrConstant(instr->MostConstantOperand());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001359 LMulI* mul = new(zone()) LMulI(left, right);
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +00001360 if (instr->CheckFlag(HValue::kCanOverflow) ||
1361 instr->CheckFlag(HValue::kBailoutOnMinusZero)) {
1362 AssignEnvironment(mul);
1363 }
1364 return DefineSameAsFirst(mul);
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00001365 } else if (instr->representation().IsDouble()) {
1366 return DoArithmeticD(Token::MUL, instr);
1367 } else {
1368 ASSERT(instr->representation().IsTagged());
1369 return DoArithmeticT(Token::MUL, instr);
1370 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001371}
1372
1373
1374LInstruction* LChunkBuilder::DoSub(HSub* instr) {
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001375 if (instr->representation().IsInteger32()) {
1376 ASSERT(instr->left()->representation().IsInteger32());
1377 ASSERT(instr->right()->representation().IsInteger32());
1378 LOperand* left = UseRegisterAtStart(instr->left());
1379 LOperand* right = UseOrConstantAtStart(instr->right());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001380 LSubI* sub = new(zone()) LSubI(left, right);
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001381 LInstruction* result = DefineSameAsFirst(sub);
1382 if (instr->CheckFlag(HValue::kCanOverflow)) {
1383 result = AssignEnvironment(result);
1384 }
1385 return result;
1386 } else if (instr->representation().IsDouble()) {
1387 return DoArithmeticD(Token::SUB, instr);
1388 } else {
1389 ASSERT(instr->representation().IsTagged());
1390 return DoArithmeticT(Token::SUB, instr);
1391 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001392}
1393
1394
1395LInstruction* LChunkBuilder::DoAdd(HAdd* instr) {
erik.corry@gmail.com0511e242011-01-19 11:11:08 +00001396 if (instr->representation().IsInteger32()) {
1397 ASSERT(instr->left()->representation().IsInteger32());
1398 ASSERT(instr->right()->representation().IsInteger32());
1399 LOperand* left = UseRegisterAtStart(instr->LeastConstantOperand());
1400 LOperand* right = UseOrConstantAtStart(instr->MostConstantOperand());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001401 LAddI* add = new(zone()) LAddI(left, right);
erik.corry@gmail.com0511e242011-01-19 11:11:08 +00001402 LInstruction* result = DefineSameAsFirst(add);
1403 if (instr->CheckFlag(HValue::kCanOverflow)) {
1404 result = AssignEnvironment(result);
1405 }
1406 return result;
1407 } else if (instr->representation().IsDouble()) {
kmillikin@chromium.org49edbdf2011-02-16 12:32:18 +00001408 return DoArithmeticD(Token::ADD, instr);
erik.corry@gmail.com0511e242011-01-19 11:11:08 +00001409 } else {
1410 ASSERT(instr->representation().IsTagged());
1411 return DoArithmeticT(Token::ADD, instr);
1412 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001413 return NULL;
1414}
1415
1416
1417LInstruction* LChunkBuilder::DoPower(HPower* instr) {
ager@chromium.org9ee27ae2011-03-02 13:43:26 +00001418 ASSERT(instr->representation().IsDouble());
1419 // We call a C function for double power. It can't trigger a GC.
1420 // We need to use fixed result register for the call.
1421 Representation exponent_type = instr->right()->representation();
1422 ASSERT(instr->left()->representation().IsDouble());
1423 LOperand* left = UseFixedDouble(instr->left(), xmm2);
1424 LOperand* right = exponent_type.IsDouble() ?
1425 UseFixedDouble(instr->right(), xmm1) :
1426#ifdef _WIN64
1427 UseFixed(instr->right(), rdx);
1428#else
1429 UseFixed(instr->right(), rdi);
1430#endif
ulan@chromium.org812308e2012-02-29 15:58:45 +00001431 LPower* result = new(zone()) LPower(left, right);
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +00001432 return MarkAsCall(DefineFixedDouble(result, xmm3), instr,
ager@chromium.org9ee27ae2011-03-02 13:43:26 +00001433 CAN_DEOPTIMIZE_EAGERLY);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001434}
1435
1436
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +00001437LInstruction* LChunkBuilder::DoRandom(HRandom* instr) {
1438 ASSERT(instr->representation().IsDouble());
1439 ASSERT(instr->global_object()->representation().IsTagged());
1440#ifdef _WIN64
1441 LOperand* global_object = UseFixed(instr->global_object(), rcx);
1442#else
1443 LOperand* global_object = UseFixed(instr->global_object(), rdi);
1444#endif
ulan@chromium.org812308e2012-02-29 15:58:45 +00001445 LRandom* result = new(zone()) LRandom(global_object);
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +00001446 return MarkAsCall(DefineFixedDouble(result, xmm1), instr);
1447}
1448
1449
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001450LInstruction* LChunkBuilder::DoCompareGeneric(HCompareGeneric* instr) {
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001451 ASSERT(instr->left()->representation().IsTagged());
1452 ASSERT(instr->right()->representation().IsTagged());
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +00001453 LOperand* left = UseFixed(instr->left(), rdx);
1454 LOperand* right = UseFixed(instr->right(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001455 LCmpT* result = new(zone()) LCmpT(left, right);
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001456 return MarkAsCall(DefineFixed(result, rax), instr);
1457}
1458
1459
1460LInstruction* LChunkBuilder::DoCompareIDAndBranch(
1461 HCompareIDAndBranch* instr) {
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00001462 Representation r = instr->GetInputRepresentation();
1463 if (r.IsInteger32()) {
1464 ASSERT(instr->left()->representation().IsInteger32());
1465 ASSERT(instr->right()->representation().IsInteger32());
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +00001466 LOperand* left = UseRegisterOrConstantAtStart(instr->left());
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00001467 LOperand* right = UseOrConstantAtStart(instr->right());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001468 return new(zone()) LCmpIDAndBranch(left, right);
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001469 } else {
1470 ASSERT(r.IsDouble());
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00001471 ASSERT(instr->left()->representation().IsDouble());
1472 ASSERT(instr->right()->representation().IsDouble());
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +00001473 LOperand* left;
1474 LOperand* right;
1475 if (instr->left()->IsConstant() && instr->right()->IsConstant()) {
1476 left = UseRegisterOrConstantAtStart(instr->left());
1477 right = UseRegisterOrConstantAtStart(instr->right());
1478 } else {
1479 left = UseRegisterAtStart(instr->left());
1480 right = UseRegisterAtStart(instr->right());
1481 }
ulan@chromium.org812308e2012-02-29 15:58:45 +00001482 return new(zone()) LCmpIDAndBranch(left, right);
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00001483 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001484}
1485
1486
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001487LInstruction* LChunkBuilder::DoCompareObjectEqAndBranch(
1488 HCompareObjectEqAndBranch* instr) {
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001489 LOperand* left = UseRegisterAtStart(instr->left());
1490 LOperand* right = UseRegisterAtStart(instr->right());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001491 return new(zone()) LCmpObjectEqAndBranch(left, right);
vegorov@chromium.org7304bca2011-05-16 12:14:13 +00001492}
1493
1494
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001495LInstruction* LChunkBuilder::DoCompareConstantEqAndBranch(
1496 HCompareConstantEqAndBranch* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001497 LOperand* value = UseRegisterAtStart(instr->value());
1498 return new(zone()) LCmpConstantEqAndBranch(value);
whesse@chromium.org7b260152011-06-20 15:33:18 +00001499}
1500
1501
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001502LInstruction* LChunkBuilder::DoIsNilAndBranch(HIsNilAndBranch* instr) {
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001503 ASSERT(instr->value()->representation().IsTagged());
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001504 LOperand* temp = instr->kind() == kStrictEquality ? NULL : TempRegister();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001505 return new(zone()) LIsNilAndBranch(UseRegisterAtStart(instr->value()), temp);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001506}
1507
1508
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001509LInstruction* LChunkBuilder::DoIsObjectAndBranch(HIsObjectAndBranch* instr) {
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001510 ASSERT(instr->value()->representation().IsTagged());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001511 return new(zone()) LIsObjectAndBranch(UseRegisterAtStart(instr->value()));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001512}
1513
1514
erikcorry0ad885c2011-11-21 13:51:57 +00001515LInstruction* LChunkBuilder::DoIsStringAndBranch(HIsStringAndBranch* instr) {
1516 ASSERT(instr->value()->representation().IsTagged());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001517 LOperand* value = UseRegisterAtStart(instr->value());
erikcorry0ad885c2011-11-21 13:51:57 +00001518 LOperand* temp = TempRegister();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001519 return new(zone()) LIsStringAndBranch(value, temp);
erikcorry0ad885c2011-11-21 13:51:57 +00001520}
1521
1522
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001523LInstruction* LChunkBuilder::DoIsSmiAndBranch(HIsSmiAndBranch* instr) {
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001524 ASSERT(instr->value()->representation().IsTagged());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001525 return new(zone()) LIsSmiAndBranch(Use(instr->value()));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001526}
1527
1528
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001529LInstruction* LChunkBuilder::DoIsUndetectableAndBranch(
1530 HIsUndetectableAndBranch* instr) {
vegorov@chromium.org7304bca2011-05-16 12:14:13 +00001531 ASSERT(instr->value()->representation().IsTagged());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001532 LOperand* value = UseRegisterAtStart(instr->value());
1533 LOperand* temp = TempRegister();
1534 return new(zone()) LIsUndetectableAndBranch(value, temp);
vegorov@chromium.org7304bca2011-05-16 12:14:13 +00001535}
1536
1537
erikcorry0ad885c2011-11-21 13:51:57 +00001538LInstruction* LChunkBuilder::DoStringCompareAndBranch(
1539 HStringCompareAndBranch* instr) {
1540
1541 ASSERT(instr->left()->representation().IsTagged());
1542 ASSERT(instr->right()->representation().IsTagged());
1543 LOperand* left = UseFixed(instr->left(), rdx);
1544 LOperand* right = UseFixed(instr->right(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001545 LStringCompareAndBranch* result =
1546 new(zone()) LStringCompareAndBranch(left, right);
erikcorry0ad885c2011-11-21 13:51:57 +00001547
1548 return MarkAsCall(result, instr);
1549}
1550
1551
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001552LInstruction* LChunkBuilder::DoHasInstanceTypeAndBranch(
1553 HHasInstanceTypeAndBranch* instr) {
vegorov@chromium.org5d6c1f52011-02-28 13:13:38 +00001554 ASSERT(instr->value()->representation().IsTagged());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001555 LOperand* value = UseRegisterAtStart(instr->value());
1556 return new(zone()) LHasInstanceTypeAndBranch(value);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001557}
1558
1559
kmillikin@chromium.org49edbdf2011-02-16 12:32:18 +00001560LInstruction* LChunkBuilder::DoGetCachedArrayIndex(
1561 HGetCachedArrayIndex* instr) {
karlklose@chromium.org8f806e82011-03-07 14:06:08 +00001562 ASSERT(instr->value()->representation().IsTagged());
1563 LOperand* value = UseRegisterAtStart(instr->value());
1564
ulan@chromium.org812308e2012-02-29 15:58:45 +00001565 return DefineAsRegister(new(zone()) LGetCachedArrayIndex(value));
kmillikin@chromium.org49edbdf2011-02-16 12:32:18 +00001566}
1567
1568
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001569LInstruction* LChunkBuilder::DoHasCachedArrayIndexAndBranch(
1570 HHasCachedArrayIndexAndBranch* instr) {
ager@chromium.org9ee27ae2011-03-02 13:43:26 +00001571 ASSERT(instr->value()->representation().IsTagged());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001572 LOperand* value = UseRegisterAtStart(instr->value());
1573 return new(zone()) LHasCachedArrayIndexAndBranch(value);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001574}
1575
1576
ricow@chromium.org4f693d62011-07-04 14:01:31 +00001577LInstruction* LChunkBuilder::DoClassOfTestAndBranch(
1578 HClassOfTestAndBranch* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001579 LOperand* value = UseRegister(instr->value());
1580 return new(zone()) LClassOfTestAndBranch(value,
1581 TempRegister(),
1582 TempRegister());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001583}
1584
1585
1586LInstruction* LChunkBuilder::DoJSArrayLength(HJSArrayLength* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001587 LOperand* array = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001588 return DefineAsRegister(new(zone()) LJSArrayLength(array));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001589}
1590
1591
whesse@chromium.org4acdc2c2011-08-15 13:01:23 +00001592LInstruction* LChunkBuilder::DoFixedArrayBaseLength(
1593 HFixedArrayBaseLength* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001594 LOperand* array = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001595 return DefineAsRegister(new(zone()) LFixedArrayBaseLength(array));
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +00001596}
1597
1598
whesse@chromium.org7b260152011-06-20 15:33:18 +00001599LInstruction* LChunkBuilder::DoElementsKind(HElementsKind* instr) {
1600 LOperand* object = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001601 return DefineAsRegister(new(zone()) LElementsKind(object));
whesse@chromium.org7b260152011-06-20 15:33:18 +00001602}
1603
1604
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001605LInstruction* LChunkBuilder::DoValueOf(HValueOf* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001606 LOperand* object = UseRegister(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001607 LValueOf* result = new(zone()) LValueOf(object);
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +00001608 return DefineSameAsFirst(result);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001609}
1610
1611
svenpanne@chromium.org4efbdb12012-03-12 08:18:42 +00001612LInstruction* LChunkBuilder::DoDateField(HDateField* instr) {
1613 LOperand* object = UseFixed(instr->value(), rax);
1614 LDateField* result = new LDateField(object, instr->index());
1615 return MarkAsCall(DefineFixed(result, rax), instr);
1616}
1617
1618
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001619LInstruction* LChunkBuilder::DoBoundsCheck(HBoundsCheck* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001620 LOperand* value = UseRegisterOrConstantAtStart(instr->index());
1621 LOperand* length = Use(instr->length());
1622 return AssignEnvironment(new(zone()) LBoundsCheck(value, length));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001623}
1624
1625
kmillikin@chromium.org31b12772011-02-02 16:08:26 +00001626LInstruction* LChunkBuilder::DoAbnormalExit(HAbnormalExit* instr) {
1627 // The control instruction marking the end of a block that completed
1628 // abruptly (e.g., threw an exception). There is nothing specific to do.
1629 return NULL;
1630}
1631
1632
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001633LInstruction* LChunkBuilder::DoThrow(HThrow* instr) {
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00001634 LOperand* value = UseFixed(instr->value(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001635 return MarkAsCall(new(zone()) LThrow(value), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001636}
1637
1638
ricow@chromium.orgd2be9012011-06-01 06:00:58 +00001639LInstruction* LChunkBuilder::DoUseConst(HUseConst* instr) {
1640 return NULL;
1641}
1642
1643
kmillikin@chromium.orgc53e10d2011-05-18 09:12:58 +00001644LInstruction* LChunkBuilder::DoForceRepresentation(HForceRepresentation* bad) {
1645 // All HForceRepresentation instructions should be eliminated in the
1646 // representation change phase of Hydrogen.
1647 UNREACHABLE();
1648 return NULL;
1649}
1650
1651
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001652LInstruction* LChunkBuilder::DoChange(HChange* instr) {
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001653 Representation from = instr->from();
1654 Representation to = instr->to();
1655 if (from.IsTagged()) {
1656 if (to.IsDouble()) {
1657 LOperand* value = UseRegister(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001658 LNumberUntagD* res = new(zone()) LNumberUntagD(value);
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001659 return AssignEnvironment(DefineAsRegister(res));
1660 } else {
1661 ASSERT(to.IsInteger32());
1662 LOperand* value = UseRegister(instr->value());
1663 bool needs_check = !instr->value()->type().IsSmi();
1664 if (needs_check) {
ricow@chromium.orgc54d3652011-05-30 09:20:16 +00001665 bool truncating = instr->CanTruncateToInt32();
1666 LOperand* xmm_temp = truncating ? NULL : FixedTemp(xmm1);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001667 LTaggedToI* res = new(zone()) LTaggedToI(value, xmm_temp);
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001668 return AssignEnvironment(DefineSameAsFirst(res));
1669 } else {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001670 return DefineSameAsFirst(new(zone()) LSmiUntag(value, needs_check));
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001671 }
1672 }
1673 } else if (from.IsDouble()) {
1674 if (to.IsTagged()) {
1675 LOperand* value = UseRegister(instr->value());
1676 LOperand* temp = TempRegister();
1677
1678 // Make sure that temp and result_temp are different registers.
1679 LUnallocated* result_temp = TempRegister();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001680 LNumberTagD* result = new(zone()) LNumberTagD(value, temp);
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001681 return AssignPointerMap(Define(result, result_temp));
1682 } else {
1683 ASSERT(to.IsInteger32());
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001684 LOperand* value = UseRegister(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001685 return AssignEnvironment(DefineAsRegister(new(zone()) LDoubleToI(value)));
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001686 }
1687 } else if (from.IsInteger32()) {
1688 if (to.IsTagged()) {
1689 HValue* val = instr->value();
1690 LOperand* value = UseRegister(val);
1691 if (val->HasRange() && val->range()->IsInSmiRange()) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001692 return DefineSameAsFirst(new(zone()) LSmiTag(value));
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001693 } else {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001694 LNumberTagI* result = new(zone()) LNumberTagI(value);
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001695 return AssignEnvironment(AssignPointerMap(DefineSameAsFirst(result)));
1696 }
1697 } else {
1698 ASSERT(to.IsDouble());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001699 LOperand* value = Use(instr->value());
1700 return DefineAsRegister(new(zone()) LInteger32ToDouble(value));
ager@chromium.org0ee099b2011-01-25 14:06:47 +00001701 }
1702 }
1703 UNREACHABLE();
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001704 return NULL;
1705}
1706
1707
1708LInstruction* LChunkBuilder::DoCheckNonSmi(HCheckNonSmi* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00001709 LOperand* value = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001710 return AssignEnvironment(new(zone()) LCheckNonSmi(value));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001711}
1712
1713
1714LInstruction* LChunkBuilder::DoCheckInstanceType(HCheckInstanceType* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001715 LOperand* value = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001716 LCheckInstanceType* result = new(zone()) LCheckInstanceType(value);
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001717 return AssignEnvironment(result);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001718}
1719
1720
1721LInstruction* LChunkBuilder::DoCheckPrototypeMaps(HCheckPrototypeMaps* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00001722 LOperand* temp = TempRegister();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001723 LCheckPrototypeMaps* result = new(zone()) LCheckPrototypeMaps(temp);
ager@chromium.org378b34e2011-01-28 08:04:38 +00001724 return AssignEnvironment(result);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001725}
1726
1727
1728LInstruction* LChunkBuilder::DoCheckSmi(HCheckSmi* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00001729 LOperand* value = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001730 return AssignEnvironment(new(zone()) LCheckSmi(value));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001731}
1732
1733
1734LInstruction* LChunkBuilder::DoCheckFunction(HCheckFunction* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00001735 LOperand* value = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001736 return AssignEnvironment(new(zone()) LCheckFunction(value));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001737}
1738
1739
jkummerow@chromium.org1456e702012-03-30 08:38:13 +00001740LInstruction* LChunkBuilder::DoCheckMaps(HCheckMaps* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00001741 LOperand* value = UseRegisterAtStart(instr->value());
jkummerow@chromium.org1456e702012-03-30 08:38:13 +00001742 LCheckMaps* result = new(zone()) LCheckMaps(value);
ager@chromium.org378b34e2011-01-28 08:04:38 +00001743 return AssignEnvironment(result);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001744}
1745
1746
kmillikin@chromium.orgc53e10d2011-05-18 09:12:58 +00001747LInstruction* LChunkBuilder::DoClampToUint8(HClampToUint8* instr) {
1748 HValue* value = instr->value();
1749 Representation input_rep = value->representation();
1750 LOperand* reg = UseRegister(value);
1751 if (input_rep.IsDouble()) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001752 return DefineAsRegister(new(zone()) LClampDToUint8(reg,
kmillikin@chromium.orgc53e10d2011-05-18 09:12:58 +00001753 TempRegister()));
1754 } else if (input_rep.IsInteger32()) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001755 return DefineSameAsFirst(new(zone()) LClampIToUint8(reg));
kmillikin@chromium.orgc53e10d2011-05-18 09:12:58 +00001756 } else {
1757 ASSERT(input_rep.IsTagged());
1758 // Register allocator doesn't (yet) support allocation of double
1759 // temps. Reserve xmm1 explicitly.
ulan@chromium.org812308e2012-02-29 15:58:45 +00001760 LClampTToUint8* result = new(zone()) LClampTToUint8(reg,
1761 TempRegister(),
1762 FixedTemp(xmm1));
kmillikin@chromium.orgc53e10d2011-05-18 09:12:58 +00001763 return AssignEnvironment(DefineSameAsFirst(result));
1764 }
1765}
1766
1767
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001768LInstruction* LChunkBuilder::DoReturn(HReturn* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001769 return new(zone()) LReturn(UseFixed(instr->value(), rax));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001770}
1771
1772
1773LInstruction* LChunkBuilder::DoConstant(HConstant* instr) {
1774 Representation r = instr->representation();
1775 if (r.IsInteger32()) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001776 return DefineAsRegister(new(zone()) LConstantI);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001777 } else if (r.IsDouble()) {
erik.corry@gmail.com0511e242011-01-19 11:11:08 +00001778 LOperand* temp = TempRegister();
ulan@chromium.org812308e2012-02-29 15:58:45 +00001779 return DefineAsRegister(new(zone()) LConstantD(temp));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001780 } else if (r.IsTagged()) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001781 return DefineAsRegister(new(zone()) LConstantT);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001782 } else {
1783 UNREACHABLE();
1784 return NULL;
1785 }
1786}
1787
1788
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +00001789LInstruction* LChunkBuilder::DoLoadGlobalCell(HLoadGlobalCell* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00001790 LLoadGlobalCell* result = new(zone()) LLoadGlobalCell;
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001791 return instr->RequiresHoleCheck()
ager@chromium.org378b34e2011-01-28 08:04:38 +00001792 ? AssignEnvironment(DefineAsRegister(result))
1793 : DefineAsRegister(result);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001794}
1795
1796
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +00001797LInstruction* LChunkBuilder::DoLoadGlobalGeneric(HLoadGlobalGeneric* instr) {
1798 LOperand* global_object = UseFixed(instr->global_object(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001799 LLoadGlobalGeneric* result = new(zone()) LLoadGlobalGeneric(global_object);
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +00001800 return MarkAsCall(DefineFixed(result, rax), instr);
1801}
1802
1803
vegorov@chromium.org74f333b2011-04-06 11:17:46 +00001804LInstruction* LChunkBuilder::DoStoreGlobalCell(HStoreGlobalCell* instr) {
danno@chromium.orge78f9fc2011-12-21 08:29:34 +00001805 LOperand* value = UseRegister(instr->value());
1806 // Use a temp to avoid reloading the cell value address in the case where
1807 // we perform a hole check.
1808 return instr->RequiresHoleCheck()
ulan@chromium.org812308e2012-02-29 15:58:45 +00001809 ? AssignEnvironment(new(zone()) LStoreGlobalCell(value, TempRegister()))
1810 : new(zone()) LStoreGlobalCell(value, NULL);
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001811}
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001812
1813
vegorov@chromium.org74f333b2011-04-06 11:17:46 +00001814LInstruction* LChunkBuilder::DoStoreGlobalGeneric(HStoreGlobalGeneric* instr) {
1815 LOperand* global_object = UseFixed(instr->global_object(), rdx);
1816 LOperand* value = UseFixed(instr->value(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001817 LStoreGlobalGeneric* result = new(zone()) LStoreGlobalGeneric(global_object,
1818 value);
vegorov@chromium.org74f333b2011-04-06 11:17:46 +00001819 return MarkAsCall(result, instr);
1820}
1821
1822
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001823LInstruction* LChunkBuilder::DoLoadContextSlot(HLoadContextSlot* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001824 LOperand* context = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001825 LInstruction* result =
1826 DefineAsRegister(new(zone()) LLoadContextSlot(context));
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +00001827 return instr->RequiresHoleCheck() ? AssignEnvironment(result) : result;
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001828}
1829
1830
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001831LInstruction* LChunkBuilder::DoStoreContextSlot(HStoreContextSlot* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001832 LOperand* context;
1833 LOperand* value;
karlklose@chromium.org8f806e82011-03-07 14:06:08 +00001834 LOperand* temp;
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001835 if (instr->NeedsWriteBarrier()) {
1836 context = UseTempRegister(instr->context());
1837 value = UseTempRegister(instr->value());
karlklose@chromium.org8f806e82011-03-07 14:06:08 +00001838 temp = TempRegister();
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001839 } else {
1840 context = UseRegister(instr->context());
1841 value = UseRegister(instr->value());
karlklose@chromium.org8f806e82011-03-07 14:06:08 +00001842 temp = NULL;
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001843 }
ulan@chromium.org812308e2012-02-29 15:58:45 +00001844 LInstruction* result = new(zone()) LStoreContextSlot(context, value, temp);
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +00001845 return instr->RequiresHoleCheck() ? AssignEnvironment(result) : result;
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001846}
1847
1848
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001849LInstruction* LChunkBuilder::DoLoadNamedField(HLoadNamedField* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00001850 ASSERT(instr->representation().IsTagged());
1851 LOperand* obj = UseRegisterAtStart(instr->object());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001852 return DefineAsRegister(new(zone()) LLoadNamedField(obj));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001853}
1854
1855
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00001856LInstruction* LChunkBuilder::DoLoadNamedFieldPolymorphic(
1857 HLoadNamedFieldPolymorphic* instr) {
1858 ASSERT(instr->representation().IsTagged());
1859 if (instr->need_generic()) {
1860 LOperand* obj = UseFixed(instr->object(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001861 LLoadNamedFieldPolymorphic* result =
1862 new(zone()) LLoadNamedFieldPolymorphic(obj);
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00001863 return MarkAsCall(DefineFixed(result, rax), instr);
1864 } else {
1865 LOperand* obj = UseRegisterAtStart(instr->object());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001866 LLoadNamedFieldPolymorphic* result =
1867 new(zone()) LLoadNamedFieldPolymorphic(obj);
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00001868 return AssignEnvironment(DefineAsRegister(result));
1869 }
1870}
1871
1872
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001873LInstruction* LChunkBuilder::DoLoadNamedGeneric(HLoadNamedGeneric* instr) {
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00001874 LOperand* object = UseFixed(instr->object(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00001875 LLoadNamedGeneric* result = new(zone()) LLoadNamedGeneric(object);
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00001876 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001877}
1878
1879
1880LInstruction* LChunkBuilder::DoLoadFunctionPrototype(
1881 HLoadFunctionPrototype* instr) {
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +00001882 return AssignEnvironment(DefineAsRegister(
ulan@chromium.org812308e2012-02-29 15:58:45 +00001883 new(zone()) LLoadFunctionPrototype(UseRegister(instr->function()))));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001884}
1885
1886
1887LInstruction* LChunkBuilder::DoLoadElements(HLoadElements* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001888 LOperand* input = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001889 return DefineAsRegister(new(zone()) LLoadElements(input));
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +00001890}
1891
1892
danno@chromium.org4d3fe4e2011-03-10 10:14:28 +00001893LInstruction* LChunkBuilder::DoLoadExternalArrayPointer(
1894 HLoadExternalArrayPointer* instr) {
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +00001895 LOperand* input = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001896 return DefineAsRegister(new(zone()) LLoadExternalArrayPointer(input));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001897}
1898
1899
1900LInstruction* LChunkBuilder::DoLoadKeyedFastElement(
1901 HLoadKeyedFastElement* instr) {
ricow@chromium.org83aa5492011-02-07 12:42:56 +00001902 ASSERT(instr->representation().IsTagged());
1903 ASSERT(instr->key()->representation().IsInteger32());
1904 LOperand* obj = UseRegisterAtStart(instr->object());
danno@chromium.orgb6451162011-08-17 14:33:23 +00001905 LOperand* key = UseRegisterOrConstantAtStart(instr->key());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001906 LLoadKeyedFastElement* result = new(zone()) LLoadKeyedFastElement(obj, key);
erik.corry@gmail.comf2038fb2012-01-16 11:42:08 +00001907 if (instr->RequiresHoleCheck()) AssignEnvironment(result);
1908 return DefineAsRegister(result);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001909}
1910
1911
rossberg@chromium.org717967f2011-07-20 13:44:42 +00001912LInstruction* LChunkBuilder::DoLoadKeyedFastDoubleElement(
1913 HLoadKeyedFastDoubleElement* instr) {
1914 ASSERT(instr->representation().IsDouble());
1915 ASSERT(instr->key()->representation().IsInteger32());
1916 LOperand* elements = UseRegisterAtStart(instr->elements());
1917 LOperand* key = UseRegisterOrConstantAtStart(instr->key());
1918 LLoadKeyedFastDoubleElement* result =
ulan@chromium.org812308e2012-02-29 15:58:45 +00001919 new(zone()) LLoadKeyedFastDoubleElement(elements, key);
rossberg@chromium.org717967f2011-07-20 13:44:42 +00001920 return AssignEnvironment(DefineAsRegister(result));
1921}
1922
1923
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00001924LInstruction* LChunkBuilder::DoLoadKeyedSpecializedArrayElement(
1925 HLoadKeyedSpecializedArrayElement* instr) {
kmillikin@chromium.org83e16822011-09-13 08:21:47 +00001926 ElementsKind elements_kind = instr->elements_kind();
erik.corry@gmail.com3847bd52011-04-27 10:38:56 +00001927 ASSERT(
yangguo@chromium.org659ceec2012-01-26 07:37:54 +00001928 (instr->representation().IsInteger32() &&
kmillikin@chromium.org83e16822011-09-13 08:21:47 +00001929 (elements_kind != EXTERNAL_FLOAT_ELEMENTS) &&
1930 (elements_kind != EXTERNAL_DOUBLE_ELEMENTS)) ||
yangguo@chromium.org659ceec2012-01-26 07:37:54 +00001931 (instr->representation().IsDouble() &&
kmillikin@chromium.org83e16822011-09-13 08:21:47 +00001932 ((elements_kind == EXTERNAL_FLOAT_ELEMENTS) ||
1933 (elements_kind == EXTERNAL_DOUBLE_ELEMENTS))));
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +00001934 ASSERT(instr->key()->representation().IsInteger32());
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00001935 LOperand* external_pointer = UseRegister(instr->external_pointer());
karlklose@chromium.org83a47282011-05-11 11:54:09 +00001936 LOperand* key = UseRegisterOrConstant(instr->key());
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00001937 LLoadKeyedSpecializedArrayElement* result =
ulan@chromium.org812308e2012-02-29 15:58:45 +00001938 new(zone()) LLoadKeyedSpecializedArrayElement(external_pointer, key);
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00001939 LInstruction* load_instr = DefineAsRegister(result);
1940 // An unsigned int array load might overflow and cause a deopt, make sure it
1941 // has an environment.
kmillikin@chromium.org83e16822011-09-13 08:21:47 +00001942 return (elements_kind == EXTERNAL_UNSIGNED_INT_ELEMENTS) ?
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00001943 AssignEnvironment(load_instr) : load_instr;
sgjesse@chromium.org496c03a2011-02-14 12:05:43 +00001944}
1945
1946
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001947LInstruction* LChunkBuilder::DoLoadKeyedGeneric(HLoadKeyedGeneric* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001948 LOperand* object = UseFixed(instr->object(), rdx);
1949 LOperand* key = UseFixed(instr->key(), rax);
1950
ulan@chromium.org812308e2012-02-29 15:58:45 +00001951 LLoadKeyedGeneric* result = new(zone()) LLoadKeyedGeneric(object, key);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001952 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001953}
1954
1955
1956LInstruction* LChunkBuilder::DoStoreKeyedFastElement(
1957 HStoreKeyedFastElement* instr) {
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00001958 bool needs_write_barrier = instr->NeedsWriteBarrier();
1959 ASSERT(instr->value()->representation().IsTagged());
1960 ASSERT(instr->object()->representation().IsTagged());
1961 ASSERT(instr->key()->representation().IsInteger32());
1962
1963 LOperand* obj = UseTempRegister(instr->object());
1964 LOperand* val = needs_write_barrier
1965 ? UseTempRegister(instr->value())
1966 : UseRegisterAtStart(instr->value());
1967 LOperand* key = needs_write_barrier
1968 ? UseTempRegister(instr->key())
1969 : UseRegisterOrConstantAtStart(instr->key());
ulan@chromium.org812308e2012-02-29 15:58:45 +00001970 return new(zone()) LStoreKeyedFastElement(obj, key, val);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00001971}
1972
1973
rossberg@chromium.org717967f2011-07-20 13:44:42 +00001974LInstruction* LChunkBuilder::DoStoreKeyedFastDoubleElement(
1975 HStoreKeyedFastDoubleElement* instr) {
1976 ASSERT(instr->value()->representation().IsDouble());
1977 ASSERT(instr->elements()->representation().IsTagged());
1978 ASSERT(instr->key()->representation().IsInteger32());
1979
1980 LOperand* elements = UseRegisterAtStart(instr->elements());
1981 LOperand* val = UseTempRegister(instr->value());
1982 LOperand* key = UseRegisterOrConstantAtStart(instr->key());
1983
ulan@chromium.org812308e2012-02-29 15:58:45 +00001984 return new(zone()) LStoreKeyedFastDoubleElement(elements, key, val);
rossberg@chromium.org717967f2011-07-20 13:44:42 +00001985}
1986
1987
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00001988LInstruction* LChunkBuilder::DoStoreKeyedSpecializedArrayElement(
1989 HStoreKeyedSpecializedArrayElement* instr) {
kmillikin@chromium.org83e16822011-09-13 08:21:47 +00001990 ElementsKind elements_kind = instr->elements_kind();
erik.corry@gmail.com3847bd52011-04-27 10:38:56 +00001991 ASSERT(
yangguo@chromium.org659ceec2012-01-26 07:37:54 +00001992 (instr->value()->representation().IsInteger32() &&
kmillikin@chromium.org83e16822011-09-13 08:21:47 +00001993 (elements_kind != EXTERNAL_FLOAT_ELEMENTS) &&
1994 (elements_kind != EXTERNAL_DOUBLE_ELEMENTS)) ||
yangguo@chromium.org659ceec2012-01-26 07:37:54 +00001995 (instr->value()->representation().IsDouble() &&
kmillikin@chromium.org83e16822011-09-13 08:21:47 +00001996 ((elements_kind == EXTERNAL_FLOAT_ELEMENTS) ||
1997 (elements_kind == EXTERNAL_DOUBLE_ELEMENTS))));
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00001998 ASSERT(instr->external_pointer()->representation().IsExternal());
1999 ASSERT(instr->key()->representation().IsInteger32());
2000
2001 LOperand* external_pointer = UseRegister(instr->external_pointer());
svenpanne@chromium.org6d786c92011-06-15 10:58:27 +00002002 bool val_is_temp_register =
kmillikin@chromium.org83e16822011-09-13 08:21:47 +00002003 elements_kind == EXTERNAL_PIXEL_ELEMENTS ||
2004 elements_kind == EXTERNAL_FLOAT_ELEMENTS;
fschneider@chromium.org7979bbb2011-03-28 10:47:03 +00002005 LOperand* val = val_is_temp_register
2006 ? UseTempRegister(instr->value())
vegorov@chromium.org74f333b2011-04-06 11:17:46 +00002007 : UseRegister(instr->value());
karlklose@chromium.org83a47282011-05-11 11:54:09 +00002008 LOperand* key = UseRegisterOrConstant(instr->key());
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002009
ulan@chromium.org812308e2012-02-29 15:58:45 +00002010 return new(zone()) LStoreKeyedSpecializedArrayElement(external_pointer,
2011 key,
2012 val);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002013}
2014
2015
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002016LInstruction* LChunkBuilder::DoStoreKeyedGeneric(HStoreKeyedGeneric* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002017 LOperand* object = UseFixed(instr->object(), rdx);
2018 LOperand* key = UseFixed(instr->key(), rcx);
2019 LOperand* value = UseFixed(instr->value(), rax);
2020
2021 ASSERT(instr->object()->representation().IsTagged());
2022 ASSERT(instr->key()->representation().IsTagged());
2023 ASSERT(instr->value()->representation().IsTagged());
2024
ulan@chromium.org812308e2012-02-29 15:58:45 +00002025 LStoreKeyedGeneric* result =
2026 new(zone()) LStoreKeyedGeneric(object, key, value);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002027 return MarkAsCall(result, instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002028}
2029
2030
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +00002031LInstruction* LChunkBuilder::DoTransitionElementsKind(
2032 HTransitionElementsKind* instr) {
2033 if (instr->original_map()->elements_kind() == FAST_SMI_ONLY_ELEMENTS &&
2034 instr->transitioned_map()->elements_kind() == FAST_ELEMENTS) {
2035 LOperand* object = UseRegister(instr->object());
2036 LOperand* new_map_reg = TempRegister();
2037 LOperand* temp_reg = TempRegister();
2038 LTransitionElementsKind* result =
ulan@chromium.org812308e2012-02-29 15:58:45 +00002039 new(zone()) LTransitionElementsKind(object, new_map_reg, temp_reg);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +00002040 return DefineSameAsFirst(result);
2041 } else {
2042 LOperand* object = UseFixed(instr->object(), rax);
2043 LOperand* fixed_object_reg = FixedTemp(rdx);
2044 LOperand* new_map_reg = FixedTemp(rbx);
2045 LTransitionElementsKind* result =
ulan@chromium.org812308e2012-02-29 15:58:45 +00002046 new(zone()) LTransitionElementsKind(object,
2047 new_map_reg,
2048 fixed_object_reg);
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +00002049 return MarkAsCall(DefineFixed(result, rax), instr);
2050 }
2051}
2052
2053
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002054LInstruction* LChunkBuilder::DoStoreNamedField(HStoreNamedField* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00002055 bool needs_write_barrier = instr->NeedsWriteBarrier();
2056
2057 LOperand* obj = needs_write_barrier
2058 ? UseTempRegister(instr->object())
2059 : UseRegisterAtStart(instr->object());
2060
2061 LOperand* val = needs_write_barrier
2062 ? UseTempRegister(instr->value())
2063 : UseRegister(instr->value());
2064
2065 // We only need a scratch register if we have a write barrier or we
2066 // have a store into the properties array (not in-object-property).
2067 LOperand* temp = (!instr->is_in_object() || needs_write_barrier)
2068 ? TempRegister() : NULL;
2069
ulan@chromium.org812308e2012-02-29 15:58:45 +00002070 return new(zone()) LStoreNamedField(obj, val, temp);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002071}
2072
2073
2074LInstruction* LChunkBuilder::DoStoreNamedGeneric(HStoreNamedGeneric* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002075 LOperand* object = UseFixed(instr->object(), rdx);
2076 LOperand* value = UseFixed(instr->value(), rax);
2077
ulan@chromium.org812308e2012-02-29 15:58:45 +00002078 LStoreNamedGeneric* result = new(zone()) LStoreNamedGeneric(object, value);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002079 return MarkAsCall(result, instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002080}
2081
2082
danno@chromium.org160a7b02011-04-18 15:51:38 +00002083LInstruction* LChunkBuilder::DoStringAdd(HStringAdd* instr) {
2084 LOperand* left = UseOrConstantAtStart(instr->left());
2085 LOperand* right = UseOrConstantAtStart(instr->right());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002086 return MarkAsCall(DefineFixed(new(zone()) LStringAdd(left, right), rax),
2087 instr);
danno@chromium.org160a7b02011-04-18 15:51:38 +00002088}
2089
2090
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00002091LInstruction* LChunkBuilder::DoStringCharCodeAt(HStringCharCodeAt* instr) {
ricow@chromium.org4668a2c2011-08-29 10:41:00 +00002092 LOperand* string = UseTempRegister(instr->string());
2093 LOperand* index = UseTempRegister(instr->index());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002094 LStringCharCodeAt* result = new(zone()) LStringCharCodeAt(string, index);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002095 return AssignEnvironment(AssignPointerMap(DefineAsRegister(result)));
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00002096}
2097
2098
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00002099LInstruction* LChunkBuilder::DoStringCharFromCode(HStringCharFromCode* instr) {
2100 LOperand* char_code = UseRegister(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002101 LStringCharFromCode* result = new(zone()) LStringCharFromCode(char_code);
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00002102 return AssignPointerMap(DefineAsRegister(result));
2103}
2104
2105
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00002106LInstruction* LChunkBuilder::DoStringLength(HStringLength* instr) {
kmillikin@chromium.org49edbdf2011-02-16 12:32:18 +00002107 LOperand* string = UseRegisterAtStart(instr->value());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002108 return DefineAsRegister(new(zone()) LStringLength(string));
vegorov@chromium.org0a4e9012011-01-24 12:33:13 +00002109}
2110
2111
ulan@chromium.org967e2702012-02-28 09:49:15 +00002112LInstruction* LChunkBuilder::DoAllocateObject(HAllocateObject* instr) {
fschneider@chromium.org35814e52012-03-01 15:43:35 +00002113 LAllocateObject* result = new LAllocateObject(TempRegister());
ulan@chromium.org967e2702012-02-28 09:49:15 +00002114 return AssignPointerMap(DefineAsRegister(result));
2115}
2116
2117
jkummerow@chromium.orgf7a58842012-02-21 10:08:21 +00002118LInstruction* LChunkBuilder::DoFastLiteral(HFastLiteral* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002119 return MarkAsCall(DefineFixed(new(zone()) LFastLiteral, rax), instr);
jkummerow@chromium.orgf7a58842012-02-21 10:08:21 +00002120}
2121
2122
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002123LInstruction* LChunkBuilder::DoArrayLiteral(HArrayLiteral* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002124 return MarkAsCall(DefineFixed(new(zone()) LArrayLiteral, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002125}
2126
2127
jkummerow@chromium.orgf7a58842012-02-21 10:08:21 +00002128LInstruction* LChunkBuilder::DoObjectLiteral(HObjectLiteral* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002129 return MarkAsCall(DefineFixed(new(zone()) LObjectLiteral, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002130}
2131
2132
2133LInstruction* LChunkBuilder::DoRegExpLiteral(HRegExpLiteral* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002134 return MarkAsCall(DefineFixed(new(zone()) LRegExpLiteral, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002135}
2136
2137
2138LInstruction* LChunkBuilder::DoFunctionLiteral(HFunctionLiteral* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002139 return MarkAsCall(DefineFixed(new(zone()) LFunctionLiteral, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002140}
2141
2142
2143LInstruction* LChunkBuilder::DoDeleteProperty(HDeleteProperty* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002144 LOperand* object = UseAtStart(instr->object());
2145 LOperand* key = UseOrConstantAtStart(instr->key());
2146 LDeleteProperty* result = new(zone()) LDeleteProperty(object, key);
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002147 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002148}
2149
2150
2151LInstruction* LChunkBuilder::DoOsrEntry(HOsrEntry* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002152 allocator_->MarkAsOsrEntry();
2153 current_block_->last_environment()->set_ast_id(instr->ast_id());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002154 return AssignEnvironment(new(zone()) LOsrEntry);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002155}
2156
2157
2158LInstruction* LChunkBuilder::DoParameter(HParameter* instr) {
2159 int spill_index = chunk()->GetParameterStackSlot(instr->index());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002160 return DefineAsSpilled(new(zone()) LParameter, spill_index);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002161}
2162
2163
2164LInstruction* LChunkBuilder::DoUnknownOSRValue(HUnknownOSRValue* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002165 int spill_index = chunk()->GetNextSpillIndex(false); // Not double-width.
ager@chromium.org04921a82011-06-27 13:21:41 +00002166 if (spill_index > LUnallocated::kMaxFixedIndex) {
2167 Abort("Too many spill slots needed for OSR");
2168 spill_index = 0;
2169 }
ulan@chromium.org812308e2012-02-29 15:58:45 +00002170 return DefineAsSpilled(new(zone()) LUnknownOSRValue, spill_index);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002171}
2172
2173
2174LInstruction* LChunkBuilder::DoCallStub(HCallStub* instr) {
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00002175 argument_count_ -= instr->argument_count();
ulan@chromium.org812308e2012-02-29 15:58:45 +00002176 return MarkAsCall(DefineFixed(new(zone()) LCallStub, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002177}
2178
2179
2180LInstruction* LChunkBuilder::DoArgumentsObject(HArgumentsObject* instr) {
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002181 // There are no real uses of the arguments object.
2182 // arguments.length and element access are supported directly on
2183 // stack arguments, and any real arguments object use causes a bailout.
2184 // So this value is never used.
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002185 return NULL;
2186}
2187
2188
2189LInstruction* LChunkBuilder::DoAccessArgumentsAt(HAccessArgumentsAt* instr) {
kmillikin@chromium.org49edbdf2011-02-16 12:32:18 +00002190 LOperand* arguments = UseRegister(instr->arguments());
2191 LOperand* length = UseTempRegister(instr->length());
2192 LOperand* index = Use(instr->index());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002193 LAccessArgumentsAt* result =
2194 new(zone()) LAccessArgumentsAt(arguments, length, index);
kmillikin@chromium.org49edbdf2011-02-16 12:32:18 +00002195 return AssignEnvironment(DefineAsRegister(result));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002196}
2197
2198
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00002199LInstruction* LChunkBuilder::DoToFastProperties(HToFastProperties* instr) {
2200 LOperand* object = UseFixed(instr->value(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00002201 LToFastProperties* result = new(zone()) LToFastProperties(object);
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00002202 return MarkAsCall(DefineFixed(result, rax), instr);
2203}
2204
2205
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002206LInstruction* LChunkBuilder::DoTypeof(HTypeof* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002207 LTypeof* result = new(zone()) LTypeof(UseAtStart(instr->value()));
fschneider@chromium.org3a5fd782011-02-24 10:10:44 +00002208 return MarkAsCall(DefineFixed(result, rax), instr);
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002209}
2210
2211
ricow@chromium.org4f693d62011-07-04 14:01:31 +00002212LInstruction* LChunkBuilder::DoTypeofIsAndBranch(HTypeofIsAndBranch* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002213 return new(zone()) LTypeofIsAndBranch(UseTempRegister(instr->value()));
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002214}
2215
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00002216
ricow@chromium.org4f693d62011-07-04 14:01:31 +00002217LInstruction* LChunkBuilder::DoIsConstructCallAndBranch(
2218 HIsConstructCallAndBranch* instr) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002219 return new(zone()) LIsConstructCallAndBranch(TempRegister());
erik.corry@gmail.comd91075f2011-02-10 07:45:38 +00002220}
2221
2222
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002223LInstruction* LChunkBuilder::DoSimulate(HSimulate* instr) {
2224 HEnvironment* env = current_block_->last_environment();
2225 ASSERT(env != NULL);
2226
2227 env->set_ast_id(instr->ast_id());
2228
2229 env->Drop(instr->pop_count());
2230 for (int i = 0; i < instr->values()->length(); ++i) {
2231 HValue* value = instr->values()->at(i);
2232 if (instr->HasAssignedIndexAt(i)) {
2233 env->Bind(instr->GetAssignedIndexAt(i), value);
2234 } else {
2235 env->Push(value);
2236 }
2237 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002238
2239 // If there is an instruction pending deoptimization environment create a
2240 // lazy bailout instruction to capture the environment.
fschneider@chromium.org1df6b472011-01-26 08:23:03 +00002241 if (pending_deoptimization_ast_id_ == instr->ast_id()) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002242 LLazyBailout* lazy_bailout = new(zone()) LLazyBailout;
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002243 LInstruction* result = AssignEnvironment(lazy_bailout);
kmillikin@chromium.org31b12772011-02-02 16:08:26 +00002244 instruction_pending_deoptimization_environment_->
fschneider@chromium.org1df6b472011-01-26 08:23:03 +00002245 set_deoptimization_environment(result->environment());
2246 ClearInstructionPendingDeoptimizationEnvironment();
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002247 return result;
2248 }
2249
2250 return NULL;
2251}
2252
2253
2254LInstruction* LChunkBuilder::DoStackCheck(HStackCheck* instr) {
ager@chromium.org04921a82011-06-27 13:21:41 +00002255 if (instr->is_function_entry()) {
ulan@chromium.org812308e2012-02-29 15:58:45 +00002256 return MarkAsCall(new(zone()) LStackCheck, instr);
ager@chromium.org04921a82011-06-27 13:21:41 +00002257 } else {
2258 ASSERT(instr->is_backwards_branch());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002259 return AssignEnvironment(AssignPointerMap(new(zone()) LStackCheck));
ager@chromium.org04921a82011-06-27 13:21:41 +00002260 }
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002261}
2262
2263
2264LInstruction* LChunkBuilder::DoEnterInlined(HEnterInlined* instr) {
ager@chromium.org378b34e2011-01-28 08:04:38 +00002265 HEnvironment* outer = current_block_->last_environment();
2266 HConstant* undefined = graph()->GetConstantUndefined();
2267 HEnvironment* inner = outer->CopyForInlining(instr->closure(),
yangguo@chromium.org659ceec2012-01-26 07:37:54 +00002268 instr->arguments_count(),
ager@chromium.org378b34e2011-01-28 08:04:38 +00002269 instr->function(),
danno@chromium.org40cb8782011-05-25 07:58:50 +00002270 undefined,
ulan@chromium.org967e2702012-02-28 09:49:15 +00002271 instr->call_kind(),
2272 instr->is_construct());
jkummerow@chromium.org28faa982012-04-13 09:58:30 +00002273 if (instr->arguments_var() != NULL) {
2274 inner->Bind(instr->arguments_var(), graph()->GetArgumentsObject());
danno@chromium.org8c0a43f2012-04-03 08:37:53 +00002275 }
ager@chromium.org378b34e2011-01-28 08:04:38 +00002276 current_block_->UpdateEnvironment(inner);
2277 chunk_->AddInlinedClosure(instr->closure());
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002278 return NULL;
2279}
2280
2281
2282LInstruction* LChunkBuilder::DoLeaveInlined(HLeaveInlined* instr) {
jkummerow@chromium.org28faa982012-04-13 09:58:30 +00002283 LInstruction* pop = NULL;
2284
2285 HEnvironment* env = current_block_->last_environment();
2286
2287 if (instr->arguments_pushed()) {
2288 int argument_count = env->arguments_environment()->parameter_count();
2289 pop = new(zone()) LDrop(argument_count);
2290 argument_count_ -= argument_count;
2291 }
2292
yangguo@chromium.org659ceec2012-01-26 07:37:54 +00002293 HEnvironment* outer = current_block_->last_environment()->
2294 DiscardInlined(false);
ricow@chromium.org83aa5492011-02-07 12:42:56 +00002295 current_block_->UpdateEnvironment(outer);
jkummerow@chromium.org28faa982012-04-13 09:58:30 +00002296
2297 return pop;
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002298}
2299
erik.corry@gmail.com3847bd52011-04-27 10:38:56 +00002300
2301LInstruction* LChunkBuilder::DoIn(HIn* instr) {
2302 LOperand* key = UseOrConstantAtStart(instr->key());
2303 LOperand* object = UseOrConstantAtStart(instr->object());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002304 LIn* result = new(zone()) LIn(key, object);
erik.corry@gmail.com3847bd52011-04-27 10:38:56 +00002305 return MarkAsCall(DefineFixed(result, rax), instr);
2306}
2307
2308
kmillikin@chromium.orgbe6bd102012-02-23 08:45:21 +00002309LInstruction* LChunkBuilder::DoForInPrepareMap(HForInPrepareMap* instr) {
2310 LOperand* object = UseFixed(instr->enumerable(), rax);
ulan@chromium.org812308e2012-02-29 15:58:45 +00002311 LForInPrepareMap* result = new(zone()) LForInPrepareMap(object);
kmillikin@chromium.orgbe6bd102012-02-23 08:45:21 +00002312 return MarkAsCall(DefineFixed(result, rax), instr, CAN_DEOPTIMIZE_EAGERLY);
2313}
2314
2315
2316LInstruction* LChunkBuilder::DoForInCacheArray(HForInCacheArray* instr) {
2317 LOperand* map = UseRegister(instr->map());
2318 return AssignEnvironment(DefineAsRegister(
ulan@chromium.org812308e2012-02-29 15:58:45 +00002319 new(zone()) LForInCacheArray(map)));
kmillikin@chromium.orgbe6bd102012-02-23 08:45:21 +00002320}
2321
2322
2323LInstruction* LChunkBuilder::DoCheckMapValue(HCheckMapValue* instr) {
2324 LOperand* value = UseRegisterAtStart(instr->value());
2325 LOperand* map = UseRegisterAtStart(instr->map());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002326 return AssignEnvironment(new(zone()) LCheckMapValue(value, map));
kmillikin@chromium.orgbe6bd102012-02-23 08:45:21 +00002327}
2328
2329
2330LInstruction* LChunkBuilder::DoLoadFieldByIndex(HLoadFieldByIndex* instr) {
2331 LOperand* object = UseRegister(instr->object());
2332 LOperand* index = UseTempRegister(instr->index());
ulan@chromium.org812308e2012-02-29 15:58:45 +00002333 return DefineSameAsFirst(new(zone()) LLoadFieldByIndex(object, index));
kmillikin@chromium.orgbe6bd102012-02-23 08:45:21 +00002334}
2335
2336
kmillikin@chromium.orgd2c22f02011-01-10 08:15:37 +00002337} } // namespace v8::internal
sgjesse@chromium.orgc6c57182011-01-17 12:24:25 +00002338
2339#endif // V8_TARGET_ARCH_X64