Index: src/arm/lithium-codegen-arm.cc |
diff --git a/src/arm/lithium-codegen-arm.cc b/src/arm/lithium-codegen-arm.cc |
index 407bb91d7df9da56c5026ff033b17e8c39d83cd7..0447468a78e7d853b1258de14abea6ff8f7dc302 100644 |
--- a/src/arm/lithium-codegen-arm.cc |
+++ b/src/arm/lithium-codegen-arm.cc |
@@ -4506,6 +4506,100 @@ void LCodeGen::DoStoreKeyedGeneric(LStoreKeyedGeneric* instr) { |
} |
+void LCodeGen::DoMaybeGrowElements(LMaybeGrowElements* instr) { |
+ class DeferredMaybeGrowElements final : public LDeferredCode { |
+ public: |
+ DeferredMaybeGrowElements(LCodeGen* codegen, LMaybeGrowElements* instr) |
+ : LDeferredCode(codegen), instr_(instr) {} |
+ void Generate() override { codegen()->DoDeferredMaybeGrowElements(instr_); } |
+ LInstruction* instr() override { return instr_; } |
+ |
+ private: |
+ LMaybeGrowElements* instr_; |
+ }; |
+ |
+ Register result = r0; |
+ DeferredMaybeGrowElements* deferred = |
+ new (zone()) DeferredMaybeGrowElements(this, instr); |
+ LOperand* key = instr->key(); |
+ LOperand* current_capacity = instr->current_capacity(); |
+ |
+ DCHECK(instr->hydrogen()->key()->representation().IsInteger32()); |
+ DCHECK(instr->hydrogen()->current_capacity()->representation().IsInteger32()); |
+ DCHECK(key->IsConstantOperand() || key->IsRegister()); |
+ DCHECK(current_capacity->IsConstantOperand() || |
+ current_capacity->IsRegister()); |
+ |
+ if (key->IsConstantOperand() && current_capacity->IsConstantOperand()) { |
+ int32_t constant_key = ToInteger32(LConstantOperand::cast(key)); |
+ int32_t constant_capacity = |
+ ToInteger32(LConstantOperand::cast(current_capacity)); |
+ if (constant_key >= constant_capacity) { |
+ // Deferred case. |
+ __ jmp(deferred->entry()); |
+ } |
+ } else if (key->IsConstantOperand()) { |
+ int32_t constant_key = ToInteger32(LConstantOperand::cast(key)); |
+ __ cmp(ToRegister(current_capacity), Operand(constant_key)); |
+ __ b(le, deferred->entry()); |
+ } else if (current_capacity->IsConstantOperand()) { |
+ int32_t constant_capacity = |
+ ToInteger32(LConstantOperand::cast(current_capacity)); |
+ __ cmp(ToRegister(key), Operand(constant_capacity)); |
+ __ b(ge, deferred->entry()); |
+ } else { |
+ __ cmp(ToRegister(key), ToRegister(current_capacity)); |
+ __ b(ge, deferred->entry()); |
+ } |
+ |
+ if (instr->elements()->IsRegister()) { |
+ __ Move(result, ToRegister(instr->elements())); |
+ } else { |
+ __ ldr(result, ToMemOperand(instr->elements())); |
+ } |
+ |
+ __ bind(deferred->exit()); |
+} |
+ |
+ |
+void LCodeGen::DoDeferredMaybeGrowElements(LMaybeGrowElements* instr) { |
+ // TODO(3095996): Get rid of this. For now, we need to make the |
+ // result register contain a valid pointer because it is already |
+ // contained in the register pointer map. |
+ Register result = r0; |
+ __ mov(result, Operand::Zero()); |
+ |
+ // We have to call a stub. |
+ { |
+ PushSafepointRegistersScope scope(this); |
+ if (instr->object()->IsRegister()) { |
+ __ Move(result, ToRegister(instr->object())); |
+ } else { |
+ __ ldr(result, ToMemOperand(instr->object())); |
+ } |
+ |
+ LOperand* key = instr->key(); |
+ if (key->IsConstantOperand()) { |
+ __ Move(r3, Operand(ToSmi(LConstantOperand::cast(key)))); |
+ } else { |
+ __ Move(r3, ToRegister(key)); |
+ __ SmiTag(r3); |
+ } |
+ |
+ GrowArrayElementsStub stub(isolate(), instr->hydrogen()->is_js_array(), |
+ instr->hydrogen()->kind()); |
+ __ CallStub(&stub); |
+ RecordSafepointWithLazyDeopt( |
+ instr, RECORD_SAFEPOINT_WITH_REGISTERS_AND_NO_ARGUMENTS); |
+ __ StoreToSafepointRegisterSlot(result, result); |
+ } |
+ |
+ // Deopt on smi, which means the elements array changed to dictionary mode. |
+ __ SmiTst(result); |
+ DeoptimizeIf(eq, instr, Deoptimizer::kSmi); |
+} |
+ |
+ |
void LCodeGen::DoTransitionElementsKind(LTransitionElementsKind* instr) { |
Register object_reg = ToRegister(instr->object()); |
Register scratch = scratch0(); |