| OLD | NEW |
| 1 // Copyright 2014 the V8 project authors. All rights reserved. | 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 | 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_PPC | 7 #if V8_TARGET_ARCH_PPC |
| 8 | 8 |
| 9 #include "src/codegen.h" | 9 #include "src/codegen.h" |
| 10 #include "src/debug.h" | 10 #include "src/debug.h" |
| (...skipping 1348 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1359 __ cmp(r5, r3); // Check formal and actual parameter counts. | 1359 __ cmp(r5, r3); // Check formal and actual parameter counts. |
| 1360 __ Jump(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), | 1360 __ Jump(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), |
| 1361 RelocInfo::CODE_TARGET, ne); | 1361 RelocInfo::CODE_TARGET, ne); |
| 1362 | 1362 |
| 1363 __ LoadP(ip, FieldMemOperand(r4, JSFunction::kCodeEntryOffset)); | 1363 __ LoadP(ip, FieldMemOperand(r4, JSFunction::kCodeEntryOffset)); |
| 1364 ParameterCount expected(0); | 1364 ParameterCount expected(0); |
| 1365 __ InvokeCode(ip, expected, expected, JUMP_FUNCTION, NullCallWrapper()); | 1365 __ InvokeCode(ip, expected, expected, JUMP_FUNCTION, NullCallWrapper()); |
| 1366 } | 1366 } |
| 1367 | 1367 |
| 1368 | 1368 |
| 1369 void Builtins::Generate_FunctionApply(MacroAssembler* masm) { | 1369 static void Generate_CheckStackOverflow(MacroAssembler* masm, |
| 1370 const int kIndexOffset = | 1370 const int calleeOffset) { |
| 1371 StandardFrameConstants::kExpressionsOffset - (2 * kPointerSize); | 1371 // Check the stack for overflow. We are not trying to catch |
| 1372 const int kLimitOffset = | 1372 // interruptions (e.g. debug break and preemption) here, so the "real stack |
| 1373 StandardFrameConstants::kExpressionsOffset - (1 * kPointerSize); | 1373 // limit" is checked. |
| 1374 const int kArgsOffset = 2 * kPointerSize; | 1374 Label okay; |
| 1375 const int kRecvOffset = 3 * kPointerSize; | 1375 __ LoadRoot(r5, Heap::kRealStackLimitRootIndex); |
| 1376 const int kFunctionOffset = 4 * kPointerSize; | 1376 // Make r5 the space we have left. The stack might already be overflowed |
| 1377 // here which will cause r5 to become negative. |
| 1378 __ sub(r5, sp, r5); |
| 1379 // Check if the arguments will overflow the stack. |
| 1380 __ SmiToPtrArrayOffset(r0, r3); |
| 1381 __ cmp(r5, r0); |
| 1382 __ bgt(&okay); // Signed comparison. |
| 1383 |
| 1384 // Out of stack space. |
| 1385 __ LoadP(r4, MemOperand(fp, calleeOffset)); |
| 1386 __ Push(r4, r3); |
| 1387 __ InvokeBuiltin(Builtins::STACK_OVERFLOW, CALL_FUNCTION); |
| 1388 |
| 1389 __ bind(&okay); |
| 1390 } |
| 1391 |
| 1392 |
| 1393 static void Generate_PushAppliedArguments(MacroAssembler* masm, |
| 1394 const int argumentsOffset, |
| 1395 const int indexOffset, |
| 1396 const int limitOffset) { |
| 1397 Label entry, loop; |
| 1398 __ LoadP(r3, MemOperand(fp, indexOffset)); |
| 1399 __ b(&entry); |
| 1400 |
| 1401 // Load the current argument from the arguments array and push it to the |
| 1402 // stack. |
| 1403 // r3: current argument index |
| 1404 __ bind(&loop); |
| 1405 __ LoadP(r4, MemOperand(fp, argumentsOffset)); |
| 1406 __ Push(r4, r3); |
| 1407 |
| 1408 // Call the runtime to access the property in the arguments array. |
| 1409 __ CallRuntime(Runtime::kGetProperty, 2); |
| 1410 __ push(r3); |
| 1411 |
| 1412 // Use inline caching to access the arguments. |
| 1413 __ LoadP(r3, MemOperand(fp, indexOffset)); |
| 1414 __ AddSmiLiteral(r3, r3, Smi::FromInt(1), r0); |
| 1415 __ StoreP(r3, MemOperand(fp, indexOffset)); |
| 1416 |
| 1417 // Test if the copy loop has finished copying all the elements from the |
| 1418 // arguments object. |
| 1419 __ bind(&entry); |
| 1420 __ LoadP(r4, MemOperand(fp, limitOffset)); |
| 1421 __ cmp(r3, r4); |
| 1422 __ bne(&loop); |
| 1423 |
| 1424 // On exit, the pushed arguments count is in r0, untagged |
| 1425 __ SmiUntag(r3); |
| 1426 } |
| 1427 |
| 1428 |
| 1429 // Used by FunctionApply and ReflectApply |
| 1430 static void Generate_ApplyHelper(MacroAssembler* masm, bool targetIsArgument) { |
| 1431 const int kFormalParameters = targetIsArgument ? 3 : 2; |
| 1432 const int kStackSize = kFormalParameters + 1; |
| 1377 | 1433 |
| 1378 { | 1434 { |
| 1379 FrameScope frame_scope(masm, StackFrame::INTERNAL); | 1435 FrameScope frame_scope(masm, StackFrame::INTERNAL); |
| 1436 const int kArgumentsOffset = kFPOnStackSize + kPCOnStackSize; |
| 1437 const int kReceiverOffset = kArgumentsOffset + kPointerSize; |
| 1438 const int kFunctionOffset = kReceiverOffset + kPointerSize; |
| 1380 | 1439 |
| 1381 __ LoadP(r3, MemOperand(fp, kFunctionOffset)); // get the function | 1440 __ LoadP(r3, MemOperand(fp, kFunctionOffset)); // get the function |
| 1382 __ push(r3); | 1441 __ push(r3); |
| 1383 __ LoadP(r3, MemOperand(fp, kArgsOffset)); // get the args array | 1442 __ LoadP(r3, MemOperand(fp, kArgumentsOffset)); // get the args array |
| 1384 __ push(r3); | 1443 __ push(r3); |
| 1385 __ InvokeBuiltin(Builtins::APPLY_PREPARE, CALL_FUNCTION); | 1444 if (targetIsArgument) { |
| 1445 __ InvokeBuiltin(Builtins::REFLECT_APPLY_PREPARE, CALL_FUNCTION); |
| 1446 } else { |
| 1447 __ InvokeBuiltin(Builtins::APPLY_PREPARE, CALL_FUNCTION); |
| 1448 } |
| 1386 | 1449 |
| 1387 // Check the stack for overflow. We are not trying to catch | 1450 Generate_CheckStackOverflow(masm, kFunctionOffset); |
| 1388 // interruptions (e.g. debug break and preemption) here, so the "real stack | |
| 1389 // limit" is checked. | |
| 1390 Label okay; | |
| 1391 __ LoadRoot(r5, Heap::kRealStackLimitRootIndex); | |
| 1392 // Make r5 the space we have left. The stack might already be overflowed | |
| 1393 // here which will cause r5 to become negative. | |
| 1394 __ sub(r5, sp, r5); | |
| 1395 // Check if the arguments will overflow the stack. | |
| 1396 __ SmiToPtrArrayOffset(r0, r3); | |
| 1397 __ cmp(r5, r0); | |
| 1398 __ bgt(&okay); // Signed comparison. | |
| 1399 | |
| 1400 // Out of stack space. | |
| 1401 __ LoadP(r4, MemOperand(fp, kFunctionOffset)); | |
| 1402 __ Push(r4, r3); | |
| 1403 __ InvokeBuiltin(Builtins::STACK_OVERFLOW, CALL_FUNCTION); | |
| 1404 // End of stack check. | |
| 1405 | 1451 |
| 1406 // Push current limit and index. | 1452 // Push current limit and index. |
| 1407 __ bind(&okay); | 1453 const int kIndexOffset = |
| 1454 StandardFrameConstants::kExpressionsOffset - (2 * kPointerSize); |
| 1455 const int kLimitOffset = |
| 1456 StandardFrameConstants::kExpressionsOffset - (1 * kPointerSize); |
| 1408 __ li(r4, Operand::Zero()); | 1457 __ li(r4, Operand::Zero()); |
| 1409 __ Push(r3, r4); // limit and initial index. | 1458 __ Push(r3, r4); // limit and initial index. |
| 1410 | 1459 |
| 1411 // Get the receiver. | 1460 // Get the receiver. |
| 1412 __ LoadP(r3, MemOperand(fp, kRecvOffset)); | 1461 __ LoadP(r3, MemOperand(fp, kReceiverOffset)); |
| 1413 | 1462 |
| 1414 // Check that the function is a JS function (otherwise it must be a proxy). | 1463 // Check that the function is a JS function (otherwise it must be a proxy). |
| 1415 Label push_receiver; | 1464 Label push_receiver; |
| 1416 __ LoadP(r4, MemOperand(fp, kFunctionOffset)); | 1465 __ LoadP(r4, MemOperand(fp, kFunctionOffset)); |
| 1417 __ CompareObjectType(r4, r5, r5, JS_FUNCTION_TYPE); | 1466 __ CompareObjectType(r4, r5, r5, JS_FUNCTION_TYPE); |
| 1418 __ bne(&push_receiver); | 1467 __ bne(&push_receiver); |
| 1419 | 1468 |
| 1420 // Change context eagerly to get the right global object if necessary. | 1469 // Change context eagerly to get the right global object if necessary. |
| 1421 __ LoadP(cp, FieldMemOperand(r4, JSFunction::kContextOffset)); | 1470 __ LoadP(cp, FieldMemOperand(r4, JSFunction::kContextOffset)); |
| 1422 // Load the shared function info while the function is still in r4. | 1471 // Load the shared function info while the function is still in r4. |
| (...skipping 47 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1470 __ bind(&use_global_proxy); | 1519 __ bind(&use_global_proxy); |
| 1471 __ LoadP(r3, ContextOperand(cp, Context::GLOBAL_OBJECT_INDEX)); | 1520 __ LoadP(r3, ContextOperand(cp, Context::GLOBAL_OBJECT_INDEX)); |
| 1472 __ LoadP(r3, FieldMemOperand(r3, GlobalObject::kGlobalProxyOffset)); | 1521 __ LoadP(r3, FieldMemOperand(r3, GlobalObject::kGlobalProxyOffset)); |
| 1473 | 1522 |
| 1474 // Push the receiver. | 1523 // Push the receiver. |
| 1475 // r3: receiver | 1524 // r3: receiver |
| 1476 __ bind(&push_receiver); | 1525 __ bind(&push_receiver); |
| 1477 __ push(r3); | 1526 __ push(r3); |
| 1478 | 1527 |
| 1479 // Copy all arguments from the array to the stack. | 1528 // Copy all arguments from the array to the stack. |
| 1480 Label entry, loop; | 1529 Generate_PushAppliedArguments(masm, kArgumentsOffset, kIndexOffset, |
| 1481 __ LoadP(r3, MemOperand(fp, kIndexOffset)); | 1530 kLimitOffset); |
| 1482 __ b(&entry); | |
| 1483 | |
| 1484 // Load the current argument from the arguments array and push it to the | |
| 1485 // stack. | |
| 1486 // r3: current argument index | |
| 1487 __ bind(&loop); | |
| 1488 __ LoadP(r4, MemOperand(fp, kArgsOffset)); | |
| 1489 __ Push(r4, r3); | |
| 1490 | |
| 1491 // Call the runtime to access the property in the arguments array. | |
| 1492 __ CallRuntime(Runtime::kGetProperty, 2); | |
| 1493 __ push(r3); | |
| 1494 | |
| 1495 // Use inline caching to access the arguments. | |
| 1496 __ LoadP(r3, MemOperand(fp, kIndexOffset)); | |
| 1497 __ AddSmiLiteral(r3, r3, Smi::FromInt(1), r0); | |
| 1498 __ StoreP(r3, MemOperand(fp, kIndexOffset)); | |
| 1499 | |
| 1500 // Test if the copy loop has finished copying all the elements from the | |
| 1501 // arguments object. | |
| 1502 __ bind(&entry); | |
| 1503 __ LoadP(r4, MemOperand(fp, kLimitOffset)); | |
| 1504 __ cmp(r3, r4); | |
| 1505 __ bne(&loop); | |
| 1506 | 1531 |
| 1507 // Call the function. | 1532 // Call the function. |
| 1508 Label call_proxy; | 1533 Label call_proxy; |
| 1509 ParameterCount actual(r3); | 1534 ParameterCount actual(r3); |
| 1510 __ SmiUntag(r3); | |
| 1511 __ LoadP(r4, MemOperand(fp, kFunctionOffset)); | 1535 __ LoadP(r4, MemOperand(fp, kFunctionOffset)); |
| 1512 __ CompareObjectType(r4, r5, r5, JS_FUNCTION_TYPE); | 1536 __ CompareObjectType(r4, r5, r5, JS_FUNCTION_TYPE); |
| 1513 __ bne(&call_proxy); | 1537 __ bne(&call_proxy); |
| 1514 __ InvokeFunction(r4, actual, CALL_FUNCTION, NullCallWrapper()); | 1538 __ InvokeFunction(r4, actual, CALL_FUNCTION, NullCallWrapper()); |
| 1515 | 1539 |
| 1516 __ LeaveFrame(StackFrame::INTERNAL, 3 * kPointerSize); | 1540 __ LeaveFrame(StackFrame::INTERNAL, kStackSize * kPointerSize); |
| 1517 __ blr(); | 1541 __ blr(); |
| 1518 | 1542 |
| 1519 // Call the function proxy. | 1543 // Call the function proxy. |
| 1520 __ bind(&call_proxy); | 1544 __ bind(&call_proxy); |
| 1521 __ push(r4); // add function proxy as last argument | 1545 __ push(r4); // add function proxy as last argument |
| 1522 __ addi(r3, r3, Operand(1)); | 1546 __ addi(r3, r3, Operand(1)); |
| 1523 __ li(r5, Operand::Zero()); | 1547 __ li(r5, Operand::Zero()); |
| 1524 __ GetBuiltinFunction(r4, Builtins::CALL_FUNCTION_PROXY); | 1548 __ GetBuiltinFunction(r4, Builtins::CALL_FUNCTION_PROXY); |
| 1525 __ Call(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), | 1549 __ Call(masm->isolate()->builtins()->ArgumentsAdaptorTrampoline(), |
| 1526 RelocInfo::CODE_TARGET); | 1550 RelocInfo::CODE_TARGET); |
| 1527 | 1551 |
| 1528 // Tear down the internal frame and remove function, receiver and args. | 1552 // Tear down the internal frame and remove function, receiver and args. |
| 1529 } | 1553 } |
| 1530 __ addi(sp, sp, Operand(3 * kPointerSize)); | 1554 __ addi(sp, sp, Operand(kStackSize * kPointerSize)); |
| 1531 __ blr(); | 1555 __ blr(); |
| 1532 } | 1556 } |
| 1533 | 1557 |
| 1534 | 1558 |
| 1559 static void Generate_ConstructHelper(MacroAssembler* masm) { |
| 1560 const int kFormalParameters = 3; |
| 1561 const int kStackSize = kFormalParameters + 1; |
| 1562 |
| 1563 { |
| 1564 FrameScope frame_scope(masm, StackFrame::INTERNAL); |
| 1565 const int kNewTargetOffset = kFPOnStackSize + kPCOnStackSize; |
| 1566 const int kArgumentsOffset = kNewTargetOffset + kPointerSize; |
| 1567 const int kFunctionOffset = kArgumentsOffset + kPointerSize; |
| 1568 |
| 1569 // If newTarget is not supplied, set it to constructor |
| 1570 Label validate_arguments; |
| 1571 __ LoadP(r3, MemOperand(fp, kNewTargetOffset)); |
| 1572 __ CompareRoot(r3, Heap::kUndefinedValueRootIndex); |
| 1573 __ bne(&validate_arguments); |
| 1574 __ LoadP(r3, MemOperand(fp, kFunctionOffset)); |
| 1575 __ StoreP(r3, MemOperand(fp, kNewTargetOffset)); |
| 1576 |
| 1577 // Validate arguments |
| 1578 __ bind(&validate_arguments); |
| 1579 __ LoadP(r3, MemOperand(fp, kFunctionOffset)); // get the function |
| 1580 __ push(r3); |
| 1581 __ LoadP(r3, MemOperand(fp, kArgumentsOffset)); // get the args array |
| 1582 __ push(r3); |
| 1583 __ LoadP(r3, MemOperand(fp, kNewTargetOffset)); // get the new.target |
| 1584 __ push(r3); |
| 1585 __ InvokeBuiltin(Builtins::REFLECT_CONSTRUCT_PREPARE, CALL_FUNCTION); |
| 1586 |
| 1587 Generate_CheckStackOverflow(masm, kFunctionOffset); |
| 1588 |
| 1589 // Push current limit and index. |
| 1590 const int kIndexOffset = |
| 1591 StandardFrameConstants::kExpressionsOffset - (2 * kPointerSize); |
| 1592 const int kLimitOffset = |
| 1593 StandardFrameConstants::kExpressionsOffset - (1 * kPointerSize); |
| 1594 __ li(r4, Operand::Zero()); |
| 1595 __ Push(r3, r4); // limit and initial index. |
| 1596 // Push newTarget and callee functions |
| 1597 __ LoadP(r3, MemOperand(fp, kNewTargetOffset)); |
| 1598 __ push(r3); |
| 1599 __ LoadP(r3, MemOperand(fp, kFunctionOffset)); |
| 1600 __ push(r3); |
| 1601 |
| 1602 // Copy all arguments from the array to the stack. |
| 1603 Generate_PushAppliedArguments(masm, kArgumentsOffset, kIndexOffset, |
| 1604 kLimitOffset); |
| 1605 |
| 1606 // Use undefined feedback vector |
| 1607 __ LoadRoot(r5, Heap::kUndefinedValueRootIndex); |
| 1608 __ LoadP(r4, MemOperand(fp, kFunctionOffset)); |
| 1609 |
| 1610 // Call the function. |
| 1611 CallConstructStub stub(masm->isolate(), SUPER_CONSTRUCTOR_CALL); |
| 1612 __ Call(stub.GetCode(), RelocInfo::CONSTRUCT_CALL); |
| 1613 |
| 1614 __ Drop(1); |
| 1615 |
| 1616 // Leave internal frame. |
| 1617 } |
| 1618 __ addi(sp, sp, Operand(kStackSize * kPointerSize)); |
| 1619 __ blr(); |
| 1620 } |
| 1621 |
| 1622 |
| 1623 void Builtins::Generate_FunctionApply(MacroAssembler* masm) { |
| 1624 Generate_ApplyHelper(masm, false); |
| 1625 } |
| 1626 |
| 1627 |
| 1628 void Builtins::Generate_ReflectApply(MacroAssembler* masm) { |
| 1629 Generate_ApplyHelper(masm, true); |
| 1630 } |
| 1631 |
| 1632 |
| 1633 void Builtins::Generate_ReflectConstruct(MacroAssembler* masm) { |
| 1634 Generate_ConstructHelper(masm); |
| 1635 } |
| 1636 |
| 1637 |
| 1535 static void ArgumentAdaptorStackCheck(MacroAssembler* masm, | 1638 static void ArgumentAdaptorStackCheck(MacroAssembler* masm, |
| 1536 Label* stack_overflow) { | 1639 Label* stack_overflow) { |
| 1537 // ----------- S t a t e ------------- | 1640 // ----------- S t a t e ------------- |
| 1538 // -- r3 : actual number of arguments | 1641 // -- r3 : actual number of arguments |
| 1539 // -- r4 : function (passed through to callee) | 1642 // -- r4 : function (passed through to callee) |
| 1540 // -- r5 : expected number of arguments | 1643 // -- r5 : expected number of arguments |
| 1541 // ----------------------------------- | 1644 // ----------------------------------- |
| 1542 // Check the stack for overflow. We are not trying to catch | 1645 // Check the stack for overflow. We are not trying to catch |
| 1543 // interruptions (e.g. debug break and preemption) here, so the "real stack | 1646 // interruptions (e.g. debug break and preemption) here, so the "real stack |
| 1544 // limit" is checked. | 1647 // limit" is checked. |
| (...skipping 155 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1700 __ bkpt(0); | 1803 __ bkpt(0); |
| 1701 } | 1804 } |
| 1702 } | 1805 } |
| 1703 | 1806 |
| 1704 | 1807 |
| 1705 #undef __ | 1808 #undef __ |
| 1706 } | 1809 } |
| 1707 } // namespace v8::internal | 1810 } // namespace v8::internal |
| 1708 | 1811 |
| 1709 #endif // V8_TARGET_ARCH_PPC | 1812 #endif // V8_TARGET_ARCH_PPC |
| OLD | NEW |