Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(1030)

Side by Side Diff: src/arm/stub-cache-arm.cc

Issue 83753002: Use the HandlerFrontendHeader in most call compilers. (Closed) Base URL: https://v8.googlecode.com/svn/branches/bleeding_edge
Patch Set: Created 7 years, 1 month ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « no previous file | src/ia32/stub-cache-ia32.cc » ('j') | src/stub-cache.h » ('J')
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright 2012 the V8 project authors. All rights reserved. 1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without 2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are 3 // modification, are permitted provided that the following conditions are
4 // met: 4 // met:
5 // 5 //
6 // * Redistributions of source code must retain the above copyright 6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer. 7 // notice, this list of conditions and the following disclaimer.
8 // * Redistributions in binary form must reproduce the above 8 // * Redistributions in binary form must reproduce the above
9 // copyright notice, this list of conditions and the following 9 // copyright notice, this list of conditions and the following
10 // disclaimer in the documentation and/or other materials provided 10 // disclaimer in the documentation and/or other materials provided
(...skipping 739 matching lines...) Expand 10 before | Expand all | Expand 10 after
750 // ----------- S t a t e ------------- 750 // ----------- S t a t e -------------
751 // -- r0: receiver 751 // -- r0: receiver
752 // -- r1: function to call 752 // -- r1: function to call
753 // ----------------------------------- 753 // -----------------------------------
754 754
755 // Check that the function really is a function. 755 // Check that the function really is a function.
756 __ JumpIfSmi(r1, miss); 756 __ JumpIfSmi(r1, miss);
757 __ CompareObjectType(r1, r3, r3, JS_FUNCTION_TYPE); 757 __ CompareObjectType(r1, r3, r3, JS_FUNCTION_TYPE);
758 __ b(ne, miss); 758 __ b(ne, miss);
759 759
760 // Patch the receiver on the stack with the global proxy if
761 // necessary.
762 if (object->IsGlobalObject()) {
763 __ ldr(r3, FieldMemOperand(r0, GlobalObject::kGlobalReceiverOffset));
764 __ str(r3, MemOperand(sp, arguments.immediate() * kPointerSize));
765 }
766
Igor Sheludko 2013/11/25 16:31:53 Non-equivalent modification for one of the Generat
Toon Verwaest 2013/11/27 15:16:55 Done.
767 // Invoke the function. 760 // Invoke the function.
768 CallKind call_kind = CallICBase::Contextual::decode(extra_ic_state) 761 CallKind call_kind = CallICBase::Contextual::decode(extra_ic_state)
769 ? CALL_AS_FUNCTION 762 ? CALL_AS_FUNCTION
770 : CALL_AS_METHOD; 763 : CALL_AS_METHOD;
771 __ InvokeFunction(r1, arguments, JUMP_FUNCTION, NullCallWrapper(), call_kind); 764 __ InvokeFunction(r1, arguments, JUMP_FUNCTION, NullCallWrapper(), call_kind);
772 } 765 }
773 766
774 767
775 static void PushInterceptorArguments(MacroAssembler* masm, 768 static void PushInterceptorArguments(MacroAssembler* masm,
776 Register receiver, 769 Register receiver,
(...skipping 765 matching lines...) Expand 10 before | Expand all | Expand 10 after
1542 1535
1543 1536
1544 void CallStubCompiler::GenerateNameCheck(Handle<Name> name, Label* miss) { 1537 void CallStubCompiler::GenerateNameCheck(Handle<Name> name, Label* miss) {
1545 if (kind_ == Code::KEYED_CALL_IC) { 1538 if (kind_ == Code::KEYED_CALL_IC) {
1546 __ cmp(r2, Operand(name)); 1539 __ cmp(r2, Operand(name));
1547 __ b(ne, miss); 1540 __ b(ne, miss);
1548 } 1541 }
1549 } 1542 }
1550 1543
1551 1544
1552 void CallStubCompiler::GenerateGlobalReceiverCheck(Handle<JSObject> object,
1553 Handle<JSObject> holder,
1554 Handle<Name> name,
1555 Label* miss) {
1556 ASSERT(holder->IsGlobalObject());
1557
1558 // Get the number of arguments.
1559 const int argc = arguments().immediate();
1560
1561 // Get the receiver from the stack.
1562 __ ldr(r0, MemOperand(sp, argc * kPointerSize));
1563
1564 // Check that the maps haven't changed.
1565 __ JumpIfSmi(r0, miss);
1566 CheckPrototypes(
1567 IC::CurrentTypeOf(object, isolate()), r0, holder, r3, r1, r4, name, miss);
1568 }
1569
1570
1571 void CallStubCompiler::GenerateLoadFunctionFromCell( 1545 void CallStubCompiler::GenerateLoadFunctionFromCell(
1572 Handle<Cell> cell, 1546 Handle<Cell> cell,
1573 Handle<JSFunction> function, 1547 Handle<JSFunction> function,
1574 Label* miss) { 1548 Label* miss) {
1575 // Get the value from the cell. 1549 // Get the value from the cell.
1576 __ mov(r3, Operand(cell)); 1550 __ mov(r3, Operand(cell));
1577 __ ldr(r1, FieldMemOperand(r3, Cell::kValueOffset)); 1551 __ ldr(r1, FieldMemOperand(r3, Cell::kValueOffset));
1578 1552
1579 // Check that the cell contains the same function. 1553 // Check that the cell contains the same function.
1580 if (heap()->InNewSpace(*function)) { 1554 if (heap()->InNewSpace(*function)) {
(...skipping 23 matching lines...) Expand all
1604 kind_, 1578 kind_,
1605 extra_state_); 1579 extra_state_);
1606 __ Jump(code, RelocInfo::CODE_TARGET); 1580 __ Jump(code, RelocInfo::CODE_TARGET);
1607 } 1581 }
1608 1582
1609 1583
1610 Handle<Code> CallStubCompiler::CompileCallField(Handle<JSObject> object, 1584 Handle<Code> CallStubCompiler::CompileCallField(Handle<JSObject> object,
1611 Handle<JSObject> holder, 1585 Handle<JSObject> holder,
1612 PropertyIndex index, 1586 PropertyIndex index,
1613 Handle<Name> name) { 1587 Handle<Name> name) {
1614 // ----------- S t a t e -------------
1615 // -- r2 : name
1616 // -- lr : return address
1617 // -----------------------------------
Igor Sheludko 2013/11/25 16:31:53 Are these (and similar) comments really useless he
Toon Verwaest 2013/11/27 15:16:55 Yes they are. Look at Load/Store, that's what I wa
1618 Label miss; 1588 Label miss;
1619 1589
1620 GenerateNameCheck(name, &miss); 1590 Register reg = HandlerFrontendHeader(
1621 1591 object, holder, name, RECEIVER_MAP_CHECK, &miss);
1622 const int argc = arguments().immediate();
1623
1624 // Get the receiver of the function from the stack into r0.
1625 __ ldr(r0, MemOperand(sp, argc * kPointerSize));
1626 // Check that the receiver isn't a smi.
1627 __ JumpIfSmi(r0, &miss);
1628
1629 // Do the right check and compute the holder register.
1630 Register reg = CheckPrototypes(
1631 IC::CurrentTypeOf(object, isolate()),
1632 r0, holder, r1, r3, r4, name, &miss);
1633 GenerateFastPropertyLoad(masm(), r1, reg, index.is_inobject(holder), 1592 GenerateFastPropertyLoad(masm(), r1, reg, index.is_inobject(holder),
1634 index.translate(holder), Representation::Tagged()); 1593 index.translate(holder), Representation::Tagged());
1635 1594
1636 GenerateCallFunction(masm(), object, arguments(), &miss, extra_state_); 1595 GenerateCallFunction(masm(), object, arguments(), &miss, extra_state_);
1637 1596
1638 // Handle call cache miss. 1597 HandlerFrontendFooter(&miss);
1639 __ bind(&miss);
1640 GenerateMissBranch();
1641 1598
1642 // Return the generated code. 1599 // Return the generated code.
1643 return GetCode(Code::FAST, name); 1600 return GetCode(Code::FAST, name);
1644 } 1601 }
1645 1602
1646 1603
1647 Handle<Code> CallStubCompiler::CompileArrayCodeCall( 1604 Handle<Code> CallStubCompiler::CompileArrayCodeCall(
1648 Handle<Object> object, 1605 Handle<Object> object,
1649 Handle<JSObject> holder, 1606 Handle<JSObject> holder,
1650 Handle<Cell> cell, 1607 Handle<Cell> cell,
1651 Handle<JSFunction> function, 1608 Handle<JSFunction> function,
1652 Handle<String> name, 1609 Handle<String> name,
1653 Code::StubType type) { 1610 Code::StubType type) {
1654 Label miss; 1611 Label miss;
1655 1612
1656 // Check that function is still array 1613 HandlerFrontendHeader(object, holder, name, RECEIVER_MAP_CHECK, &miss);
Igor Sheludko 2013/11/25 16:31:53 Eagle's eye!
1657 const int argc = arguments().immediate(); 1614 if (!cell.is_null()) {
1658 GenerateNameCheck(name, &miss);
1659 Register receiver = r1;
1660
1661 if (cell.is_null()) {
1662 __ ldr(receiver, MemOperand(sp, argc * kPointerSize));
1663
1664 // Check that the receiver isn't a smi.
1665 __ JumpIfSmi(receiver, &miss);
1666
1667 // Check that the maps haven't changed.
1668 CheckPrototypes(
1669 IC::CurrentTypeOf(object, isolate()), receiver, holder,
1670 r3, r0, r4, name, &miss);
1671 } else {
1672 ASSERT(cell->value() == *function); 1615 ASSERT(cell->value() == *function);
1673 GenerateGlobalReceiverCheck(Handle<JSObject>::cast(object), holder, name,
1674 &miss);
1675 GenerateLoadFunctionFromCell(cell, function, &miss); 1616 GenerateLoadFunctionFromCell(cell, function, &miss);
1676 } 1617 }
1677 1618
1678 Handle<AllocationSite> site = isolate()->factory()->NewAllocationSite(); 1619 Handle<AllocationSite> site = isolate()->factory()->NewAllocationSite();
1679 site->SetElementsKind(GetInitialFastElementsKind()); 1620 site->SetElementsKind(GetInitialFastElementsKind());
1680 Handle<Cell> site_feedback_cell = isolate()->factory()->NewCell(site); 1621 Handle<Cell> site_feedback_cell = isolate()->factory()->NewCell(site);
1622 const int argc = arguments().immediate();
1681 __ mov(r0, Operand(argc)); 1623 __ mov(r0, Operand(argc));
1682 __ mov(r2, Operand(site_feedback_cell)); 1624 __ mov(r2, Operand(site_feedback_cell));
1683 __ mov(r1, Operand(function)); 1625 __ mov(r1, Operand(function));
1684 1626
1685 ArrayConstructorStub stub(isolate()); 1627 ArrayConstructorStub stub(isolate());
1686 __ TailCallStub(&stub); 1628 __ TailCallStub(&stub);
1687 1629
1688 __ bind(&miss); 1630 HandlerFrontendFooter(&miss);
1689 GenerateMissBranch();
1690 1631
1691 // Return the generated code. 1632 // Return the generated code.
1692 return GetCode(type, name); 1633 return GetCode(type, name);
1693 } 1634 }
1694 1635
1695 1636
1696 Handle<Code> CallStubCompiler::CompileArrayPushCall( 1637 Handle<Code> CallStubCompiler::CompileArrayPushCall(
1697 Handle<Object> object, 1638 Handle<Object> object,
1698 Handle<JSObject> holder, 1639 Handle<JSObject> holder,
1699 Handle<Cell> cell, 1640 Handle<Cell> cell,
1700 Handle<JSFunction> function, 1641 Handle<JSFunction> function,
1701 Handle<String> name, 1642 Handle<String> name,
1702 Code::StubType type) { 1643 Code::StubType type) {
1703 // ----------- S t a t e -------------
1704 // -- r2 : name
1705 // -- lr : return address
1706 // -- sp[(argc - n - 1) * 4] : arg[n] (zero-based)
1707 // -- ...
1708 // -- sp[argc * 4] : receiver
1709 // -----------------------------------
1710
1711 // If object is not an array or is observed, bail out to regular call. 1644 // If object is not an array or is observed, bail out to regular call.
1712 if (!object->IsJSArray() || 1645 if (!object->IsJSArray() ||
1713 !cell.is_null() || 1646 !cell.is_null() ||
1714 Handle<JSArray>::cast(object)->map()->is_observed()) { 1647 Handle<JSArray>::cast(object)->map()->is_observed()) {
1715 return Handle<Code>::null(); 1648 return Handle<Code>::null();
1716 } 1649 }
1717 1650
1718 Label miss; 1651 Label miss;
1719 GenerateNameCheck(name, &miss);
1720 1652
1653 HandlerFrontendHeader(object, holder, name, RECEIVER_MAP_CHECK, &miss);
1721 Register receiver = r1; 1654 Register receiver = r1;
1722 // Get the receiver from the stack 1655
1723 const int argc = arguments().immediate(); 1656 const int argc = arguments().immediate();
1724 __ ldr(receiver, MemOperand(sp, argc * kPointerSize));
1725
1726 // Check that the receiver isn't a smi.
1727 __ JumpIfSmi(receiver, &miss);
1728
1729 // Check that the maps haven't changed.
1730 CheckPrototypes(
1731 IC::CurrentTypeOf(object, isolate()), receiver, holder,
1732 r3, r0, r4, name, &miss);
1733
1734 if (argc == 0) { 1657 if (argc == 0) {
1735 // Nothing to do, just return the length. 1658 // Nothing to do, just return the length.
1736 __ ldr(r0, FieldMemOperand(receiver, JSArray::kLengthOffset)); 1659 __ ldr(r0, FieldMemOperand(receiver, JSArray::kLengthOffset));
1737 __ Drop(argc + 1); 1660 __ Drop(argc + 1);
1738 __ Ret(); 1661 __ Ret();
1739 } else { 1662 } else {
1740 Label call_builtin; 1663 Label call_builtin;
1741 1664
1742 if (argc == 1) { // Otherwise fall through to call the builtin. 1665 if (argc == 1) { // Otherwise fall through to call the builtin.
1743 Label attempt_to_grow_elements, with_write_barrier, check_double; 1666 Label attempt_to_grow_elements, with_write_barrier, check_double;
(...skipping 192 matching lines...) Expand 10 before | Expand all | Expand 10 after
1936 1859
1937 // Elements are in new space, so write barrier is not required. 1860 // Elements are in new space, so write barrier is not required.
1938 __ Drop(argc + 1); 1861 __ Drop(argc + 1);
1939 __ Ret(); 1862 __ Ret();
1940 } 1863 }
1941 __ bind(&call_builtin); 1864 __ bind(&call_builtin);
1942 __ TailCallExternalReference( 1865 __ TailCallExternalReference(
1943 ExternalReference(Builtins::c_ArrayPush, isolate()), argc + 1, 1); 1866 ExternalReference(Builtins::c_ArrayPush, isolate()), argc + 1, 1);
1944 } 1867 }
1945 1868
1946 // Handle call cache miss. 1869 HandlerFrontendFooter(&miss);
1947 __ bind(&miss);
1948 GenerateMissBranch();
1949 1870
1950 // Return the generated code. 1871 // Return the generated code.
1951 return GetCode(type, name); 1872 return GetCode(type, name);
1952 } 1873 }
1953 1874
1954 1875
1955 Handle<Code> CallStubCompiler::CompileArrayPopCall( 1876 Handle<Code> CallStubCompiler::CompileArrayPopCall(
1956 Handle<Object> object, 1877 Handle<Object> object,
1957 Handle<JSObject> holder, 1878 Handle<JSObject> holder,
1958 Handle<Cell> cell, 1879 Handle<Cell> cell,
1959 Handle<JSFunction> function, 1880 Handle<JSFunction> function,
1960 Handle<String> name, 1881 Handle<String> name,
1961 Code::StubType type) { 1882 Code::StubType type) {
1962 // ----------- S t a t e -------------
1963 // -- r2 : name
1964 // -- lr : return address
1965 // -- sp[(argc - n - 1) * 4] : arg[n] (zero-based)
1966 // -- ...
1967 // -- sp[argc * 4] : receiver
1968 // -----------------------------------
1969
1970 // If object is not an array or is observed, bail out to regular call. 1883 // If object is not an array or is observed, bail out to regular call.
1971 if (!object->IsJSArray() || 1884 if (!object->IsJSArray() ||
1972 !cell.is_null() || 1885 !cell.is_null() ||
1973 Handle<JSArray>::cast(object)->map()->is_observed()) { 1886 Handle<JSArray>::cast(object)->map()->is_observed()) {
1974 return Handle<Code>::null(); 1887 return Handle<Code>::null();
1975 } 1888 }
1976 1889
1977 Label miss, return_undefined, call_builtin; 1890 Label miss, return_undefined, call_builtin;
1978 Register receiver = r1; 1891 Register receiver = r1;
1979 Register elements = r3; 1892 Register elements = r3;
1980 GenerateNameCheck(name, &miss);
1981 1893
1982 // Get the receiver from the stack 1894 HandlerFrontendHeader(object, holder, name, RECEIVER_MAP_CHECK, &miss);
1983 const int argc = arguments().immediate();
1984 __ ldr(receiver, MemOperand(sp, argc * kPointerSize));
1985 // Check that the receiver isn't a smi.
1986 __ JumpIfSmi(receiver, &miss);
1987
1988 // Check that the maps haven't changed.
1989 CheckPrototypes(IC::CurrentTypeOf(object, isolate()), receiver, holder,
1990 elements, r4, r0, name, &miss);
1991 1895
1992 // Get the elements array of the object. 1896 // Get the elements array of the object.
1993 __ ldr(elements, FieldMemOperand(receiver, JSArray::kElementsOffset)); 1897 __ ldr(elements, FieldMemOperand(receiver, JSArray::kElementsOffset));
1994 1898
1995 // Check that the elements are in fast mode and writable. 1899 // Check that the elements are in fast mode and writable.
1996 __ CheckMap(elements, 1900 __ CheckMap(elements,
1997 r0, 1901 r0,
1998 Heap::kFixedArrayMapRootIndex, 1902 Heap::kFixedArrayMapRootIndex,
1999 &call_builtin, 1903 &call_builtin,
2000 DONT_DO_SMI_CHECK); 1904 DONT_DO_SMI_CHECK);
(...skipping 10 matching lines...) Expand all
2011 __ add(elements, elements, Operand::PointerOffsetFromSmiKey(r4)); 1915 __ add(elements, elements, Operand::PointerOffsetFromSmiKey(r4));
2012 __ ldr(r0, FieldMemOperand(elements, FixedArray::kHeaderSize)); 1916 __ ldr(r0, FieldMemOperand(elements, FixedArray::kHeaderSize));
2013 __ cmp(r0, r6); 1917 __ cmp(r0, r6);
2014 __ b(eq, &call_builtin); 1918 __ b(eq, &call_builtin);
2015 1919
2016 // Set the array's length. 1920 // Set the array's length.
2017 __ str(r4, FieldMemOperand(receiver, JSArray::kLengthOffset)); 1921 __ str(r4, FieldMemOperand(receiver, JSArray::kLengthOffset));
2018 1922
2019 // Fill with the hole. 1923 // Fill with the hole.
2020 __ str(r6, FieldMemOperand(elements, FixedArray::kHeaderSize)); 1924 __ str(r6, FieldMemOperand(elements, FixedArray::kHeaderSize));
1925 const int argc = arguments().immediate();
2021 __ Drop(argc + 1); 1926 __ Drop(argc + 1);
2022 __ Ret(); 1927 __ Ret();
2023 1928
2024 __ bind(&return_undefined); 1929 __ bind(&return_undefined);
2025 __ LoadRoot(r0, Heap::kUndefinedValueRootIndex); 1930 __ LoadRoot(r0, Heap::kUndefinedValueRootIndex);
2026 __ Drop(argc + 1); 1931 __ Drop(argc + 1);
2027 __ Ret(); 1932 __ Ret();
2028 1933
2029 __ bind(&call_builtin); 1934 __ bind(&call_builtin);
2030 __ TailCallExternalReference( 1935 __ TailCallExternalReference(
2031 ExternalReference(Builtins::c_ArrayPop, isolate()), argc + 1, 1); 1936 ExternalReference(Builtins::c_ArrayPop, isolate()), argc + 1, 1);
2032 1937
2033 // Handle call cache miss. 1938 HandlerFrontendFooter(&miss);
2034 __ bind(&miss);
2035 GenerateMissBranch();
2036 1939
2037 // Return the generated code. 1940 // Return the generated code.
2038 return GetCode(type, name); 1941 return GetCode(type, name);
2039 } 1942 }
2040 1943
2041 1944
2042 Handle<Code> CallStubCompiler::CompileStringCharCodeAtCall( 1945 Handle<Code> CallStubCompiler::CompileStringCharCodeAtCall(
2043 Handle<Object> object, 1946 Handle<Object> object,
2044 Handle<JSObject> holder, 1947 Handle<JSObject> holder,
2045 Handle<Cell> cell, 1948 Handle<Cell> cell,
2046 Handle<JSFunction> function, 1949 Handle<JSFunction> function,
2047 Handle<String> name, 1950 Handle<String> name,
2048 Code::StubType type) { 1951 Code::StubType type) {
2049 // ----------- S t a t e -------------
2050 // -- r2 : function name
2051 // -- lr : return address
2052 // -- sp[(argc - n - 1) * 4] : arg[n] (zero-based)
2053 // -- ...
2054 // -- sp[argc * 4] : receiver
2055 // -----------------------------------
2056
2057 // If object is not a string, bail out to regular call. 1952 // If object is not a string, bail out to regular call.
2058 if (!object->IsString() || !cell.is_null()) return Handle<Code>::null(); 1953 if (!object->IsString() || !cell.is_null()) return Handle<Code>::null();
2059 1954
2060 const int argc = arguments().immediate();
2061 Label miss; 1955 Label miss;
2062 Label name_miss; 1956 Label name_miss;
2063 Label index_out_of_range; 1957 Label index_out_of_range;
2064 Label* index_out_of_range_label = &index_out_of_range; 1958 Label* index_out_of_range_label = &index_out_of_range;
2065 1959
2066 if (kind_ == Code::CALL_IC && 1960 if (kind_ == Code::CALL_IC &&
2067 (CallICBase::StringStubState::decode(extra_state_) == 1961 (CallICBase::StringStubState::decode(extra_state_) ==
2068 DEFAULT_STRING_STUB)) { 1962 DEFAULT_STRING_STUB)) {
2069 index_out_of_range_label = &miss; 1963 index_out_of_range_label = &miss;
2070 } 1964 }
2071 GenerateNameCheck(name, &name_miss);
2072 1965
2073 // Check that the maps starting from the prototype haven't changed. 1966 HandlerFrontendHeader(object, holder, name, STRING_CHECK, &name_miss);
2074 GenerateDirectLoadGlobalFunctionPrototype(masm(),
2075 Context::STRING_FUNCTION_INDEX,
2076 r0,
2077 &miss);
2078 ASSERT(!object.is_identical_to(holder));
2079 Handle<JSObject> prototype(JSObject::cast(object->GetPrototype(isolate())));
2080 CheckPrototypes(
2081 IC::CurrentTypeOf(prototype, isolate()),
2082 r0, holder, r1, r3, r4, name, &miss);
2083 1967
2084 Register receiver = r1; 1968 Register receiver = r1;
2085 Register index = r4; 1969 Register index = r4;
2086 Register result = r0; 1970 Register result = r0;
1971 const int argc = arguments().immediate();
2087 __ ldr(receiver, MemOperand(sp, argc * kPointerSize)); 1972 __ ldr(receiver, MemOperand(sp, argc * kPointerSize));
2088 if (argc > 0) { 1973 if (argc > 0) {
2089 __ ldr(index, MemOperand(sp, (argc - 1) * kPointerSize)); 1974 __ ldr(index, MemOperand(sp, (argc - 1) * kPointerSize));
2090 } else { 1975 } else {
2091 __ LoadRoot(index, Heap::kUndefinedValueRootIndex); 1976 __ LoadRoot(index, Heap::kUndefinedValueRootIndex);
2092 } 1977 }
2093 1978
2094 StringCharCodeAtGenerator generator(receiver, 1979 StringCharCodeAtGenerator generator(receiver,
2095 index, 1980 index,
2096 result, 1981 result,
(...skipping 26 matching lines...) Expand all
2123 } 2008 }
2124 2009
2125 2010
2126 Handle<Code> CallStubCompiler::CompileStringCharAtCall( 2011 Handle<Code> CallStubCompiler::CompileStringCharAtCall(
2127 Handle<Object> object, 2012 Handle<Object> object,
2128 Handle<JSObject> holder, 2013 Handle<JSObject> holder,
2129 Handle<Cell> cell, 2014 Handle<Cell> cell,
2130 Handle<JSFunction> function, 2015 Handle<JSFunction> function,
2131 Handle<String> name, 2016 Handle<String> name,
2132 Code::StubType type) { 2017 Code::StubType type) {
2133 // ----------- S t a t e -------------
2134 // -- r2 : function name
2135 // -- lr : return address
2136 // -- sp[(argc - n - 1) * 4] : arg[n] (zero-based)
2137 // -- ...
2138 // -- sp[argc * 4] : receiver
2139 // -----------------------------------
2140
2141 // If object is not a string, bail out to regular call. 2018 // If object is not a string, bail out to regular call.
2142 if (!object->IsString() || !cell.is_null()) return Handle<Code>::null(); 2019 if (!object->IsString() || !cell.is_null()) return Handle<Code>::null();
2143 2020
2144 const int argc = arguments().immediate(); 2021 const int argc = arguments().immediate();
2145 Label miss; 2022 Label miss;
2146 Label name_miss; 2023 Label name_miss;
2147 Label index_out_of_range; 2024 Label index_out_of_range;
2148 Label* index_out_of_range_label = &index_out_of_range; 2025 Label* index_out_of_range_label = &index_out_of_range;
2149 if (kind_ == Code::CALL_IC && 2026 if (kind_ == Code::CALL_IC &&
2150 (CallICBase::StringStubState::decode(extra_state_) == 2027 (CallICBase::StringStubState::decode(extra_state_) ==
2151 DEFAULT_STRING_STUB)) { 2028 DEFAULT_STRING_STUB)) {
2152 index_out_of_range_label = &miss; 2029 index_out_of_range_label = &miss;
2153 } 2030 }
2154 GenerateNameCheck(name, &name_miss);
2155 2031
2156 // Check that the maps starting from the prototype haven't changed. 2032 HandlerFrontendHeader(object, holder, name, STRING_CHECK, &name_miss);
2157 GenerateDirectLoadGlobalFunctionPrototype(masm(),
2158 Context::STRING_FUNCTION_INDEX,
2159 r0,
2160 &miss);
2161 ASSERT(!object.is_identical_to(holder));
2162 Handle<JSObject> prototype(JSObject::cast(object->GetPrototype(isolate())));
2163 CheckPrototypes(
2164 IC::CurrentTypeOf(prototype, isolate()),
2165 r0, holder, r1, r3, r4, name, &miss);
2166 2033
2167 Register receiver = r0; 2034 Register receiver = r0;
2168 Register index = r4; 2035 Register index = r4;
2169 Register scratch = r3; 2036 Register scratch = r3;
2170 Register result = r0; 2037 Register result = r0;
2171 __ ldr(receiver, MemOperand(sp, argc * kPointerSize)); 2038 __ ldr(receiver, MemOperand(sp, argc * kPointerSize));
2172 if (argc > 0) { 2039 if (argc > 0) {
2173 __ ldr(index, MemOperand(sp, (argc - 1) * kPointerSize)); 2040 __ ldr(index, MemOperand(sp, (argc - 1) * kPointerSize));
2174 } else { 2041 } else {
2175 __ LoadRoot(index, Heap::kUndefinedValueRootIndex); 2042 __ LoadRoot(index, Heap::kUndefinedValueRootIndex);
(...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after
2208 } 2075 }
2209 2076
2210 2077
2211 Handle<Code> CallStubCompiler::CompileStringFromCharCodeCall( 2078 Handle<Code> CallStubCompiler::CompileStringFromCharCodeCall(
2212 Handle<Object> object, 2079 Handle<Object> object,
2213 Handle<JSObject> holder, 2080 Handle<JSObject> holder,
2214 Handle<Cell> cell, 2081 Handle<Cell> cell,
2215 Handle<JSFunction> function, 2082 Handle<JSFunction> function,
2216 Handle<String> name, 2083 Handle<String> name,
2217 Code::StubType type) { 2084 Code::StubType type) {
2218 // ----------- S t a t e -------------
2219 // -- r2 : function name
2220 // -- lr : return address
2221 // -- sp[(argc - n - 1) * 4] : arg[n] (zero-based)
2222 // -- ...
2223 // -- sp[argc * 4] : receiver
2224 // -----------------------------------
2225
2226 const int argc = arguments().immediate(); 2085 const int argc = arguments().immediate();
2227 2086
2228 // If the object is not a JSObject or we got an unexpected number of 2087 // If the object is not a JSObject or we got an unexpected number of
2229 // arguments, bail out to the regular call. 2088 // arguments, bail out to the regular call.
2230 if (!object->IsJSObject() || argc != 1) return Handle<Code>::null(); 2089 if (!object->IsJSObject() || argc != 1) return Handle<Code>::null();
2231 2090
2232 Label miss; 2091 Label miss;
2233 GenerateNameCheck(name, &miss);
2234 2092
2235 if (cell.is_null()) { 2093 HandlerFrontendHeader(object, holder, name, RECEIVER_MAP_CHECK, &miss);
2236 __ ldr(r1, MemOperand(sp, 1 * kPointerSize)); 2094 if (!cell.is_null()) {
2237
2238 __ JumpIfSmi(r1, &miss);
2239
2240 CheckPrototypes(
2241 IC::CurrentTypeOf(object, isolate()),
2242 r1, holder, r0, r3, r4, name, &miss);
2243 } else {
2244 ASSERT(cell->value() == *function); 2095 ASSERT(cell->value() == *function);
2245 GenerateGlobalReceiverCheck(Handle<JSObject>::cast(object), holder, name,
2246 &miss);
2247 GenerateLoadFunctionFromCell(cell, function, &miss); 2096 GenerateLoadFunctionFromCell(cell, function, &miss);
2248 } 2097 }
2249 2098
2250 // Load the char code argument. 2099 // Load the char code argument.
2251 Register code = r1; 2100 Register code = r1;
2252 __ ldr(code, MemOperand(sp, 0 * kPointerSize)); 2101 __ ldr(code, MemOperand(sp, 0 * kPointerSize));
2253 2102
2254 // Check the code is a smi. 2103 // Check the code is a smi.
2255 Label slow; 2104 Label slow;
2256 __ JumpIfNotSmi(code, &slow); 2105 __ JumpIfNotSmi(code, &slow);
2257 2106
2258 // Convert the smi code to uint16. 2107 // Convert the smi code to uint16.
2259 __ and_(code, code, Operand(Smi::FromInt(0xffff))); 2108 __ and_(code, code, Operand(Smi::FromInt(0xffff)));
2260 2109
2261 StringCharFromCodeGenerator generator(code, r0); 2110 StringCharFromCodeGenerator generator(code, r0);
2262 generator.GenerateFast(masm()); 2111 generator.GenerateFast(masm());
2263 __ Drop(argc + 1); 2112 __ Drop(argc + 1);
2264 __ Ret(); 2113 __ Ret();
2265 2114
2266 StubRuntimeCallHelper call_helper; 2115 StubRuntimeCallHelper call_helper;
2267 generator.GenerateSlow(masm(), call_helper); 2116 generator.GenerateSlow(masm(), call_helper);
2268 2117
2269 // Tail call the full function. We do not have to patch the receiver 2118 // Tail call the full function. We do not have to patch the receiver
2270 // because the function makes no use of it. 2119 // because the function makes no use of it.
2271 __ bind(&slow); 2120 __ bind(&slow);
2272 ParameterCount expected(function); 2121 ParameterCount expected(function);
2273 __ InvokeFunction(function, expected, arguments(), 2122 __ InvokeFunction(function, expected, arguments(),
2274 JUMP_FUNCTION, NullCallWrapper(), CALL_AS_METHOD); 2123 JUMP_FUNCTION, NullCallWrapper(), CALL_AS_METHOD);
2275 2124
2276 __ bind(&miss); 2125 HandlerFrontendFooter(&miss);
2277 // r2: function name.
2278 GenerateMissBranch();
2279 2126
2280 // Return the generated code. 2127 // Return the generated code.
2281 return GetCode(type, name); 2128 return GetCode(type, name);
2282 } 2129 }
2283 2130
2284 2131
2285 Handle<Code> CallStubCompiler::CompileMathFloorCall( 2132 Handle<Code> CallStubCompiler::CompileMathFloorCall(
2286 Handle<Object> object, 2133 Handle<Object> object,
2287 Handle<JSObject> holder, 2134 Handle<JSObject> holder,
2288 Handle<Cell> cell, 2135 Handle<Cell> cell,
2289 Handle<JSFunction> function, 2136 Handle<JSFunction> function,
2290 Handle<String> name, 2137 Handle<String> name,
2291 Code::StubType type) { 2138 Code::StubType type) {
2292 // ----------- S t a t e -------------
2293 // -- r2 : function name
2294 // -- lr : return address
2295 // -- sp[(argc - n - 1) * 4] : arg[n] (zero-based)
2296 // -- ...
2297 // -- sp[argc * 4] : receiver
2298 // -----------------------------------
2299
2300 const int argc = arguments().immediate(); 2139 const int argc = arguments().immediate();
2301 // If the object is not a JSObject or we got an unexpected number of 2140 // If the object is not a JSObject or we got an unexpected number of
2302 // arguments, bail out to the regular call. 2141 // arguments, bail out to the regular call.
2303 if (!object->IsJSObject() || argc != 1) return Handle<Code>::null(); 2142 if (!object->IsJSObject() || argc != 1) return Handle<Code>::null();
2304 2143
2305 Label miss, slow; 2144 Label miss, slow;
2306 GenerateNameCheck(name, &miss);
2307 2145
2308 if (cell.is_null()) { 2146 HandlerFrontendHeader(object, holder, name, RECEIVER_MAP_CHECK, &miss);
2309 __ ldr(r1, MemOperand(sp, 1 * kPointerSize)); 2147 if (!cell.is_null()) {
2310 __ JumpIfSmi(r1, &miss);
2311 CheckPrototypes(
2312 IC::CurrentTypeOf(object, isolate()),
2313 r1, holder, r0, r3, r4, name, &miss);
2314 } else {
2315 ASSERT(cell->value() == *function); 2148 ASSERT(cell->value() == *function);
2316 GenerateGlobalReceiverCheck(Handle<JSObject>::cast(object), holder, name,
2317 &miss);
2318 GenerateLoadFunctionFromCell(cell, function, &miss); 2149 GenerateLoadFunctionFromCell(cell, function, &miss);
2319 } 2150 }
2320 2151
2321 // Load the (only) argument into r0. 2152 // Load the (only) argument into r0.
2322 __ ldr(r0, MemOperand(sp, 0 * kPointerSize)); 2153 __ ldr(r0, MemOperand(sp, 0 * kPointerSize));
2323 2154
2324 // If the argument is a smi, just return. 2155 // If the argument is a smi, just return.
2325 __ SmiTst(r0); 2156 __ SmiTst(r0);
2326 __ Drop(argc + 1, eq); 2157 __ Drop(argc + 1, eq);
2327 __ Ret(eq); 2158 __ Ret(eq);
(...skipping 48 matching lines...) Expand 10 before | Expand all | Expand 10 after
2376 __ Drop(argc + 1); 2207 __ Drop(argc + 1);
2377 __ Ret(); 2208 __ Ret();
2378 2209
2379 __ bind(&slow); 2210 __ bind(&slow);
2380 // Tail call the full function. We do not have to patch the receiver 2211 // Tail call the full function. We do not have to patch the receiver
2381 // because the function makes no use of it. 2212 // because the function makes no use of it.
2382 ParameterCount expected(function); 2213 ParameterCount expected(function);
2383 __ InvokeFunction(function, expected, arguments(), 2214 __ InvokeFunction(function, expected, arguments(),
2384 JUMP_FUNCTION, NullCallWrapper(), CALL_AS_METHOD); 2215 JUMP_FUNCTION, NullCallWrapper(), CALL_AS_METHOD);
2385 2216
2386 __ bind(&miss); 2217 HandlerFrontendFooter(&miss);
2387 // r2: function name.
2388 GenerateMissBranch();
2389 2218
2390 // Return the generated code. 2219 // Return the generated code.
2391 return GetCode(type, name); 2220 return GetCode(type, name);
2392 } 2221 }
2393 2222
2394 2223
2395 Handle<Code> CallStubCompiler::CompileMathAbsCall( 2224 Handle<Code> CallStubCompiler::CompileMathAbsCall(
2396 Handle<Object> object, 2225 Handle<Object> object,
2397 Handle<JSObject> holder, 2226 Handle<JSObject> holder,
2398 Handle<Cell> cell, 2227 Handle<Cell> cell,
2399 Handle<JSFunction> function, 2228 Handle<JSFunction> function,
2400 Handle<String> name, 2229 Handle<String> name,
2401 Code::StubType type) { 2230 Code::StubType type) {
2402 // ----------- S t a t e -------------
2403 // -- r2 : function name
2404 // -- lr : return address
2405 // -- sp[(argc - n - 1) * 4] : arg[n] (zero-based)
2406 // -- ...
2407 // -- sp[argc * 4] : receiver
2408 // -----------------------------------
2409
2410 const int argc = arguments().immediate(); 2231 const int argc = arguments().immediate();
2411 // If the object is not a JSObject or we got an unexpected number of 2232 // If the object is not a JSObject or we got an unexpected number of
2412 // arguments, bail out to the regular call. 2233 // arguments, bail out to the regular call.
2413 if (!object->IsJSObject() || argc != 1) return Handle<Code>::null(); 2234 if (!object->IsJSObject() || argc != 1) return Handle<Code>::null();
2414 2235
2415 Label miss; 2236 Label miss;
2416 GenerateNameCheck(name, &miss); 2237
2417 if (cell.is_null()) { 2238 HandlerFrontendHeader(object, holder, name, RECEIVER_MAP_CHECK, &miss);
2418 __ ldr(r1, MemOperand(sp, 1 * kPointerSize)); 2239 if (!cell.is_null()) {
2419 __ JumpIfSmi(r1, &miss);
2420 CheckPrototypes(
2421 IC::CurrentTypeOf(object, isolate()),
2422 r1, holder, r0, r3, r4, name, &miss);
2423 } else {
2424 ASSERT(cell->value() == *function); 2240 ASSERT(cell->value() == *function);
2425 GenerateGlobalReceiverCheck(Handle<JSObject>::cast(object), holder, name,
2426 &miss);
2427 GenerateLoadFunctionFromCell(cell, function, &miss); 2241 GenerateLoadFunctionFromCell(cell, function, &miss);
2428 } 2242 }
2429 2243
2430 // Load the (only) argument into r0. 2244 // Load the (only) argument into r0.
2431 __ ldr(r0, MemOperand(sp, 0 * kPointerSize)); 2245 __ ldr(r0, MemOperand(sp, 0 * kPointerSize));
2432 2246
2433 // Check if the argument is a smi. 2247 // Check if the argument is a smi.
2434 Label not_smi; 2248 Label not_smi;
2435 __ JumpIfNotSmi(r0, &not_smi); 2249 __ JumpIfNotSmi(r0, &not_smi);
2436 2250
(...skipping 39 matching lines...) Expand 10 before | Expand all | Expand 10 after
2476 __ Drop(argc + 1); 2290 __ Drop(argc + 1);
2477 __ Ret(); 2291 __ Ret();
2478 2292
2479 // Tail call the full function. We do not have to patch the receiver 2293 // Tail call the full function. We do not have to patch the receiver
2480 // because the function makes no use of it. 2294 // because the function makes no use of it.
2481 __ bind(&slow); 2295 __ bind(&slow);
2482 ParameterCount expected(function); 2296 ParameterCount expected(function);
2483 __ InvokeFunction(function, expected, arguments(), 2297 __ InvokeFunction(function, expected, arguments(),
2484 JUMP_FUNCTION, NullCallWrapper(), CALL_AS_METHOD); 2298 JUMP_FUNCTION, NullCallWrapper(), CALL_AS_METHOD);
2485 2299
2486 __ bind(&miss); 2300 HandlerFrontendFooter(&miss);
2487 // r2: function name.
2488 GenerateMissBranch();
2489 2301
2490 // Return the generated code. 2302 // Return the generated code.
2491 return GetCode(type, name); 2303 return GetCode(type, name);
2492 } 2304 }
2493 2305
2494 2306
2495 Handle<Code> CallStubCompiler::CompileFastApiCall( 2307 Handle<Code> CallStubCompiler::CompileFastApiCall(
2496 const CallOptimization& optimization, 2308 const CallOptimization& optimization,
2497 Handle<Object> object, 2309 Handle<Object> object,
2498 Handle<JSObject> holder, 2310 Handle<JSObject> holder,
(...skipping 51 matching lines...) Expand 10 before | Expand all | Expand 10 after
2550 __ LoadRoot(ip, Heap::kTrueValueRootIndex); 2362 __ LoadRoot(ip, Heap::kTrueValueRootIndex);
2551 __ cmp(object, ip); 2363 __ cmp(object, ip);
2552 __ b(eq, &success); 2364 __ b(eq, &success);
2553 __ LoadRoot(ip, Heap::kFalseValueRootIndex); 2365 __ LoadRoot(ip, Heap::kFalseValueRootIndex);
2554 __ cmp(object, ip); 2366 __ cmp(object, ip);
2555 __ b(ne, miss); 2367 __ b(ne, miss);
2556 __ bind(&success); 2368 __ bind(&success);
2557 } 2369 }
2558 2370
2559 2371
2560 void CallStubCompiler::CompileHandlerFrontend(Handle<Object> object, 2372 Register CallStubCompiler::HandlerFrontendHeader(Handle<Object> object,
2561 Handle<JSObject> holder, 2373 Handle<JSObject> holder,
2562 Handle<Name> name, 2374 Handle<Name> name,
2563 CheckType check) { 2375 CheckType check,
2376 Label* miss) {
2564 // ----------- S t a t e ------------- 2377 // ----------- S t a t e -------------
2565 // -- r2 : name 2378 // -- r2 : name
2566 // -- lr : return address 2379 // -- lr : return address
2567 // ----------------------------------- 2380 // -----------------------------------
Igor Sheludko 2013/11/25 16:31:53 What do you think about putting these comments her
Toon Verwaest 2013/11/27 15:16:55 I removed the comment altogether. I prefer to add
2568 Label miss; 2381 GenerateNameCheck(name, miss);
2569 GenerateNameCheck(name, &miss); 2382
2383 Register reg = r1;
Igor Sheludko 2013/11/25 16:31:53 It looks like the other usages of "r1" below shoul
Toon Verwaest 2013/11/27 15:16:55 Done.
2570 2384
2571 // Get the receiver from the stack 2385 // Get the receiver from the stack
2572 const int argc = arguments().immediate(); 2386 const int argc = arguments().immediate();
2573 __ ldr(r1, MemOperand(sp, argc * kPointerSize)); 2387 const int receiver_offset = argc * kPointerSize;
2388 __ ldr(r1, MemOperand(sp, receiver_offset));
2574 2389
2575 // Check that the receiver isn't a smi. 2390 // Check that the receiver isn't a smi.
2576 if (check != NUMBER_CHECK) { 2391 if (check != NUMBER_CHECK) {
2577 __ JumpIfSmi(r1, &miss); 2392 __ JumpIfSmi(r1, miss);
2578 } 2393 }
2579 2394
2580 // Make sure that it's okay not to patch the on stack receiver 2395 // Make sure that it's okay not to patch the on stack receiver
2581 // unless we're doing a receiver map check. 2396 // unless we're doing a receiver map check.
2582 ASSERT(!object->IsGlobalObject() || check == RECEIVER_MAP_CHECK); 2397 ASSERT(!object->IsGlobalObject() || check == RECEIVER_MAP_CHECK);
2583 switch (check) { 2398 switch (check) {
2584 case RECEIVER_MAP_CHECK: 2399 case RECEIVER_MAP_CHECK:
2585 __ IncrementCounter(isolate()->counters()->call_const(), 1, r0, r3); 2400 __ IncrementCounter(isolate()->counters()->call_const(), 1, r0, r3);
2586 2401
2587 // Check that the maps haven't changed. 2402 // Check that the maps haven't changed.
2588 CheckPrototypes( 2403 reg = CheckPrototypes(
2589 IC::CurrentTypeOf(object, isolate()), 2404 IC::CurrentTypeOf(object, isolate()),
2590 r1, holder, r0, r3, r4, name, &miss); 2405 r1, holder, r0, r3, r4, name, miss);
2591 2406
2592 // Patch the receiver on the stack with the global proxy if 2407 // Patch the receiver on the stack with the global proxy if
2593 // necessary. 2408 // necessary.
2594 if (object->IsGlobalObject()) { 2409 if (object->IsGlobalObject()) {
2595 __ ldr(r3, FieldMemOperand(r1, GlobalObject::kGlobalReceiverOffset)); 2410 __ ldr(r3, FieldMemOperand(r1, GlobalObject::kGlobalReceiverOffset));
2596 __ str(r3, MemOperand(sp, argc * kPointerSize)); 2411 __ str(r3, MemOperand(sp, receiver_offset));
2597 } 2412 }
2598 break; 2413 break;
2599 2414
2600 case STRING_CHECK: { 2415 case STRING_CHECK: {
2601 // Check that the object is a string. 2416 // Check that the object is a string.
2602 __ CompareObjectType(r1, r3, r3, FIRST_NONSTRING_TYPE); 2417 __ CompareObjectType(r1, r3, r3, FIRST_NONSTRING_TYPE);
2603 __ b(ge, &miss); 2418 __ b(ge, miss);
2604 // Check that the maps starting from the prototype haven't changed. 2419 // Check that the maps starting from the prototype haven't changed.
2605 GenerateDirectLoadGlobalFunctionPrototype( 2420 GenerateDirectLoadGlobalFunctionPrototype(
2606 masm(), Context::STRING_FUNCTION_INDEX, r0, &miss); 2421 masm(), Context::STRING_FUNCTION_INDEX, r0, miss);
2607 Handle<Object> prototype(object->GetPrototype(isolate()), isolate());
2608 CheckPrototypes(
2609 IC::CurrentTypeOf(prototype, isolate()),
2610 r0, holder, r3, r1, r4, name, &miss);
2611 break; 2422 break;
2612 } 2423 }
2613 case SYMBOL_CHECK: { 2424 case SYMBOL_CHECK: {
2614 // Check that the object is a symbol. 2425 // Check that the object is a symbol.
2615 __ CompareObjectType(r1, r1, r3, SYMBOL_TYPE); 2426 __ CompareObjectType(r1, r1, r3, SYMBOL_TYPE);
2616 __ b(ne, &miss); 2427 __ b(ne, miss);
2617 // Check that the maps starting from the prototype haven't changed. 2428 // Check that the maps starting from the prototype haven't changed.
2618 GenerateDirectLoadGlobalFunctionPrototype( 2429 GenerateDirectLoadGlobalFunctionPrototype(
2619 masm(), Context::SYMBOL_FUNCTION_INDEX, r0, &miss); 2430 masm(), Context::SYMBOL_FUNCTION_INDEX, r0, miss);
2620 Handle<Object> prototype(object->GetPrototype(isolate()), isolate());
2621 CheckPrototypes(
2622 IC::CurrentTypeOf(prototype, isolate()),
2623 r0, holder, r3, r1, r4, name, &miss);
2624 break; 2431 break;
2625 } 2432 }
2626 case NUMBER_CHECK: { 2433 case NUMBER_CHECK: {
2627 Label fast; 2434 Label fast;
2628 // Check that the object is a smi or a heap number. 2435 // Check that the object is a smi or a heap number.
2629 __ JumpIfSmi(r1, &fast); 2436 __ JumpIfSmi(r1, &fast);
2630 __ CompareObjectType(r1, r0, r0, HEAP_NUMBER_TYPE); 2437 __ CompareObjectType(r1, r0, r0, HEAP_NUMBER_TYPE);
2631 __ b(ne, &miss); 2438 __ b(ne, miss);
2632 __ bind(&fast); 2439 __ bind(&fast);
2633 // Check that the maps starting from the prototype haven't changed. 2440 // Check that the maps starting from the prototype haven't changed.
2634 GenerateDirectLoadGlobalFunctionPrototype( 2441 GenerateDirectLoadGlobalFunctionPrototype(
2635 masm(), Context::NUMBER_FUNCTION_INDEX, r0, &miss); 2442 masm(), Context::NUMBER_FUNCTION_INDEX, r0, miss);
2636 Handle<Object> prototype(object->GetPrototype(isolate()), isolate());
2637 CheckPrototypes(
2638 IC::CurrentTypeOf(prototype, isolate()),
2639 r0, holder, r3, r1, r4, name, &miss);
2640 break; 2443 break;
2641 } 2444 }
2642 case BOOLEAN_CHECK: { 2445 case BOOLEAN_CHECK: {
2643 GenerateBooleanCheck(r1, &miss); 2446 GenerateBooleanCheck(r1, miss);
2644 2447
2645 // Check that the maps starting from the prototype haven't changed. 2448 // Check that the maps starting from the prototype haven't changed.
2646 GenerateDirectLoadGlobalFunctionPrototype( 2449 GenerateDirectLoadGlobalFunctionPrototype(
2647 masm(), Context::BOOLEAN_FUNCTION_INDEX, r0, &miss); 2450 masm(), Context::BOOLEAN_FUNCTION_INDEX, r0, miss);
2648 Handle<Object> prototype(object->GetPrototype(isolate()), isolate());
2649 CheckPrototypes(
2650 IC::CurrentTypeOf(prototype, isolate()),
2651 r0, holder, r3, r1, r4, name, &miss);
2652 break; 2451 break;
2653 } 2452 }
2654 } 2453 }
2655 2454
2656 Label success; 2455 if (check != RECEIVER_MAP_CHECK) {
2657 __ b(&success); 2456 Handle<Object> prototype(object->GetPrototype(isolate()), isolate());
2457 reg = CheckPrototypes(
2458 IC::CurrentTypeOf(prototype, isolate()),
2459 r0, holder, r3, r1, r4, name, miss);
2460 }
2658 2461
2659 // Handle call cache miss. 2462 return reg;
2660 __ bind(&miss);
2661 GenerateMissBranch();
2662
2663 __ bind(&success);
2664 } 2463 }
2665 2464
2666 2465
2667 void CallStubCompiler::CompileHandlerBackend(Handle<JSFunction> function) { 2466 void CallStubCompiler::CompileHandlerBackend(Handle<JSFunction> function) {
2668 CallKind call_kind = CallICBase::Contextual::decode(extra_state_) 2467 CallKind call_kind = CallICBase::Contextual::decode(extra_state_)
2669 ? CALL_AS_FUNCTION 2468 ? CALL_AS_FUNCTION
2670 : CALL_AS_METHOD; 2469 : CALL_AS_METHOD;
2671 ParameterCount expected(function); 2470 ParameterCount expected(function);
2672 __ InvokeFunction(function, expected, arguments(), 2471 __ InvokeFunction(function, expected, arguments(),
2673 JUMP_FUNCTION, NullCallWrapper(), call_kind); 2472 JUMP_FUNCTION, NullCallWrapper(), call_kind);
2674 } 2473 }
2675 2474
2676 2475
2677 Handle<Code> CallStubCompiler::CompileCallConstant( 2476 Handle<Code> CallStubCompiler::CompileCallConstant(
2678 Handle<Object> object, 2477 Handle<Object> object,
2679 Handle<JSObject> holder, 2478 Handle<JSObject> holder,
2680 Handle<Name> name, 2479 Handle<Name> name,
2681 CheckType check, 2480 CheckType check,
2682 Handle<JSFunction> function) { 2481 Handle<JSFunction> function) {
2683 if (HasCustomCallGenerator(function)) { 2482 if (HasCustomCallGenerator(function)) {
2684 Handle<Code> code = CompileCustomCall(object, holder, 2483 Handle<Code> code = CompileCustomCall(object, holder,
2685 Handle<Cell>::null(), 2484 Handle<Cell>::null(),
2686 function, Handle<String>::cast(name), 2485 function, Handle<String>::cast(name),
2687 Code::FAST); 2486 Code::FAST);
2688 // A null handle means bail out to the regular compiler code below. 2487 // A null handle means bail out to the regular compiler code below.
2689 if (!code.is_null()) return code; 2488 if (!code.is_null()) return code;
2690 } 2489 }
2691 2490
2692 CompileHandlerFrontend(object, holder, name, check); 2491 Label miss;
2492 HandlerFrontendHeader(object, holder, name, check, &miss);
2693 CompileHandlerBackend(function); 2493 CompileHandlerBackend(function);
2494 HandlerFrontendFooter(&miss);
2694 2495
2695 // Return the generated code. 2496 // Return the generated code.
2696 return GetCode(function); 2497 return GetCode(function);
2697 } 2498 }
2698 2499
2699 2500
2700 Handle<Code> CallStubCompiler::CompileCallInterceptor(Handle<JSObject> object, 2501 Handle<Code> CallStubCompiler::CompileCallInterceptor(Handle<JSObject> object,
2701 Handle<JSObject> holder, 2502 Handle<JSObject> holder,
2702 Handle<Name> name) { 2503 Handle<Name> name) {
2703 // ----------- S t a t e -------------
2704 // -- r2 : name
2705 // -- lr : return address
2706 // -----------------------------------
2707 Label miss; 2504 Label miss;
2708 GenerateNameCheck(name, &miss); 2505 GenerateNameCheck(name, &miss);
2709 2506
2710 // Get the number of arguments. 2507 // Get the number of arguments.
2711 const int argc = arguments().immediate(); 2508 const int argc = arguments().immediate();
2712 LookupResult lookup(isolate()); 2509 LookupResult lookup(isolate());
2713 LookupPostInterceptor(holder, name, &lookup); 2510 LookupPostInterceptor(holder, name, &lookup);
2714 2511
2715 // Get the receiver from the stack. 2512 // Get the receiver from the stack.
2716 __ ldr(r1, MemOperand(sp, argc * kPointerSize)); 2513 __ ldr(r1, MemOperand(sp, argc * kPointerSize));
2717 2514
2718 CallInterceptorCompiler compiler(this, arguments(), r2, extra_state_); 2515 CallInterceptorCompiler compiler(this, arguments(), r2, extra_state_);
2719 compiler.Compile(masm(), object, holder, name, &lookup, r1, r3, r4, r0, 2516 compiler.Compile(masm(), object, holder, name, &lookup, r1, r3, r4, r0,
2720 &miss); 2517 &miss);
2721 2518
2722 // Move returned value, the function to call, to r1. 2519 // Move returned value, the function to call, to r1.
2723 __ mov(r1, r0); 2520 __ mov(r1, r0);
2724 // Restore receiver. 2521 // Restore receiver.
2725 __ ldr(r0, MemOperand(sp, argc * kPointerSize)); 2522 __ ldr(r0, MemOperand(sp, argc * kPointerSize));
2726 2523
2727 GenerateCallFunction(masm(), object, arguments(), &miss, extra_state_); 2524 GenerateCallFunction(masm(), object, arguments(), &miss, extra_state_);
Igor Sheludko 2013/11/25 16:31:53 Modified GenerateCallFunction() is no longer patch
Toon Verwaest 2013/11/27 15:16:55 Done.
2728 2525
2729 // Handle call cache miss. 2526 // Handle call cache miss.
2730 __ bind(&miss); 2527 __ bind(&miss);
2731 GenerateMissBranch(); 2528 GenerateMissBranch();
2732 2529
2733 // Return the generated code. 2530 // Return the generated code.
2734 return GetCode(Code::FAST, name); 2531 return GetCode(Code::FAST, name);
2735 } 2532 }
2736 2533
2737 2534
2738 Handle<Code> CallStubCompiler::CompileCallGlobal( 2535 Handle<Code> CallStubCompiler::CompileCallGlobal(
2739 Handle<JSObject> object, 2536 Handle<JSObject> object,
2740 Handle<GlobalObject> holder, 2537 Handle<GlobalObject> holder,
2741 Handle<PropertyCell> cell, 2538 Handle<PropertyCell> cell,
2742 Handle<JSFunction> function, 2539 Handle<JSFunction> function,
2743 Handle<Name> name) { 2540 Handle<Name> name) {
2744 // ----------- S t a t e -------------
2745 // -- r2 : name
2746 // -- lr : return address
2747 // -----------------------------------
2748 if (HasCustomCallGenerator(function)) { 2541 if (HasCustomCallGenerator(function)) {
2749 Handle<Code> code = CompileCustomCall( 2542 Handle<Code> code = CompileCustomCall(
2750 object, holder, cell, function, Handle<String>::cast(name), 2543 object, holder, cell, function, Handle<String>::cast(name),
2751 Code::NORMAL); 2544 Code::NORMAL);
2752 // A null handle means bail out to the regular compiler code below. 2545 // A null handle means bail out to the regular compiler code below.
2753 if (!code.is_null()) return code; 2546 if (!code.is_null()) return code;
2754 } 2547 }
2755 2548
2756 Label miss; 2549 Label miss;
2757 GenerateNameCheck(name, &miss); 2550 HandlerFrontendHeader(object, holder, name, RECEIVER_MAP_CHECK, &miss);
2758
2759 // Get the number of arguments.
2760 const int argc = arguments().immediate();
2761 GenerateGlobalReceiverCheck(object, holder, name, &miss);
2762 GenerateLoadFunctionFromCell(cell, function, &miss); 2551 GenerateLoadFunctionFromCell(cell, function, &miss);
2763 2552
2764 // Patch the receiver on the stack with the global proxy if
2765 // necessary.
2766 if (object->IsGlobalObject()) {
2767 __ ldr(r3, FieldMemOperand(r0, GlobalObject::kGlobalReceiverOffset));
2768 __ str(r3, MemOperand(sp, argc * kPointerSize));
2769 }
2770
2771 // Set up the context (function already in r1). 2553 // Set up the context (function already in r1).
2772 __ ldr(cp, FieldMemOperand(r1, JSFunction::kContextOffset)); 2554 __ ldr(cp, FieldMemOperand(r1, JSFunction::kContextOffset));
2773 2555
2774 // Jump to the cached code (tail call). 2556 // Jump to the cached code (tail call).
2775 Counters* counters = isolate()->counters(); 2557 Counters* counters = isolate()->counters();
2776 __ IncrementCounter(counters->call_global_inline(), 1, r3, r4); 2558 __ IncrementCounter(counters->call_global_inline(), 1, r3, r4);
2777 ParameterCount expected(function->shared()->formal_parameter_count()); 2559 ParameterCount expected(function->shared()->formal_parameter_count());
2778 CallKind call_kind = CallICBase::Contextual::decode(extra_state_) 2560 CallKind call_kind = CallICBase::Contextual::decode(extra_state_)
2779 ? CALL_AS_FUNCTION 2561 ? CALL_AS_FUNCTION
2780 : CALL_AS_METHOD; 2562 : CALL_AS_METHOD;
2781 // We call indirectly through the code field in the function to 2563 // We call indirectly through the code field in the function to
2782 // allow recompilation to take effect without changing any of the 2564 // allow recompilation to take effect without changing any of the
2783 // call sites. 2565 // call sites.
2784 __ ldr(r3, FieldMemOperand(r1, JSFunction::kCodeEntryOffset)); 2566 __ ldr(r3, FieldMemOperand(r1, JSFunction::kCodeEntryOffset));
2785 __ InvokeCode(r3, expected, arguments(), JUMP_FUNCTION, 2567 __ InvokeCode(r3, expected, arguments(), JUMP_FUNCTION,
2786 NullCallWrapper(), call_kind); 2568 NullCallWrapper(), call_kind);
2787 2569
2788 // Handle call cache miss. 2570 HandlerFrontendFooter(&miss);
2789 __ bind(&miss);
2790 __ IncrementCounter(counters->call_global_inline_miss(), 1, r1, r3);
2791 GenerateMissBranch();
2792 2571
2793 // Return the generated code. 2572 // Return the generated code.
2794 return GetCode(Code::NORMAL, name); 2573 return GetCode(Code::NORMAL, name);
2795 } 2574 }
2796 2575
2797 2576
2798 Handle<Code> StoreStubCompiler::CompileStoreCallback( 2577 Handle<Code> StoreStubCompiler::CompileStoreCallback(
2799 Handle<JSObject> object, 2578 Handle<JSObject> object,
2800 Handle<JSObject> holder, 2579 Handle<JSObject> holder,
2801 Handle<Name> name, 2580 Handle<Name> name,
(...skipping 373 matching lines...) Expand 10 before | Expand all | Expand 10 after
3175 // ----------------------------------- 2954 // -----------------------------------
3176 TailCallBuiltin(masm, Builtins::kKeyedLoadIC_Miss); 2955 TailCallBuiltin(masm, Builtins::kKeyedLoadIC_Miss);
3177 } 2956 }
3178 2957
3179 2958
3180 #undef __ 2959 #undef __
3181 2960
3182 } } // namespace v8::internal 2961 } } // namespace v8::internal
3183 2962
3184 #endif // V8_TARGET_ARCH_ARM 2963 #endif // V8_TARGET_ARCH_ARM
OLDNEW
« no previous file with comments | « no previous file | src/ia32/stub-cache-ia32.cc » ('j') | src/stub-cache.h » ('J')

Powered by Google App Engine
This is Rietveld 408576698