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
3 // found in the LICENSE file.
10 #include "src/compiler.h"
11 #include "src/debug.h"
12 #include "src/macro-assembler.h"
13 #include "src/prototype.h"
19 Address IC::address() const {
20 // Get the address of the call.
21 return Assembler::target_address_from_return_address(pc());
25 Address IC::constant_pool() const {
26 if (FLAG_enable_embedded_constant_pool) {
27 return raw_constant_pool();
34 Address IC::raw_constant_pool() const {
35 if (FLAG_enable_embedded_constant_pool) {
36 return *constant_pool_address_;
43 Code* IC::GetTargetAtAddress(Address address, Address constant_pool) {
44 // Get the target address of the IC.
45 Address target = Assembler::target_address_at(address, constant_pool);
46 // Convert target address to the code object. Code::GetCodeFromTargetAddress
47 // is safe for use during GC where the map might be marked.
48 Code* result = Code::GetCodeFromTargetAddress(target);
49 DCHECK(result->is_inline_cache_stub());
54 void IC::SetTargetAtAddress(Address address, Code* target,
55 Address constant_pool) {
56 if (AddressIsDeoptimizedCode(target->GetIsolate(), address)) return;
58 DCHECK(target->is_inline_cache_stub() || target->is_compare_ic_stub());
60 DCHECK(!target->is_inline_cache_stub() ||
61 (target->kind() != Code::LOAD_IC &&
62 target->kind() != Code::KEYED_LOAD_IC &&
63 (!FLAG_vector_stores || (target->kind() != Code::STORE_IC &&
64 target->kind() != Code::KEYED_STORE_IC))));
66 Heap* heap = target->GetHeap();
67 Code* old_target = GetTargetAtAddress(address, constant_pool);
69 // STORE_IC and KEYED_STORE_IC use Code::extra_ic_state() to mark
70 // ICs as language mode. The language mode of the IC must be preserved.
71 if (old_target->kind() == Code::STORE_IC ||
72 old_target->kind() == Code::KEYED_STORE_IC) {
73 DCHECK(StoreICState::GetLanguageMode(old_target->extra_ic_state()) ==
74 StoreICState::GetLanguageMode(target->extra_ic_state()));
77 Assembler::set_target_address_at(address, constant_pool,
78 target->instruction_start());
79 if (heap->gc_state() == Heap::MARK_COMPACT) {
80 heap->mark_compact_collector()->RecordCodeTargetPatch(address, target);
82 heap->incremental_marking()->RecordCodeTargetPatch(address, target);
84 PostPatching(address, target, old_target);
88 void IC::set_target(Code* code) {
89 SetTargetAtAddress(address(), code, constant_pool());
94 void LoadIC::set_target(Code* code) {
95 // The contextual mode must be preserved across IC patching.
96 DCHECK(LoadICState::GetTypeofMode(code->extra_ic_state()) ==
97 LoadICState::GetTypeofMode(target()->extra_ic_state()));
98 // Strongness must be preserved across IC patching.
99 DCHECK(LoadICState::GetLanguageMode(code->extra_ic_state()) ==
100 LoadICState::GetLanguageMode(target()->extra_ic_state()));
102 IC::set_target(code);
106 void StoreIC::set_target(Code* code) {
107 // Language mode must be preserved across IC patching.
108 DCHECK(StoreICState::GetLanguageMode(code->extra_ic_state()) ==
109 StoreICState::GetLanguageMode(target()->extra_ic_state()));
110 IC::set_target(code);
114 void KeyedStoreIC::set_target(Code* code) {
115 // Language mode must be preserved across IC patching.
116 DCHECK(StoreICState::GetLanguageMode(code->extra_ic_state()) ==
118 IC::set_target(code);
122 Code* IC::raw_target() const {
123 return GetTargetAtAddress(address(), constant_pool());
126 void IC::UpdateTarget() { target_ = handle(raw_target(), isolate_); }
129 JSFunction* IC::GetRootConstructor(Map* receiver_map, Context* native_context) {
130 Isolate* isolate = receiver_map->GetIsolate();
131 if (receiver_map == isolate->heap()->boolean_map()) {
132 return native_context->boolean_function();
133 } else if (receiver_map->instance_type() == HEAP_NUMBER_TYPE) {
134 return native_context->number_function();
135 } else if (receiver_map->instance_type() < FIRST_NONSTRING_TYPE) {
136 return native_context->string_function();
137 } else if (receiver_map->instance_type() == SYMBOL_TYPE) {
138 return native_context->symbol_function();
139 } else if (receiver_map->instance_type() == FLOAT32X4_TYPE) {
140 return native_context->float32x4_function();
147 Handle<Map> IC::GetHandlerCacheHolder(Handle<Map> receiver_map,
148 bool receiver_is_holder, Isolate* isolate,
149 CacheHolderFlag* flag) {
150 if (receiver_is_holder) {
151 *flag = kCacheOnReceiver;
154 Context* native_context = *isolate->native_context();
155 JSFunction* builtin_ctor = GetRootConstructor(*receiver_map, native_context);
156 if (builtin_ctor != NULL) {
157 *flag = kCacheOnPrototypeReceiverIsPrimitive;
158 return handle(HeapObject::cast(builtin_ctor->instance_prototype())->map());
160 *flag = receiver_map->is_dictionary_map()
161 ? kCacheOnPrototypeReceiverIsDictionary
163 // Callers must ensure that the prototype is non-null.
164 return handle(JSObject::cast(receiver_map->prototype())->map());
168 Handle<Map> IC::GetICCacheHolder(Handle<Map> map, Isolate* isolate,
169 CacheHolderFlag* flag) {
170 Context* native_context = *isolate->native_context();
171 JSFunction* builtin_ctor = GetRootConstructor(*map, native_context);
172 if (builtin_ctor != NULL) {
173 *flag = kCacheOnPrototype;
174 return handle(builtin_ctor->initial_map());
176 *flag = kCacheOnReceiver;
181 Code* IC::get_host() {
183 ->inner_pointer_to_code_cache()
184 ->GetCacheEntry(address())
189 bool IC::AddressIsDeoptimizedCode() const {
190 return AddressIsDeoptimizedCode(isolate(), address());
194 bool IC::AddressIsDeoptimizedCode(Isolate* isolate, Address address) {
196 isolate->inner_pointer_to_code_cache()->GetCacheEntry(address)->code;
197 return (host->kind() == Code::OPTIMIZED_FUNCTION &&
198 host->marked_for_deoptimization());
201 } // namespace v8::internal
203 #endif // V8_IC_INL_H_