OLD | NEW |
(Empty) | |
| 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/machine-operator-reducer.h" |
| 6 |
| 7 #include "src/compiler/common-node-cache.h" |
| 8 #include "src/compiler/generic-node-inl.h" |
| 9 #include "src/compiler/graph.h" |
| 10 #include "src/compiler/node-matchers.h" |
| 11 |
| 12 namespace v8 { |
| 13 namespace internal { |
| 14 namespace compiler { |
| 15 |
| 16 MachineOperatorReducer::MachineOperatorReducer(Graph* graph) |
| 17 : graph_(graph), |
| 18 cache_(new (graph->zone()) CommonNodeCache(graph->zone())), |
| 19 common_(graph->zone()), |
| 20 machine_(graph->zone()) {} |
| 21 |
| 22 |
| 23 MachineOperatorReducer::MachineOperatorReducer(Graph* graph, |
| 24 CommonNodeCache* cache) |
| 25 : graph_(graph), |
| 26 cache_(cache), |
| 27 common_(graph->zone()), |
| 28 machine_(graph->zone()) {} |
| 29 |
| 30 |
| 31 Node* MachineOperatorReducer::Int32Constant(int32_t value) { |
| 32 Node** loc = cache_->FindInt32Constant(value); |
| 33 if (*loc == NULL) { |
| 34 *loc = graph_->NewNode(common_.Int32Constant(value)); |
| 35 } |
| 36 return *loc; |
| 37 } |
| 38 |
| 39 |
| 40 Node* MachineOperatorReducer::Float64Constant(volatile double value) { |
| 41 Node** loc = cache_->FindFloat64Constant(value); |
| 42 if (*loc == NULL) { |
| 43 *loc = graph_->NewNode(common_.Float64Constant(value)); |
| 44 } |
| 45 return *loc; |
| 46 } |
| 47 |
| 48 |
| 49 // Perform constant folding and strength reduction on machine operators. |
| 50 Reduction MachineOperatorReducer::Reduce(Node* node) { |
| 51 switch (node->opcode()) { |
| 52 case IrOpcode::kWord32And: { |
| 53 Int32BinopMatcher m(node); |
| 54 if (m.right().Is(0)) return Replace(m.right().node()); // x & 0 => 0 |
| 55 if (m.right().Is(-1)) return Replace(m.left().node()); // x & -1 => x |
| 56 if (m.IsFoldable()) { // K & K => K |
| 57 return ReplaceInt32(m.left().Value() & m.right().Value()); |
| 58 } |
| 59 if (m.LeftEqualsRight()) return Replace(m.left().node()); // x & x => x |
| 60 break; |
| 61 } |
| 62 case IrOpcode::kWord32Or: { |
| 63 Int32BinopMatcher m(node); |
| 64 if (m.right().Is(0)) return Replace(m.left().node()); // x | 0 => x |
| 65 if (m.right().Is(-1)) return Replace(m.right().node()); // x | -1 => -1 |
| 66 if (m.IsFoldable()) { // K | K => K |
| 67 return ReplaceInt32(m.left().Value() | m.right().Value()); |
| 68 } |
| 69 if (m.LeftEqualsRight()) return Replace(m.left().node()); // x | x => x |
| 70 break; |
| 71 } |
| 72 case IrOpcode::kWord32Xor: { |
| 73 Int32BinopMatcher m(node); |
| 74 if (m.right().Is(0)) return Replace(m.left().node()); // x ^ 0 => x |
| 75 if (m.IsFoldable()) { // K ^ K => K |
| 76 return ReplaceInt32(m.left().Value() ^ m.right().Value()); |
| 77 } |
| 78 if (m.LeftEqualsRight()) return ReplaceInt32(0); // x ^ x => 0 |
| 79 break; |
| 80 } |
| 81 case IrOpcode::kWord32Shl: { |
| 82 Int32BinopMatcher m(node); |
| 83 if (m.right().Is(0)) return Replace(m.left().node()); // x << 0 => x |
| 84 if (m.IsFoldable()) { // K << K => K |
| 85 return ReplaceInt32(m.left().Value() << m.right().Value()); |
| 86 } |
| 87 break; |
| 88 } |
| 89 case IrOpcode::kWord32Shr: { |
| 90 Uint32BinopMatcher m(node); |
| 91 if (m.right().Is(0)) return Replace(m.left().node()); // x >>> 0 => x |
| 92 if (m.IsFoldable()) { // K >>> K => K |
| 93 return ReplaceInt32(m.left().Value() >> m.right().Value()); |
| 94 } |
| 95 break; |
| 96 } |
| 97 case IrOpcode::kWord32Sar: { |
| 98 Int32BinopMatcher m(node); |
| 99 if (m.right().Is(0)) return Replace(m.left().node()); // x >> 0 => x |
| 100 if (m.IsFoldable()) { // K >> K => K |
| 101 return ReplaceInt32(m.left().Value() >> m.right().Value()); |
| 102 } |
| 103 break; |
| 104 } |
| 105 case IrOpcode::kWord32Equal: { |
| 106 Int32BinopMatcher m(node); |
| 107 if (m.IsFoldable()) { // K == K => K |
| 108 return ReplaceBool(m.left().Value() == m.right().Value()); |
| 109 } |
| 110 if (m.left().IsInt32Sub() && m.right().Is(0)) { // x - y == 0 => x == y |
| 111 Int32BinopMatcher msub(m.left().node()); |
| 112 node->ReplaceInput(0, msub.left().node()); |
| 113 node->ReplaceInput(1, msub.right().node()); |
| 114 return Changed(node); |
| 115 } |
| 116 // TODO(turbofan): fold HeapConstant, ExternalReference, pointer compares |
| 117 if (m.LeftEqualsRight()) return ReplaceBool(true); // x == x => true |
| 118 break; |
| 119 } |
| 120 case IrOpcode::kInt32Add: { |
| 121 Int32BinopMatcher m(node); |
| 122 if (m.right().Is(0)) return Replace(m.left().node()); // x + 0 => x |
| 123 if (m.IsFoldable()) { // K + K => K |
| 124 return ReplaceInt32(static_cast<uint32_t>(m.left().Value()) + |
| 125 static_cast<uint32_t>(m.right().Value())); |
| 126 } |
| 127 break; |
| 128 } |
| 129 case IrOpcode::kInt32Sub: { |
| 130 Int32BinopMatcher m(node); |
| 131 if (m.right().Is(0)) return Replace(m.left().node()); // x - 0 => x |
| 132 if (m.IsFoldable()) { // K - K => K |
| 133 return ReplaceInt32(static_cast<uint32_t>(m.left().Value()) - |
| 134 static_cast<uint32_t>(m.right().Value())); |
| 135 } |
| 136 if (m.LeftEqualsRight()) return ReplaceInt32(0); // x - x => 0 |
| 137 break; |
| 138 } |
| 139 case IrOpcode::kInt32Mul: { |
| 140 Int32BinopMatcher m(node); |
| 141 if (m.right().Is(0)) return Replace(m.right().node()); // x * 0 => 0 |
| 142 if (m.right().Is(1)) return Replace(m.left().node()); // x * 1 => x |
| 143 if (m.IsFoldable()) { // K * K => K |
| 144 return ReplaceInt32(m.left().Value() * m.right().Value()); |
| 145 } |
| 146 if (m.right().Is(-1)) { // x * -1 => 0 - x |
| 147 graph_->ChangeOperator(node, machine_.Int32Sub()); |
| 148 node->ReplaceInput(0, Int32Constant(0)); |
| 149 node->ReplaceInput(1, m.left().node()); |
| 150 return Changed(node); |
| 151 } |
| 152 if (m.right().IsPowerOf2()) { // x * 2^n => x << n |
| 153 graph_->ChangeOperator(node, machine_.Word32Shl()); |
| 154 node->ReplaceInput(1, Int32Constant(WhichPowerOf2(m.right().Value()))); |
| 155 return Changed(node); |
| 156 } |
| 157 break; |
| 158 } |
| 159 case IrOpcode::kInt32Div: { |
| 160 Int32BinopMatcher m(node); |
| 161 if (m.right().Is(1)) return Replace(m.left().node()); // x / 1 => x |
| 162 // TODO(turbofan): if (m.left().Is(0)) |
| 163 // TODO(turbofan): if (m.right().IsPowerOf2()) |
| 164 // TODO(turbofan): if (m.right().Is(0)) |
| 165 // TODO(turbofan): if (m.LeftEqualsRight()) |
| 166 if (m.IsFoldable() && !m.right().Is(0)) { // K / K => K |
| 167 if (m.right().Is(-1)) return ReplaceInt32(-m.left().Value()); |
| 168 return ReplaceInt32(m.left().Value() / m.right().Value()); |
| 169 } |
| 170 if (m.right().Is(-1)) { // x / -1 => 0 - x |
| 171 graph_->ChangeOperator(node, machine_.Int32Sub()); |
| 172 node->ReplaceInput(0, Int32Constant(0)); |
| 173 node->ReplaceInput(1, m.left().node()); |
| 174 return Changed(node); |
| 175 } |
| 176 break; |
| 177 } |
| 178 case IrOpcode::kInt32UDiv: { |
| 179 Uint32BinopMatcher m(node); |
| 180 if (m.right().Is(1)) return Replace(m.left().node()); // x / 1 => x |
| 181 // TODO(turbofan): if (m.left().Is(0)) |
| 182 // TODO(turbofan): if (m.right().Is(0)) |
| 183 // TODO(turbofan): if (m.LeftEqualsRight()) |
| 184 if (m.IsFoldable() && !m.right().Is(0)) { // K / K => K |
| 185 return ReplaceInt32(m.left().Value() / m.right().Value()); |
| 186 } |
| 187 if (m.right().IsPowerOf2()) { // x / 2^n => x >> n |
| 188 graph_->ChangeOperator(node, machine_.Word32Shr()); |
| 189 node->ReplaceInput(1, Int32Constant(WhichPowerOf2(m.right().Value()))); |
| 190 return Changed(node); |
| 191 } |
| 192 break; |
| 193 } |
| 194 case IrOpcode::kInt32Mod: { |
| 195 Int32BinopMatcher m(node); |
| 196 if (m.right().Is(1)) return ReplaceInt32(0); // x % 1 => 0 |
| 197 if (m.right().Is(-1)) return ReplaceInt32(0); // x % -1 => 0 |
| 198 // TODO(turbofan): if (m.left().Is(0)) |
| 199 // TODO(turbofan): if (m.right().IsPowerOf2()) |
| 200 // TODO(turbofan): if (m.right().Is(0)) |
| 201 // TODO(turbofan): if (m.LeftEqualsRight()) |
| 202 if (m.IsFoldable() && !m.right().Is(0)) { // K % K => K |
| 203 return ReplaceInt32(m.left().Value() % m.right().Value()); |
| 204 } |
| 205 break; |
| 206 } |
| 207 case IrOpcode::kInt32UMod: { |
| 208 Uint32BinopMatcher m(node); |
| 209 if (m.right().Is(1)) return ReplaceInt32(0); // x % 1 => 0 |
| 210 // TODO(turbofan): if (m.left().Is(0)) |
| 211 // TODO(turbofan): if (m.right().Is(0)) |
| 212 // TODO(turbofan): if (m.LeftEqualsRight()) |
| 213 if (m.IsFoldable() && !m.right().Is(0)) { // K % K => K |
| 214 return ReplaceInt32(m.left().Value() % m.right().Value()); |
| 215 } |
| 216 if (m.right().IsPowerOf2()) { // x % 2^n => x & 2^n-1 |
| 217 graph_->ChangeOperator(node, machine_.Word32And()); |
| 218 node->ReplaceInput(1, Int32Constant(m.right().Value() - 1)); |
| 219 return Changed(node); |
| 220 } |
| 221 break; |
| 222 } |
| 223 case IrOpcode::kInt32LessThan: { |
| 224 Int32BinopMatcher m(node); |
| 225 if (m.IsFoldable()) { // K < K => K |
| 226 return ReplaceBool(m.left().Value() < m.right().Value()); |
| 227 } |
| 228 if (m.left().IsInt32Sub() && m.right().Is(0)) { // x - y < 0 => x < y |
| 229 Int32BinopMatcher msub(m.left().node()); |
| 230 node->ReplaceInput(0, msub.left().node()); |
| 231 node->ReplaceInput(1, msub.right().node()); |
| 232 return Changed(node); |
| 233 } |
| 234 if (m.left().Is(0) && m.right().IsInt32Sub()) { // 0 < x - y => y < x |
| 235 Int32BinopMatcher msub(m.right().node()); |
| 236 node->ReplaceInput(0, msub.right().node()); |
| 237 node->ReplaceInput(1, msub.left().node()); |
| 238 return Changed(node); |
| 239 } |
| 240 if (m.LeftEqualsRight()) return ReplaceBool(false); // x < x => false |
| 241 break; |
| 242 } |
| 243 case IrOpcode::kInt32LessThanOrEqual: { |
| 244 Int32BinopMatcher m(node); |
| 245 if (m.IsFoldable()) { // K <= K => K |
| 246 return ReplaceBool(m.left().Value() <= m.right().Value()); |
| 247 } |
| 248 if (m.left().IsInt32Sub() && m.right().Is(0)) { // x - y <= 0 => x <= y |
| 249 Int32BinopMatcher msub(m.left().node()); |
| 250 node->ReplaceInput(0, msub.left().node()); |
| 251 node->ReplaceInput(1, msub.right().node()); |
| 252 return Changed(node); |
| 253 } |
| 254 if (m.left().Is(0) && m.right().IsInt32Sub()) { // 0 <= x - y => y <= x |
| 255 Int32BinopMatcher msub(m.right().node()); |
| 256 node->ReplaceInput(0, msub.right().node()); |
| 257 node->ReplaceInput(1, msub.left().node()); |
| 258 return Changed(node); |
| 259 } |
| 260 if (m.LeftEqualsRight()) return ReplaceBool(true); // x <= x => true |
| 261 break; |
| 262 } |
| 263 case IrOpcode::kUint32LessThan: { |
| 264 Uint32BinopMatcher m(node); |
| 265 if (m.left().Is(kMaxUInt32)) return ReplaceBool(false); // M < x => false |
| 266 if (m.right().Is(0)) return ReplaceBool(false); // x < 0 => false |
| 267 if (m.IsFoldable()) { // K < K => K |
| 268 return ReplaceBool(m.left().Value() < m.right().Value()); |
| 269 } |
| 270 if (m.LeftEqualsRight()) return ReplaceBool(false); // x < x => false |
| 271 break; |
| 272 } |
| 273 case IrOpcode::kUint32LessThanOrEqual: { |
| 274 Uint32BinopMatcher m(node); |
| 275 if (m.left().Is(0)) return ReplaceBool(true); // 0 <= x => true |
| 276 if (m.right().Is(kMaxUInt32)) return ReplaceBool(true); // x <= M => true |
| 277 if (m.IsFoldable()) { // K <= K => K |
| 278 return ReplaceBool(m.left().Value() <= m.right().Value()); |
| 279 } |
| 280 if (m.LeftEqualsRight()) return ReplaceBool(true); // x <= x => true |
| 281 break; |
| 282 } |
| 283 case IrOpcode::kFloat64Add: { |
| 284 Float64BinopMatcher m(node); |
| 285 if (m.IsFoldable()) { // K + K => K |
| 286 return ReplaceFloat64(m.left().Value() + m.right().Value()); |
| 287 } |
| 288 break; |
| 289 } |
| 290 case IrOpcode::kFloat64Sub: { |
| 291 Float64BinopMatcher m(node); |
| 292 if (m.IsFoldable()) { // K - K => K |
| 293 return ReplaceFloat64(m.left().Value() - m.right().Value()); |
| 294 } |
| 295 break; |
| 296 } |
| 297 case IrOpcode::kFloat64Mul: { |
| 298 Float64BinopMatcher m(node); |
| 299 if (m.right().Is(1)) return Replace(m.left().node()); // x * 1.0 => x |
| 300 if (m.right().IsNaN()) { // x * NaN => NaN |
| 301 return Replace(m.right().node()); |
| 302 } |
| 303 if (m.IsFoldable()) { // K * K => K |
| 304 return ReplaceFloat64(m.left().Value() * m.right().Value()); |
| 305 } |
| 306 break; |
| 307 } |
| 308 case IrOpcode::kFloat64Div: { |
| 309 Float64BinopMatcher m(node); |
| 310 if (m.right().Is(1)) return Replace(m.left().node()); // x / 1.0 => x |
| 311 if (m.right().IsNaN()) { // x / NaN => NaN |
| 312 return Replace(m.right().node()); |
| 313 } |
| 314 if (m.left().IsNaN()) { // NaN / x => NaN |
| 315 return Replace(m.left().node()); |
| 316 } |
| 317 if (m.IsFoldable()) { // K / K => K |
| 318 return ReplaceFloat64(m.left().Value() / m.right().Value()); |
| 319 } |
| 320 break; |
| 321 } |
| 322 case IrOpcode::kFloat64Mod: { |
| 323 Float64BinopMatcher m(node); |
| 324 if (m.right().IsNaN()) { // x % NaN => NaN |
| 325 return Replace(m.right().node()); |
| 326 } |
| 327 if (m.left().IsNaN()) { // NaN % x => NaN |
| 328 return Replace(m.left().node()); |
| 329 } |
| 330 if (m.IsFoldable()) { // K % K => K |
| 331 return ReplaceFloat64(modulo(m.left().Value(), m.right().Value())); |
| 332 } |
| 333 break; |
| 334 } |
| 335 // TODO(turbofan): strength-reduce and fold floating point operations. |
| 336 default: |
| 337 break; |
| 338 } |
| 339 return NoChange(); |
| 340 } |
| 341 } |
| 342 } |
| 343 } // namespace v8::internal::compiler |
OLD | NEW |