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/generic-node-inl.h"
6 #include "src/compiler/js-graph.h"
7 #include "src/compiler/machine-operator.h"
8 #include "src/compiler/node-matchers.h"
9 #include "src/compiler/simplified-operator-reducer.h"
10 
11 namespace v8 {
12 namespace internal {
13 namespace compiler {
14 
~SimplifiedOperatorReducer()15 SimplifiedOperatorReducer::~SimplifiedOperatorReducer() {}
16 
17 
Reduce(Node * node)18 Reduction SimplifiedOperatorReducer::Reduce(Node* node) {
19   switch (node->opcode()) {
20     case IrOpcode::kBooleanNot: {
21       HeapObjectMatcher<HeapObject> m(node->InputAt(0));
22       if (m.Is(Unique<HeapObject>::CreateImmovable(factory()->false_value()))) {
23         return Replace(jsgraph()->TrueConstant());
24       }
25       if (m.Is(Unique<HeapObject>::CreateImmovable(factory()->true_value()))) {
26         return Replace(jsgraph()->FalseConstant());
27       }
28       if (m.IsBooleanNot()) return Replace(m.node()->InputAt(0));
29       break;
30     }
31     case IrOpcode::kChangeBitToBool: {
32       Int32Matcher m(node->InputAt(0));
33       if (m.Is(0)) return Replace(jsgraph()->FalseConstant());
34       if (m.Is(1)) return Replace(jsgraph()->TrueConstant());
35       if (m.IsChangeBoolToBit()) return Replace(m.node()->InputAt(0));
36       break;
37     }
38     case IrOpcode::kChangeBoolToBit: {
39       HeapObjectMatcher<HeapObject> m(node->InputAt(0));
40       if (m.Is(Unique<HeapObject>::CreateImmovable(factory()->false_value()))) {
41         return ReplaceInt32(0);
42       }
43       if (m.Is(Unique<HeapObject>::CreateImmovable(factory()->true_value()))) {
44         return ReplaceInt32(1);
45       }
46       if (m.IsChangeBitToBool()) return Replace(m.node()->InputAt(0));
47       break;
48     }
49     case IrOpcode::kChangeFloat64ToTagged: {
50       Float64Matcher m(node->InputAt(0));
51       if (m.HasValue()) return ReplaceNumber(m.Value());
52       break;
53     }
54     case IrOpcode::kChangeInt32ToTagged: {
55       Int32Matcher m(node->InputAt(0));
56       if (m.HasValue()) return ReplaceNumber(m.Value());
57       break;
58     }
59     case IrOpcode::kChangeTaggedToFloat64: {
60       NumberMatcher m(node->InputAt(0));
61       if (m.HasValue()) return ReplaceFloat64(m.Value());
62       if (m.IsChangeFloat64ToTagged()) return Replace(m.node()->InputAt(0));
63       if (m.IsChangeInt32ToTagged()) {
64         return Change(node, machine()->ChangeInt32ToFloat64(),
65                       m.node()->InputAt(0));
66       }
67       if (m.IsChangeUint32ToTagged()) {
68         return Change(node, machine()->ChangeUint32ToFloat64(),
69                       m.node()->InputAt(0));
70       }
71       break;
72     }
73     case IrOpcode::kChangeTaggedToInt32: {
74       NumberMatcher m(node->InputAt(0));
75       if (m.HasValue()) return ReplaceInt32(DoubleToInt32(m.Value()));
76       if (m.IsChangeFloat64ToTagged()) {
77         return Change(node, machine()->ChangeFloat64ToInt32(),
78                       m.node()->InputAt(0));
79       }
80       if (m.IsChangeInt32ToTagged()) return Replace(m.node()->InputAt(0));
81       break;
82     }
83     case IrOpcode::kChangeTaggedToUint32: {
84       NumberMatcher m(node->InputAt(0));
85       if (m.HasValue()) return ReplaceUint32(DoubleToUint32(m.Value()));
86       if (m.IsChangeFloat64ToTagged()) {
87         return Change(node, machine()->ChangeFloat64ToUint32(),
88                       m.node()->InputAt(0));
89       }
90       if (m.IsChangeUint32ToTagged()) return Replace(m.node()->InputAt(0));
91       break;
92     }
93     case IrOpcode::kChangeUint32ToTagged: {
94       Uint32Matcher m(node->InputAt(0));
95       if (m.HasValue()) return ReplaceNumber(FastUI2D(m.Value()));
96       break;
97     }
98     default:
99       break;
100   }
101   return NoChange();
102 }
103 
104 
Change(Node * node,const Operator * op,Node * a)105 Reduction SimplifiedOperatorReducer::Change(Node* node, const Operator* op,
106                                             Node* a) {
107   node->set_op(op);
108   node->ReplaceInput(0, a);
109   return Changed(node);
110 }
111 
112 
ReplaceFloat64(double value)113 Reduction SimplifiedOperatorReducer::ReplaceFloat64(double value) {
114   return Replace(jsgraph()->Float64Constant(value));
115 }
116 
117 
ReplaceInt32(int32_t value)118 Reduction SimplifiedOperatorReducer::ReplaceInt32(int32_t value) {
119   return Replace(jsgraph()->Int32Constant(value));
120 }
121 
122 
ReplaceNumber(double value)123 Reduction SimplifiedOperatorReducer::ReplaceNumber(double value) {
124   return Replace(jsgraph()->Constant(value));
125 }
126 
127 
ReplaceNumber(int32_t value)128 Reduction SimplifiedOperatorReducer::ReplaceNumber(int32_t value) {
129   return Replace(jsgraph()->Constant(value));
130 }
131 
132 
graph() const133 Graph* SimplifiedOperatorReducer::graph() const { return jsgraph()->graph(); }
134 
135 
factory() const136 Factory* SimplifiedOperatorReducer::factory() const {
137   return jsgraph()->isolate()->factory();
138 }
139 
140 
machine() const141 MachineOperatorBuilder* SimplifiedOperatorReducer::machine() const {
142   return jsgraph()->machine();
143 }
144 
145 }  // namespace compiler
146 }  // namespace internal
147 }  // namespace v8
148