| OLD | NEW |
| 1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "src/v8.h" | 5 #include "src/v8.h" |
| 6 | 6 |
| 7 #if V8_TARGET_ARCH_MIPS | 7 #if V8_TARGET_ARCH_MIPS |
| 8 | 8 |
| 9 // Note on Mips implementation: | 9 // Note on Mips implementation: |
| 10 // | 10 // |
| (...skipping 1334 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1345 context()->Plug(v0); | 1345 context()->Plug(v0); |
| 1346 } | 1346 } |
| 1347 | 1347 |
| 1348 | 1348 |
| 1349 void FullCodeGenerator::VisitVariableProxy(VariableProxy* expr) { | 1349 void FullCodeGenerator::VisitVariableProxy(VariableProxy* expr) { |
| 1350 Comment cmnt(masm_, "[ VariableProxy"); | 1350 Comment cmnt(masm_, "[ VariableProxy"); |
| 1351 EmitVariableLoad(expr); | 1351 EmitVariableLoad(expr); |
| 1352 } | 1352 } |
| 1353 | 1353 |
| 1354 | 1354 |
| 1355 void FullCodeGenerator::EmitLoadGlobalCheckExtensions(Variable* var, | 1355 void FullCodeGenerator::EmitLoadGlobalCheckExtensions(VariableProxy* proxy, |
| 1356 TypeofState typeof_state, | 1356 TypeofState typeof_state, |
| 1357 Label* slow) { | 1357 Label* slow) { |
| 1358 Register current = cp; | 1358 Register current = cp; |
| 1359 Register next = a1; | 1359 Register next = a1; |
| 1360 Register temp = a2; | 1360 Register temp = a2; |
| 1361 | 1361 |
| 1362 Scope* s = scope(); | 1362 Scope* s = scope(); |
| 1363 while (s != NULL) { | 1363 while (s != NULL) { |
| 1364 if (s->num_heap_slots() > 0) { | 1364 if (s->num_heap_slots() > 0) { |
| 1365 if (s->calls_sloppy_eval()) { | 1365 if (s->calls_sloppy_eval()) { |
| (...skipping 25 matching lines...) Expand all Loading... |
| 1391 // Check that extension is NULL. | 1391 // Check that extension is NULL. |
| 1392 __ lw(temp, ContextOperand(next, Context::EXTENSION_INDEX)); | 1392 __ lw(temp, ContextOperand(next, Context::EXTENSION_INDEX)); |
| 1393 __ Branch(slow, ne, temp, Operand(zero_reg)); | 1393 __ Branch(slow, ne, temp, Operand(zero_reg)); |
| 1394 // Load next context in chain. | 1394 // Load next context in chain. |
| 1395 __ lw(next, ContextOperand(next, Context::PREVIOUS_INDEX)); | 1395 __ lw(next, ContextOperand(next, Context::PREVIOUS_INDEX)); |
| 1396 __ Branch(&loop); | 1396 __ Branch(&loop); |
| 1397 __ bind(&fast); | 1397 __ bind(&fast); |
| 1398 } | 1398 } |
| 1399 | 1399 |
| 1400 __ lw(LoadIC::ReceiverRegister(), GlobalObjectOperand()); | 1400 __ lw(LoadIC::ReceiverRegister(), GlobalObjectOperand()); |
| 1401 __ li(LoadIC::NameRegister(), Operand(var->name())); | 1401 __ li(LoadIC::NameRegister(), Operand(proxy->var()->name())); |
| 1402 if (FLAG_vector_ics) { |
| 1403 __ li(LoadIC::SlotRegister(), |
| 1404 Operand(Smi::FromInt(proxy->VariableFeedbackSlot()))); |
| 1405 } |
| 1406 |
| 1402 ContextualMode mode = (typeof_state == INSIDE_TYPEOF) | 1407 ContextualMode mode = (typeof_state == INSIDE_TYPEOF) |
| 1403 ? NOT_CONTEXTUAL | 1408 ? NOT_CONTEXTUAL |
| 1404 : CONTEXTUAL; | 1409 : CONTEXTUAL; |
| 1405 CallLoadIC(mode); | 1410 CallLoadIC(mode); |
| 1406 } | 1411 } |
| 1407 | 1412 |
| 1408 | 1413 |
| 1409 MemOperand FullCodeGenerator::ContextSlotOperandCheckExtensions(Variable* var, | 1414 MemOperand FullCodeGenerator::ContextSlotOperandCheckExtensions(Variable* var, |
| 1410 Label* slow) { | 1415 Label* slow) { |
| 1411 ASSERT(var->IsContextSlot()); | 1416 ASSERT(var->IsContextSlot()); |
| (...skipping 17 matching lines...) Expand all Loading... |
| 1429 __ lw(temp, ContextOperand(context, Context::EXTENSION_INDEX)); | 1434 __ lw(temp, ContextOperand(context, Context::EXTENSION_INDEX)); |
| 1430 __ Branch(slow, ne, temp, Operand(zero_reg)); | 1435 __ Branch(slow, ne, temp, Operand(zero_reg)); |
| 1431 | 1436 |
| 1432 // This function is used only for loads, not stores, so it's safe to | 1437 // This function is used only for loads, not stores, so it's safe to |
| 1433 // return an cp-based operand (the write barrier cannot be allowed to | 1438 // return an cp-based operand (the write barrier cannot be allowed to |
| 1434 // destroy the cp register). | 1439 // destroy the cp register). |
| 1435 return ContextOperand(context, var->index()); | 1440 return ContextOperand(context, var->index()); |
| 1436 } | 1441 } |
| 1437 | 1442 |
| 1438 | 1443 |
| 1439 void FullCodeGenerator::EmitDynamicLookupFastCase(Variable* var, | 1444 void FullCodeGenerator::EmitDynamicLookupFastCase(VariableProxy* proxy, |
| 1440 TypeofState typeof_state, | 1445 TypeofState typeof_state, |
| 1441 Label* slow, | 1446 Label* slow, |
| 1442 Label* done) { | 1447 Label* done) { |
| 1443 // Generate fast-case code for variables that might be shadowed by | 1448 // Generate fast-case code for variables that might be shadowed by |
| 1444 // eval-introduced variables. Eval is used a lot without | 1449 // eval-introduced variables. Eval is used a lot without |
| 1445 // introducing variables. In those cases, we do not want to | 1450 // introducing variables. In those cases, we do not want to |
| 1446 // perform a runtime call for all variables in the scope | 1451 // perform a runtime call for all variables in the scope |
| 1447 // containing the eval. | 1452 // containing the eval. |
| 1453 Variable* var = proxy->var(); |
| 1448 if (var->mode() == DYNAMIC_GLOBAL) { | 1454 if (var->mode() == DYNAMIC_GLOBAL) { |
| 1449 EmitLoadGlobalCheckExtensions(var, typeof_state, slow); | 1455 EmitLoadGlobalCheckExtensions(proxy, typeof_state, slow); |
| 1450 __ Branch(done); | 1456 __ Branch(done); |
| 1451 } else if (var->mode() == DYNAMIC_LOCAL) { | 1457 } else if (var->mode() == DYNAMIC_LOCAL) { |
| 1452 Variable* local = var->local_if_not_shadowed(); | 1458 Variable* local = var->local_if_not_shadowed(); |
| 1453 __ lw(v0, ContextSlotOperandCheckExtensions(local, slow)); | 1459 __ lw(v0, ContextSlotOperandCheckExtensions(local, slow)); |
| 1454 if (local->mode() == LET || local->mode() == CONST || | 1460 if (local->mode() == LET || local->mode() == CONST || |
| 1455 local->mode() == CONST_LEGACY) { | 1461 local->mode() == CONST_LEGACY) { |
| 1456 __ LoadRoot(at, Heap::kTheHoleValueRootIndex); | 1462 __ LoadRoot(at, Heap::kTheHoleValueRootIndex); |
| 1457 __ subu(at, v0, at); // Sub as compare: at == 0 on eq. | 1463 __ subu(at, v0, at); // Sub as compare: at == 0 on eq. |
| 1458 if (local->mode() == CONST_LEGACY) { | 1464 if (local->mode() == CONST_LEGACY) { |
| 1459 __ LoadRoot(a0, Heap::kUndefinedValueRootIndex); | 1465 __ LoadRoot(a0, Heap::kUndefinedValueRootIndex); |
| (...skipping 15 matching lines...) Expand all Loading... |
| 1475 SetSourcePosition(proxy->position()); | 1481 SetSourcePosition(proxy->position()); |
| 1476 Variable* var = proxy->var(); | 1482 Variable* var = proxy->var(); |
| 1477 | 1483 |
| 1478 // Three cases: global variables, lookup variables, and all other types of | 1484 // Three cases: global variables, lookup variables, and all other types of |
| 1479 // variables. | 1485 // variables. |
| 1480 switch (var->location()) { | 1486 switch (var->location()) { |
| 1481 case Variable::UNALLOCATED: { | 1487 case Variable::UNALLOCATED: { |
| 1482 Comment cmnt(masm_, "[ Global variable"); | 1488 Comment cmnt(masm_, "[ Global variable"); |
| 1483 __ lw(LoadIC::ReceiverRegister(), GlobalObjectOperand()); | 1489 __ lw(LoadIC::ReceiverRegister(), GlobalObjectOperand()); |
| 1484 __ li(LoadIC::NameRegister(), Operand(var->name())); | 1490 __ li(LoadIC::NameRegister(), Operand(var->name())); |
| 1491 if (FLAG_vector_ics) { |
| 1492 __ li(LoadIC::SlotRegister(), |
| 1493 Operand(Smi::FromInt(proxy->VariableFeedbackSlot()))); |
| 1494 } |
| 1485 CallLoadIC(CONTEXTUAL); | 1495 CallLoadIC(CONTEXTUAL); |
| 1486 context()->Plug(v0); | 1496 context()->Plug(v0); |
| 1487 break; | 1497 break; |
| 1488 } | 1498 } |
| 1489 | 1499 |
| 1490 case Variable::PARAMETER: | 1500 case Variable::PARAMETER: |
| 1491 case Variable::LOCAL: | 1501 case Variable::LOCAL: |
| 1492 case Variable::CONTEXT: { | 1502 case Variable::CONTEXT: { |
| 1493 Comment cmnt(masm_, var->IsContextSlot() ? "[ Context variable" | 1503 Comment cmnt(masm_, var->IsContextSlot() ? "[ Context variable" |
| 1494 : "[ Stack variable"); | 1504 : "[ Stack variable"); |
| (...skipping 58 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1553 } | 1563 } |
| 1554 context()->Plug(var); | 1564 context()->Plug(var); |
| 1555 break; | 1565 break; |
| 1556 } | 1566 } |
| 1557 | 1567 |
| 1558 case Variable::LOOKUP: { | 1568 case Variable::LOOKUP: { |
| 1559 Comment cmnt(masm_, "[ Lookup variable"); | 1569 Comment cmnt(masm_, "[ Lookup variable"); |
| 1560 Label done, slow; | 1570 Label done, slow; |
| 1561 // Generate code for loading from variables potentially shadowed | 1571 // Generate code for loading from variables potentially shadowed |
| 1562 // by eval-introduced variables. | 1572 // by eval-introduced variables. |
| 1563 EmitDynamicLookupFastCase(var, NOT_INSIDE_TYPEOF, &slow, &done); | 1573 EmitDynamicLookupFastCase(proxy, NOT_INSIDE_TYPEOF, &slow, &done); |
| 1564 __ bind(&slow); | 1574 __ bind(&slow); |
| 1565 __ li(a1, Operand(var->name())); | 1575 __ li(a1, Operand(var->name())); |
| 1566 __ Push(cp, a1); // Context and name. | 1576 __ Push(cp, a1); // Context and name. |
| 1567 __ CallRuntime(Runtime::kLoadLookupSlot, 2); | 1577 __ CallRuntime(Runtime::kLoadLookupSlot, 2); |
| 1568 __ bind(&done); | 1578 __ bind(&done); |
| 1569 context()->Plug(v0); | 1579 context()->Plug(v0); |
| 1570 } | 1580 } |
| 1571 } | 1581 } |
| 1572 } | 1582 } |
| 1573 | 1583 |
| (...skipping 500 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2074 __ bind(&l_next); | 2084 __ bind(&l_next); |
| 2075 | 2085 |
| 2076 __ LoadRoot(load_name, Heap::knext_stringRootIndex); // "next" | 2086 __ LoadRoot(load_name, Heap::knext_stringRootIndex); // "next" |
| 2077 __ lw(a3, MemOperand(sp, 1 * kPointerSize)); // iter | 2087 __ lw(a3, MemOperand(sp, 1 * kPointerSize)); // iter |
| 2078 __ Push(load_name, a3, a0); // "next", iter, received | 2088 __ Push(load_name, a3, a0); // "next", iter, received |
| 2079 | 2089 |
| 2080 // result = receiver[f](arg); | 2090 // result = receiver[f](arg); |
| 2081 __ bind(&l_call); | 2091 __ bind(&l_call); |
| 2082 __ lw(load_receiver, MemOperand(sp, kPointerSize)); | 2092 __ lw(load_receiver, MemOperand(sp, kPointerSize)); |
| 2083 __ lw(load_name, MemOperand(sp, 2 * kPointerSize)); | 2093 __ lw(load_name, MemOperand(sp, 2 * kPointerSize)); |
| 2094 if (FLAG_vector_ics) { |
| 2095 __ li(LoadIC::SlotRegister(), |
| 2096 Operand(Smi::FromInt(expr->KeyedLoadFeedbackSlot()))); |
| 2097 } |
| 2084 Handle<Code> ic = isolate()->builtins()->KeyedLoadIC_Initialize(); | 2098 Handle<Code> ic = isolate()->builtins()->KeyedLoadIC_Initialize(); |
| 2085 CallIC(ic, TypeFeedbackId::None()); | 2099 CallIC(ic, TypeFeedbackId::None()); |
| 2086 __ mov(a0, v0); | 2100 __ mov(a0, v0); |
| 2087 __ mov(a1, a0); | 2101 __ mov(a1, a0); |
| 2088 __ sw(a1, MemOperand(sp, 2 * kPointerSize)); | 2102 __ sw(a1, MemOperand(sp, 2 * kPointerSize)); |
| 2089 CallFunctionStub stub(isolate(), 1, CALL_AS_METHOD); | 2103 CallFunctionStub stub(isolate(), 1, CALL_AS_METHOD); |
| 2090 __ CallStub(&stub); | 2104 __ CallStub(&stub); |
| 2091 | 2105 |
| 2092 __ lw(cp, MemOperand(fp, StandardFrameConstants::kContextOffset)); | 2106 __ lw(cp, MemOperand(fp, StandardFrameConstants::kContextOffset)); |
| 2093 __ Drop(1); // The function is still on the stack; drop it. | 2107 __ Drop(1); // The function is still on the stack; drop it. |
| 2094 | 2108 |
| 2095 // if (!result.done) goto l_try; | 2109 // if (!result.done) goto l_try; |
| 2096 __ Move(load_receiver, v0); | 2110 __ Move(load_receiver, v0); |
| 2097 | 2111 |
| 2098 __ push(load_receiver); // save result | 2112 __ push(load_receiver); // save result |
| 2099 __ LoadRoot(load_name, Heap::kdone_stringRootIndex); // "done" | 2113 __ LoadRoot(load_name, Heap::kdone_stringRootIndex); // "done" |
| 2114 if (FLAG_vector_ics) { |
| 2115 __ li(LoadIC::SlotRegister(), |
| 2116 Operand(Smi::FromInt(expr->DoneFeedbackSlot()))); |
| 2117 } |
| 2100 CallLoadIC(NOT_CONTEXTUAL); // v0=result.done | 2118 CallLoadIC(NOT_CONTEXTUAL); // v0=result.done |
| 2101 __ mov(a0, v0); | 2119 __ mov(a0, v0); |
| 2102 Handle<Code> bool_ic = ToBooleanStub::GetUninitialized(isolate()); | 2120 Handle<Code> bool_ic = ToBooleanStub::GetUninitialized(isolate()); |
| 2103 CallIC(bool_ic); | 2121 CallIC(bool_ic); |
| 2104 __ Branch(&l_try, eq, v0, Operand(zero_reg)); | 2122 __ Branch(&l_try, eq, v0, Operand(zero_reg)); |
| 2105 | 2123 |
| 2106 // result.value | 2124 // result.value |
| 2107 __ pop(load_receiver); // result | 2125 __ pop(load_receiver); // result |
| 2108 __ LoadRoot(load_name, Heap::kvalue_stringRootIndex); // "value" | 2126 __ LoadRoot(load_name, Heap::kvalue_stringRootIndex); // "value" |
| 2127 if (FLAG_vector_ics) { |
| 2128 __ li(LoadIC::SlotRegister(), |
| 2129 Operand(Smi::FromInt(expr->ValueFeedbackSlot()))); |
| 2130 } |
| 2109 CallLoadIC(NOT_CONTEXTUAL); // v0=result.value | 2131 CallLoadIC(NOT_CONTEXTUAL); // v0=result.value |
| 2110 context()->DropAndPlug(2, v0); // drop iter and g | 2132 context()->DropAndPlug(2, v0); // drop iter and g |
| 2111 break; | 2133 break; |
| 2112 } | 2134 } |
| 2113 } | 2135 } |
| 2114 } | 2136 } |
| 2115 | 2137 |
| 2116 | 2138 |
| 2117 void FullCodeGenerator::EmitGeneratorResume(Expression *generator, | 2139 void FullCodeGenerator::EmitGeneratorResume(Expression *generator, |
| 2118 Expression *value, | 2140 Expression *value, |
| (...skipping 143 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2262 // root set. | 2284 // root set. |
| 2263 __ RecordWriteField(v0, JSGeneratorObject::kResultValuePropertyOffset, | 2285 __ RecordWriteField(v0, JSGeneratorObject::kResultValuePropertyOffset, |
| 2264 a2, a3, kRAHasBeenSaved, kDontSaveFPRegs); | 2286 a2, a3, kRAHasBeenSaved, kDontSaveFPRegs); |
| 2265 } | 2287 } |
| 2266 | 2288 |
| 2267 | 2289 |
| 2268 void FullCodeGenerator::EmitNamedPropertyLoad(Property* prop) { | 2290 void FullCodeGenerator::EmitNamedPropertyLoad(Property* prop) { |
| 2269 SetSourcePosition(prop->position()); | 2291 SetSourcePosition(prop->position()); |
| 2270 Literal* key = prop->key()->AsLiteral(); | 2292 Literal* key = prop->key()->AsLiteral(); |
| 2271 __ li(LoadIC::NameRegister(), Operand(key->value())); | 2293 __ li(LoadIC::NameRegister(), Operand(key->value())); |
| 2272 // Call load IC. It has register arguments receiver and property. | 2294 if (FLAG_vector_ics) { |
| 2273 CallLoadIC(NOT_CONTEXTUAL, prop->PropertyFeedbackId()); | 2295 __ li(LoadIC::SlotRegister(), |
| 2296 Operand(Smi::FromInt(prop->PropertyFeedbackSlot()))); |
| 2297 CallLoadIC(NOT_CONTEXTUAL); |
| 2298 } else { |
| 2299 CallLoadIC(NOT_CONTEXTUAL, prop->PropertyFeedbackId()); |
| 2300 } |
| 2274 } | 2301 } |
| 2275 | 2302 |
| 2276 | 2303 |
| 2277 void FullCodeGenerator::EmitKeyedPropertyLoad(Property* prop) { | 2304 void FullCodeGenerator::EmitKeyedPropertyLoad(Property* prop) { |
| 2278 SetSourcePosition(prop->position()); | 2305 SetSourcePosition(prop->position()); |
| 2279 // Call keyed load IC. It has register arguments receiver and key. | |
| 2280 Handle<Code> ic = isolate()->builtins()->KeyedLoadIC_Initialize(); | 2306 Handle<Code> ic = isolate()->builtins()->KeyedLoadIC_Initialize(); |
| 2281 CallIC(ic, prop->PropertyFeedbackId()); | 2307 if (FLAG_vector_ics) { |
| 2308 __ li(LoadIC::SlotRegister(), |
| 2309 Operand(Smi::FromInt(prop->PropertyFeedbackSlot()))); |
| 2310 CallIC(ic); |
| 2311 } else { |
| 2312 CallIC(ic, prop->PropertyFeedbackId()); |
| 2313 } |
| 2282 } | 2314 } |
| 2283 | 2315 |
| 2284 | 2316 |
| 2285 void FullCodeGenerator::EmitInlineSmiBinaryOp(BinaryOperation* expr, | 2317 void FullCodeGenerator::EmitInlineSmiBinaryOp(BinaryOperation* expr, |
| 2286 Token::Value op, | 2318 Token::Value op, |
| 2287 OverwriteMode mode, | 2319 OverwriteMode mode, |
| 2288 Expression* left_expr, | 2320 Expression* left_expr, |
| 2289 Expression* right_expr) { | 2321 Expression* right_expr) { |
| 2290 Label done, smi_case, stub_call; | 2322 Label done, smi_case, stub_call; |
| 2291 | 2323 |
| (...skipping 457 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2749 } else if (call_type == Call::GLOBAL_CALL) { | 2781 } else if (call_type == Call::GLOBAL_CALL) { |
| 2750 EmitCallWithLoadIC(expr); | 2782 EmitCallWithLoadIC(expr); |
| 2751 } else if (call_type == Call::LOOKUP_SLOT_CALL) { | 2783 } else if (call_type == Call::LOOKUP_SLOT_CALL) { |
| 2752 // Call to a lookup slot (dynamically introduced variable). | 2784 // Call to a lookup slot (dynamically introduced variable). |
| 2753 VariableProxy* proxy = callee->AsVariableProxy(); | 2785 VariableProxy* proxy = callee->AsVariableProxy(); |
| 2754 Label slow, done; | 2786 Label slow, done; |
| 2755 | 2787 |
| 2756 { PreservePositionScope scope(masm()->positions_recorder()); | 2788 { PreservePositionScope scope(masm()->positions_recorder()); |
| 2757 // Generate code for loading from variables potentially shadowed | 2789 // Generate code for loading from variables potentially shadowed |
| 2758 // by eval-introduced variables. | 2790 // by eval-introduced variables. |
| 2759 EmitDynamicLookupFastCase(proxy->var(), NOT_INSIDE_TYPEOF, &slow, &done); | 2791 EmitDynamicLookupFastCase(proxy, NOT_INSIDE_TYPEOF, &slow, &done); |
| 2760 } | 2792 } |
| 2761 | 2793 |
| 2762 __ bind(&slow); | 2794 __ bind(&slow); |
| 2763 // Call the runtime to find the function to call (returned in v0) | 2795 // Call the runtime to find the function to call (returned in v0) |
| 2764 // and the object holding it (returned in v1). | 2796 // and the object holding it (returned in v1). |
| 2765 ASSERT(!context_register().is(a2)); | 2797 ASSERT(!context_register().is(a2)); |
| 2766 __ li(a2, Operand(proxy->name())); | 2798 __ li(a2, Operand(proxy->name())); |
| 2767 __ Push(context_register(), a2); | 2799 __ Push(context_register(), a2); |
| 2768 __ CallRuntime(Runtime::kLoadLookupSlot, 2); | 2800 __ CallRuntime(Runtime::kLoadLookupSlot, 2); |
| 2769 __ Push(v0, v1); // Function, receiver. | 2801 __ Push(v0, v1); // Function, receiver. |
| (...skipping 1309 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4079 | 4111 |
| 4080 if (expr->is_jsruntime()) { | 4112 if (expr->is_jsruntime()) { |
| 4081 // Push the builtins object as the receiver. | 4113 // Push the builtins object as the receiver. |
| 4082 Register receiver = LoadIC::ReceiverRegister(); | 4114 Register receiver = LoadIC::ReceiverRegister(); |
| 4083 __ lw(receiver, GlobalObjectOperand()); | 4115 __ lw(receiver, GlobalObjectOperand()); |
| 4084 __ lw(receiver, FieldMemOperand(receiver, GlobalObject::kBuiltinsOffset)); | 4116 __ lw(receiver, FieldMemOperand(receiver, GlobalObject::kBuiltinsOffset)); |
| 4085 __ push(receiver); | 4117 __ push(receiver); |
| 4086 | 4118 |
| 4087 // Load the function from the receiver. | 4119 // Load the function from the receiver. |
| 4088 __ li(LoadIC::NameRegister(), Operand(expr->name())); | 4120 __ li(LoadIC::NameRegister(), Operand(expr->name())); |
| 4089 CallLoadIC(NOT_CONTEXTUAL, expr->CallRuntimeFeedbackId()); | 4121 if (FLAG_vector_ics) { |
| 4122 __ li(LoadIC::SlotRegister(), |
| 4123 Operand(Smi::FromInt(expr->CallRuntimeFeedbackSlot()))); |
| 4124 CallLoadIC(NOT_CONTEXTUAL); |
| 4125 } else { |
| 4126 CallLoadIC(NOT_CONTEXTUAL, expr->CallRuntimeFeedbackId()); |
| 4127 } |
| 4090 | 4128 |
| 4091 // Push the target function under the receiver. | 4129 // Push the target function under the receiver. |
| 4092 __ lw(at, MemOperand(sp, 0)); | 4130 __ lw(at, MemOperand(sp, 0)); |
| 4093 __ push(at); | 4131 __ push(at); |
| 4094 __ sw(v0, MemOperand(sp, kPointerSize)); | 4132 __ sw(v0, MemOperand(sp, kPointerSize)); |
| 4095 | 4133 |
| 4096 // Push the arguments ("left-to-right"). | 4134 // Push the arguments ("left-to-right"). |
| 4097 int arg_count = args->length(); | 4135 int arg_count = args->length(); |
| 4098 for (int i = 0; i < arg_count; i++) { | 4136 for (int i = 0; i < arg_count; i++) { |
| 4099 VisitForStackValue(args->at(i)); | 4137 VisitForStackValue(args->at(i)); |
| (...skipping 318 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4418 | 4456 |
| 4419 | 4457 |
| 4420 void FullCodeGenerator::VisitForTypeofValue(Expression* expr) { | 4458 void FullCodeGenerator::VisitForTypeofValue(Expression* expr) { |
| 4421 ASSERT(!context()->IsEffect()); | 4459 ASSERT(!context()->IsEffect()); |
| 4422 ASSERT(!context()->IsTest()); | 4460 ASSERT(!context()->IsTest()); |
| 4423 VariableProxy* proxy = expr->AsVariableProxy(); | 4461 VariableProxy* proxy = expr->AsVariableProxy(); |
| 4424 if (proxy != NULL && proxy->var()->IsUnallocated()) { | 4462 if (proxy != NULL && proxy->var()->IsUnallocated()) { |
| 4425 Comment cmnt(masm_, "[ Global variable"); | 4463 Comment cmnt(masm_, "[ Global variable"); |
| 4426 __ lw(LoadIC::ReceiverRegister(), GlobalObjectOperand()); | 4464 __ lw(LoadIC::ReceiverRegister(), GlobalObjectOperand()); |
| 4427 __ li(LoadIC::NameRegister(), Operand(proxy->name())); | 4465 __ li(LoadIC::NameRegister(), Operand(proxy->name())); |
| 4466 if (FLAG_vector_ics) { |
| 4467 __ li(LoadIC::SlotRegister(), |
| 4468 Operand(Smi::FromInt(proxy->VariableFeedbackSlot()))); |
| 4469 } |
| 4428 // Use a regular load, not a contextual load, to avoid a reference | 4470 // Use a regular load, not a contextual load, to avoid a reference |
| 4429 // error. | 4471 // error. |
| 4430 CallLoadIC(NOT_CONTEXTUAL); | 4472 CallLoadIC(NOT_CONTEXTUAL); |
| 4431 PrepareForBailout(expr, TOS_REG); | 4473 PrepareForBailout(expr, TOS_REG); |
| 4432 context()->Plug(v0); | 4474 context()->Plug(v0); |
| 4433 } else if (proxy != NULL && proxy->var()->IsLookupSlot()) { | 4475 } else if (proxy != NULL && proxy->var()->IsLookupSlot()) { |
| 4434 Comment cmnt(masm_, "[ Lookup slot"); | 4476 Comment cmnt(masm_, "[ Lookup slot"); |
| 4435 Label done, slow; | 4477 Label done, slow; |
| 4436 | 4478 |
| 4437 // Generate code for loading from variables potentially shadowed | 4479 // Generate code for loading from variables potentially shadowed |
| 4438 // by eval-introduced variables. | 4480 // by eval-introduced variables. |
| 4439 EmitDynamicLookupFastCase(proxy->var(), INSIDE_TYPEOF, &slow, &done); | 4481 EmitDynamicLookupFastCase(proxy, INSIDE_TYPEOF, &slow, &done); |
| 4440 | 4482 |
| 4441 __ bind(&slow); | 4483 __ bind(&slow); |
| 4442 __ li(a0, Operand(proxy->name())); | 4484 __ li(a0, Operand(proxy->name())); |
| 4443 __ Push(cp, a0); | 4485 __ Push(cp, a0); |
| 4444 __ CallRuntime(Runtime::kLoadLookupSlotNoReferenceError, 2); | 4486 __ CallRuntime(Runtime::kLoadLookupSlotNoReferenceError, 2); |
| 4445 PrepareForBailout(expr, TOS_REG); | 4487 PrepareForBailout(expr, TOS_REG); |
| 4446 __ bind(&done); | 4488 __ bind(&done); |
| 4447 | 4489 |
| 4448 context()->Plug(v0); | 4490 context()->Plug(v0); |
| 4449 } else { | 4491 } else { |
| (...skipping 398 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 4848 Assembler::target_address_at(pc_immediate_load_address)) == | 4890 Assembler::target_address_at(pc_immediate_load_address)) == |
| 4849 reinterpret_cast<uint32_t>( | 4891 reinterpret_cast<uint32_t>( |
| 4850 isolate->builtins()->OsrAfterStackCheck()->entry())); | 4892 isolate->builtins()->OsrAfterStackCheck()->entry())); |
| 4851 return OSR_AFTER_STACK_CHECK; | 4893 return OSR_AFTER_STACK_CHECK; |
| 4852 } | 4894 } |
| 4853 | 4895 |
| 4854 | 4896 |
| 4855 } } // namespace v8::internal | 4897 } } // namespace v8::internal |
| 4856 | 4898 |
| 4857 #endif // V8_TARGET_ARCH_MIPS | 4899 #endif // V8_TARGET_ARCH_MIPS |
| OLD | NEW |