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

Side by Side Diff: src/ia32/code-stubs-ia32.cc

Issue 1980483003: [es6] Reintroduce the instanceof operator in the backends. (Closed) Base URL: https://chromium.googlesource.com/v8/v8.git@master
Patch Set: Igors comments. Created 4 years, 7 months 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
« no previous file with comments | « src/ia32/builtins-ia32.cc ('k') | src/ia32/interface-descriptors-ia32.cc » ('j') | no next file with comments »
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 // 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
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
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
OLDNEW
« no previous file with comments | « src/ia32/builtins-ia32.cc ('k') | src/ia32/interface-descriptors-ia32.cc » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698