js-intrinsic-lowering.cc 12.6 KB
Newer Older
1 2 3 4 5 6
// Copyright 2015 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "src/compiler/js-intrinsic-lowering.h"

7 8
#include <stack>

9
#include "src/code-factory.h"
10 11
#include "src/compiler/access-builder.h"
#include "src/compiler/js-graph.h"
12
#include "src/compiler/linkage.h"
13
#include "src/compiler/node-matchers.h"
14
#include "src/compiler/node-properties.h"
15
#include "src/compiler/operator-properties.h"
16 17
#include "src/counters.h"
#include "src/objects-inl.h"
18 19 20 21 22

namespace v8 {
namespace internal {
namespace compiler {

23 24
JSIntrinsicLowering::JSIntrinsicLowering(Editor* editor, JSGraph* jsgraph,
                                         DeoptimizationMode mode)
25
    : AdvancedReducer(editor), jsgraph_(jsgraph), mode_(mode) {}
26 27 28 29 30

Reduction JSIntrinsicLowering::Reduce(Node* node) {
  if (node->opcode() != IrOpcode::kJSCallRuntime) return NoChange();
  const Runtime::Function* const f =
      Runtime::FunctionForId(CallRuntimeParametersOf(node->op()).id());
31
  if (f->intrinsic_type != Runtime::IntrinsicType::INLINE) return NoChange();
32
  switch (f->function_id) {
33 34
    case Runtime::kInlineCreateIterResultObject:
      return ReduceCreateIterResultObject(node);
35
    case Runtime::kInlineDeoptimizeNow:
36
      return ReduceDeoptimizeNow(node);
37 38
    case Runtime::kInlineGeneratorClose:
      return ReduceGeneratorClose(node);
39 40
    case Runtime::kInlineGeneratorGetInputOrDebugPos:
      return ReduceGeneratorGetInputOrDebugPos(node);
41 42
    case Runtime::kInlineGeneratorGetResumeMode:
      return ReduceGeneratorGetResumeMode(node);
43
    case Runtime::kInlineIsArray:
44
      return ReduceIsInstanceType(node, JS_ARRAY_TYPE);
45 46
    case Runtime::kInlineIsTypedArray:
      return ReduceIsInstanceType(node, JS_TYPED_ARRAY_TYPE);
47
    case Runtime::kInlineIsRegExp:
48
      return ReduceIsInstanceType(node, JS_REGEXP_TYPE);
49 50
    case Runtime::kInlineIsJSReceiver:
      return ReduceIsJSReceiver(node);
51 52
    case Runtime::kInlineIsSmi:
      return ReduceIsSmi(node);
53 54
    case Runtime::kInlineFixedArrayGet:
      return ReduceFixedArrayGet(node);
55 56
    case Runtime::kInlineFixedArraySet:
      return ReduceFixedArraySet(node);
57 58 59 60
    case Runtime::kInlineRegExpExec:
      return ReduceRegExpExec(node);
    case Runtime::kInlineSubString:
      return ReduceSubString(node);
61 62 63 64 65 66
    case Runtime::kInlineToInteger:
      return ReduceToInteger(node);
    case Runtime::kInlineToLength:
      return ReduceToLength(node);
    case Runtime::kInlineToNumber:
      return ReduceToNumber(node);
67 68
    case Runtime::kInlineToObject:
      return ReduceToObject(node);
69 70
    case Runtime::kInlineToString:
      return ReduceToString(node);
71 72
    case Runtime::kInlineCall:
      return ReduceCall(node);
cbruni's avatar
cbruni committed
73 74
    case Runtime::kInlineNewObject:
      return ReduceNewObject(node);
75 76
    case Runtime::kInlineGetSuperConstructor:
      return ReduceGetSuperConstructor(node);
77 78 79 80 81 82 83
    default:
      break;
  }
  return NoChange();
}


84 85 86 87 88 89 90 91 92 93
Reduction JSIntrinsicLowering::ReduceCreateIterResultObject(Node* node) {
  Node* const value = NodeProperties::GetValueInput(node, 0);
  Node* const done = NodeProperties::GetValueInput(node, 1);
  Node* const context = NodeProperties::GetContextInput(node);
  Node* const effect = NodeProperties::GetEffectInput(node);
  return Change(node, javascript()->CreateIterResultObject(), value, done,
                context, effect);
}


94
Reduction JSIntrinsicLowering::ReduceDeoptimizeNow(Node* node) {
95
  if (mode() != kDeoptimizationEnabled) return NoChange();
96
  Node* const frame_state = NodeProperties::GetFrameStateInput(node);
97 98
  Node* const effect = NodeProperties::GetEffectInput(node);
  Node* const control = NodeProperties::GetControlInput(node);
99

100
  // TODO(bmeurer): Move MergeControlToEnd() to the AdvancedReducer.
101 102 103
  Node* deoptimize = graph()->NewNode(
      common()->Deoptimize(DeoptimizeKind::kEager, DeoptimizeReason::kNoReason),
      frame_state, effect, control);
104
  NodeProperties::MergeControlToEnd(graph(), common(), deoptimize);
105
  Revisit(graph()->end());
106

107
  node->TrimInputCount(0);
108
  NodeProperties::ChangeOp(node, common()->Dead());
109
  return Changed(node);
110 111
}

112 113 114 115 116 117 118 119 120 121 122 123 124
Reduction JSIntrinsicLowering::ReduceGeneratorClose(Node* node) {
  Node* const generator = NodeProperties::GetValueInput(node, 0);
  Node* const effect = NodeProperties::GetEffectInput(node);
  Node* const control = NodeProperties::GetControlInput(node);
  Node* const closed = jsgraph()->Constant(JSGeneratorObject::kGeneratorClosed);
  Node* const undefined = jsgraph()->UndefinedConstant();
  Operator const* const op = simplified()->StoreField(
      AccessBuilder::ForJSGeneratorObjectContinuation());

  ReplaceWithValue(node, undefined, node);
  NodeProperties::RemoveType(node);
  return Change(node, op, generator, closed, effect, control);
}
125

126
Reduction JSIntrinsicLowering::ReduceGeneratorGetInputOrDebugPos(Node* node) {
127 128 129
  Node* const generator = NodeProperties::GetValueInput(node, 0);
  Node* const effect = NodeProperties::GetEffectInput(node);
  Node* const control = NodeProperties::GetControlInput(node);
130 131
  Operator const* const op = simplified()->LoadField(
      AccessBuilder::ForJSGeneratorObjectInputOrDebugPos());
132 133 134 135 136 137 138 139 140 141 142 143 144 145

  return Change(node, op, generator, effect, control);
}

Reduction JSIntrinsicLowering::ReduceGeneratorGetResumeMode(Node* node) {
  Node* const generator = NodeProperties::GetValueInput(node, 0);
  Node* const effect = NodeProperties::GetEffectInput(node);
  Node* const control = NodeProperties::GetControlInput(node);
  Operator const* const op =
      simplified()->LoadField(AccessBuilder::ForJSGeneratorObjectResumeMode());

  return Change(node, op, generator, effect, control);
}

146
Reduction JSIntrinsicLowering::ReduceIsInstanceType(
147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169
    Node* node, InstanceType instance_type) {
  // if (%_IsSmi(value)) {
  //   return false;
  // } else {
  //   return %_GetInstanceType(%_GetMap(value)) == instance_type;
  // }
  Node* value = NodeProperties::GetValueInput(node, 0);
  Node* effect = NodeProperties::GetEffectInput(node);
  Node* control = NodeProperties::GetControlInput(node);

  Node* check = graph()->NewNode(simplified()->ObjectIsSmi(), value);
  Node* branch = graph()->NewNode(common()->Branch(), check, control);

  Node* if_true = graph()->NewNode(common()->IfTrue(), branch);
  Node* etrue = effect;
  Node* vtrue = jsgraph()->FalseConstant();

  Node* if_false = graph()->NewNode(common()->IfFalse(), branch);
  Node* efalse = graph()->NewNode(
      simplified()->LoadField(AccessBuilder::ForMapInstanceType()),
      graph()->NewNode(simplified()->LoadField(AccessBuilder::ForMap()), value,
                       effect, if_false),
      effect, if_false);
170 171
  Node* vfalse = graph()->NewNode(simplified()->NumberEqual(), efalse,
                                  jsgraph()->Constant(instance_type));
172 173 174 175 176

  Node* merge = graph()->NewNode(common()->Merge(2), if_true, if_false);

  // Replace all effect uses of {node} with the {ephi}.
  Node* ephi = graph()->NewNode(common()->EffectPhi(2), etrue, efalse, merge);
177
  ReplaceWithValue(node, node, ephi);
178 179

  // Turn the {node} into a Phi.
180 181
  return Change(node, common()->Phi(MachineRepresentation::kTagged, 2), vtrue,
                vfalse, merge);
182 183 184
}


185
Reduction JSIntrinsicLowering::ReduceIsJSReceiver(Node* node) {
186
  return Change(node, simplified()->ObjectIsReceiver());
187 188 189
}


190 191 192 193 194
Reduction JSIntrinsicLowering::ReduceIsSmi(Node* node) {
  return Change(node, simplified()->ObjectIsSmi());
}


195
Reduction JSIntrinsicLowering::Change(Node* node, const Operator* op) {
196
  // Replace all effect uses of {node} with the effect dependency.
197
  RelaxEffectsAndControls(node);
198 199 200
  // Remove the inputs corresponding to context, effect and control.
  NodeProperties::RemoveNonValueInputs(node);
  // Finally update the operator to the new one.
201
  NodeProperties::ChangeOp(node, op);
202 203 204 205
  return Changed(node);
}


206 207 208 209 210 211 212 213 214 215 216
Reduction JSIntrinsicLowering::ReduceFixedArrayGet(Node* node) {
  Node* base = node->InputAt(0);
  Node* index = node->InputAt(1);
  Node* effect = NodeProperties::GetEffectInput(node);
  Node* control = NodeProperties::GetControlInput(node);
  return Change(
      node, simplified()->LoadElement(AccessBuilder::ForFixedArrayElement()),
      base, index, effect, control);
}


217 218 219 220 221 222 223 224 225
Reduction JSIntrinsicLowering::ReduceFixedArraySet(Node* node) {
  Node* base = node->InputAt(0);
  Node* index = node->InputAt(1);
  Node* value = node->InputAt(2);
  Node* effect = NodeProperties::GetEffectInput(node);
  Node* control = NodeProperties::GetControlInput(node);
  Node* store = (graph()->NewNode(
      simplified()->StoreElement(AccessBuilder::ForFixedArrayElement()), base,
      index, value, effect, control));
226
  ReplaceWithValue(node, value, store);
227 228 229 230
  return Changed(store);
}


231 232 233 234 235 236 237 238 239 240
Reduction JSIntrinsicLowering::ReduceRegExpExec(Node* node) {
  return Change(node, CodeFactory::RegExpExec(isolate()), 4);
}


Reduction JSIntrinsicLowering::ReduceSubString(Node* node) {
  return Change(node, CodeFactory::SubString(isolate()), 3);
}


241
Reduction JSIntrinsicLowering::ReduceToInteger(Node* node) {
242 243
  NodeProperties::ChangeOp(node, javascript()->ToInteger());
  return Changed(node);
244 245 246 247 248 249 250 251 252 253
}


Reduction JSIntrinsicLowering::ReduceToNumber(Node* node) {
  NodeProperties::ChangeOp(node, javascript()->ToNumber());
  return Changed(node);
}


Reduction JSIntrinsicLowering::ReduceToLength(Node* node) {
254 255
  NodeProperties::ChangeOp(node, javascript()->ToLength());
  return Changed(node);
256 257 258
}


259
Reduction JSIntrinsicLowering::ReduceToObject(Node* node) {
260
  NodeProperties::ChangeOp(node, javascript()->ToObject());
261 262 263 264
  return Changed(node);
}


265 266 267 268 269 270
Reduction JSIntrinsicLowering::ReduceToString(Node* node) {
  NodeProperties::ChangeOp(node, javascript()->ToString());
  return Changed(node);
}


271
Reduction JSIntrinsicLowering::ReduceCall(Node* node) {
272
  size_t const arity = CallRuntimeParametersOf(node->op()).arity();
273 274 275 276
  NodeProperties::ChangeOp(
      node, javascript()->CallFunction(arity, 0.0f, VectorSlotPair(),
                                       ConvertReceiverMode::kAny,
                                       TailCallMode::kDisallow));
277 278 279
  return Changed(node);
}

cbruni's avatar
cbruni committed
280
Reduction JSIntrinsicLowering::ReduceNewObject(Node* node) {
281
  return Change(node, CodeFactory::FastNewObject(isolate()), 0);
cbruni's avatar
cbruni committed
282
}
283

284
Reduction JSIntrinsicLowering::ReduceGetSuperConstructor(Node* node) {
285 286 287 288 289 290 291 292
  Node* active_function = NodeProperties::GetValueInput(node, 0);
  Node* effect = NodeProperties::GetEffectInput(node);
  Node* control = NodeProperties::GetControlInput(node);
  Node* active_function_map = effect =
      graph()->NewNode(simplified()->LoadField(AccessBuilder::ForMap()),
                       active_function, effect, control);
  return Change(node, simplified()->LoadField(AccessBuilder::ForMapPrototype()),
                active_function_map, effect, control);
293 294
}

295 296
Reduction JSIntrinsicLowering::Change(Node* node, const Operator* op, Node* a,
                                      Node* b) {
297
  RelaxControls(node);
298 299 300
  node->ReplaceInput(0, a);
  node->ReplaceInput(1, b);
  node->TrimInputCount(2);
301
  NodeProperties::ChangeOp(node, op);
302 303 304 305
  return Changed(node);
}


306 307
Reduction JSIntrinsicLowering::Change(Node* node, const Operator* op, Node* a,
                                      Node* b, Node* c) {
308
  RelaxControls(node);
309 310 311 312
  node->ReplaceInput(0, a);
  node->ReplaceInput(1, b);
  node->ReplaceInput(2, c);
  node->TrimInputCount(3);
313
  NodeProperties::ChangeOp(node, op);
314 315 316 317
  return Changed(node);
}


318 319
Reduction JSIntrinsicLowering::Change(Node* node, const Operator* op, Node* a,
                                      Node* b, Node* c, Node* d) {
320
  RelaxControls(node);
321 322 323 324 325
  node->ReplaceInput(0, a);
  node->ReplaceInput(1, b);
  node->ReplaceInput(2, c);
  node->ReplaceInput(3, d);
  node->TrimInputCount(4);
326
  NodeProperties::ChangeOp(node, op);
327 328 329 330
  return Changed(node);
}


331 332 333 334 335 336 337 338 339 340 341 342
Reduction JSIntrinsicLowering::Change(Node* node, Callable const& callable,
                                      int stack_parameter_count) {
  CallDescriptor const* const desc = Linkage::GetStubCallDescriptor(
      isolate(), graph()->zone(), callable.descriptor(), stack_parameter_count,
      CallDescriptor::kNeedsFrameState, node->op()->properties());
  node->InsertInput(graph()->zone(), 0,
                    jsgraph()->HeapConstant(callable.code()));
  NodeProperties::ChangeOp(node, common()->Call(desc));
  return Changed(node);
}


343 344 345
Graph* JSIntrinsicLowering::graph() const { return jsgraph()->graph(); }


346 347 348
Isolate* JSIntrinsicLowering::isolate() const { return jsgraph()->isolate(); }


349 350 351 352
CommonOperatorBuilder* JSIntrinsicLowering::common() const {
  return jsgraph()->common();
}

353 354 355 356
JSOperatorBuilder* JSIntrinsicLowering::javascript() const {
  return jsgraph_->javascript();
}

357 358 359 360
SimplifiedOperatorBuilder* JSIntrinsicLowering::simplified() const {
  return jsgraph()->simplified();
}

361 362 363
}  // namespace compiler
}  // namespace internal
}  // namespace v8