| Index: src/arm/full-codegen-arm.cc
|
| ===================================================================
|
| --- src/arm/full-codegen-arm.cc (revision 8618)
|
| +++ src/arm/full-codegen-arm.cc (working copy)
|
| @@ -93,17 +93,19 @@
|
| }
|
|
|
| void EmitPatchInfo() {
|
| - int delta_to_patch_site = masm_->InstructionsGeneratedSince(&patch_site_);
|
| - Register reg;
|
| - reg.set_code(delta_to_patch_site / kOff12Mask);
|
| - __ cmp_raw_immediate(reg, delta_to_patch_site % kOff12Mask);
|
| + if (patch_site_.is_bound()) {
|
| + int delta_to_patch_site = masm_->InstructionsGeneratedSince(&patch_site_);
|
| + Register reg;
|
| + reg.set_code(delta_to_patch_site / kOff12Mask);
|
| + __ cmp_raw_immediate(reg, delta_to_patch_site % kOff12Mask);
|
| #ifdef DEBUG
|
| - info_emitted_ = true;
|
| + info_emitted_ = true;
|
| #endif
|
| + } else {
|
| + __ nop(); // Signals no inlined code.
|
| + }
|
| }
|
|
|
| - bool is_bound() const { return patch_site_.is_bound(); }
|
| -
|
| private:
|
| MacroAssembler* masm_;
|
| Label patch_site_;
|
| @@ -130,6 +132,7 @@
|
| void FullCodeGenerator::Generate(CompilationInfo* info) {
|
| ASSERT(info_ == NULL);
|
| info_ = info;
|
| + scope_ = info->scope();
|
| SetFunctionPosition(function());
|
| Comment cmnt(masm_, "[ function compiled by full code generator");
|
|
|
| @@ -140,21 +143,21 @@
|
| }
|
| #endif
|
|
|
| - // Strict mode functions need to replace the receiver with undefined
|
| - // when called as functions (without an explicit receiver
|
| - // object). r5 is zero for method calls and non-zero for function
|
| - // calls.
|
| - if (info->is_strict_mode()) {
|
| + // Strict mode functions and builtins need to replace the receiver
|
| + // with undefined when called as functions (without an explicit
|
| + // receiver object). r5 is zero for method calls and non-zero for
|
| + // function calls.
|
| + if (info->is_strict_mode() || info->is_native()) {
|
| Label ok;
|
| __ cmp(r5, Operand(0));
|
| __ b(eq, &ok);
|
| - int receiver_offset = scope()->num_parameters() * kPointerSize;
|
| + int receiver_offset = info->scope()->num_parameters() * kPointerSize;
|
| __ LoadRoot(r2, Heap::kUndefinedValueRootIndex);
|
| __ str(r2, MemOperand(sp, receiver_offset));
|
| __ bind(&ok);
|
| }
|
|
|
| - int locals_count = scope()->num_stack_slots();
|
| + int locals_count = info->scope()->num_stack_slots();
|
|
|
| __ Push(lr, fp, cp, r1);
|
| if (locals_count > 0) {
|
| @@ -174,7 +177,7 @@
|
| bool function_in_register = true;
|
|
|
| // Possibly allocate a local context.
|
| - int heap_slots = scope()->num_heap_slots() - Context::MIN_CONTEXT_SLOTS;
|
| + int heap_slots = info->scope()->num_heap_slots() - Context::MIN_CONTEXT_SLOTS;
|
| if (heap_slots > 0) {
|
| Comment cmnt(masm_, "[ Allocate local context");
|
| // Argument to NewContext is the function, which is in r1.
|
| @@ -183,14 +186,14 @@
|
| FastNewContextStub stub(heap_slots);
|
| __ CallStub(&stub);
|
| } else {
|
| - __ CallRuntime(Runtime::kNewContext, 1);
|
| + __ CallRuntime(Runtime::kNewFunctionContext, 1);
|
| }
|
| function_in_register = false;
|
| // Context is returned in both r0 and cp. It replaces the context
|
| // passed to us. It's saved in the stack and kept live in cp.
|
| __ str(cp, MemOperand(fp, StandardFrameConstants::kContextOffset));
|
| // Copy any necessary parameters into the context.
|
| - int num_parameters = scope()->num_parameters();
|
| + int num_parameters = info->scope()->num_parameters();
|
| for (int i = 0; i < num_parameters; i++) {
|
| Slot* slot = scope()->parameter(i)->AsSlot();
|
| if (slot != NULL && slot->type() == Slot::CONTEXT) {
|
| @@ -220,27 +223,28 @@
|
| __ mov(r3, r1);
|
| }
|
| // Receiver is just before the parameters on the caller's stack.
|
| - int offset = scope()->num_parameters() * kPointerSize;
|
| + int num_parameters = info->scope()->num_parameters();
|
| + int offset = num_parameters * kPointerSize;
|
| __ add(r2, fp,
|
| Operand(StandardFrameConstants::kCallerSPOffset + offset));
|
| - __ mov(r1, Operand(Smi::FromInt(scope()->num_parameters())));
|
| + __ mov(r1, Operand(Smi::FromInt(num_parameters)));
|
| __ Push(r3, r2, r1);
|
|
|
| // Arguments to ArgumentsAccessStub:
|
| // function, receiver address, parameter count.
|
| // The stub will rewrite receiever and parameter count if the previous
|
| // stack frame was an arguments adapter frame.
|
| - ArgumentsAccessStub stub(
|
| - is_strict_mode() ? ArgumentsAccessStub::NEW_STRICT
|
| - : ArgumentsAccessStub::NEW_NON_STRICT);
|
| + ArgumentsAccessStub::Type type;
|
| + if (is_strict_mode()) {
|
| + type = ArgumentsAccessStub::NEW_STRICT;
|
| + } else if (function()->has_duplicate_parameters()) {
|
| + type = ArgumentsAccessStub::NEW_NON_STRICT_SLOW;
|
| + } else {
|
| + type = ArgumentsAccessStub::NEW_NON_STRICT_FAST;
|
| + }
|
| + ArgumentsAccessStub stub(type);
|
| __ CallStub(&stub);
|
|
|
| - Variable* arguments_shadow = scope()->arguments_shadow();
|
| - if (arguments_shadow != NULL) {
|
| - // Duplicate the value; move-to-slot operation might clobber registers.
|
| - __ mov(r3, r0);
|
| - Move(arguments_shadow->AsSlot(), r3, r1, r2);
|
| - }
|
| Move(arguments->AsSlot(), r0, r1, r2);
|
| }
|
|
|
| @@ -345,7 +349,7 @@
|
| { Assembler::BlockConstPoolScope block_const_pool(masm_);
|
| // Here we use masm_-> instead of the __ macro to avoid the code coverage
|
| // tool from instrumenting as we rely on the code size here.
|
| - int32_t sp_delta = (scope()->num_parameters() + 1) * kPointerSize;
|
| + int32_t sp_delta = (info_->scope()->num_parameters() + 1) * kPointerSize;
|
| CodeGenerator::RecordPositions(masm_, function()->end_position() - 1);
|
| __ RecordJSReturn();
|
| masm_->mov(sp, fp);
|
| @@ -383,7 +387,7 @@
|
| // For simplicity we always test the accumulator register.
|
| codegen()->Move(result_register(), slot);
|
| codegen()->PrepareForBailoutBeforeSplit(TOS_REG, false, NULL, NULL);
|
| - codegen()->DoTest(true_label_, false_label_, fall_through_);
|
| + codegen()->DoTest(this);
|
| }
|
|
|
|
|
| @@ -417,7 +421,7 @@
|
| if (true_label_ != fall_through_) __ b(true_label_);
|
| } else {
|
| __ LoadRoot(result_register(), index);
|
| - codegen()->DoTest(true_label_, false_label_, fall_through_);
|
| + codegen()->DoTest(this);
|
| }
|
| }
|
|
|
| @@ -464,7 +468,7 @@
|
| } else {
|
| // For simplicity we always test the accumulator register.
|
| __ mov(result_register(), Operand(lit));
|
| - codegen()->DoTest(true_label_, false_label_, fall_through_);
|
| + codegen()->DoTest(this);
|
| }
|
| }
|
|
|
| @@ -500,7 +504,7 @@
|
| __ Drop(count);
|
| __ Move(result_register(), reg);
|
| codegen()->PrepareForBailoutBeforeSplit(TOS_REG, false, NULL, NULL);
|
| - codegen()->DoTest(true_label_, false_label_, fall_through_);
|
| + codegen()->DoTest(this);
|
| }
|
|
|
|
|
| @@ -578,7 +582,8 @@
|
| }
|
|
|
|
|
| -void FullCodeGenerator::DoTest(Label* if_true,
|
| +void FullCodeGenerator::DoTest(Expression* condition,
|
| + Label* if_true,
|
| Label* if_false,
|
| Label* fall_through) {
|
| if (CpuFeatures::IsSupported(VFP3)) {
|
| @@ -715,10 +720,12 @@
|
| // context.
|
| ASSERT_EQ(0, scope()->ContextChainLength(variable->scope()));
|
| if (FLAG_debug_code) {
|
| - // Check that we're not inside a 'with'.
|
| - __ ldr(r1, ContextOperand(cp, Context::FCONTEXT_INDEX));
|
| - __ cmp(r1, cp);
|
| - __ Check(eq, "Unexpected declaration in current context.");
|
| + // Check that we're not inside a with or catch context.
|
| + __ ldr(r1, FieldMemOperand(cp, HeapObject::kMapOffset));
|
| + __ CompareRoot(r1, Heap::kWithContextMapRootIndex);
|
| + __ Check(ne, "Declaration in with context.");
|
| + __ CompareRoot(r1, Heap::kCatchContextMapRootIndex);
|
| + __ Check(ne, "Declaration in catch context.");
|
| }
|
| if (mode == Variable::CONST) {
|
| __ LoadRoot(ip, Heap::kTheHoleValueRootIndex);
|
| @@ -793,7 +800,7 @@
|
| Handle<Code> ic = is_strict_mode()
|
| ? isolate()->builtins()->KeyedStoreIC_Initialize_Strict()
|
| : isolate()->builtins()->KeyedStoreIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, AstNode::kNoNumber);
|
| + __ Call(ic);
|
| // Value in r0 is ignored (declarations are statements).
|
| }
|
| }
|
| @@ -867,7 +874,8 @@
|
| // Record position before stub call for type feedback.
|
| SetSourcePosition(clause->position());
|
| Handle<Code> ic = CompareIC::GetUninitialized(Token::EQ_STRICT);
|
| - EmitCallIC(ic, &patch_site, clause->CompareId());
|
| + __ Call(ic, RelocInfo::CODE_TARGET, clause->CompareId());
|
| + patch_site.EmitPatchInfo();
|
|
|
| __ cmp(r0, Operand(0));
|
| __ b(ne, &next_test);
|
| @@ -922,8 +930,8 @@
|
| // Convert the object to a JS object.
|
| Label convert, done_convert;
|
| __ JumpIfSmi(r0, &convert);
|
| - __ CompareObjectType(r0, r1, r1, FIRST_JS_OBJECT_TYPE);
|
| - __ b(hs, &done_convert);
|
| + __ CompareObjectType(r0, r1, r1, FIRST_SPEC_OBJECT_TYPE);
|
| + __ b(ge, &done_convert);
|
| __ bind(&convert);
|
| __ push(r0);
|
| __ InvokeBuiltin(Builtins::TO_OBJECT, CALL_FUNCTION);
|
| @@ -1137,8 +1145,7 @@
|
| __ b(ne, slow);
|
| }
|
| // Load next context in chain.
|
| - __ ldr(next, ContextOperand(current, Context::CLOSURE_INDEX));
|
| - __ ldr(next, FieldMemOperand(next, JSFunction::kContextOffset));
|
| + __ ldr(next, ContextOperand(current, Context::PREVIOUS_INDEX));
|
| // Walk the rest of the chain without clobbering cp.
|
| current = next;
|
| }
|
| @@ -1164,8 +1171,7 @@
|
| __ tst(temp, temp);
|
| __ b(ne, slow);
|
| // Load next context in chain.
|
| - __ ldr(next, ContextOperand(next, Context::CLOSURE_INDEX));
|
| - __ ldr(next, FieldMemOperand(next, JSFunction::kContextOffset));
|
| + __ ldr(next, ContextOperand(next, Context::PREVIOUS_INDEX));
|
| __ b(&loop);
|
| __ bind(&fast);
|
| }
|
| @@ -1176,7 +1182,7 @@
|
| ? RelocInfo::CODE_TARGET
|
| : RelocInfo::CODE_TARGET_CONTEXT;
|
| Handle<Code> ic = isolate()->builtins()->LoadIC_Initialize();
|
| - EmitCallIC(ic, mode, AstNode::kNoNumber);
|
| + __ Call(ic, mode);
|
| }
|
|
|
|
|
| @@ -1196,8 +1202,7 @@
|
| __ tst(temp, temp);
|
| __ b(ne, slow);
|
| }
|
| - __ ldr(next, ContextOperand(context, Context::CLOSURE_INDEX));
|
| - __ ldr(next, FieldMemOperand(next, JSFunction::kContextOffset));
|
| + __ ldr(next, ContextOperand(context, Context::PREVIOUS_INDEX));
|
| // Walk the rest of the chain without clobbering cp.
|
| context = next;
|
| }
|
| @@ -1258,7 +1263,7 @@
|
| __ mov(r0, Operand(key_literal->handle()));
|
| Handle<Code> ic =
|
| isolate()->builtins()->KeyedLoadIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, GetPropertyId(property));
|
| + __ Call(ic, RelocInfo::CODE_TARGET, GetPropertyId(property));
|
| __ jmp(done);
|
| }
|
| }
|
| @@ -1268,23 +1273,22 @@
|
|
|
|
|
| void FullCodeGenerator::EmitVariableLoad(Variable* var) {
|
| - // Four cases: non-this global variables, lookup slots, all other
|
| - // types of slots, and parameters that rewrite to explicit property
|
| - // accesses on the arguments object.
|
| + // Three cases: non-this global variables, lookup slots, and all other
|
| + // types of slots.
|
| Slot* slot = var->AsSlot();
|
| - Property* property = var->AsProperty();
|
| + ASSERT((var->is_global() && !var->is_this()) == (slot == NULL));
|
|
|
| - if (var->is_global() && !var->is_this()) {
|
| + if (slot == NULL) {
|
| Comment cmnt(masm_, "Global variable");
|
| // Use inline caching. Variable name is passed in r2 and the global
|
| // object (receiver) in r0.
|
| __ ldr(r0, GlobalObjectOperand());
|
| __ mov(r2, Operand(var->name()));
|
| Handle<Code> ic = isolate()->builtins()->LoadIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET_CONTEXT, AstNode::kNoNumber);
|
| + __ Call(ic, RelocInfo::CODE_TARGET_CONTEXT);
|
| context()->Plug(r0);
|
|
|
| - } else if (slot != NULL && slot->type() == Slot::LOOKUP) {
|
| + } else if (slot->type() == Slot::LOOKUP) {
|
| Label done, slow;
|
|
|
| // Generate code for loading from variables potentially shadowed
|
| @@ -1300,7 +1304,7 @@
|
|
|
| context()->Plug(r0);
|
|
|
| - } else if (slot != NULL) {
|
| + } else {
|
| Comment cmnt(masm_, (slot->type() == Slot::CONTEXT)
|
| ? "Context slot"
|
| : "Stack slot");
|
| @@ -1316,32 +1320,6 @@
|
| } else {
|
| context()->Plug(slot);
|
| }
|
| - } else {
|
| - Comment cmnt(masm_, "Rewritten parameter");
|
| - ASSERT_NOT_NULL(property);
|
| - // Rewritten parameter accesses are of the form "slot[literal]".
|
| -
|
| - // Assert that the object is in a slot.
|
| - Variable* object_var = property->obj()->AsVariableProxy()->AsVariable();
|
| - ASSERT_NOT_NULL(object_var);
|
| - Slot* object_slot = object_var->AsSlot();
|
| - ASSERT_NOT_NULL(object_slot);
|
| -
|
| - // Load the object.
|
| - Move(r1, object_slot);
|
| -
|
| - // Assert that the key is a smi.
|
| - Literal* key_literal = property->key()->AsLiteral();
|
| - ASSERT_NOT_NULL(key_literal);
|
| - ASSERT(key_literal->handle()->IsSmi());
|
| -
|
| - // Load the key.
|
| - __ mov(r0, Operand(key_literal->handle()));
|
| -
|
| - // Call keyed load IC. It has arguments key and receiver in r0 and r1.
|
| - Handle<Code> ic = isolate()->builtins()->KeyedLoadIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, GetPropertyId(property));
|
| - context()->Plug(r0);
|
| }
|
| }
|
|
|
| @@ -1451,7 +1429,7 @@
|
| Handle<Code> ic = is_strict_mode()
|
| ? isolate()->builtins()->StoreIC_Initialize_Strict()
|
| : isolate()->builtins()->StoreIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, key->id());
|
| + __ Call(ic, RelocInfo::CODE_TARGET, key->id());
|
| PrepareForBailoutForId(key->id(), NO_REGISTERS);
|
| } else {
|
| VisitForEffect(value);
|
| @@ -1583,7 +1561,7 @@
|
| }
|
|
|
| // Left-hand side can only be a property, a global or a (parameter or local)
|
| - // slot. Variables with rewrite to .arguments are treated as KEYED_PROPERTY.
|
| + // slot.
|
| enum LhsKind { VARIABLE, NAMED_PROPERTY, KEYED_PROPERTY };
|
| LhsKind assign_type = VARIABLE;
|
| Property* property = expr->target()->AsProperty();
|
| @@ -1609,27 +1587,13 @@
|
| break;
|
| case KEYED_PROPERTY:
|
| if (expr->is_compound()) {
|
| - if (property->is_arguments_access()) {
|
| - VariableProxy* obj_proxy = property->obj()->AsVariableProxy();
|
| - __ ldr(r0, EmitSlotSearch(obj_proxy->var()->AsSlot(), r0));
|
| - __ push(r0);
|
| - __ mov(r0, Operand(property->key()->AsLiteral()->handle()));
|
| - } else {
|
| - VisitForStackValue(property->obj());
|
| - VisitForAccumulatorValue(property->key());
|
| - }
|
| + VisitForStackValue(property->obj());
|
| + VisitForAccumulatorValue(property->key());
|
| __ ldr(r1, MemOperand(sp, 0));
|
| __ push(r0);
|
| } else {
|
| - if (property->is_arguments_access()) {
|
| - VariableProxy* obj_proxy = property->obj()->AsVariableProxy();
|
| - __ ldr(r1, EmitSlotSearch(obj_proxy->var()->AsSlot(), r0));
|
| - __ mov(r0, Operand(property->key()->AsLiteral()->handle()));
|
| - __ Push(r1, r0);
|
| - } else {
|
| - VisitForStackValue(property->obj());
|
| - VisitForStackValue(property->key());
|
| - }
|
| + VisitForStackValue(property->obj());
|
| + VisitForStackValue(property->key());
|
| }
|
| break;
|
| }
|
| @@ -1706,7 +1670,7 @@
|
| __ mov(r2, Operand(key->handle()));
|
| // Call load IC. It has arguments receiver and property name r0 and r2.
|
| Handle<Code> ic = isolate()->builtins()->LoadIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, GetPropertyId(prop));
|
| + __ Call(ic, RelocInfo::CODE_TARGET, GetPropertyId(prop));
|
| }
|
|
|
|
|
| @@ -1714,7 +1678,7 @@
|
| SetSourcePosition(prop->position());
|
| // Call keyed load IC. It has arguments key and receiver in r0 and r1.
|
| Handle<Code> ic = isolate()->builtins()->KeyedLoadIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, GetPropertyId(prop));
|
| + __ Call(ic, RelocInfo::CODE_TARGET, GetPropertyId(prop));
|
| }
|
|
|
|
|
| @@ -1741,7 +1705,8 @@
|
|
|
| __ bind(&stub_call);
|
| BinaryOpStub stub(op, mode);
|
| - EmitCallIC(stub.GetCode(), &patch_site, expr->id());
|
| + __ Call(stub.GetCode(), RelocInfo::CODE_TARGET, expr->id());
|
| + patch_site.EmitPatchInfo();
|
| __ jmp(&done);
|
|
|
| __ bind(&smi_case);
|
| @@ -1822,7 +1787,9 @@
|
| OverwriteMode mode) {
|
| __ pop(r1);
|
| BinaryOpStub stub(op, mode);
|
| - EmitCallIC(stub.GetCode(), NULL, expr->id());
|
| + JumpPatchSite patch_site(masm_); // unbound, signals no inlined smi code.
|
| + __ Call(stub.GetCode(), RelocInfo::CODE_TARGET, expr->id());
|
| + patch_site.EmitPatchInfo();
|
| context()->Plug(r0);
|
| }
|
|
|
| @@ -1836,7 +1803,7 @@
|
| }
|
|
|
| // Left-hand side can only be a property, a global or a (parameter or local)
|
| - // slot. Variables with rewrite to .arguments are treated as KEYED_PROPERTY.
|
| + // slot.
|
| enum LhsKind { VARIABLE, NAMED_PROPERTY, KEYED_PROPERTY };
|
| LhsKind assign_type = VARIABLE;
|
| Property* prop = expr->AsProperty();
|
| @@ -1862,30 +1829,20 @@
|
| Handle<Code> ic = is_strict_mode()
|
| ? isolate()->builtins()->StoreIC_Initialize_Strict()
|
| : isolate()->builtins()->StoreIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, AstNode::kNoNumber);
|
| + __ Call(ic);
|
| break;
|
| }
|
| case KEYED_PROPERTY: {
|
| __ push(r0); // Preserve value.
|
| - if (prop->is_synthetic()) {
|
| - ASSERT(prop->obj()->AsVariableProxy() != NULL);
|
| - ASSERT(prop->key()->AsLiteral() != NULL);
|
| - { AccumulatorValueContext for_object(this);
|
| - EmitVariableLoad(prop->obj()->AsVariableProxy()->var());
|
| - }
|
| - __ mov(r2, r0);
|
| - __ mov(r1, Operand(prop->key()->AsLiteral()->handle()));
|
| - } else {
|
| - VisitForStackValue(prop->obj());
|
| - VisitForAccumulatorValue(prop->key());
|
| - __ mov(r1, r0);
|
| - __ pop(r2);
|
| - }
|
| + VisitForStackValue(prop->obj());
|
| + VisitForAccumulatorValue(prop->key());
|
| + __ mov(r1, r0);
|
| + __ pop(r2);
|
| __ pop(r0); // Restore value.
|
| Handle<Code> ic = is_strict_mode()
|
| ? isolate()->builtins()->KeyedStoreIC_Initialize_Strict()
|
| : isolate()->builtins()->KeyedStoreIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, AstNode::kNoNumber);
|
| + __ Call(ic);
|
| break;
|
| }
|
| }
|
| @@ -1896,8 +1853,6 @@
|
|
|
| void FullCodeGenerator::EmitVariableAssignment(Variable* var,
|
| Token::Value op) {
|
| - // Left-hand sides that rewrite to explicit property accesses do not reach
|
| - // here.
|
| ASSERT(var != NULL);
|
| ASSERT(var->is_global() || var->AsSlot() != NULL);
|
|
|
| @@ -1911,7 +1866,7 @@
|
| Handle<Code> ic = is_strict_mode()
|
| ? isolate()->builtins()->StoreIC_Initialize_Strict()
|
| : isolate()->builtins()->StoreIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET_CONTEXT, AstNode::kNoNumber);
|
| + __ Call(ic, RelocInfo::CODE_TARGET_CONTEXT);
|
|
|
| } else if (op == Token::INIT_CONST) {
|
| // Like var declarations, const declarations are hoisted to function
|
| @@ -1933,19 +1888,8 @@
|
| __ b(ne, &skip);
|
| __ str(result_register(), MemOperand(fp, SlotOffset(slot)));
|
| break;
|
| - case Slot::CONTEXT: {
|
| - __ ldr(r1, ContextOperand(cp, Context::FCONTEXT_INDEX));
|
| - __ ldr(r2, ContextOperand(r1, slot->index()));
|
| - __ LoadRoot(ip, Heap::kTheHoleValueRootIndex);
|
| - __ cmp(r2, ip);
|
| - __ b(ne, &skip);
|
| - MemOperand target = ContextOperand(r1, slot->index());
|
| - __ str(r0, target);
|
| - __ mov(r3, r0); // Preserve the stored value in r0.
|
| - __ RecordWriteContextSlot(
|
| - r1, target.offset(), r3, r2, kLRHasBeenSaved, kDontSaveFPRegs);
|
| - break;
|
| - }
|
| +
|
| + case Slot::CONTEXT:
|
| case Slot::LOOKUP:
|
| __ push(r0);
|
| __ mov(r0, Operand(slot->var()->name()));
|
| @@ -2022,7 +1966,7 @@
|
| Handle<Code> ic = is_strict_mode()
|
| ? isolate()->builtins()->StoreIC_Initialize_Strict()
|
| : isolate()->builtins()->StoreIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, expr->id());
|
| + __ Call(ic, RelocInfo::CODE_TARGET, expr->id());
|
|
|
| // If the assignment ends an initialization block, revert to fast case.
|
| if (expr->ends_initialization_block()) {
|
| @@ -2068,7 +2012,7 @@
|
| Handle<Code> ic = is_strict_mode()
|
| ? isolate()->builtins()->KeyedStoreIC_Initialize_Strict()
|
| : isolate()->builtins()->KeyedStoreIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, expr->id());
|
| + __ Call(ic, RelocInfo::CODE_TARGET, expr->id());
|
|
|
| // If the assignment ends an initialization block, revert to fast case.
|
| if (expr->ends_initialization_block()) {
|
| @@ -2120,7 +2064,7 @@
|
| InLoopFlag in_loop = (loop_depth() > 0) ? IN_LOOP : NOT_IN_LOOP;
|
| Handle<Code> ic =
|
| isolate()->stub_cache()->ComputeCallInitialize(arg_count, in_loop, mode);
|
| - EmitCallIC(ic, mode, expr->id());
|
| + __ Call(ic, mode, expr->id());
|
| RecordJSReturnSite(expr);
|
| // Restore context register.
|
| __ ldr(cp, MemOperand(fp, StandardFrameConstants::kContextOffset));
|
| @@ -2154,7 +2098,7 @@
|
| Handle<Code> ic =
|
| isolate()->stub_cache()->ComputeKeyedCallInitialize(arg_count, in_loop);
|
| __ ldr(r2, MemOperand(sp, (arg_count + 1) * kPointerSize)); // Key.
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, expr->id());
|
| + __ Call(ic, RelocInfo::CODE_TARGET, expr->id());
|
| RecordJSReturnSite(expr);
|
| // Restore context register.
|
| __ ldr(cp, MemOperand(fp, StandardFrameConstants::kContextOffset));
|
| @@ -2194,7 +2138,8 @@
|
| __ push(r1);
|
|
|
| // Push the receiver of the enclosing function and do runtime call.
|
| - __ ldr(r1, MemOperand(fp, (2 + scope()->num_parameters()) * kPointerSize));
|
| + int receiver_offset = 2 + info_->scope()->num_parameters();
|
| + __ ldr(r1, MemOperand(fp, receiver_offset * kPointerSize));
|
| __ push(r1);
|
| // Push the strict mode flag.
|
| __ mov(r1, Operand(Smi::FromInt(strict_mode_flag())));
|
| @@ -2313,9 +2258,9 @@
|
| __ bind(&done);
|
| // Push function.
|
| __ push(r0);
|
| - // Push global receiver.
|
| - __ ldr(r1, GlobalObjectOperand());
|
| - __ ldr(r1, FieldMemOperand(r1, GlobalObject::kGlobalReceiverOffset));
|
| + // The receiver is implicitly the global receiver. Indicate this
|
| + // by passing the hole to the call function stub.
|
| + __ LoadRoot(r1, Heap::kTheHoleValueRootIndex);
|
| __ push(r1);
|
| __ bind(&call);
|
| }
|
| @@ -2337,7 +2282,7 @@
|
| } else {
|
| // Call to a keyed property.
|
| // For a synthetic property use keyed load IC followed by function call,
|
| - // for a regular property use keyed EmitCallIC.
|
| + // for a regular property use EmitKeyedCallWithIC.
|
| if (prop->is_synthetic()) {
|
| // Do not visit the object and key subexpressions (they are shared
|
| // by all occurrences of the same rewritten parameter).
|
| @@ -2355,7 +2300,7 @@
|
| SetSourcePosition(prop->position());
|
|
|
| Handle<Code> ic = isolate()->builtins()->KeyedLoadIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, GetPropertyId(prop));
|
| + __ Call(ic, RelocInfo::CODE_TARGET, GetPropertyId(prop));
|
| __ ldr(r1, GlobalObjectOperand());
|
| __ ldr(r1, FieldMemOperand(r1, GlobalObject::kGlobalReceiverOffset));
|
| __ Push(r0, r1); // Function, receiver.
|
| @@ -2481,9 +2426,9 @@
|
| __ tst(r1, Operand(1 << Map::kIsUndetectable));
|
| __ b(ne, if_false);
|
| __ ldrb(r1, FieldMemOperand(r2, Map::kInstanceTypeOffset));
|
| - __ cmp(r1, Operand(FIRST_JS_OBJECT_TYPE));
|
| + __ cmp(r1, Operand(FIRST_NONCALLABLE_SPEC_OBJECT_TYPE));
|
| __ b(lt, if_false);
|
| - __ cmp(r1, Operand(LAST_JS_OBJECT_TYPE));
|
| + __ cmp(r1, Operand(LAST_NONCALLABLE_SPEC_OBJECT_TYPE));
|
| PrepareForBailoutBeforeSplit(TOS_REG, true, if_true, if_false);
|
| Split(le, if_true, if_false, fall_through);
|
|
|
| @@ -2504,7 +2449,7 @@
|
| &if_true, &if_false, &fall_through);
|
|
|
| __ JumpIfSmi(r0, if_false);
|
| - __ CompareObjectType(r0, r1, r1, FIRST_JS_OBJECT_TYPE);
|
| + __ CompareObjectType(r0, r1, r1, FIRST_SPEC_OBJECT_TYPE);
|
| PrepareForBailoutBeforeSplit(TOS_REG, true, if_true, if_false);
|
| Split(ge, if_true, if_false, fall_through);
|
|
|
| @@ -2601,8 +2546,7 @@
|
| // If a valueOf property is not found on the object check that it's
|
| // prototype is the un-modified String prototype. If not result is false.
|
| __ ldr(r2, FieldMemOperand(r1, Map::kPrototypeOffset));
|
| - __ tst(r2, Operand(kSmiTagMask));
|
| - __ b(eq, if_false);
|
| + __ JumpIfSmi(r2, if_false);
|
| __ ldr(r2, FieldMemOperand(r2, HeapObject::kMapOffset));
|
| __ ldr(r3, ContextOperand(cp, Context::GLOBAL_INDEX));
|
| __ ldr(r3, FieldMemOperand(r3, GlobalObject::kGlobalContextOffset));
|
| @@ -2747,7 +2691,7 @@
|
| // parameter count in r0.
|
| VisitForAccumulatorValue(args->at(0));
|
| __ mov(r1, r0);
|
| - __ mov(r0, Operand(Smi::FromInt(scope()->num_parameters())));
|
| + __ mov(r0, Operand(Smi::FromInt(info_->scope()->num_parameters())));
|
| ArgumentsAccessStub stub(ArgumentsAccessStub::READ_ELEMENT);
|
| __ CallStub(&stub);
|
| context()->Plug(r0);
|
| @@ -2759,7 +2703,7 @@
|
|
|
| Label exit;
|
| // Get the number of formal parameters.
|
| - __ mov(r0, Operand(Smi::FromInt(scope()->num_parameters())));
|
| + __ mov(r0, Operand(Smi::FromInt(info_->scope()->num_parameters())));
|
|
|
| // Check if the calling frame is an arguments adaptor frame.
|
| __ ldr(r2, MemOperand(fp, StandardFrameConstants::kCallerFPOffset));
|
| @@ -2787,16 +2731,18 @@
|
|
|
| // Check that the object is a JS object but take special care of JS
|
| // functions to make sure they have 'Function' as their class.
|
| - __ CompareObjectType(r0, r0, r1, FIRST_JS_OBJECT_TYPE); // Map is now in r0.
|
| + __ CompareObjectType(r0, r0, r1, FIRST_SPEC_OBJECT_TYPE);
|
| + // Map is now in r0.
|
| __ b(lt, &null);
|
|
|
| - // As long as JS_FUNCTION_TYPE is the last instance type and it is
|
| - // right after LAST_JS_OBJECT_TYPE, we can avoid checking for
|
| - // LAST_JS_OBJECT_TYPE.
|
| - ASSERT(LAST_TYPE == JS_FUNCTION_TYPE);
|
| - ASSERT(JS_FUNCTION_TYPE == LAST_JS_OBJECT_TYPE + 1);
|
| - __ cmp(r1, Operand(JS_FUNCTION_TYPE));
|
| - __ b(eq, &function);
|
| + // As long as LAST_CALLABLE_SPEC_OBJECT_TYPE is the last instance type, and
|
| + // FIRST_CALLABLE_SPEC_OBJECT_TYPE comes right after
|
| + // LAST_NONCALLABLE_SPEC_OBJECT_TYPE, we can avoid checking for the latter.
|
| + STATIC_ASSERT(LAST_TYPE == LAST_CALLABLE_SPEC_OBJECT_TYPE);
|
| + STATIC_ASSERT(FIRST_CALLABLE_SPEC_OBJECT_TYPE ==
|
| + LAST_NONCALLABLE_SPEC_OBJECT_TYPE + 1);
|
| + __ cmp(r1, Operand(FIRST_CALLABLE_SPEC_OBJECT_TYPE));
|
| + __ b(ge, &function);
|
|
|
| // Check if the constructor in the map is a function.
|
| __ ldr(r0, FieldMemOperand(r0, Map::kConstructorOffset));
|
| @@ -3190,7 +3136,8 @@
|
| // InvokeFunction requires the function in r1. Move it in there.
|
| __ mov(r1, result_register());
|
| ParameterCount count(arg_count);
|
| - __ InvokeFunction(r1, count, CALL_FUNCTION);
|
| + __ InvokeFunction(r1, count, CALL_FUNCTION,
|
| + NullCallWrapper(), CALL_AS_METHOD);
|
| __ ldr(cp, MemOperand(fp, StandardFrameConstants::kContextOffset));
|
| context()->Plug(r0);
|
| }
|
| @@ -3366,8 +3313,7 @@
|
| __ b(eq, &ok);
|
| // Fail if either is a non-HeapObject.
|
| __ and_(tmp, left, Operand(right));
|
| - __ tst(tmp, Operand(kSmiTagMask));
|
| - __ b(eq, &fail);
|
| + __ JumpIfSmi(tmp, &fail);
|
| __ ldr(tmp, FieldMemOperand(left, HeapObject::kMapOffset));
|
| __ ldrb(tmp2, FieldMemOperand(tmp, Map::kInstanceTypeOffset));
|
| __ cmp(tmp2, Operand(JS_REGEXP_TYPE));
|
| @@ -3457,9 +3403,7 @@
|
| __ b(ne, &bailout);
|
|
|
| // Check that the array has fast elements.
|
| - __ ldrb(scratch2, FieldMemOperand(scratch1, Map::kBitField2Offset));
|
| - __ tst(scratch2, Operand(1 << Map::kHasFastElements));
|
| - __ b(eq, &bailout);
|
| + __ CheckFastElements(scratch1, scratch2, &bailout);
|
|
|
| // If the array has length zero, return the empty string.
|
| __ ldr(array_length, FieldMemOperand(array, JSArray::kLengthOffset));
|
| @@ -3657,6 +3601,39 @@
|
| }
|
|
|
|
|
| +void FullCodeGenerator::EmitIsNativeOrStrictMode(ZoneList<Expression*>* args) {
|
| + ASSERT(args->length() == 1);
|
| +
|
| + // Load the function into r0.
|
| + VisitForAccumulatorValue(args->at(0));
|
| +
|
| + // Prepare for the test.
|
| + Label materialize_true, materialize_false;
|
| + Label* if_true = NULL;
|
| + Label* if_false = NULL;
|
| + Label* fall_through = NULL;
|
| + context()->PrepareTest(&materialize_true, &materialize_false,
|
| + &if_true, &if_false, &fall_through);
|
| +
|
| + // Test for strict mode function.
|
| + __ ldr(r1, FieldMemOperand(r0, JSFunction::kSharedFunctionInfoOffset));
|
| + __ ldr(r1, FieldMemOperand(r1, SharedFunctionInfo::kCompilerHintsOffset));
|
| + __ tst(r1, Operand(1 << (SharedFunctionInfo::kStrictModeFunction +
|
| + kSmiTagSize)));
|
| + __ b(ne, if_true);
|
| +
|
| + // Test for native function.
|
| + __ tst(r1, Operand(1 << (SharedFunctionInfo::kNative + kSmiTagSize)));
|
| + __ b(ne, if_true);
|
| +
|
| + // Not native or strict-mode function.
|
| + __ b(if_false);
|
| +
|
| + PrepareForBailoutBeforeSplit(TOS_REG, true, if_true, if_false);
|
| + context()->Plug(if_true, if_false);
|
| +}
|
| +
|
| +
|
| void FullCodeGenerator::VisitCallRuntime(CallRuntime* expr) {
|
| Handle<String> name = expr->name();
|
| if (name->length() > 0 && name->Get(0) == '_') {
|
| @@ -3689,7 +3666,7 @@
|
| isolate()->stub_cache()->ComputeCallInitialize(arg_count,
|
| NOT_IN_LOOP,
|
| mode);
|
| - EmitCallIC(ic, mode, expr->id());
|
| + __ Call(ic, mode, expr->id());
|
| // Restore context register.
|
| __ ldr(cp, MemOperand(fp, StandardFrameConstants::kContextOffset));
|
| } else {
|
| @@ -3797,8 +3774,7 @@
|
| Comment cmt(masm_, "[ UnaryOperation (ADD)");
|
| VisitForAccumulatorValue(expr->expression());
|
| Label no_conversion;
|
| - __ tst(result_register(), Operand(kSmiTagMask));
|
| - __ b(eq, &no_conversion);
|
| + __ JumpIfSmi(result_register(), &no_conversion);
|
| ToNumberStub convert_stub;
|
| __ CallStub(&convert_stub);
|
| __ bind(&no_conversion);
|
| @@ -3832,7 +3808,7 @@
|
| // accumulator register r0.
|
| VisitForAccumulatorValue(expr->expression());
|
| SetSourcePosition(expr->position());
|
| - EmitCallIC(stub.GetCode(), NULL, expr->id());
|
| + __ Call(stub.GetCode(), RelocInfo::CODE_TARGET, expr->id());
|
| context()->Plug(r0);
|
| }
|
|
|
| @@ -3849,7 +3825,7 @@
|
| }
|
|
|
| // Expression can only be a property, a global or a (parameter or local)
|
| - // slot. Variables with rewrite to .arguments are treated as KEYED_PROPERTY.
|
| + // slot.
|
| enum LhsKind { VARIABLE, NAMED_PROPERTY, KEYED_PROPERTY };
|
| LhsKind assign_type = VARIABLE;
|
| Property* prop = expr->expression()->AsProperty();
|
| @@ -3877,15 +3853,8 @@
|
| __ push(r0);
|
| EmitNamedPropertyLoad(prop);
|
| } else {
|
| - if (prop->is_arguments_access()) {
|
| - VariableProxy* obj_proxy = prop->obj()->AsVariableProxy();
|
| - __ ldr(r0, EmitSlotSearch(obj_proxy->var()->AsSlot(), r0));
|
| - __ push(r0);
|
| - __ mov(r0, Operand(prop->key()->AsLiteral()->handle()));
|
| - } else {
|
| - VisitForStackValue(prop->obj());
|
| - VisitForAccumulatorValue(prop->key());
|
| - }
|
| + VisitForStackValue(prop->obj());
|
| + VisitForAccumulatorValue(prop->key());
|
| __ ldr(r1, MemOperand(sp, 0));
|
| __ push(r0);
|
| EmitKeyedPropertyLoad(prop);
|
| @@ -3950,7 +3919,8 @@
|
| SetSourcePosition(expr->position());
|
|
|
| BinaryOpStub stub(Token::ADD, NO_OVERWRITE);
|
| - EmitCallIC(stub.GetCode(), &patch_site, expr->CountId());
|
| + __ Call(stub.GetCode(), RelocInfo::CODE_TARGET, expr->CountId());
|
| + patch_site.EmitPatchInfo();
|
| __ bind(&done);
|
|
|
| // Store the value returned in r0.
|
| @@ -3981,7 +3951,7 @@
|
| Handle<Code> ic = is_strict_mode()
|
| ? isolate()->builtins()->StoreIC_Initialize_Strict()
|
| : isolate()->builtins()->StoreIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, expr->id());
|
| + __ Call(ic, RelocInfo::CODE_TARGET, expr->id());
|
| PrepareForBailoutForId(expr->AssignmentId(), TOS_REG);
|
| if (expr->is_postfix()) {
|
| if (!context()->IsEffect()) {
|
| @@ -3998,7 +3968,7 @@
|
| Handle<Code> ic = is_strict_mode()
|
| ? isolate()->builtins()->KeyedStoreIC_Initialize_Strict()
|
| : isolate()->builtins()->KeyedStoreIC_Initialize();
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, expr->id());
|
| + __ Call(ic, RelocInfo::CODE_TARGET, expr->id());
|
| PrepareForBailoutForId(expr->AssignmentId(), TOS_REG);
|
| if (expr->is_postfix()) {
|
| if (!context()->IsEffect()) {
|
| @@ -4024,7 +3994,7 @@
|
| Handle<Code> ic = isolate()->builtins()->LoadIC_Initialize();
|
| // Use a regular load, not a contextual load, to avoid a reference
|
| // error.
|
| - EmitCallIC(ic, RelocInfo::CODE_TARGET, AstNode::kNoNumber);
|
| + __ Call(ic);
|
| PrepareForBailout(expr, TOS_REG);
|
| context()->Plug(r0);
|
| } else if (proxy != NULL &&
|
| @@ -4047,30 +4017,18 @@
|
| context()->Plug(r0);
|
| } else {
|
| // This expression cannot throw a reference error at the top level.
|
| - context()->HandleExpression(expr);
|
| + VisitInCurrentContext(expr);
|
| }
|
| }
|
|
|
|
|
| -bool FullCodeGenerator::TryLiteralCompare(Token::Value op,
|
| - Expression* left,
|
| - Expression* right,
|
| - Label* if_true,
|
| - Label* if_false,
|
| - Label* fall_through) {
|
| - if (op != Token::EQ && op != Token::EQ_STRICT) return false;
|
| -
|
| - // Check for the pattern: typeof <expression> == <string literal>.
|
| - Literal* right_literal = right->AsLiteral();
|
| - if (right_literal == NULL) return false;
|
| - Handle<Object> right_literal_value = right_literal->handle();
|
| - if (!right_literal_value->IsString()) return false;
|
| - UnaryOperation* left_unary = left->AsUnaryOperation();
|
| - if (left_unary == NULL || left_unary->op() != Token::TYPEOF) return false;
|
| - Handle<String> check = Handle<String>::cast(right_literal_value);
|
| -
|
| +void FullCodeGenerator::EmitLiteralCompareTypeof(Expression* expr,
|
| + Handle<String> check,
|
| + Label* if_true,
|
| + Label* if_false,
|
| + Label* fall_through) {
|
| { AccumulatorValueContext context(this);
|
| - VisitForTypeofValue(left_unary->expression());
|
| + VisitForTypeofValue(expr);
|
| }
|
| PrepareForBailoutBeforeSplit(TOS_REG, true, if_true, if_false);
|
|
|
| @@ -4105,7 +4063,7 @@
|
|
|
| } else if (check->Equals(isolate()->heap()->function_symbol())) {
|
| __ JumpIfSmi(r0, if_false);
|
| - __ CompareObjectType(r0, r1, r0, FIRST_FUNCTION_CLASS_TYPE);
|
| + __ CompareObjectType(r0, r1, r0, FIRST_CALLABLE_SPEC_OBJECT_TYPE);
|
| Split(ge, if_true, if_false, fall_through);
|
|
|
| } else if (check->Equals(isolate()->heap()->object_symbol())) {
|
| @@ -4113,10 +4071,10 @@
|
| __ CompareRoot(r0, Heap::kNullValueRootIndex);
|
| __ b(eq, if_true);
|
| // Check for JS objects => true.
|
| - __ CompareObjectType(r0, r0, r1, FIRST_JS_OBJECT_TYPE);
|
| - __ b(lo, if_false);
|
| - __ CompareInstanceType(r0, r1, FIRST_FUNCTION_CLASS_TYPE);
|
| - __ b(hs, if_false);
|
| + __ CompareObjectType(r0, r0, r1, FIRST_NONCALLABLE_SPEC_OBJECT_TYPE);
|
| + __ b(lt, if_false);
|
| + __ CompareInstanceType(r0, r1, LAST_NONCALLABLE_SPEC_OBJECT_TYPE);
|
| + __ b(gt, if_false);
|
| // Check for undetectable objects => false.
|
| __ ldrb(r1, FieldMemOperand(r0, Map::kBitFieldOffset));
|
| __ tst(r1, Operand(1 << Map::kIsUndetectable));
|
| @@ -4124,8 +4082,18 @@
|
| } else {
|
| if (if_false != fall_through) __ jmp(if_false);
|
| }
|
| +}
|
|
|
| - return true;
|
| +
|
| +void FullCodeGenerator::EmitLiteralCompareUndefined(Expression* expr,
|
| + Label* if_true,
|
| + Label* if_false,
|
| + Label* fall_through) {
|
| + VisitForAccumulatorValue(expr);
|
| + PrepareForBailoutBeforeSplit(TOS_REG, true, if_true, if_false);
|
| +
|
| + __ CompareRoot(r0, Heap::kUndefinedValueRootIndex);
|
| + Split(eq, if_true, if_false, fall_through);
|
| }
|
|
|
|
|
| @@ -4145,14 +4113,12 @@
|
|
|
| // First we try a fast inlined version of the compare when one of
|
| // the operands is a literal.
|
| - Token::Value op = expr->op();
|
| - Expression* left = expr->left();
|
| - Expression* right = expr->right();
|
| - if (TryLiteralCompare(op, left, right, if_true, if_false, fall_through)) {
|
| + if (TryLiteralCompare(expr, if_true, if_false, fall_through)) {
|
| context()->Plug(if_true, if_false);
|
| return;
|
| }
|
|
|
| + Token::Value op = expr->op();
|
| VisitForStackValue(expr->left());
|
| switch (op) {
|
| case Token::IN:
|
| @@ -4227,7 +4193,8 @@
|
| // Record position and call the compare IC.
|
| SetSourcePosition(expr->position());
|
| Handle<Code> ic = CompareIC::GetUninitialized(op);
|
| - EmitCallIC(ic, &patch_site, expr->id());
|
| + __ Call(ic, RelocInfo::CODE_TARGET, expr->id());
|
| + patch_site.EmitPatchInfo();
|
| PrepareForBailoutBeforeSplit(TOS_REG, true, if_true, if_false);
|
| __ cmp(r0, Operand(0));
|
| Split(cond, if_true, if_false, fall_through);
|
| @@ -4260,8 +4227,7 @@
|
| __ LoadRoot(r1, Heap::kUndefinedValueRootIndex);
|
| __ cmp(r0, r1);
|
| __ b(eq, if_true);
|
| - __ tst(r0, Operand(kSmiTagMask));
|
| - __ b(eq, if_false);
|
| + __ JumpIfSmi(r0, if_false);
|
| // It can be an undetectable object.
|
| __ ldr(r1, FieldMemOperand(r0, HeapObject::kMapOffset));
|
| __ ldrb(r1, FieldMemOperand(r1, Map::kBitFieldOffset));
|
| @@ -4289,70 +4255,6 @@
|
| }
|
|
|
|
|
| -void FullCodeGenerator::EmitCallIC(Handle<Code> ic,
|
| - RelocInfo::Mode mode,
|
| - unsigned ast_id) {
|
| - ASSERT(mode == RelocInfo::CODE_TARGET ||
|
| - mode == RelocInfo::CODE_TARGET_CONTEXT);
|
| - Counters* counters = isolate()->counters();
|
| - switch (ic->kind()) {
|
| - case Code::LOAD_IC:
|
| - __ IncrementCounter(counters->named_load_full(), 1, r1, r2);
|
| - break;
|
| - case Code::KEYED_LOAD_IC:
|
| - __ IncrementCounter(counters->keyed_load_full(), 1, r1, r2);
|
| - break;
|
| - case Code::STORE_IC:
|
| - __ IncrementCounter(counters->named_store_full(), 1, r1, r2);
|
| - break;
|
| - case Code::KEYED_STORE_IC:
|
| - __ IncrementCounter(counters->keyed_store_full(), 1, r1, r2);
|
| - default:
|
| - break;
|
| - }
|
| - if (ast_id == kNoASTId || mode == RelocInfo::CODE_TARGET_CONTEXT) {
|
| - __ Call(ic, mode);
|
| - } else {
|
| - ASSERT(mode == RelocInfo::CODE_TARGET);
|
| - mode = RelocInfo::CODE_TARGET_WITH_ID;
|
| - __ CallWithAstId(ic, mode, ast_id);
|
| - }
|
| -}
|
| -
|
| -
|
| -void FullCodeGenerator::EmitCallIC(Handle<Code> ic,
|
| - JumpPatchSite* patch_site,
|
| - unsigned ast_id) {
|
| - Counters* counters = isolate()->counters();
|
| - switch (ic->kind()) {
|
| - case Code::LOAD_IC:
|
| - __ IncrementCounter(counters->named_load_full(), 1, r1, r2);
|
| - break;
|
| - case Code::KEYED_LOAD_IC:
|
| - __ IncrementCounter(counters->keyed_load_full(), 1, r1, r2);
|
| - break;
|
| - case Code::STORE_IC:
|
| - __ IncrementCounter(counters->named_store_full(), 1, r1, r2);
|
| - break;
|
| - case Code::KEYED_STORE_IC:
|
| - __ IncrementCounter(counters->keyed_store_full(), 1, r1, r2);
|
| - default:
|
| - break;
|
| - }
|
| -
|
| - if (ast_id == kNoASTId) {
|
| - __ Call(ic, RelocInfo::CODE_TARGET);
|
| - } else {
|
| - __ CallWithAstId(ic, RelocInfo::CODE_TARGET_WITH_ID, ast_id);
|
| - }
|
| - if (patch_site != NULL && patch_site->is_bound()) {
|
| - patch_site->EmitPatchInfo();
|
| - } else {
|
| - __ nop(); // Signals no inlined code.
|
| - }
|
| -}
|
| -
|
| -
|
| void FullCodeGenerator::StoreToFrameField(int frame_offset, Register value) {
|
| ASSERT_EQ(POINTER_SIZE_ALIGN(frame_offset), frame_offset);
|
| __ str(value, MemOperand(fp, frame_offset));
|
| @@ -4364,6 +4266,27 @@
|
| }
|
|
|
|
|
| +void FullCodeGenerator::PushFunctionArgumentForContextAllocation() {
|
| + Scope* declaration_scope = scope()->DeclarationScope();
|
| + if (declaration_scope->is_global_scope()) {
|
| + // Contexts nested in the global context have a canonical empty function
|
| + // as their closure, not the anonymous closure containing the global
|
| + // code. Pass a smi sentinel and let the runtime look up the empty
|
| + // function.
|
| + __ mov(ip, Operand(Smi::FromInt(0)));
|
| + } else if (declaration_scope->is_eval_scope()) {
|
| + // Contexts created by a call to eval have the same closure as the
|
| + // context calling eval, not the anonymous closure containing the eval
|
| + // code. Fetch it from the context.
|
| + __ ldr(ip, ContextOperand(cp, Context::CLOSURE_INDEX));
|
| + } else {
|
| + ASSERT(declaration_scope->is_function_scope());
|
| + __ ldr(ip, MemOperand(fp, JavaScriptFrameConstants::kFunctionOffset));
|
| + }
|
| + __ push(ip);
|
| +}
|
| +
|
| +
|
| // ----------------------------------------------------------------------------
|
| // Non-local control flow support.
|
|
|
|
|