| 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 #if V8_TARGET_ARCH_IA32 | 5 #if V8_TARGET_ARCH_IA32 |
| 6 | 6 |
| 7 #include "src/code-stubs.h" | 7 #include "src/code-stubs.h" |
| 8 #include "src/api-arguments.h" | 8 #include "src/api-arguments.h" |
| 9 #include "src/base/bits.h" | 9 #include "src/base/bits.h" |
| 10 #include "src/bootstrapper.h" | 10 #include "src/bootstrapper.h" |
| (...skipping 2028 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 2039 __ pop(esi); | 2039 __ pop(esi); |
| 2040 __ pop(edi); | 2040 __ pop(edi); |
| 2041 __ add(esp, Immediate(2 * kPointerSize)); // remove markers | 2041 __ add(esp, Immediate(2 * kPointerSize)); // remove markers |
| 2042 | 2042 |
| 2043 // Restore frame pointer and return. | 2043 // Restore frame pointer and return. |
| 2044 __ pop(ebp); | 2044 __ pop(ebp); |
| 2045 __ ret(0); | 2045 __ ret(0); |
| 2046 } | 2046 } |
| 2047 | 2047 |
| 2048 | 2048 |
| 2049 void InstanceOfStub::Generate(MacroAssembler* masm) { | |
| 2050 Register const object = edx; // Object (lhs). | |
| 2051 Register const function = eax; // Function (rhs). | |
| 2052 Register const object_map = ecx; // Map of {object}. | |
| 2053 Register const function_map = ebx; // Map of {function}. | |
| 2054 Register const function_prototype = function_map; // Prototype of {function}. | |
| 2055 Register const scratch = edi; | |
| 2056 | |
| 2057 DCHECK(object.is(InstanceOfDescriptor::LeftRegister())); | |
| 2058 DCHECK(function.is(InstanceOfDescriptor::RightRegister())); | |
| 2059 | |
| 2060 // Check if {object} is a smi. | |
| 2061 Label object_is_smi; | |
| 2062 __ JumpIfSmi(object, &object_is_smi, Label::kNear); | |
| 2063 | |
| 2064 // Lookup the {function} and the {object} map in the global instanceof cache. | |
| 2065 // Note: This is safe because we clear the global instanceof cache whenever | |
| 2066 // we change the prototype of any object. | |
| 2067 Label fast_case, slow_case; | |
| 2068 __ mov(object_map, FieldOperand(object, HeapObject::kMapOffset)); | |
| 2069 __ CompareRoot(function, scratch, Heap::kInstanceofCacheFunctionRootIndex); | |
| 2070 __ j(not_equal, &fast_case, Label::kNear); | |
| 2071 __ CompareRoot(object_map, scratch, Heap::kInstanceofCacheMapRootIndex); | |
| 2072 __ j(not_equal, &fast_case, Label::kNear); | |
| 2073 __ LoadRoot(eax, Heap::kInstanceofCacheAnswerRootIndex); | |
| 2074 __ ret(0); | |
| 2075 | |
| 2076 // If {object} is a smi we can safely return false if {function} is a JS | |
| 2077 // function, otherwise we have to miss to the runtime and throw an exception. | |
| 2078 __ bind(&object_is_smi); | |
| 2079 __ JumpIfSmi(function, &slow_case); | |
| 2080 __ CmpObjectType(function, JS_FUNCTION_TYPE, function_map); | |
| 2081 __ j(not_equal, &slow_case); | |
| 2082 __ LoadRoot(eax, Heap::kFalseValueRootIndex); | |
| 2083 __ ret(0); | |
| 2084 | |
| 2085 // Fast-case: The {function} must be a valid JSFunction. | |
| 2086 __ bind(&fast_case); | |
| 2087 __ JumpIfSmi(function, &slow_case); | |
| 2088 __ CmpObjectType(function, JS_FUNCTION_TYPE, function_map); | |
| 2089 __ j(not_equal, &slow_case); | |
| 2090 | |
| 2091 // Go to the runtime if the function is not a constructor. | |
| 2092 __ test_b(FieldOperand(function_map, Map::kBitFieldOffset), | |
| 2093 Immediate(1 << Map::kIsConstructor)); | |
| 2094 __ j(zero, &slow_case); | |
| 2095 | |
| 2096 // Ensure that {function} has an instance prototype. | |
| 2097 __ test_b(FieldOperand(function_map, Map::kBitFieldOffset), | |
| 2098 Immediate(1 << Map::kHasNonInstancePrototype)); | |
| 2099 __ j(not_zero, &slow_case); | |
| 2100 | |
| 2101 // Get the "prototype" (or initial map) of the {function}. | |
| 2102 __ mov(function_prototype, | |
| 2103 FieldOperand(function, JSFunction::kPrototypeOrInitialMapOffset)); | |
| 2104 __ AssertNotSmi(function_prototype); | |
| 2105 | |
| 2106 // Resolve the prototype if the {function} has an initial map. Afterwards the | |
| 2107 // {function_prototype} will be either the JSReceiver prototype object or the | |
| 2108 // hole value, which means that no instances of the {function} were created so | |
| 2109 // far and hence we should return false. | |
| 2110 Label function_prototype_valid; | |
| 2111 Register const function_prototype_map = scratch; | |
| 2112 __ CmpObjectType(function_prototype, MAP_TYPE, function_prototype_map); | |
| 2113 __ j(not_equal, &function_prototype_valid, Label::kNear); | |
| 2114 __ mov(function_prototype, | |
| 2115 FieldOperand(function_prototype, Map::kPrototypeOffset)); | |
| 2116 __ bind(&function_prototype_valid); | |
| 2117 __ AssertNotSmi(function_prototype); | |
| 2118 | |
| 2119 // Update the global instanceof cache with the current {object} map and | |
| 2120 // {function}. The cached answer will be set when it is known below. | |
| 2121 __ StoreRoot(function, scratch, Heap::kInstanceofCacheFunctionRootIndex); | |
| 2122 __ StoreRoot(object_map, scratch, Heap::kInstanceofCacheMapRootIndex); | |
| 2123 | |
| 2124 // Loop through the prototype chain looking for the {function} prototype. | |
| 2125 // Assume true, and change to false if not found. | |
| 2126 Label done, loop, fast_runtime_fallback; | |
| 2127 __ mov(eax, isolate()->factory()->true_value()); | |
| 2128 __ bind(&loop); | |
| 2129 | |
| 2130 // Check if the object needs to be access checked. | |
| 2131 __ test_b(FieldOperand(object_map, Map::kBitFieldOffset), | |
| 2132 Immediate(1 << Map::kIsAccessCheckNeeded)); | |
| 2133 __ j(not_zero, &fast_runtime_fallback, Label::kNear); | |
| 2134 // Check if the current object is a Proxy. | |
| 2135 __ CmpInstanceType(object_map, JS_PROXY_TYPE); | |
| 2136 __ j(equal, &fast_runtime_fallback, Label::kNear); | |
| 2137 | |
| 2138 __ mov(object, FieldOperand(object_map, Map::kPrototypeOffset)); | |
| 2139 __ cmp(object, function_prototype); | |
| 2140 __ j(equal, &done, Label::kNear); | |
| 2141 __ mov(object_map, FieldOperand(object, HeapObject::kMapOffset)); | |
| 2142 __ cmp(object, isolate()->factory()->null_value()); | |
| 2143 __ j(not_equal, &loop); | |
| 2144 __ mov(eax, isolate()->factory()->false_value()); | |
| 2145 | |
| 2146 __ bind(&done); | |
| 2147 __ StoreRoot(eax, scratch, Heap::kInstanceofCacheAnswerRootIndex); | |
| 2148 __ ret(0); | |
| 2149 | |
| 2150 // Found Proxy or access check needed: Call the runtime. | |
| 2151 __ bind(&fast_runtime_fallback); | |
| 2152 __ PopReturnAddressTo(scratch); | |
| 2153 __ Push(object); | |
| 2154 __ Push(function_prototype); | |
| 2155 __ PushReturnAddressFrom(scratch); | |
| 2156 // Invalidate the instanceof cache. | |
| 2157 __ Move(eax, Immediate(Smi::FromInt(0))); | |
| 2158 __ StoreRoot(eax, scratch, Heap::kInstanceofCacheFunctionRootIndex); | |
| 2159 __ TailCallRuntime(Runtime::kHasInPrototypeChain); | |
| 2160 | |
| 2161 // Slow-case: Call the %InstanceOf runtime function. | |
| 2162 __ bind(&slow_case); | |
| 2163 __ PopReturnAddressTo(scratch); | |
| 2164 __ Push(object); | |
| 2165 __ Push(function); | |
| 2166 __ PushReturnAddressFrom(scratch); | |
| 2167 __ TailCallRuntime(is_es6_instanceof() ? Runtime::kOrdinaryHasInstance | |
| 2168 : Runtime::kInstanceOf); | |
| 2169 } | |
| 2170 | |
| 2171 | |
| 2172 // ------------------------------------------------------------------------- | 2049 // ------------------------------------------------------------------------- |
| 2173 // StringCharCodeAtGenerator | 2050 // StringCharCodeAtGenerator |
| 2174 | 2051 |
| 2175 void StringCharCodeAtGenerator::GenerateFast(MacroAssembler* masm) { | 2052 void StringCharCodeAtGenerator::GenerateFast(MacroAssembler* masm) { |
| 2176 // If the receiver is a smi trigger the non-string case. | 2053 // If the receiver is a smi trigger the non-string case. |
| 2177 STATIC_ASSERT(kSmiTag == 0); | 2054 STATIC_ASSERT(kSmiTag == 0); |
| 2178 if (check_mode_ == RECEIVER_IS_UNKNOWN) { | 2055 if (check_mode_ == RECEIVER_IS_UNKNOWN) { |
| 2179 __ JumpIfSmi(object_, receiver_not_string_); | 2056 __ JumpIfSmi(object_, receiver_not_string_); |
| 2180 | 2057 |
| 2181 // Fetch the instance type of the receiver into result register. | 2058 // Fetch the instance type of the receiver into result register. |
| (...skipping 3663 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 5845 kStackUnwindSpace, nullptr, return_value_operand, | 5722 kStackUnwindSpace, nullptr, return_value_operand, |
| 5846 NULL); | 5723 NULL); |
| 5847 } | 5724 } |
| 5848 | 5725 |
| 5849 #undef __ | 5726 #undef __ |
| 5850 | 5727 |
| 5851 } // namespace internal | 5728 } // namespace internal |
| 5852 } // namespace v8 | 5729 } // namespace v8 |
| 5853 | 5730 |
| 5854 #endif // V8_TARGET_ARCH_IA32 | 5731 #endif // V8_TARGET_ARCH_IA32 |
| OLD | NEW |