• Home
  • History
  • Annotate
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright 2014 the V8 project authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
5 #include "src/compiler/simplified-lowering.h"
6 
7 #include "src/base/bits.h"
8 #include "src/code-factory.h"
9 #include "src/compiler/common-operator.h"
10 #include "src/compiler/graph-inl.h"
11 #include "src/compiler/node-properties-inl.h"
12 #include "src/compiler/representation-change.h"
13 #include "src/compiler/simplified-lowering.h"
14 #include "src/compiler/simplified-operator.h"
15 #include "src/objects.h"
16 
17 namespace v8 {
18 namespace internal {
19 namespace compiler {
20 
21 // Macro for outputting trace information from representation inference.
22 #define TRACE(x) \
23   if (FLAG_trace_representation) PrintF x
24 
25 // Representation selection and lowering of {Simplified} operators to machine
26 // operators are interwined. We use a fixpoint calculation to compute both the
27 // output representation and the best possible lowering for {Simplified} nodes.
28 // Representation change insertion ensures that all values are in the correct
29 // machine representation after this phase, as dictated by the machine
30 // operators themselves.
31 enum Phase {
32   // 1.) PROPAGATE: Traverse the graph from the end, pushing usage information
33   //     backwards from uses to definitions, around cycles in phis, according
34   //     to local rules for each operator.
35   //     During this phase, the usage information for a node determines the best
36   //     possible lowering for each operator so far, and that in turn determines
37   //     the output representation.
38   //     Therefore, to be correct, this phase must iterate to a fixpoint before
39   //     the next phase can begin.
40   PROPAGATE,
41 
42   // 2.) LOWER: perform lowering for all {Simplified} nodes by replacing some
43   //     operators for some nodes, expanding some nodes to multiple nodes, or
44   //     removing some (redundant) nodes.
45   //     During this phase, use the {RepresentationChanger} to insert
46   //     representation changes between uses that demand a particular
47   //     representation and nodes that produce a different representation.
48   LOWER
49 };
50 
51 
52 class RepresentationSelector {
53  public:
54   // Information for each node tracked during the fixpoint.
55   struct NodeInfo {
56     MachineTypeUnion use : 15;     // Union of all usages for the node.
57     bool queued : 1;           // Bookkeeping for the traversal.
58     bool visited : 1;          // Bookkeeping for the traversal.
59     MachineTypeUnion output : 15;  // Output type of the node.
60   };
61 
RepresentationSelector(JSGraph * jsgraph,Zone * zone,RepresentationChanger * changer)62   RepresentationSelector(JSGraph* jsgraph, Zone* zone,
63                          RepresentationChanger* changer)
64       : jsgraph_(jsgraph),
65         count_(jsgraph->graph()->NodeCount()),
66         info_(zone->NewArray<NodeInfo>(count_)),
67         nodes_(zone),
68         replacements_(zone),
69         contains_js_nodes_(false),
70         phase_(PROPAGATE),
71         changer_(changer),
72         queue_(zone) {
73     memset(info_, 0, sizeof(NodeInfo) * count_);
74   }
75 
Run(SimplifiedLowering * lowering)76   void Run(SimplifiedLowering* lowering) {
77     // Run propagation phase to a fixpoint.
78     TRACE(("--{Propagation phase}--\n"));
79     phase_ = PROPAGATE;
80     Enqueue(jsgraph_->graph()->end());
81     // Process nodes from the queue until it is empty.
82     while (!queue_.empty()) {
83       Node* node = queue_.front();
84       NodeInfo* info = GetInfo(node);
85       queue_.pop();
86       info->queued = false;
87       TRACE((" visit #%d: %s\n", node->id(), node->op()->mnemonic()));
88       VisitNode(node, info->use, NULL);
89       TRACE(("  ==> output "));
90       PrintInfo(info->output);
91       TRACE(("\n"));
92     }
93 
94     // Run lowering and change insertion phase.
95     TRACE(("--{Simplified lowering phase}--\n"));
96     phase_ = LOWER;
97     // Process nodes from the collected {nodes_} vector.
98     for (NodeVector::iterator i = nodes_.begin(); i != nodes_.end(); ++i) {
99       Node* node = *i;
100       TRACE((" visit #%d: %s\n", node->id(), node->op()->mnemonic()));
101       // Reuse {VisitNode()} so the representation rules are in one place.
102       VisitNode(node, GetUseInfo(node), lowering);
103     }
104 
105     // Perform the final replacements.
106     for (NodeVector::iterator i = replacements_.begin();
107          i != replacements_.end(); ++i) {
108       Node* node = *i;
109       Node* replacement = *(++i);
110       node->ReplaceUses(replacement);
111     }
112   }
113 
114   // Enqueue {node} if the {use} contains new information for that node.
115   // Add {node} to {nodes_} if this is the first time it's been visited.
Enqueue(Node * node,MachineTypeUnion use=0)116   void Enqueue(Node* node, MachineTypeUnion use = 0) {
117     if (phase_ != PROPAGATE) return;
118     NodeInfo* info = GetInfo(node);
119     if (!info->visited) {
120       // First visit of this node.
121       info->visited = true;
122       info->queued = true;
123       nodes_.push_back(node);
124       queue_.push(node);
125       TRACE(("  initial: "));
126       info->use |= use;
127       PrintUseInfo(node);
128       return;
129     }
130     TRACE(("   queue?: "));
131     PrintUseInfo(node);
132     if ((info->use & use) != use) {
133       // New usage information for the node is available.
134       if (!info->queued) {
135         queue_.push(node);
136         info->queued = true;
137         TRACE(("   added: "));
138       } else {
139         TRACE((" inqueue: "));
140       }
141       info->use |= use;
142       PrintUseInfo(node);
143     }
144   }
145 
lower()146   bool lower() { return phase_ == LOWER; }
147 
Enqueue(Node * node,MachineType use)148   void Enqueue(Node* node, MachineType use) {
149     Enqueue(node, static_cast<MachineTypeUnion>(use));
150   }
151 
SetOutput(Node * node,MachineTypeUnion output)152   void SetOutput(Node* node, MachineTypeUnion output) {
153     // Every node should have at most one output representation. Note that
154     // phis can have 0, if they have not been used in a representation-inducing
155     // instruction.
156     DCHECK((output & kRepMask) == 0 ||
157            base::bits::IsPowerOfTwo32(output & kRepMask));
158     GetInfo(node)->output = output;
159   }
160 
BothInputsAre(Node * node,Type * type)161   bool BothInputsAre(Node* node, Type* type) {
162     DCHECK_EQ(2, node->InputCount());
163     return NodeProperties::GetBounds(node->InputAt(0)).upper->Is(type) &&
164            NodeProperties::GetBounds(node->InputAt(1)).upper->Is(type);
165   }
166 
ProcessInput(Node * node,int index,MachineTypeUnion use)167   void ProcessInput(Node* node, int index, MachineTypeUnion use) {
168     Node* input = node->InputAt(index);
169     if (phase_ == PROPAGATE) {
170       // In the propagate phase, propagate the usage information backward.
171       Enqueue(input, use);
172     } else {
173       // In the change phase, insert a change before the use if necessary.
174       if ((use & kRepMask) == 0) return;  // No input requirement on the use.
175       MachineTypeUnion output = GetInfo(input)->output;
176       if ((output & kRepMask & use) == 0) {
177         // Output representation doesn't match usage.
178         TRACE(("  change: #%d:%s(@%d #%d:%s) ", node->id(),
179                node->op()->mnemonic(), index, input->id(),
180                input->op()->mnemonic()));
181         TRACE((" from "));
182         PrintInfo(output);
183         TRACE((" to "));
184         PrintInfo(use);
185         TRACE(("\n"));
186         Node* n = changer_->GetRepresentationFor(input, output, use);
187         node->ReplaceInput(index, n);
188       }
189     }
190   }
191 
ProcessRemainingInputs(Node * node,int index)192   void ProcessRemainingInputs(Node* node, int index) {
193     DCHECK_GE(index, NodeProperties::PastValueIndex(node));
194     DCHECK_GE(index, NodeProperties::PastContextIndex(node));
195     for (int i = std::max(index, NodeProperties::FirstEffectIndex(node));
196          i < NodeProperties::PastEffectIndex(node); ++i) {
197       Enqueue(node->InputAt(i));  // Effect inputs: just visit
198     }
199     for (int i = std::max(index, NodeProperties::FirstControlIndex(node));
200          i < NodeProperties::PastControlIndex(node); ++i) {
201       Enqueue(node->InputAt(i));  // Control inputs: just visit
202     }
203   }
204 
205   // The default, most general visitation case. For {node}, process all value,
206   // context, effect, and control inputs, assuming that value inputs should have
207   // {kRepTagged} representation and can observe all output values {kTypeAny}.
VisitInputs(Node * node)208   void VisitInputs(Node* node) {
209     InputIter i = node->inputs().begin();
210     for (int j = OperatorProperties::GetValueInputCount(node->op()); j > 0;
211          ++i, j--) {
212       ProcessInput(node, i.index(), kMachAnyTagged);  // Value inputs
213     }
214     for (int j = OperatorProperties::GetContextInputCount(node->op()); j > 0;
215          ++i, j--) {
216       ProcessInput(node, i.index(), kMachAnyTagged);  // Context inputs
217     }
218     for (int j = OperatorProperties::GetEffectInputCount(node->op()); j > 0;
219          ++i, j--) {
220       Enqueue(*i);  // Effect inputs: just visit
221     }
222     for (int j = OperatorProperties::GetControlInputCount(node->op()); j > 0;
223          ++i, j--) {
224       Enqueue(*i);  // Control inputs: just visit
225     }
226     SetOutput(node, kMachAnyTagged);
227   }
228 
229   // Helper for binops of the I x I -> O variety.
VisitBinop(Node * node,MachineTypeUnion input_use,MachineTypeUnion output)230   void VisitBinop(Node* node, MachineTypeUnion input_use,
231                   MachineTypeUnion output) {
232     DCHECK_EQ(2, node->InputCount());
233     ProcessInput(node, 0, input_use);
234     ProcessInput(node, 1, input_use);
235     SetOutput(node, output);
236   }
237 
238   // Helper for unops of the I -> O variety.
VisitUnop(Node * node,MachineTypeUnion input_use,MachineTypeUnion output)239   void VisitUnop(Node* node, MachineTypeUnion input_use,
240                  MachineTypeUnion output) {
241     DCHECK_EQ(1, node->InputCount());
242     ProcessInput(node, 0, input_use);
243     SetOutput(node, output);
244   }
245 
246   // Helper for leaf nodes.
VisitLeaf(Node * node,MachineTypeUnion output)247   void VisitLeaf(Node* node, MachineTypeUnion output) {
248     DCHECK_EQ(0, node->InputCount());
249     SetOutput(node, output);
250   }
251 
252   // Helpers for specific types of binops.
VisitFloat64Binop(Node * node)253   void VisitFloat64Binop(Node* node) {
254     VisitBinop(node, kMachFloat64, kMachFloat64);
255   }
VisitInt32Binop(Node * node)256   void VisitInt32Binop(Node* node) { VisitBinop(node, kMachInt32, kMachInt32); }
VisitUint32Binop(Node * node)257   void VisitUint32Binop(Node* node) {
258     VisitBinop(node, kMachUint32, kMachUint32);
259   }
VisitInt64Binop(Node * node)260   void VisitInt64Binop(Node* node) { VisitBinop(node, kMachInt64, kMachInt64); }
VisitUint64Binop(Node * node)261   void VisitUint64Binop(Node* node) {
262     VisitBinop(node, kMachUint64, kMachUint64);
263   }
VisitFloat64Cmp(Node * node)264   void VisitFloat64Cmp(Node* node) { VisitBinop(node, kMachFloat64, kRepBit); }
VisitInt32Cmp(Node * node)265   void VisitInt32Cmp(Node* node) { VisitBinop(node, kMachInt32, kRepBit); }
VisitUint32Cmp(Node * node)266   void VisitUint32Cmp(Node* node) { VisitBinop(node, kMachUint32, kRepBit); }
VisitInt64Cmp(Node * node)267   void VisitInt64Cmp(Node* node) { VisitBinop(node, kMachInt64, kRepBit); }
VisitUint64Cmp(Node * node)268   void VisitUint64Cmp(Node* node) { VisitBinop(node, kMachUint64, kRepBit); }
269 
270   // Helper for handling phis.
VisitPhi(Node * node,MachineTypeUnion use,SimplifiedLowering * lowering)271   void VisitPhi(Node* node, MachineTypeUnion use,
272                 SimplifiedLowering* lowering) {
273     // First, propagate the usage information to inputs of the phi.
274     if (!lower()) {
275       int values = OperatorProperties::GetValueInputCount(node->op());
276       // Propagate {use} of the phi to value inputs, and 0 to control.
277       Node::Inputs inputs = node->inputs();
278       for (Node::Inputs::iterator iter(inputs.begin()); iter != inputs.end();
279            ++iter, --values) {
280         // TODO(titzer): it'd be nice to have distinguished edge kinds here.
281         ProcessInput(node, iter.index(), values > 0 ? use : 0);
282       }
283     }
284     // Phis adapt to whatever output representation their uses demand,
285     // pushing representation changes to their inputs.
286     MachineTypeUnion use_rep = GetUseInfo(node) & kRepMask;
287     MachineTypeUnion use_type = GetUseInfo(node) & kTypeMask;
288     MachineTypeUnion rep = 0;
289     if (use_rep & kRepTagged) {
290       rep = kRepTagged;  // Tagged overrides everything.
291     } else if (use_rep & kRepFloat64) {
292       rep = kRepFloat64;
293     } else if (use_rep & kRepWord64) {
294       rep = kRepWord64;
295     } else if (use_rep & kRepWord32) {
296       rep = kRepWord32;
297     } else if (use_rep & kRepBit) {
298       rep = kRepBit;
299     } else {
300       // There was no representation associated with any of the uses.
301       // TODO(titzer): Select the best rep using phi's type, not the usage type?
302       if (use_type & kTypeAny) {
303         rep = kRepTagged;
304       } else if (use_type & kTypeNumber) {
305         rep = kRepFloat64;
306       } else if (use_type & kTypeInt64 || use_type & kTypeUint64) {
307         rep = kRepWord64;
308       } else if (use_type & kTypeInt32 || use_type & kTypeUint32) {
309         rep = kRepWord32;
310       } else if (use_type & kTypeBool) {
311         rep = kRepBit;
312       } else {
313         UNREACHABLE();  // should have at least a usage type!
314       }
315     }
316     // Preserve the usage type, but set the representation.
317     Type* upper = NodeProperties::GetBounds(node).upper;
318     MachineTypeUnion output_type = rep | changer_->TypeFromUpperBound(upper);
319     SetOutput(node, output_type);
320 
321     if (lower()) {
322       int values = OperatorProperties::GetValueInputCount(node->op());
323 
324       // Update the phi operator.
325       MachineType type = static_cast<MachineType>(output_type);
326       if (type != OpParameter<MachineType>(node)) {
327         node->set_op(lowering->common()->Phi(type, values));
328       }
329 
330       // Convert inputs to the output representation of this phi.
331       Node::Inputs inputs = node->inputs();
332       for (Node::Inputs::iterator iter(inputs.begin()); iter != inputs.end();
333            ++iter, --values) {
334         // TODO(titzer): it'd be nice to have distinguished edge kinds here.
335         ProcessInput(node, iter.index(), values > 0 ? output_type : 0);
336       }
337     }
338   }
339 
Int32Op(Node * node)340   const Operator* Int32Op(Node* node) {
341     return changer_->Int32OperatorFor(node->opcode());
342   }
343 
Uint32Op(Node * node)344   const Operator* Uint32Op(Node* node) {
345     return changer_->Uint32OperatorFor(node->opcode());
346   }
347 
Float64Op(Node * node)348   const Operator* Float64Op(Node* node) {
349     return changer_->Float64OperatorFor(node->opcode());
350   }
351 
AssumeImplicitFloat32Change(MachineType type)352   static MachineType AssumeImplicitFloat32Change(MachineType type) {
353     // TODO(titzer): Assume loads of float32 change representation to float64.
354     // Fix this with full support for float32 representations.
355     if (type & kRepFloat32) {
356       return static_cast<MachineType>((type & ~kRepFloat32) | kRepFloat64);
357     }
358     return type;
359   }
360 
361   // Dispatching routine for visiting the node {node} with the usage {use}.
362   // Depending on the operator, propagate new usage info to the inputs.
VisitNode(Node * node,MachineTypeUnion use,SimplifiedLowering * lowering)363   void VisitNode(Node* node, MachineTypeUnion use,
364                  SimplifiedLowering* lowering) {
365     switch (node->opcode()) {
366       //------------------------------------------------------------------
367       // Common operators.
368       //------------------------------------------------------------------
369       case IrOpcode::kStart:
370       case IrOpcode::kDead:
371         return VisitLeaf(node, 0);
372       case IrOpcode::kParameter: {
373         // TODO(titzer): use representation from linkage.
374         Type* upper = NodeProperties::GetBounds(node).upper;
375         ProcessInput(node, 0, 0);
376         SetOutput(node, kRepTagged | changer_->TypeFromUpperBound(upper));
377         return;
378       }
379       case IrOpcode::kInt32Constant:
380         return VisitLeaf(node, kRepWord32);
381       case IrOpcode::kInt64Constant:
382         return VisitLeaf(node, kRepWord64);
383       case IrOpcode::kFloat64Constant:
384         return VisitLeaf(node, kRepFloat64);
385       case IrOpcode::kExternalConstant:
386         return VisitLeaf(node, kMachPtr);
387       case IrOpcode::kNumberConstant:
388         return VisitLeaf(node, kRepTagged);
389       case IrOpcode::kHeapConstant:
390         return VisitLeaf(node, kRepTagged);
391 
392       case IrOpcode::kEnd:
393       case IrOpcode::kIfTrue:
394       case IrOpcode::kIfFalse:
395       case IrOpcode::kReturn:
396       case IrOpcode::kMerge:
397       case IrOpcode::kThrow:
398         return VisitInputs(node);  // default visit for all node inputs.
399 
400       case IrOpcode::kBranch:
401         ProcessInput(node, 0, kRepBit);
402         Enqueue(NodeProperties::GetControlInput(node, 0));
403         break;
404       case IrOpcode::kPhi:
405         return VisitPhi(node, use, lowering);
406 
407 //------------------------------------------------------------------
408 // JavaScript operators.
409 //------------------------------------------------------------------
410 // For now, we assume that all JS operators were too complex to lower
411 // to Simplified and that they will always require tagged value inputs
412 // and produce tagged value outputs.
413 // TODO(turbofan): it might be possible to lower some JSOperators here,
414 // but that responsibility really lies in the typed lowering phase.
415 #define DEFINE_JS_CASE(x) case IrOpcode::k##x:
416         JS_OP_LIST(DEFINE_JS_CASE)
417 #undef DEFINE_JS_CASE
418         contains_js_nodes_ = true;
419         VisitInputs(node);
420         return SetOutput(node, kRepTagged);
421 
422       //------------------------------------------------------------------
423       // Simplified operators.
424       //------------------------------------------------------------------
425       case IrOpcode::kBooleanNot: {
426         if (lower()) {
427           MachineTypeUnion input = GetInfo(node->InputAt(0))->output;
428           if (input & kRepBit) {
429             // BooleanNot(x: kRepBit) => WordEqual(x, #0)
430             node->set_op(lowering->machine()->WordEqual());
431             node->AppendInput(jsgraph_->zone(), jsgraph_->Int32Constant(0));
432           } else {
433             // BooleanNot(x: kRepTagged) => WordEqual(x, #false)
434             node->set_op(lowering->machine()->WordEqual());
435             node->AppendInput(jsgraph_->zone(), jsgraph_->FalseConstant());
436           }
437         } else {
438           // No input representation requirement; adapt during lowering.
439           ProcessInput(node, 0, kTypeBool);
440           SetOutput(node, kRepBit);
441         }
442         break;
443       }
444       case IrOpcode::kBooleanToNumber: {
445         if (lower()) {
446           MachineTypeUnion input = GetInfo(node->InputAt(0))->output;
447           if (input & kRepBit) {
448             // BooleanToNumber(x: kRepBit) => x
449             DeferReplacement(node, node->InputAt(0));
450           } else {
451             // BooleanToNumber(x: kRepTagged) => WordEqual(x, #true)
452             node->set_op(lowering->machine()->WordEqual());
453             node->AppendInput(jsgraph_->zone(), jsgraph_->TrueConstant());
454           }
455         } else {
456           // No input representation requirement; adapt during lowering.
457           ProcessInput(node, 0, kTypeBool);
458           SetOutput(node, kMachInt32);
459         }
460         break;
461       }
462       case IrOpcode::kNumberEqual:
463       case IrOpcode::kNumberLessThan:
464       case IrOpcode::kNumberLessThanOrEqual: {
465         // Number comparisons reduce to integer comparisons for integer inputs.
466         if (BothInputsAre(node, Type::Signed32())) {
467           // => signed Int32Cmp
468           VisitInt32Cmp(node);
469           if (lower()) node->set_op(Int32Op(node));
470         } else if (BothInputsAre(node, Type::Unsigned32())) {
471           // => unsigned Int32Cmp
472           VisitUint32Cmp(node);
473           if (lower()) node->set_op(Uint32Op(node));
474         } else {
475           // => Float64Cmp
476           VisitFloat64Cmp(node);
477           if (lower()) node->set_op(Float64Op(node));
478         }
479         break;
480       }
481       case IrOpcode::kNumberAdd:
482       case IrOpcode::kNumberSubtract: {
483         // Add and subtract reduce to Int32Add/Sub if the inputs
484         // are already integers and all uses are truncating.
485         if (BothInputsAre(node, Type::Signed32()) &&
486             (use & (kTypeUint32 | kTypeNumber | kTypeAny)) == 0) {
487           // => signed Int32Add/Sub
488           VisitInt32Binop(node);
489           if (lower()) node->set_op(Int32Op(node));
490         } else if (BothInputsAre(node, Type::Unsigned32()) &&
491                    (use & (kTypeInt32 | kTypeNumber | kTypeAny)) == 0) {
492           // => unsigned Int32Add/Sub
493           VisitUint32Binop(node);
494           if (lower()) node->set_op(Uint32Op(node));
495         } else {
496           // => Float64Add/Sub
497           VisitFloat64Binop(node);
498           if (lower()) node->set_op(Float64Op(node));
499         }
500         break;
501       }
502       case IrOpcode::kNumberMultiply:
503       case IrOpcode::kNumberDivide:
504       case IrOpcode::kNumberModulus: {
505         // Float64Mul/Div/Mod
506         VisitFloat64Binop(node);
507         if (lower()) node->set_op(Float64Op(node));
508         break;
509       }
510       case IrOpcode::kNumberToInt32: {
511         MachineTypeUnion use_rep = use & kRepMask;
512         if (lower()) {
513           MachineTypeUnion in = GetInfo(node->InputAt(0))->output;
514           if ((in & kTypeMask) == kTypeInt32 || (in & kRepMask) == kRepWord32) {
515             // If the input has type int32, or is already a word32, just change
516             // representation if necessary.
517             VisitUnop(node, kTypeInt32 | use_rep, kTypeInt32 | use_rep);
518             DeferReplacement(node, node->InputAt(0));
519           } else {
520             // Require the input in float64 format and perform truncation.
521             // TODO(turbofan): avoid a truncation with a smi check.
522             VisitUnop(node, kTypeInt32 | kRepFloat64, kTypeInt32 | kRepWord32);
523             node->set_op(lowering->machine()->TruncateFloat64ToInt32());
524           }
525         } else {
526           // Propagate a type to the input, but pass through representation.
527           VisitUnop(node, kTypeInt32, kTypeInt32 | use_rep);
528         }
529         break;
530       }
531       case IrOpcode::kNumberToUint32: {
532         MachineTypeUnion use_rep = use & kRepMask;
533         if (lower()) {
534           MachineTypeUnion in = GetInfo(node->InputAt(0))->output;
535           if ((in & kTypeMask) == kTypeUint32 ||
536               (in & kRepMask) == kRepWord32) {
537             // The input has type int32, just change representation.
538             VisitUnop(node, kTypeUint32 | use_rep, kTypeUint32 | use_rep);
539             DeferReplacement(node, node->InputAt(0));
540           } else {
541             // Require the input in float64 format to perform truncation.
542             // TODO(turbofan): avoid the truncation with a smi check.
543             VisitUnop(node, kTypeUint32 | kRepFloat64,
544                       kTypeUint32 | kRepWord32);
545             node->set_op(lowering->machine()->TruncateFloat64ToInt32());
546           }
547         } else {
548           // Propagate a type to the input, but pass through representation.
549           VisitUnop(node, kTypeUint32, kTypeUint32 | use_rep);
550         }
551         break;
552       }
553       case IrOpcode::kReferenceEqual: {
554         VisitBinop(node, kMachAnyTagged, kRepBit);
555         if (lower()) node->set_op(lowering->machine()->WordEqual());
556         break;
557       }
558       case IrOpcode::kStringEqual: {
559         VisitBinop(node, kMachAnyTagged, kRepBit);
560         if (lower()) lowering->DoStringEqual(node);
561         break;
562       }
563       case IrOpcode::kStringLessThan: {
564         VisitBinop(node, kMachAnyTagged, kRepBit);
565         if (lower()) lowering->DoStringLessThan(node);
566         break;
567       }
568       case IrOpcode::kStringLessThanOrEqual: {
569         VisitBinop(node, kMachAnyTagged, kRepBit);
570         if (lower()) lowering->DoStringLessThanOrEqual(node);
571         break;
572       }
573       case IrOpcode::kStringAdd: {
574         VisitBinop(node, kMachAnyTagged, kMachAnyTagged);
575         if (lower()) lowering->DoStringAdd(node);
576         break;
577       }
578       case IrOpcode::kLoadField: {
579         FieldAccess access = FieldAccessOf(node->op());
580         ProcessInput(node, 0, changer_->TypeForBasePointer(access));
581         ProcessRemainingInputs(node, 1);
582         SetOutput(node, AssumeImplicitFloat32Change(access.machine_type));
583         if (lower()) lowering->DoLoadField(node);
584         break;
585       }
586       case IrOpcode::kStoreField: {
587         FieldAccess access = FieldAccessOf(node->op());
588         ProcessInput(node, 0, changer_->TypeForBasePointer(access));
589         ProcessInput(node, 1, AssumeImplicitFloat32Change(access.machine_type));
590         ProcessRemainingInputs(node, 2);
591         SetOutput(node, 0);
592         if (lower()) lowering->DoStoreField(node);
593         break;
594       }
595       case IrOpcode::kLoadElement: {
596         ElementAccess access = ElementAccessOf(node->op());
597         ProcessInput(node, 0, changer_->TypeForBasePointer(access));
598         ProcessInput(node, 1, kMachInt32);  // element index
599         ProcessInput(node, 2, kMachInt32);  // length
600         ProcessRemainingInputs(node, 3);
601         SetOutput(node, AssumeImplicitFloat32Change(access.machine_type));
602         if (lower()) lowering->DoLoadElement(node);
603         break;
604       }
605       case IrOpcode::kStoreElement: {
606         ElementAccess access = ElementAccessOf(node->op());
607         ProcessInput(node, 0, changer_->TypeForBasePointer(access));
608         ProcessInput(node, 1, kMachInt32);  // element index
609         ProcessInput(node, 2, kMachInt32);  // length
610         ProcessInput(node, 3, AssumeImplicitFloat32Change(access.machine_type));
611         ProcessRemainingInputs(node, 4);
612         SetOutput(node, 0);
613         if (lower()) lowering->DoStoreElement(node);
614         break;
615       }
616 
617       //------------------------------------------------------------------
618       // Machine-level operators.
619       //------------------------------------------------------------------
620       case IrOpcode::kLoad: {
621         // TODO(titzer): machine loads/stores need to know BaseTaggedness!?
622         MachineType tBase = kRepTagged;
623         LoadRepresentation rep = OpParameter<LoadRepresentation>(node);
624         ProcessInput(node, 0, tBase);   // pointer or object
625         ProcessInput(node, 1, kMachInt32);  // index
626         ProcessRemainingInputs(node, 2);
627         SetOutput(node, rep);
628         break;
629       }
630       case IrOpcode::kStore: {
631         // TODO(titzer): machine loads/stores need to know BaseTaggedness!?
632         MachineType tBase = kRepTagged;
633         StoreRepresentation rep = OpParameter<StoreRepresentation>(node);
634         ProcessInput(node, 0, tBase);   // pointer or object
635         ProcessInput(node, 1, kMachInt32);  // index
636         ProcessInput(node, 2, rep.machine_type());
637         ProcessRemainingInputs(node, 3);
638         SetOutput(node, 0);
639         break;
640       }
641       case IrOpcode::kWord32Shr:
642         // We output unsigned int32 for shift right because JavaScript.
643         return VisitBinop(node, kRepWord32, kRepWord32 | kTypeUint32);
644       case IrOpcode::kWord32And:
645       case IrOpcode::kWord32Or:
646       case IrOpcode::kWord32Xor:
647       case IrOpcode::kWord32Shl:
648       case IrOpcode::kWord32Sar:
649         // We use signed int32 as the output type for these word32 operations,
650         // though the machine bits are the same for either signed or unsigned,
651         // because JavaScript considers the result from these operations signed.
652         return VisitBinop(node, kRepWord32, kRepWord32 | kTypeInt32);
653       case IrOpcode::kWord32Equal:
654         return VisitBinop(node, kRepWord32, kRepBit);
655 
656       case IrOpcode::kInt32Add:
657       case IrOpcode::kInt32Sub:
658       case IrOpcode::kInt32Mul:
659       case IrOpcode::kInt32Div:
660       case IrOpcode::kInt32Mod:
661         return VisitInt32Binop(node);
662       case IrOpcode::kInt32UDiv:
663       case IrOpcode::kInt32UMod:
664         return VisitUint32Binop(node);
665       case IrOpcode::kInt32LessThan:
666       case IrOpcode::kInt32LessThanOrEqual:
667         return VisitInt32Cmp(node);
668 
669       case IrOpcode::kUint32LessThan:
670       case IrOpcode::kUint32LessThanOrEqual:
671         return VisitUint32Cmp(node);
672 
673       case IrOpcode::kInt64Add:
674       case IrOpcode::kInt64Sub:
675       case IrOpcode::kInt64Mul:
676       case IrOpcode::kInt64Div:
677       case IrOpcode::kInt64Mod:
678         return VisitInt64Binop(node);
679       case IrOpcode::kInt64LessThan:
680       case IrOpcode::kInt64LessThanOrEqual:
681         return VisitInt64Cmp(node);
682 
683       case IrOpcode::kInt64UDiv:
684       case IrOpcode::kInt64UMod:
685         return VisitUint64Binop(node);
686 
687       case IrOpcode::kWord64And:
688       case IrOpcode::kWord64Or:
689       case IrOpcode::kWord64Xor:
690       case IrOpcode::kWord64Shl:
691       case IrOpcode::kWord64Shr:
692       case IrOpcode::kWord64Sar:
693         return VisitBinop(node, kRepWord64, kRepWord64);
694       case IrOpcode::kWord64Equal:
695         return VisitBinop(node, kRepWord64, kRepBit);
696 
697       case IrOpcode::kChangeInt32ToInt64:
698         return VisitUnop(node, kTypeInt32 | kRepWord32,
699                          kTypeInt32 | kRepWord64);
700       case IrOpcode::kChangeUint32ToUint64:
701         return VisitUnop(node, kTypeUint32 | kRepWord32,
702                          kTypeUint32 | kRepWord64);
703       case IrOpcode::kTruncateInt64ToInt32:
704         // TODO(titzer): Is kTypeInt32 correct here?
705         return VisitUnop(node, kTypeInt32 | kRepWord64,
706                          kTypeInt32 | kRepWord32);
707 
708       case IrOpcode::kChangeInt32ToFloat64:
709         return VisitUnop(node, kTypeInt32 | kRepWord32,
710                          kTypeInt32 | kRepFloat64);
711       case IrOpcode::kChangeUint32ToFloat64:
712         return VisitUnop(node, kTypeUint32 | kRepWord32,
713                          kTypeUint32 | kRepFloat64);
714       case IrOpcode::kChangeFloat64ToInt32:
715         return VisitUnop(node, kTypeInt32 | kRepFloat64,
716                          kTypeInt32 | kRepWord32);
717       case IrOpcode::kChangeFloat64ToUint32:
718         return VisitUnop(node, kTypeUint32 | kRepFloat64,
719                          kTypeUint32 | kRepWord32);
720 
721       case IrOpcode::kFloat64Add:
722       case IrOpcode::kFloat64Sub:
723       case IrOpcode::kFloat64Mul:
724       case IrOpcode::kFloat64Div:
725       case IrOpcode::kFloat64Mod:
726         return VisitFloat64Binop(node);
727       case IrOpcode::kFloat64Sqrt:
728         return VisitUnop(node, kMachFloat64, kMachFloat64);
729       case IrOpcode::kFloat64Equal:
730       case IrOpcode::kFloat64LessThan:
731       case IrOpcode::kFloat64LessThanOrEqual:
732         return VisitFloat64Cmp(node);
733       default:
734         VisitInputs(node);
735         break;
736     }
737   }
738 
DeferReplacement(Node * node,Node * replacement)739   void DeferReplacement(Node* node, Node* replacement) {
740     if (replacement->id() < count_) {
741       // Replace with a previously existing node eagerly.
742       node->ReplaceUses(replacement);
743     } else {
744       // Otherwise, we are replacing a node with a representation change.
745       // Such a substitution must be done after all lowering is done, because
746       // new nodes do not have {NodeInfo} entries, and that would confuse
747       // the representation change insertion for uses of it.
748       replacements_.push_back(node);
749       replacements_.push_back(replacement);
750     }
751     // TODO(titzer) node->RemoveAllInputs();  // Node is now dead.
752   }
753 
PrintUseInfo(Node * node)754   void PrintUseInfo(Node* node) {
755     TRACE(("#%d:%-20s ", node->id(), node->op()->mnemonic()));
756     PrintInfo(GetUseInfo(node));
757     TRACE(("\n"));
758   }
759 
PrintInfo(MachineTypeUnion info)760   void PrintInfo(MachineTypeUnion info) {
761     if (FLAG_trace_representation) {
762       OFStream os(stdout);
763       os << static_cast<MachineType>(info);
764     }
765   }
766 
767  private:
768   JSGraph* jsgraph_;
769   int count_;                       // number of nodes in the graph
770   NodeInfo* info_;                  // node id -> usage information
771   NodeVector nodes_;                // collected nodes
772   NodeVector replacements_;         // replacements to be done after lowering
773   bool contains_js_nodes_;          // {true} if a JS operator was seen
774   Phase phase_;                     // current phase of algorithm
775   RepresentationChanger* changer_;  // for inserting representation changes
776   ZoneQueue<Node*> queue_;          // queue for traversing the graph
777 
GetInfo(Node * node)778   NodeInfo* GetInfo(Node* node) {
779     DCHECK(node->id() >= 0);
780     DCHECK(node->id() < count_);
781     return &info_[node->id()];
782   }
783 
GetUseInfo(Node * node)784   MachineTypeUnion GetUseInfo(Node* node) { return GetInfo(node)->use; }
785 };
786 
787 
IsTagged(Node * node)788 Node* SimplifiedLowering::IsTagged(Node* node) {
789   // TODO(titzer): factor this out to a TaggingScheme abstraction.
790   STATIC_ASSERT(kSmiTagMask == 1);  // Only works if tag is the low bit.
791   return graph()->NewNode(machine()->WordAnd(), node,
792                           jsgraph()->Int32Constant(kSmiTagMask));
793 }
794 
795 
LowerAllNodes()796 void SimplifiedLowering::LowerAllNodes() {
797   SimplifiedOperatorBuilder simplified(graph()->zone());
798   RepresentationChanger changer(jsgraph(), &simplified,
799                                 graph()->zone()->isolate());
800   RepresentationSelector selector(jsgraph(), zone(), &changer);
801   selector.Run(this);
802 }
803 
804 
Untag(Node * node)805 Node* SimplifiedLowering::Untag(Node* node) {
806   // TODO(titzer): factor this out to a TaggingScheme abstraction.
807   Node* shift_amount = jsgraph()->Int32Constant(kSmiTagSize + kSmiShiftSize);
808   return graph()->NewNode(machine()->WordSar(), node, shift_amount);
809 }
810 
811 
SmiTag(Node * node)812 Node* SimplifiedLowering::SmiTag(Node* node) {
813   // TODO(titzer): factor this out to a TaggingScheme abstraction.
814   Node* shift_amount = jsgraph()->Int32Constant(kSmiTagSize + kSmiShiftSize);
815   return graph()->NewNode(machine()->WordShl(), node, shift_amount);
816 }
817 
818 
OffsetMinusTagConstant(int32_t offset)819 Node* SimplifiedLowering::OffsetMinusTagConstant(int32_t offset) {
820   return jsgraph()->Int32Constant(offset - kHeapObjectTag);
821 }
822 
823 
ComputeWriteBarrierKind(BaseTaggedness base_is_tagged,MachineType representation,Type * type)824 static WriteBarrierKind ComputeWriteBarrierKind(BaseTaggedness base_is_tagged,
825                                                 MachineType representation,
826                                                 Type* type) {
827   // TODO(turbofan): skip write barriers for Smis, etc.
828   if (base_is_tagged == kTaggedBase &&
829       RepresentationOf(representation) == kRepTagged) {
830     // Write barriers are only for writes into heap objects (i.e. tagged base).
831     return kFullWriteBarrier;
832   }
833   return kNoWriteBarrier;
834 }
835 
836 
DoLoadField(Node * node)837 void SimplifiedLowering::DoLoadField(Node* node) {
838   const FieldAccess& access = FieldAccessOf(node->op());
839   node->set_op(machine()->Load(access.machine_type));
840   Node* offset = jsgraph()->Int32Constant(access.offset - access.tag());
841   node->InsertInput(zone(), 1, offset);
842 }
843 
844 
DoStoreField(Node * node)845 void SimplifiedLowering::DoStoreField(Node* node) {
846   const FieldAccess& access = FieldAccessOf(node->op());
847   WriteBarrierKind kind = ComputeWriteBarrierKind(
848       access.base_is_tagged, access.machine_type, access.type);
849   node->set_op(
850       machine()->Store(StoreRepresentation(access.machine_type, kind)));
851   Node* offset = jsgraph()->Int32Constant(access.offset - access.tag());
852   node->InsertInput(zone(), 1, offset);
853 }
854 
855 
ComputeIndex(const ElementAccess & access,Node * index)856 Node* SimplifiedLowering::ComputeIndex(const ElementAccess& access,
857                                        Node* index) {
858   int element_size = ElementSizeOf(access.machine_type);
859   if (element_size != 1) {
860     index = graph()->NewNode(machine()->Int32Mul(),
861                              jsgraph()->Int32Constant(element_size), index);
862   }
863   int fixed_offset = access.header_size - access.tag();
864   if (fixed_offset == 0) return index;
865   return graph()->NewNode(machine()->Int32Add(), index,
866                           jsgraph()->Int32Constant(fixed_offset));
867 }
868 
869 
DoLoadElement(Node * node)870 void SimplifiedLowering::DoLoadElement(Node* node) {
871   const ElementAccess& access = ElementAccessOf(node->op());
872   node->set_op(machine()->Load(access.machine_type));
873   node->ReplaceInput(1, ComputeIndex(access, node->InputAt(1)));
874   node->RemoveInput(2);
875 }
876 
877 
DoStoreElement(Node * node)878 void SimplifiedLowering::DoStoreElement(Node* node) {
879   const ElementAccess& access = ElementAccessOf(node->op());
880   WriteBarrierKind kind = ComputeWriteBarrierKind(
881       access.base_is_tagged, access.machine_type, access.type);
882   node->set_op(
883       machine()->Store(StoreRepresentation(access.machine_type, kind)));
884   node->ReplaceInput(1, ComputeIndex(access, node->InputAt(1)));
885   node->RemoveInput(2);
886 }
887 
888 
DoStringAdd(Node * node)889 void SimplifiedLowering::DoStringAdd(Node* node) {
890   Callable callable = CodeFactory::StringAdd(
891       zone()->isolate(), STRING_ADD_CHECK_NONE, NOT_TENURED);
892   CallDescriptor::Flags flags = CallDescriptor::kNoFlags;
893   CallDescriptor* desc =
894       Linkage::GetStubCallDescriptor(callable.descriptor(), 0, flags, zone());
895   node->set_op(common()->Call(desc));
896   node->InsertInput(zone(), 0, jsgraph()->HeapConstant(callable.code()));
897   node->AppendInput(zone(), jsgraph()->UndefinedConstant());
898   node->AppendInput(zone(), graph()->start());
899   node->AppendInput(zone(), graph()->start());
900 }
901 
902 
StringComparison(Node * node,bool requires_ordering)903 Node* SimplifiedLowering::StringComparison(Node* node, bool requires_ordering) {
904   CEntryStub stub(zone()->isolate(), 1);
905   Runtime::FunctionId f =
906       requires_ordering ? Runtime::kStringCompare : Runtime::kStringEquals;
907   ExternalReference ref(f, zone()->isolate());
908   Operator::Properties props = node->op()->properties();
909   // TODO(mstarzinger): We should call StringCompareStub here instead, once an
910   // interface descriptor is available for it.
911   CallDescriptor* desc = Linkage::GetRuntimeCallDescriptor(f, 2, props, zone());
912   return graph()->NewNode(common()->Call(desc),
913                           jsgraph()->HeapConstant(stub.GetCode()),
914                           NodeProperties::GetValueInput(node, 0),
915                           NodeProperties::GetValueInput(node, 1),
916                           jsgraph()->ExternalConstant(ref),
917                           jsgraph()->Int32Constant(2),
918                           jsgraph()->UndefinedConstant());
919 }
920 
921 
DoStringEqual(Node * node)922 void SimplifiedLowering::DoStringEqual(Node* node) {
923   node->set_op(machine()->WordEqual());
924   node->ReplaceInput(0, StringComparison(node, false));
925   node->ReplaceInput(1, jsgraph()->SmiConstant(EQUAL));
926 }
927 
928 
DoStringLessThan(Node * node)929 void SimplifiedLowering::DoStringLessThan(Node* node) {
930   node->set_op(machine()->IntLessThan());
931   node->ReplaceInput(0, StringComparison(node, true));
932   node->ReplaceInput(1, jsgraph()->SmiConstant(EQUAL));
933 }
934 
935 
DoStringLessThanOrEqual(Node * node)936 void SimplifiedLowering::DoStringLessThanOrEqual(Node* node) {
937   node->set_op(machine()->IntLessThanOrEqual());
938   node->ReplaceInput(0, StringComparison(node, true));
939   node->ReplaceInput(1, jsgraph()->SmiConstant(EQUAL));
940 }
941 
942 
943 }  // namespace compiler
944 }  // namespace internal
945 }  // namespace v8
946