| OLD | NEW |
| 1 // Copyright 2006-2009 the V8 project authors. All rights reserved. | 1 // Copyright 2006-2009 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 75 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 86 *secondary = *primary; | 86 *secondary = *primary; |
| 87 } | 87 } |
| 88 | 88 |
| 89 // Update primary cache. | 89 // Update primary cache. |
| 90 primary->key = name; | 90 primary->key = name; |
| 91 primary->value = code; | 91 primary->value = code; |
| 92 return code; | 92 return code; |
| 93 } | 93 } |
| 94 | 94 |
| 95 | 95 |
| 96 Object* StubCache::ComputeLoadNonexistent(String* name, JSObject* receiver) { | 96 MaybeObject* StubCache::ComputeLoadNonexistent(String* name, |
| 97 JSObject* receiver) { |
| 97 ASSERT(receiver->IsGlobalObject() || receiver->HasFastProperties()); | 98 ASSERT(receiver->IsGlobalObject() || receiver->HasFastProperties()); |
| 98 // If no global objects are present in the prototype chain, the load | 99 // If no global objects are present in the prototype chain, the load |
| 99 // nonexistent IC stub can be shared for all names for a given map | 100 // nonexistent IC stub can be shared for all names for a given map |
| 100 // and we use the empty string for the map cache in that case. If | 101 // and we use the empty string for the map cache in that case. If |
| 101 // there are global objects involved, we need to check global | 102 // there are global objects involved, we need to check global |
| 102 // property cells in the stub and therefore the stub will be | 103 // property cells in the stub and therefore the stub will be |
| 103 // specific to the name. | 104 // specific to the name. |
| 104 String* cache_name = Heap::empty_string(); | 105 String* cache_name = Heap::empty_string(); |
| 105 if (receiver->IsGlobalObject()) cache_name = name; | 106 if (receiver->IsGlobalObject()) cache_name = name; |
| 106 JSObject* last = receiver; | 107 JSObject* last = receiver; |
| 107 while (last->GetPrototype() != Heap::null_value()) { | 108 while (last->GetPrototype() != Heap::null_value()) { |
| 108 last = JSObject::cast(last->GetPrototype()); | 109 last = JSObject::cast(last->GetPrototype()); |
| 109 if (last->IsGlobalObject()) cache_name = name; | 110 if (last->IsGlobalObject()) cache_name = name; |
| 110 } | 111 } |
| 111 // Compile the stub that is either shared for all names or | 112 // Compile the stub that is either shared for all names or |
| 112 // name specific if there are global objects involved. | 113 // name specific if there are global objects involved. |
| 113 Code::Flags flags = | 114 Code::Flags flags = |
| 114 Code::ComputeMonomorphicFlags(Code::LOAD_IC, NONEXISTENT); | 115 Code::ComputeMonomorphicFlags(Code::LOAD_IC, NONEXISTENT); |
| 115 Object* code = receiver->map()->FindInCodeCache(cache_name, flags); | 116 Object* code = receiver->map()->FindInCodeCache(cache_name, flags); |
| 116 if (code->IsUndefined()) { | 117 if (code->IsUndefined()) { |
| 117 LoadStubCompiler compiler; | 118 LoadStubCompiler compiler; |
| 118 code = compiler.CompileLoadNonexistent(cache_name, receiver, last); | 119 { MaybeObject* maybe_code = |
| 119 if (code->IsFailure()) return code; | 120 compiler.CompileLoadNonexistent(cache_name, receiver, last); |
| 121 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 122 } |
| 120 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), cache_name)); | 123 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), cache_name)); |
| 121 Object* result = | 124 Object* result; |
| 122 receiver->UpdateMapCodeCache(cache_name, Code::cast(code)); | 125 { MaybeObject* maybe_result = |
| 123 if (result->IsFailure()) return result; | 126 receiver->UpdateMapCodeCache(cache_name, Code::cast(code)); |
| 124 } | 127 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 125 return code; | 128 } |
| 126 } | 129 } |
| 127 | 130 return code; |
| 128 | 131 } |
| 129 Object* StubCache::ComputeLoadField(String* name, | 132 |
| 130 JSObject* receiver, | 133 |
| 131 JSObject* holder, | 134 MaybeObject* StubCache::ComputeLoadField(String* name, |
| 132 int field_index) { | |
| 133 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); | |
| 134 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, FIELD); | |
| 135 Object* code = receiver->map()->FindInCodeCache(name, flags); | |
| 136 if (code->IsUndefined()) { | |
| 137 LoadStubCompiler compiler; | |
| 138 code = compiler.CompileLoadField(receiver, holder, field_index, name); | |
| 139 if (code->IsFailure()) return code; | |
| 140 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); | |
| 141 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | |
| 142 if (result->IsFailure()) return result; | |
| 143 } | |
| 144 return code; | |
| 145 } | |
| 146 | |
| 147 | |
| 148 Object* StubCache::ComputeLoadCallback(String* name, | |
| 149 JSObject* receiver, | |
| 150 JSObject* holder, | |
| 151 AccessorInfo* callback) { | |
| 152 ASSERT(v8::ToCData<Address>(callback->getter()) != 0); | |
| 153 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); | |
| 154 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, CALLBACKS); | |
| 155 Object* code = receiver->map()->FindInCodeCache(name, flags); | |
| 156 if (code->IsUndefined()) { | |
| 157 LoadStubCompiler compiler; | |
| 158 code = compiler.CompileLoadCallback(name, receiver, holder, callback); | |
| 159 if (code->IsFailure()) return code; | |
| 160 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); | |
| 161 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | |
| 162 if (result->IsFailure()) return result; | |
| 163 } | |
| 164 return code; | |
| 165 } | |
| 166 | |
| 167 | |
| 168 Object* StubCache::ComputeLoadConstant(String* name, | |
| 169 JSObject* receiver, | |
| 170 JSObject* holder, | |
| 171 Object* value) { | |
| 172 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); | |
| 173 Code::Flags flags = | |
| 174 Code::ComputeMonomorphicFlags(Code::LOAD_IC, CONSTANT_FUNCTION); | |
| 175 Object* code = receiver->map()->FindInCodeCache(name, flags); | |
| 176 if (code->IsUndefined()) { | |
| 177 LoadStubCompiler compiler; | |
| 178 code = compiler.CompileLoadConstant(receiver, holder, value, name); | |
| 179 if (code->IsFailure()) return code; | |
| 180 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); | |
| 181 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | |
| 182 if (result->IsFailure()) return result; | |
| 183 } | |
| 184 return code; | |
| 185 } | |
| 186 | |
| 187 | |
| 188 Object* StubCache::ComputeLoadInterceptor(String* name, | |
| 189 JSObject* receiver, | |
| 190 JSObject* holder) { | |
| 191 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); | |
| 192 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, INTERCEPTOR); | |
| 193 Object* code = receiver->map()->FindInCodeCache(name, flags); | |
| 194 if (code->IsUndefined()) { | |
| 195 LoadStubCompiler compiler; | |
| 196 code = compiler.CompileLoadInterceptor(receiver, holder, name); | |
| 197 if (code->IsFailure()) return code; | |
| 198 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); | |
| 199 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | |
| 200 if (result->IsFailure()) return result; | |
| 201 } | |
| 202 return code; | |
| 203 } | |
| 204 | |
| 205 | |
| 206 Object* StubCache::ComputeLoadNormal() { | |
| 207 return Builtins::builtin(Builtins::LoadIC_Normal); | |
| 208 } | |
| 209 | |
| 210 | |
| 211 Object* StubCache::ComputeLoadGlobal(String* name, | |
| 212 JSObject* receiver, | |
| 213 GlobalObject* holder, | |
| 214 JSGlobalPropertyCell* cell, | |
| 215 bool is_dont_delete) { | |
| 216 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); | |
| 217 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, NORMAL); | |
| 218 Object* code = receiver->map()->FindInCodeCache(name, flags); | |
| 219 if (code->IsUndefined()) { | |
| 220 LoadStubCompiler compiler; | |
| 221 code = compiler.CompileLoadGlobal(receiver, | |
| 222 holder, | |
| 223 cell, | |
| 224 name, | |
| 225 is_dont_delete); | |
| 226 if (code->IsFailure()) return code; | |
| 227 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); | |
| 228 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | |
| 229 if (result->IsFailure()) return result; | |
| 230 } | |
| 231 return code; | |
| 232 } | |
| 233 | |
| 234 | |
| 235 Object* StubCache::ComputeKeyedLoadField(String* name, | |
| 236 JSObject* receiver, | 135 JSObject* receiver, |
| 237 JSObject* holder, | 136 JSObject* holder, |
| 238 int field_index) { | 137 int field_index) { |
| 239 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); | 138 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); |
| 240 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, FIELD); | 139 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, FIELD); |
| 241 Object* code = receiver->map()->FindInCodeCache(name, flags); | 140 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 242 if (code->IsUndefined()) { | 141 if (code->IsUndefined()) { |
| 243 KeyedLoadStubCompiler compiler; | 142 LoadStubCompiler compiler; |
| 244 code = compiler.CompileLoadField(name, receiver, holder, field_index); | 143 { MaybeObject* maybe_code = |
| 245 if (code->IsFailure()) return code; | 144 compiler.CompileLoadField(receiver, holder, field_index, name); |
| 246 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); | 145 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 247 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 146 } |
| 248 if (result->IsFailure()) return result; | 147 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); |
| 249 } | 148 Object* result; |
| 250 return code; | 149 { MaybeObject* maybe_result = |
| 251 } | 150 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 252 | 151 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 253 | 152 } |
| 254 Object* StubCache::ComputeKeyedLoadConstant(String* name, | 153 } |
| 154 return code; |
| 155 } |
| 156 |
| 157 |
| 158 MaybeObject* StubCache::ComputeLoadCallback(String* name, |
| 159 JSObject* receiver, |
| 160 JSObject* holder, |
| 161 AccessorInfo* callback) { |
| 162 ASSERT(v8::ToCData<Address>(callback->getter()) != 0); |
| 163 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); |
| 164 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, CALLBACKS); |
| 165 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 166 if (code->IsUndefined()) { |
| 167 LoadStubCompiler compiler; |
| 168 { MaybeObject* maybe_code = |
| 169 compiler.CompileLoadCallback(name, receiver, holder, callback); |
| 170 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 171 } |
| 172 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); |
| 173 Object* result; |
| 174 { MaybeObject* maybe_result = |
| 175 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 176 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 177 } |
| 178 } |
| 179 return code; |
| 180 } |
| 181 |
| 182 |
| 183 MaybeObject* StubCache::ComputeLoadConstant(String* name, |
| 255 JSObject* receiver, | 184 JSObject* receiver, |
| 256 JSObject* holder, | 185 JSObject* holder, |
| 257 Object* value) { | 186 Object* value) { |
| 258 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); | 187 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); |
| 259 Code::Flags flags = | 188 Code::Flags flags = |
| 260 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CONSTANT_FUNCTION); | 189 Code::ComputeMonomorphicFlags(Code::LOAD_IC, CONSTANT_FUNCTION); |
| 261 Object* code = receiver->map()->FindInCodeCache(name, flags); | 190 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 262 if (code->IsUndefined()) { | 191 if (code->IsUndefined()) { |
| 263 KeyedLoadStubCompiler compiler; | 192 LoadStubCompiler compiler; |
| 264 code = compiler.CompileLoadConstant(name, receiver, holder, value); | 193 { MaybeObject* maybe_code = |
| 265 if (code->IsFailure()) return code; | 194 compiler.CompileLoadConstant(receiver, holder, value, name); |
| 266 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); | 195 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 267 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 196 } |
| 268 if (result->IsFailure()) return result; | 197 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); |
| 269 } | 198 Object* result; |
| 270 return code; | 199 { MaybeObject* maybe_result = |
| 271 } | 200 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 272 | 201 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 273 | 202 } |
| 274 Object* StubCache::ComputeKeyedLoadInterceptor(String* name, | 203 } |
| 204 return code; |
| 205 } |
| 206 |
| 207 |
| 208 MaybeObject* StubCache::ComputeLoadInterceptor(String* name, |
| 275 JSObject* receiver, | 209 JSObject* receiver, |
| 276 JSObject* holder) { | 210 JSObject* holder) { |
| 277 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); | 211 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); |
| 212 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, INTERCEPTOR); |
| 213 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 214 if (code->IsUndefined()) { |
| 215 LoadStubCompiler compiler; |
| 216 { MaybeObject* maybe_code = |
| 217 compiler.CompileLoadInterceptor(receiver, holder, name); |
| 218 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 219 } |
| 220 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); |
| 221 Object* result; |
| 222 { MaybeObject* maybe_result = |
| 223 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 224 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 225 } |
| 226 } |
| 227 return code; |
| 228 } |
| 229 |
| 230 |
| 231 MaybeObject* StubCache::ComputeLoadNormal() { |
| 232 return Builtins::builtin(Builtins::LoadIC_Normal); |
| 233 } |
| 234 |
| 235 |
| 236 MaybeObject* StubCache::ComputeLoadGlobal(String* name, |
| 237 JSObject* receiver, |
| 238 GlobalObject* holder, |
| 239 JSGlobalPropertyCell* cell, |
| 240 bool is_dont_delete) { |
| 241 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); |
| 242 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, NORMAL); |
| 243 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 244 if (code->IsUndefined()) { |
| 245 LoadStubCompiler compiler; |
| 246 { MaybeObject* maybe_code = compiler.CompileLoadGlobal(receiver, |
| 247 holder, |
| 248 cell, |
| 249 name, |
| 250 is_dont_delete); |
| 251 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 252 } |
| 253 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(code), name)); |
| 254 Object* result; |
| 255 { MaybeObject* maybe_result = |
| 256 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 257 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 258 } |
| 259 } |
| 260 return code; |
| 261 } |
| 262 |
| 263 |
| 264 MaybeObject* StubCache::ComputeKeyedLoadField(String* name, |
| 265 JSObject* receiver, |
| 266 JSObject* holder, |
| 267 int field_index) { |
| 268 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); |
| 269 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, FIELD); |
| 270 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 271 if (code->IsUndefined()) { |
| 272 KeyedLoadStubCompiler compiler; |
| 273 { MaybeObject* maybe_code = |
| 274 compiler.CompileLoadField(name, receiver, holder, field_index); |
| 275 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 276 } |
| 277 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); |
| 278 Object* result; |
| 279 { MaybeObject* maybe_result = |
| 280 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 281 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 282 } |
| 283 } |
| 284 return code; |
| 285 } |
| 286 |
| 287 |
| 288 MaybeObject* StubCache::ComputeKeyedLoadConstant(String* name, |
| 289 JSObject* receiver, |
| 290 JSObject* holder, |
| 291 Object* value) { |
| 292 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); |
| 293 Code::Flags flags = |
| 294 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CONSTANT_FUNCTION); |
| 295 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 296 if (code->IsUndefined()) { |
| 297 KeyedLoadStubCompiler compiler; |
| 298 { MaybeObject* maybe_code = |
| 299 compiler.CompileLoadConstant(name, receiver, holder, value); |
| 300 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 301 } |
| 302 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); |
| 303 Object* result; |
| 304 { MaybeObject* maybe_result = |
| 305 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 306 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 307 } |
| 308 } |
| 309 return code; |
| 310 } |
| 311 |
| 312 |
| 313 MaybeObject* StubCache::ComputeKeyedLoadInterceptor(String* name, |
| 314 JSObject* receiver, |
| 315 JSObject* holder) { |
| 316 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); |
| 278 Code::Flags flags = | 317 Code::Flags flags = |
| 279 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, INTERCEPTOR); | 318 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, INTERCEPTOR); |
| 280 Object* code = receiver->map()->FindInCodeCache(name, flags); | 319 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 281 if (code->IsUndefined()) { | 320 if (code->IsUndefined()) { |
| 282 KeyedLoadStubCompiler compiler; | 321 KeyedLoadStubCompiler compiler; |
| 283 code = compiler.CompileLoadInterceptor(receiver, holder, name); | 322 { MaybeObject* maybe_code = |
| 284 if (code->IsFailure()) return code; | 323 compiler.CompileLoadInterceptor(receiver, holder, name); |
| 285 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); | 324 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 286 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 325 } |
| 287 if (result->IsFailure()) return result; | 326 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); |
| 288 } | 327 Object* result; |
| 289 return code; | 328 { MaybeObject* maybe_result = |
| 290 } | 329 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 291 | 330 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 292 | 331 } |
| 293 Object* StubCache::ComputeKeyedLoadCallback(String* name, | 332 } |
| 294 JSObject* receiver, | 333 return code; |
| 295 JSObject* holder, | 334 } |
| 296 AccessorInfo* callback) { | 335 |
| 336 |
| 337 MaybeObject* StubCache::ComputeKeyedLoadCallback(String* name, |
| 338 JSObject* receiver, |
| 339 JSObject* holder, |
| 340 AccessorInfo* callback) { |
| 297 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); | 341 ASSERT(IC::GetCodeCacheForObject(receiver, holder) == OWN_MAP); |
| 298 Code::Flags flags = | 342 Code::Flags flags = |
| 299 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CALLBACKS); | 343 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CALLBACKS); |
| 300 Object* code = receiver->map()->FindInCodeCache(name, flags); | 344 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 301 if (code->IsUndefined()) { | 345 if (code->IsUndefined()) { |
| 302 KeyedLoadStubCompiler compiler; | 346 KeyedLoadStubCompiler compiler; |
| 303 code = compiler.CompileLoadCallback(name, receiver, holder, callback); | 347 { MaybeObject* maybe_code = |
| 304 if (code->IsFailure()) return code; | 348 compiler.CompileLoadCallback(name, receiver, holder, callback); |
| 305 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); | 349 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 306 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 350 } |
| 307 if (result->IsFailure()) return result; | 351 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); |
| 308 } | 352 Object* result; |
| 309 return code; | 353 { MaybeObject* maybe_result = |
| 310 } | 354 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 311 | 355 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 312 | 356 } |
| 313 | 357 } |
| 314 Object* StubCache::ComputeKeyedLoadArrayLength(String* name, | 358 return code; |
| 315 JSArray* receiver) { | 359 } |
| 360 |
| 361 |
| 362 |
| 363 MaybeObject* StubCache::ComputeKeyedLoadArrayLength(String* name, |
| 364 JSArray* receiver) { |
| 316 Code::Flags flags = | 365 Code::Flags flags = |
| 317 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CALLBACKS); | 366 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CALLBACKS); |
| 318 ASSERT(receiver->IsJSObject()); | 367 ASSERT(receiver->IsJSObject()); |
| 319 Object* code = receiver->map()->FindInCodeCache(name, flags); | 368 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 320 if (code->IsUndefined()) { | 369 if (code->IsUndefined()) { |
| 321 KeyedLoadStubCompiler compiler; | 370 KeyedLoadStubCompiler compiler; |
| 322 code = compiler.CompileLoadArrayLength(name); | 371 { MaybeObject* maybe_code = compiler.CompileLoadArrayLength(name); |
| 323 if (code->IsFailure()) return code; | 372 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 324 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); | 373 } |
| 325 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 374 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); |
| 326 if (result->IsFailure()) return result; | 375 Object* result; |
| 327 } | 376 { MaybeObject* maybe_result = |
| 328 return code; | 377 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 329 } | 378 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 330 | 379 } |
| 331 | 380 } |
| 332 Object* StubCache::ComputeKeyedLoadStringLength(String* name, | 381 return code; |
| 333 String* receiver) { | 382 } |
| 383 |
| 384 |
| 385 MaybeObject* StubCache::ComputeKeyedLoadStringLength(String* name, |
| 386 String* receiver) { |
| 334 Code::Flags flags = | 387 Code::Flags flags = |
| 335 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CALLBACKS); | 388 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CALLBACKS); |
| 336 Map* map = receiver->map(); | 389 Map* map = receiver->map(); |
| 337 Object* code = map->FindInCodeCache(name, flags); | 390 Object* code = map->FindInCodeCache(name, flags); |
| 338 if (code->IsUndefined()) { | 391 if (code->IsUndefined()) { |
| 339 KeyedLoadStubCompiler compiler; | 392 KeyedLoadStubCompiler compiler; |
| 340 code = compiler.CompileLoadStringLength(name); | 393 { MaybeObject* maybe_code = compiler.CompileLoadStringLength(name); |
| 341 if (code->IsFailure()) return code; | 394 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 342 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); | 395 } |
| 343 Object* result = map->UpdateCodeCache(name, Code::cast(code)); | 396 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); |
| 344 if (result->IsFailure()) return result; | 397 Object* result; |
| 345 } | 398 { MaybeObject* maybe_result = map->UpdateCodeCache(name, Code::cast(code)); |
| 346 return code; | 399 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 347 } | 400 } |
| 348 | 401 } |
| 349 | 402 return code; |
| 350 Object* StubCache::ComputeKeyedLoadFunctionPrototype(String* name, | 403 } |
| 351 JSFunction* receiver) { | 404 |
| 405 |
| 406 MaybeObject* StubCache::ComputeKeyedLoadFunctionPrototype( |
| 407 String* name, |
| 408 JSFunction* receiver) { |
| 352 Code::Flags flags = | 409 Code::Flags flags = |
| 353 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CALLBACKS); | 410 Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, CALLBACKS); |
| 354 Object* code = receiver->map()->FindInCodeCache(name, flags); | 411 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 355 if (code->IsUndefined()) { | 412 if (code->IsUndefined()) { |
| 356 KeyedLoadStubCompiler compiler; | 413 KeyedLoadStubCompiler compiler; |
| 357 code = compiler.CompileLoadFunctionPrototype(name); | 414 { MaybeObject* maybe_code = compiler.CompileLoadFunctionPrototype(name); |
| 358 if (code->IsFailure()) return code; | 415 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 359 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); | 416 } |
| 360 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 417 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(code), name)); |
| 361 if (result->IsFailure()) return result; | 418 Object* result; |
| 362 } | 419 { MaybeObject* maybe_result = |
| 363 return code; | 420 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 364 } | 421 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 365 | 422 } |
| 366 | 423 } |
| 367 Object* StubCache::ComputeStoreField(String* name, | 424 return code; |
| 368 JSObject* receiver, | 425 } |
| 369 int field_index, | 426 |
| 370 Map* transition) { | 427 |
| 428 MaybeObject* StubCache::ComputeStoreField(String* name, |
| 429 JSObject* receiver, |
| 430 int field_index, |
| 431 Map* transition) { |
| 371 PropertyType type = (transition == NULL) ? FIELD : MAP_TRANSITION; | 432 PropertyType type = (transition == NULL) ? FIELD : MAP_TRANSITION; |
| 372 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::STORE_IC, type); | 433 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::STORE_IC, type); |
| 373 Object* code = receiver->map()->FindInCodeCache(name, flags); | 434 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 374 if (code->IsUndefined()) { | 435 if (code->IsUndefined()) { |
| 375 StoreStubCompiler compiler; | 436 StoreStubCompiler compiler; |
| 376 code = compiler.CompileStoreField(receiver, field_index, transition, name); | 437 { MaybeObject* maybe_code = |
| 377 if (code->IsFailure()) return code; | 438 compiler.CompileStoreField(receiver, field_index, transition, name); |
| 439 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 440 } |
| 378 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, Code::cast(code), name)); | 441 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, Code::cast(code), name)); |
| 379 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 442 Object* result; |
| 380 if (result->IsFailure()) return result; | 443 { MaybeObject* maybe_result = |
| 381 } | 444 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 382 return code; | 445 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 383 } | 446 } |
| 384 | 447 } |
| 385 | 448 return code; |
| 386 Object* StubCache::ComputeStoreNormal() { | 449 } |
| 450 |
| 451 |
| 452 MaybeObject* StubCache::ComputeStoreNormal() { |
| 387 return Builtins::builtin(Builtins::StoreIC_Normal); | 453 return Builtins::builtin(Builtins::StoreIC_Normal); |
| 388 } | 454 } |
| 389 | 455 |
| 390 | 456 |
| 391 Object* StubCache::ComputeStoreGlobal(String* name, | 457 MaybeObject* StubCache::ComputeStoreGlobal(String* name, |
| 392 GlobalObject* receiver, | 458 GlobalObject* receiver, |
| 393 JSGlobalPropertyCell* cell) { | 459 JSGlobalPropertyCell* cell) { |
| 394 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::STORE_IC, NORMAL); | 460 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::STORE_IC, NORMAL); |
| 395 Object* code = receiver->map()->FindInCodeCache(name, flags); | 461 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 396 if (code->IsUndefined()) { | 462 if (code->IsUndefined()) { |
| 397 StoreStubCompiler compiler; | 463 StoreStubCompiler compiler; |
| 398 code = compiler.CompileStoreGlobal(receiver, cell, name); | 464 { MaybeObject* maybe_code = |
| 399 if (code->IsFailure()) return code; | 465 compiler.CompileStoreGlobal(receiver, cell, name); |
| 466 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 467 } |
| 400 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, Code::cast(code), name)); | 468 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, Code::cast(code), name)); |
| 401 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 469 Object* result; |
| 402 if (result->IsFailure()) return result; | 470 { MaybeObject* maybe_result = |
| 403 } | 471 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 404 return code; | 472 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 405 } | 473 } |
| 406 | 474 } |
| 407 | 475 return code; |
| 408 Object* StubCache::ComputeStoreCallback(String* name, | 476 } |
| 409 JSObject* receiver, | 477 |
| 410 AccessorInfo* callback) { | 478 |
| 479 MaybeObject* StubCache::ComputeStoreCallback(String* name, |
| 480 JSObject* receiver, |
| 481 AccessorInfo* callback) { |
| 411 ASSERT(v8::ToCData<Address>(callback->setter()) != 0); | 482 ASSERT(v8::ToCData<Address>(callback->setter()) != 0); |
| 412 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::STORE_IC, CALLBACKS); | 483 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::STORE_IC, CALLBACKS); |
| 413 Object* code = receiver->map()->FindInCodeCache(name, flags); | 484 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 414 if (code->IsUndefined()) { | 485 if (code->IsUndefined()) { |
| 415 StoreStubCompiler compiler; | 486 StoreStubCompiler compiler; |
| 416 code = compiler.CompileStoreCallback(receiver, callback, name); | 487 { MaybeObject* maybe_code = |
| 417 if (code->IsFailure()) return code; | 488 compiler.CompileStoreCallback(receiver, callback, name); |
| 489 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 490 } |
| 418 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, Code::cast(code), name)); | 491 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, Code::cast(code), name)); |
| 419 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 492 Object* result; |
| 420 if (result->IsFailure()) return result; | 493 { MaybeObject* maybe_result = |
| 421 } | 494 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 422 return code; | 495 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 423 } | 496 } |
| 424 | 497 } |
| 425 | 498 return code; |
| 426 Object* StubCache::ComputeStoreInterceptor(String* name, | 499 } |
| 427 JSObject* receiver) { | 500 |
| 501 |
| 502 MaybeObject* StubCache::ComputeStoreInterceptor(String* name, |
| 503 JSObject* receiver) { |
| 428 Code::Flags flags = | 504 Code::Flags flags = |
| 429 Code::ComputeMonomorphicFlags(Code::STORE_IC, INTERCEPTOR); | 505 Code::ComputeMonomorphicFlags(Code::STORE_IC, INTERCEPTOR); |
| 430 Object* code = receiver->map()->FindInCodeCache(name, flags); | 506 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 431 if (code->IsUndefined()) { | 507 if (code->IsUndefined()) { |
| 432 StoreStubCompiler compiler; | 508 StoreStubCompiler compiler; |
| 433 code = compiler.CompileStoreInterceptor(receiver, name); | 509 { MaybeObject* maybe_code = |
| 434 if (code->IsFailure()) return code; | 510 compiler.CompileStoreInterceptor(receiver, name); |
| 511 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 512 } |
| 435 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, Code::cast(code), name)); | 513 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, Code::cast(code), name)); |
| 436 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 514 Object* result; |
| 437 if (result->IsFailure()) return result; | 515 { MaybeObject* maybe_result = |
| 438 } | 516 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 439 return code; | 517 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 440 } | 518 } |
| 441 | 519 } |
| 442 | 520 return code; |
| 443 Object* StubCache::ComputeKeyedStoreField(String* name, JSObject* receiver, | 521 } |
| 444 int field_index, Map* transition) { | 522 |
| 523 |
| 524 MaybeObject* StubCache::ComputeKeyedStoreField(String* name, |
| 525 JSObject* receiver, |
| 526 int field_index, |
| 527 Map* transition) { |
| 445 PropertyType type = (transition == NULL) ? FIELD : MAP_TRANSITION; | 528 PropertyType type = (transition == NULL) ? FIELD : MAP_TRANSITION; |
| 446 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::KEYED_STORE_IC, type); | 529 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::KEYED_STORE_IC, type); |
| 447 Object* code = receiver->map()->FindInCodeCache(name, flags); | 530 Object* code = receiver->map()->FindInCodeCache(name, flags); |
| 448 if (code->IsUndefined()) { | 531 if (code->IsUndefined()) { |
| 449 KeyedStoreStubCompiler compiler; | 532 KeyedStoreStubCompiler compiler; |
| 450 code = compiler.CompileStoreField(receiver, field_index, transition, name); | 533 { MaybeObject* maybe_code = |
| 451 if (code->IsFailure()) return code; | 534 compiler.CompileStoreField(receiver, field_index, transition, name); |
| 535 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 536 } |
| 452 PROFILE(CodeCreateEvent( | 537 PROFILE(CodeCreateEvent( |
| 453 Logger::KEYED_STORE_IC_TAG, Code::cast(code), name)); | 538 Logger::KEYED_STORE_IC_TAG, Code::cast(code), name)); |
| 454 Object* result = receiver->UpdateMapCodeCache(name, Code::cast(code)); | 539 Object* result; |
| 455 if (result->IsFailure()) return result; | 540 { MaybeObject* maybe_result = |
| 456 } | 541 receiver->UpdateMapCodeCache(name, Code::cast(code)); |
| 457 return code; | 542 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 458 } | 543 } |
| 544 } |
| 545 return code; |
| 546 } |
| 459 | 547 |
| 460 #define CALL_LOGGER_TAG(kind, type) \ | 548 #define CALL_LOGGER_TAG(kind, type) \ |
| 461 (kind == Code::CALL_IC ? Logger::type : Logger::KEYED_##type) | 549 (kind == Code::CALL_IC ? Logger::type : Logger::KEYED_##type) |
| 462 | 550 |
| 463 Object* StubCache::ComputeCallConstant(int argc, | 551 MaybeObject* StubCache::ComputeCallConstant(int argc, |
| 464 InLoopFlag in_loop, | 552 InLoopFlag in_loop, |
| 465 Code::Kind kind, | 553 Code::Kind kind, |
| 466 String* name, | 554 String* name, |
| 467 Object* object, | 555 Object* object, |
| 468 JSObject* holder, | 556 JSObject* holder, |
| 469 JSFunction* function) { | 557 JSFunction* function) { |
| 470 // Compute the check type and the map. | 558 // Compute the check type and the map. |
| 471 InlineCacheHolderFlag cache_holder = | 559 InlineCacheHolderFlag cache_holder = |
| 472 IC::GetCodeCacheForObject(object, holder); | 560 IC::GetCodeCacheForObject(object, holder); |
| 473 JSObject* map_holder = IC::GetCodeCacheHolder(object, cache_holder); | 561 JSObject* map_holder = IC::GetCodeCacheHolder(object, cache_holder); |
| 474 | 562 |
| 475 // Compute check type based on receiver/holder. | 563 // Compute check type based on receiver/holder. |
| 476 StubCompiler::CheckType check = StubCompiler::RECEIVER_MAP_CHECK; | 564 StubCompiler::CheckType check = StubCompiler::RECEIVER_MAP_CHECK; |
| 477 if (object->IsString()) { | 565 if (object->IsString()) { |
| 478 check = StubCompiler::STRING_CHECK; | 566 check = StubCompiler::STRING_CHECK; |
| 479 } else if (object->IsNumber()) { | 567 } else if (object->IsNumber()) { |
| (...skipping 10 matching lines...) Expand all Loading... |
| 490 argc); | 578 argc); |
| 491 Object* code = map_holder->map()->FindInCodeCache(name, flags); | 579 Object* code = map_holder->map()->FindInCodeCache(name, flags); |
| 492 if (code->IsUndefined()) { | 580 if (code->IsUndefined()) { |
| 493 // If the function hasn't been compiled yet, we cannot do it now | 581 // If the function hasn't been compiled yet, we cannot do it now |
| 494 // because it may cause GC. To avoid this issue, we return an | 582 // because it may cause GC. To avoid this issue, we return an |
| 495 // internal error which will make sure we do not update any | 583 // internal error which will make sure we do not update any |
| 496 // caches. | 584 // caches. |
| 497 if (!function->is_compiled()) return Failure::InternalError(); | 585 if (!function->is_compiled()) return Failure::InternalError(); |
| 498 // Compile the stub - only create stubs for fully compiled functions. | 586 // Compile the stub - only create stubs for fully compiled functions. |
| 499 CallStubCompiler compiler(argc, in_loop, kind, cache_holder); | 587 CallStubCompiler compiler(argc, in_loop, kind, cache_holder); |
| 500 code = compiler.CompileCallConstant(object, holder, function, name, check); | 588 { MaybeObject* maybe_code = |
| 501 if (code->IsFailure()) return code; | 589 compiler.CompileCallConstant(object, holder, function, name, check); |
| 590 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 591 } |
| 502 ASSERT_EQ(flags, Code::cast(code)->flags()); | 592 ASSERT_EQ(flags, Code::cast(code)->flags()); |
| 503 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_IC_TAG), | 593 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_IC_TAG), |
| 504 Code::cast(code), name)); | 594 Code::cast(code), name)); |
| 505 Object* result = map_holder->UpdateMapCodeCache(name, Code::cast(code)); | 595 Object* result; |
| 506 if (result->IsFailure()) return result; | 596 { MaybeObject* maybe_result = |
| 597 map_holder->UpdateMapCodeCache(name, Code::cast(code)); |
| 598 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 599 } |
| 507 } | 600 } |
| 508 return code; | 601 return code; |
| 509 } | 602 } |
| 510 | 603 |
| 511 | 604 |
| 512 Object* StubCache::ComputeCallField(int argc, | 605 MaybeObject* StubCache::ComputeCallField(int argc, |
| 513 InLoopFlag in_loop, | 606 InLoopFlag in_loop, |
| 514 Code::Kind kind, | 607 Code::Kind kind, |
| 515 String* name, | 608 String* name, |
| 516 Object* object, | 609 Object* object, |
| 517 JSObject* holder, | 610 JSObject* holder, |
| 518 int index) { | 611 int index) { |
| 519 // Compute the check type and the map. | 612 // Compute the check type and the map. |
| 520 InlineCacheHolderFlag cache_holder = | 613 InlineCacheHolderFlag cache_holder = |
| 521 IC::GetCodeCacheForObject(object, holder); | 614 IC::GetCodeCacheForObject(object, holder); |
| 522 JSObject* map_holder = IC::GetCodeCacheHolder(object, cache_holder); | 615 JSObject* map_holder = IC::GetCodeCacheHolder(object, cache_holder); |
| 523 | 616 |
| 524 // TODO(1233596): We cannot do receiver map check for non-JS objects | 617 // TODO(1233596): We cannot do receiver map check for non-JS objects |
| 525 // because they may be represented as immediates without a | 618 // because they may be represented as immediates without a |
| 526 // map. Instead, we check against the map in the holder. | 619 // map. Instead, we check against the map in the holder. |
| 527 if (object->IsNumber() || object->IsBoolean() || object->IsString()) { | 620 if (object->IsNumber() || object->IsBoolean() || object->IsString()) { |
| 528 object = holder; | 621 object = holder; |
| 529 } | 622 } |
| 530 | 623 |
| 531 Code::Flags flags = Code::ComputeMonomorphicFlags(kind, | 624 Code::Flags flags = Code::ComputeMonomorphicFlags(kind, |
| 532 FIELD, | 625 FIELD, |
| 533 cache_holder, | 626 cache_holder, |
| 534 in_loop, | 627 in_loop, |
| 535 argc); | 628 argc); |
| 536 Object* code = map_holder->map()->FindInCodeCache(name, flags); | 629 Object* code = map_holder->map()->FindInCodeCache(name, flags); |
| 537 if (code->IsUndefined()) { | 630 if (code->IsUndefined()) { |
| 538 CallStubCompiler compiler(argc, in_loop, kind, cache_holder); | 631 CallStubCompiler compiler(argc, in_loop, kind, cache_holder); |
| 539 code = compiler.CompileCallField(JSObject::cast(object), | 632 { MaybeObject* maybe_code = |
| 540 holder, | 633 compiler.CompileCallField(JSObject::cast(object), |
| 541 index, | 634 holder, |
| 542 name); | 635 index, |
| 543 if (code->IsFailure()) return code; | 636 name); |
| 637 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 638 } |
| 544 ASSERT_EQ(flags, Code::cast(code)->flags()); | 639 ASSERT_EQ(flags, Code::cast(code)->flags()); |
| 545 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_IC_TAG), | 640 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_IC_TAG), |
| 546 Code::cast(code), name)); | 641 Code::cast(code), name)); |
| 547 Object* result = map_holder->UpdateMapCodeCache(name, Code::cast(code)); | 642 Object* result; |
| 548 if (result->IsFailure()) return result; | 643 { MaybeObject* maybe_result = |
| 644 map_holder->UpdateMapCodeCache(name, Code::cast(code)); |
| 645 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 646 } |
| 549 } | 647 } |
| 550 return code; | 648 return code; |
| 551 } | 649 } |
| 552 | 650 |
| 553 | 651 |
| 554 Object* StubCache::ComputeCallInterceptor(int argc, | 652 MaybeObject* StubCache::ComputeCallInterceptor(int argc, |
| 555 Code::Kind kind, | 653 Code::Kind kind, |
| 556 String* name, | 654 String* name, |
| 557 Object* object, | 655 Object* object, |
| 558 JSObject* holder) { | 656 JSObject* holder) { |
| 559 // Compute the check type and the map. | 657 // Compute the check type and the map. |
| 560 InlineCacheHolderFlag cache_holder = | 658 InlineCacheHolderFlag cache_holder = |
| 561 IC::GetCodeCacheForObject(object, holder); | 659 IC::GetCodeCacheForObject(object, holder); |
| 562 JSObject* map_holder = IC::GetCodeCacheHolder(object, cache_holder); | 660 JSObject* map_holder = IC::GetCodeCacheHolder(object, cache_holder); |
| 563 | 661 |
| 564 // TODO(1233596): We cannot do receiver map check for non-JS objects | 662 // TODO(1233596): We cannot do receiver map check for non-JS objects |
| 565 // because they may be represented as immediates without a | 663 // because they may be represented as immediates without a |
| 566 // map. Instead, we check against the map in the holder. | 664 // map. Instead, we check against the map in the holder. |
| 567 if (object->IsNumber() || object->IsBoolean() || object->IsString()) { | 665 if (object->IsNumber() || object->IsBoolean() || object->IsString()) { |
| 568 object = holder; | 666 object = holder; |
| 569 } | 667 } |
| 570 | 668 |
| 571 Code::Flags flags = | 669 Code::Flags flags = |
| 572 Code::ComputeMonomorphicFlags(kind, | 670 Code::ComputeMonomorphicFlags(kind, |
| 573 INTERCEPTOR, | 671 INTERCEPTOR, |
| 574 cache_holder, | 672 cache_holder, |
| 575 NOT_IN_LOOP, | 673 NOT_IN_LOOP, |
| 576 argc); | 674 argc); |
| 577 Object* code = map_holder->map()->FindInCodeCache(name, flags); | 675 Object* code = map_holder->map()->FindInCodeCache(name, flags); |
| 578 if (code->IsUndefined()) { | 676 if (code->IsUndefined()) { |
| 579 CallStubCompiler compiler(argc, NOT_IN_LOOP, kind, cache_holder); | 677 CallStubCompiler compiler(argc, NOT_IN_LOOP, kind, cache_holder); |
| 580 code = compiler.CompileCallInterceptor(JSObject::cast(object), | 678 { MaybeObject* maybe_code = |
| 581 holder, | 679 compiler.CompileCallInterceptor(JSObject::cast(object), holder, name); |
| 582 name); | 680 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 583 if (code->IsFailure()) return code; | 681 } |
| 584 ASSERT_EQ(flags, Code::cast(code)->flags()); | 682 ASSERT_EQ(flags, Code::cast(code)->flags()); |
| 585 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_IC_TAG), | 683 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_IC_TAG), |
| 586 Code::cast(code), name)); | 684 Code::cast(code), name)); |
| 587 Object* result = map_holder->UpdateMapCodeCache(name, Code::cast(code)); | 685 Object* result; |
| 588 if (result->IsFailure()) return result; | 686 { MaybeObject* maybe_result = |
| 687 map_holder->UpdateMapCodeCache(name, Code::cast(code)); |
| 688 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 689 } |
| 589 } | 690 } |
| 590 return code; | 691 return code; |
| 591 } | 692 } |
| 592 | 693 |
| 593 | 694 |
| 594 Object* StubCache::ComputeCallNormal(int argc, | 695 MaybeObject* StubCache::ComputeCallNormal(int argc, |
| 595 InLoopFlag in_loop, | 696 InLoopFlag in_loop, |
| 596 Code::Kind kind, | 697 Code::Kind kind, |
| 597 String* name, | 698 String* name, |
| 598 JSObject* receiver) { | 699 JSObject* receiver) { |
| 599 Object* code = ComputeCallNormal(argc, in_loop, kind); | 700 Object* code; |
| 600 if (code->IsFailure()) return code; | 701 { MaybeObject* maybe_code = ComputeCallNormal(argc, in_loop, kind); |
| 702 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 703 } |
| 601 return code; | 704 return code; |
| 602 } | 705 } |
| 603 | 706 |
| 604 | 707 |
| 605 Object* StubCache::ComputeCallGlobal(int argc, | 708 MaybeObject* StubCache::ComputeCallGlobal(int argc, |
| 606 InLoopFlag in_loop, | 709 InLoopFlag in_loop, |
| 607 Code::Kind kind, | 710 Code::Kind kind, |
| 608 String* name, | 711 String* name, |
| 609 JSObject* receiver, | 712 JSObject* receiver, |
| 610 GlobalObject* holder, | 713 GlobalObject* holder, |
| 611 JSGlobalPropertyCell* cell, | 714 JSGlobalPropertyCell* cell, |
| 612 JSFunction* function) { | 715 JSFunction* function) { |
| 613 InlineCacheHolderFlag cache_holder = | 716 InlineCacheHolderFlag cache_holder = |
| 614 IC::GetCodeCacheForObject(receiver, holder); | 717 IC::GetCodeCacheForObject(receiver, holder); |
| 615 JSObject* map_holder = IC::GetCodeCacheHolder(receiver, cache_holder); | 718 JSObject* map_holder = IC::GetCodeCacheHolder(receiver, cache_holder); |
| 616 Code::Flags flags = | 719 Code::Flags flags = |
| 617 Code::ComputeMonomorphicFlags(kind, | 720 Code::ComputeMonomorphicFlags(kind, |
| 618 NORMAL, | 721 NORMAL, |
| 619 cache_holder, | 722 cache_holder, |
| 620 in_loop, | 723 in_loop, |
| 621 argc); | 724 argc); |
| 622 Object* code = map_holder->map()->FindInCodeCache(name, flags); | 725 Object* code = map_holder->map()->FindInCodeCache(name, flags); |
| 623 if (code->IsUndefined()) { | 726 if (code->IsUndefined()) { |
| 624 // If the function hasn't been compiled yet, we cannot do it now | 727 // If the function hasn't been compiled yet, we cannot do it now |
| 625 // because it may cause GC. To avoid this issue, we return an | 728 // because it may cause GC. To avoid this issue, we return an |
| 626 // internal error which will make sure we do not update any | 729 // internal error which will make sure we do not update any |
| 627 // caches. | 730 // caches. |
| 628 if (!function->is_compiled()) return Failure::InternalError(); | 731 if (!function->is_compiled()) return Failure::InternalError(); |
| 629 CallStubCompiler compiler(argc, in_loop, kind, cache_holder); | 732 CallStubCompiler compiler(argc, in_loop, kind, cache_holder); |
| 630 code = compiler.CompileCallGlobal(receiver, holder, cell, function, name); | 733 { MaybeObject* maybe_code = |
| 631 if (code->IsFailure()) return code; | 734 compiler.CompileCallGlobal(receiver, holder, cell, function, name); |
| 735 if (!maybe_code->ToObject(&code)) return maybe_code; |
| 736 } |
| 632 ASSERT_EQ(flags, Code::cast(code)->flags()); | 737 ASSERT_EQ(flags, Code::cast(code)->flags()); |
| 633 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_IC_TAG), | 738 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_IC_TAG), |
| 634 Code::cast(code), name)); | 739 Code::cast(code), name)); |
| 635 Object* result = map_holder->UpdateMapCodeCache(name, Code::cast(code)); | 740 Object* result; |
| 636 if (result->IsFailure()) return result; | 741 { MaybeObject* maybe_result = |
| 742 map_holder->UpdateMapCodeCache(name, Code::cast(code)); |
| 743 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 744 } |
| 637 } | 745 } |
| 638 return code; | 746 return code; |
| 639 } | 747 } |
| 640 | 748 |
| 641 | 749 |
| 642 static Object* GetProbeValue(Code::Flags flags) { | 750 static Object* GetProbeValue(Code::Flags flags) { |
| 643 // Use raw_unchecked... so we don't get assert failures during GC. | 751 // Use raw_unchecked... so we don't get assert failures during GC. |
| 644 NumberDictionary* dictionary = Heap::raw_unchecked_non_monomorphic_cache(); | 752 NumberDictionary* dictionary = Heap::raw_unchecked_non_monomorphic_cache(); |
| 645 int entry = dictionary->FindEntry(flags); | 753 int entry = dictionary->FindEntry(flags); |
| 646 if (entry != -1) return dictionary->ValueAt(entry); | 754 if (entry != -1) return dictionary->ValueAt(entry); |
| 647 return Heap::raw_unchecked_undefined_value(); | 755 return Heap::raw_unchecked_undefined_value(); |
| 648 } | 756 } |
| 649 | 757 |
| 650 | 758 |
| 651 static Object* ProbeCache(Code::Flags flags) { | 759 MUST_USE_RESULT static MaybeObject* ProbeCache(Code::Flags flags) { |
| 652 Object* probe = GetProbeValue(flags); | 760 Object* probe = GetProbeValue(flags); |
| 653 if (probe != Heap::undefined_value()) return probe; | 761 if (probe != Heap::undefined_value()) return probe; |
| 654 // Seed the cache with an undefined value to make sure that any | 762 // Seed the cache with an undefined value to make sure that any |
| 655 // generated code object can always be inserted into the cache | 763 // generated code object can always be inserted into the cache |
| 656 // without causing allocation failures. | 764 // without causing allocation failures. |
| 657 Object* result = | 765 Object* result; |
| 658 Heap::non_monomorphic_cache()->AtNumberPut(flags, | 766 { MaybeObject* maybe_result = |
| 659 Heap::undefined_value()); | 767 Heap::non_monomorphic_cache()->AtNumberPut(flags, |
| 660 if (result->IsFailure()) return result; | 768 Heap::undefined_value()); |
| 769 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 770 } |
| 661 Heap::public_set_non_monomorphic_cache(NumberDictionary::cast(result)); | 771 Heap::public_set_non_monomorphic_cache(NumberDictionary::cast(result)); |
| 662 return probe; | 772 return probe; |
| 663 } | 773 } |
| 664 | 774 |
| 665 | 775 |
| 666 static Object* FillCache(Object* code) { | 776 static MaybeObject* FillCache(MaybeObject* maybe_code) { |
| 667 if (code->IsCode()) { | 777 Object* code; |
| 668 int entry = | 778 if (maybe_code->ToObject(&code)) { |
| 669 Heap::non_monomorphic_cache()->FindEntry( | 779 if (code->IsCode()) { |
| 670 Code::cast(code)->flags()); | 780 int entry = |
| 671 // The entry must be present see comment in ProbeCache. | 781 Heap::non_monomorphic_cache()->FindEntry( |
| 672 ASSERT(entry != -1); | 782 Code::cast(code)->flags()); |
| 673 ASSERT(Heap::non_monomorphic_cache()->ValueAt(entry) == | 783 // The entry must be present see comment in ProbeCache. |
| 674 Heap::undefined_value()); | 784 ASSERT(entry != -1); |
| 675 Heap::non_monomorphic_cache()->ValueAtPut(entry, code); | 785 ASSERT(Heap::non_monomorphic_cache()->ValueAt(entry) == |
| 676 CHECK(GetProbeValue(Code::cast(code)->flags()) == code); | 786 Heap::undefined_value()); |
| 787 Heap::non_monomorphic_cache()->ValueAtPut(entry, code); |
| 788 CHECK(GetProbeValue(Code::cast(code)->flags()) == code); |
| 789 } |
| 677 } | 790 } |
| 678 return code; | 791 return maybe_code; |
| 679 } | 792 } |
| 680 | 793 |
| 681 | 794 |
| 682 Code* StubCache::FindCallInitialize(int argc, | 795 Code* StubCache::FindCallInitialize(int argc, |
| 683 InLoopFlag in_loop, | 796 InLoopFlag in_loop, |
| 684 Code::Kind kind) { | 797 Code::Kind kind) { |
| 685 Code::Flags flags = | 798 Code::Flags flags = |
| 686 Code::ComputeFlags(kind, in_loop, UNINITIALIZED, NORMAL, argc); | 799 Code::ComputeFlags(kind, in_loop, UNINITIALIZED, NORMAL, argc); |
| 687 Object* result = ProbeCache(flags); | 800 Object* result = ProbeCache(flags)->ToObjectUnchecked(); |
| 688 ASSERT(!result->IsUndefined()); | 801 ASSERT(!result->IsUndefined()); |
| 689 // This might be called during the marking phase of the collector | 802 // This might be called during the marking phase of the collector |
| 690 // hence the unchecked cast. | 803 // hence the unchecked cast. |
| 691 return reinterpret_cast<Code*>(result); | 804 return reinterpret_cast<Code*>(result); |
| 692 } | 805 } |
| 693 | 806 |
| 694 | 807 |
| 695 Object* StubCache::ComputeCallInitialize(int argc, | 808 MaybeObject* StubCache::ComputeCallInitialize(int argc, |
| 696 InLoopFlag in_loop, | 809 InLoopFlag in_loop, |
| 697 Code::Kind kind) { | 810 Code::Kind kind) { |
| 698 Code::Flags flags = | 811 Code::Flags flags = |
| 699 Code::ComputeFlags(kind, in_loop, UNINITIALIZED, NORMAL, argc); | 812 Code::ComputeFlags(kind, in_loop, UNINITIALIZED, NORMAL, argc); |
| 700 Object* probe = ProbeCache(flags); | 813 Object* probe; |
| 814 { MaybeObject* maybe_probe = ProbeCache(flags); |
| 815 if (!maybe_probe->ToObject(&probe)) return maybe_probe; |
| 816 } |
| 701 if (!probe->IsUndefined()) return probe; | 817 if (!probe->IsUndefined()) return probe; |
| 702 StubCompiler compiler; | 818 StubCompiler compiler; |
| 703 return FillCache(compiler.CompileCallInitialize(flags)); | 819 return FillCache(compiler.CompileCallInitialize(flags)); |
| 704 } | 820 } |
| 705 | 821 |
| 706 | 822 |
| 707 Object* StubCache::ComputeCallPreMonomorphic(int argc, | 823 MaybeObject* StubCache::ComputeCallPreMonomorphic(int argc, |
| 708 InLoopFlag in_loop, | 824 InLoopFlag in_loop, |
| 709 Code::Kind kind) { | 825 Code::Kind kind) { |
| 710 Code::Flags flags = | 826 Code::Flags flags = |
| 711 Code::ComputeFlags(kind, in_loop, PREMONOMORPHIC, NORMAL, argc); | 827 Code::ComputeFlags(kind, in_loop, PREMONOMORPHIC, NORMAL, argc); |
| 712 Object* probe = ProbeCache(flags); | 828 Object* probe; |
| 829 { MaybeObject* maybe_probe = ProbeCache(flags); |
| 830 if (!maybe_probe->ToObject(&probe)) return maybe_probe; |
| 831 } |
| 713 if (!probe->IsUndefined()) return probe; | 832 if (!probe->IsUndefined()) return probe; |
| 714 StubCompiler compiler; | 833 StubCompiler compiler; |
| 715 return FillCache(compiler.CompileCallPreMonomorphic(flags)); | 834 return FillCache(compiler.CompileCallPreMonomorphic(flags)); |
| 716 } | 835 } |
| 717 | 836 |
| 718 | 837 |
| 719 Object* StubCache::ComputeCallNormal(int argc, | 838 MaybeObject* StubCache::ComputeCallNormal(int argc, |
| 720 InLoopFlag in_loop, | 839 InLoopFlag in_loop, |
| 721 Code::Kind kind) { | 840 Code::Kind kind) { |
| 722 Code::Flags flags = | 841 Code::Flags flags = |
| 723 Code::ComputeFlags(kind, in_loop, MONOMORPHIC, NORMAL, argc); | 842 Code::ComputeFlags(kind, in_loop, MONOMORPHIC, NORMAL, argc); |
| 724 Object* probe = ProbeCache(flags); | 843 Object* probe; |
| 844 { MaybeObject* maybe_probe = ProbeCache(flags); |
| 845 if (!maybe_probe->ToObject(&probe)) return maybe_probe; |
| 846 } |
| 725 if (!probe->IsUndefined()) return probe; | 847 if (!probe->IsUndefined()) return probe; |
| 726 StubCompiler compiler; | 848 StubCompiler compiler; |
| 727 return FillCache(compiler.CompileCallNormal(flags)); | 849 return FillCache(compiler.CompileCallNormal(flags)); |
| 728 } | 850 } |
| 729 | 851 |
| 730 | 852 |
| 731 Object* StubCache::ComputeCallMegamorphic(int argc, | 853 MaybeObject* StubCache::ComputeCallMegamorphic(int argc, |
| 732 InLoopFlag in_loop, | 854 InLoopFlag in_loop, |
| 733 Code::Kind kind) { | 855 Code::Kind kind) { |
| 734 Code::Flags flags = | 856 Code::Flags flags = |
| 735 Code::ComputeFlags(kind, in_loop, MEGAMORPHIC, NORMAL, argc); | 857 Code::ComputeFlags(kind, in_loop, MEGAMORPHIC, NORMAL, argc); |
| 736 Object* probe = ProbeCache(flags); | 858 Object* probe; |
| 859 { MaybeObject* maybe_probe = ProbeCache(flags); |
| 860 if (!maybe_probe->ToObject(&probe)) return maybe_probe; |
| 861 } |
| 737 if (!probe->IsUndefined()) return probe; | 862 if (!probe->IsUndefined()) return probe; |
| 738 StubCompiler compiler; | 863 StubCompiler compiler; |
| 739 return FillCache(compiler.CompileCallMegamorphic(flags)); | 864 return FillCache(compiler.CompileCallMegamorphic(flags)); |
| 740 } | 865 } |
| 741 | 866 |
| 742 | 867 |
| 743 Object* StubCache::ComputeCallMiss(int argc, Code::Kind kind) { | 868 MaybeObject* StubCache::ComputeCallMiss(int argc, Code::Kind kind) { |
| 744 // MONOMORPHIC_PROTOTYPE_FAILURE state is used to make sure that miss stubs | 869 // MONOMORPHIC_PROTOTYPE_FAILURE state is used to make sure that miss stubs |
| 745 // and monomorphic stubs are not mixed up together in the stub cache. | 870 // and monomorphic stubs are not mixed up together in the stub cache. |
| 746 Code::Flags flags = Code::ComputeFlags( | 871 Code::Flags flags = Code::ComputeFlags( |
| 747 kind, NOT_IN_LOOP, MONOMORPHIC_PROTOTYPE_FAILURE, NORMAL, argc); | 872 kind, NOT_IN_LOOP, MONOMORPHIC_PROTOTYPE_FAILURE, NORMAL, argc); |
| 748 Object* probe = ProbeCache(flags); | 873 Object* probe; |
| 874 { MaybeObject* maybe_probe = ProbeCache(flags); |
| 875 if (!maybe_probe->ToObject(&probe)) return maybe_probe; |
| 876 } |
| 749 if (!probe->IsUndefined()) return probe; | 877 if (!probe->IsUndefined()) return probe; |
| 750 StubCompiler compiler; | 878 StubCompiler compiler; |
| 751 return FillCache(compiler.CompileCallMiss(flags)); | 879 return FillCache(compiler.CompileCallMiss(flags)); |
| 752 } | 880 } |
| 753 | 881 |
| 754 | 882 |
| 755 #ifdef ENABLE_DEBUGGER_SUPPORT | 883 #ifdef ENABLE_DEBUGGER_SUPPORT |
| 756 Object* StubCache::ComputeCallDebugBreak(int argc, Code::Kind kind) { | 884 MaybeObject* StubCache::ComputeCallDebugBreak(int argc, Code::Kind kind) { |
| 757 Code::Flags flags = | 885 Code::Flags flags = |
| 758 Code::ComputeFlags(kind, NOT_IN_LOOP, DEBUG_BREAK, NORMAL, argc); | 886 Code::ComputeFlags(kind, NOT_IN_LOOP, DEBUG_BREAK, NORMAL, argc); |
| 759 Object* probe = ProbeCache(flags); | 887 Object* probe; |
| 888 { MaybeObject* maybe_probe = ProbeCache(flags); |
| 889 if (!maybe_probe->ToObject(&probe)) return maybe_probe; |
| 890 } |
| 760 if (!probe->IsUndefined()) return probe; | 891 if (!probe->IsUndefined()) return probe; |
| 761 StubCompiler compiler; | 892 StubCompiler compiler; |
| 762 return FillCache(compiler.CompileCallDebugBreak(flags)); | 893 return FillCache(compiler.CompileCallDebugBreak(flags)); |
| 763 } | 894 } |
| 764 | 895 |
| 765 | 896 |
| 766 Object* StubCache::ComputeCallDebugPrepareStepIn(int argc, Code::Kind kind) { | 897 MaybeObject* StubCache::ComputeCallDebugPrepareStepIn(int argc, |
| 898 Code::Kind kind) { |
| 767 Code::Flags flags = | 899 Code::Flags flags = |
| 768 Code::ComputeFlags(kind, | 900 Code::ComputeFlags(kind, |
| 769 NOT_IN_LOOP, | 901 NOT_IN_LOOP, |
| 770 DEBUG_PREPARE_STEP_IN, | 902 DEBUG_PREPARE_STEP_IN, |
| 771 NORMAL, | 903 NORMAL, |
| 772 argc); | 904 argc); |
| 773 Object* probe = ProbeCache(flags); | 905 Object* probe; |
| 906 { MaybeObject* maybe_probe = ProbeCache(flags); |
| 907 if (!maybe_probe->ToObject(&probe)) return maybe_probe; |
| 908 } |
| 774 if (!probe->IsUndefined()) return probe; | 909 if (!probe->IsUndefined()) return probe; |
| 775 StubCompiler compiler; | 910 StubCompiler compiler; |
| 776 return FillCache(compiler.CompileCallDebugPrepareStepIn(flags)); | 911 return FillCache(compiler.CompileCallDebugPrepareStepIn(flags)); |
| 777 } | 912 } |
| 778 #endif | 913 #endif |
| 779 | 914 |
| 780 | 915 |
| 781 void StubCache::Clear() { | 916 void StubCache::Clear() { |
| 782 for (int i = 0; i < kPrimaryTableSize; i++) { | 917 for (int i = 0; i < kPrimaryTableSize; i++) { |
| 783 primary_[i].key = Heap::empty_string(); | 918 primary_[i].key = Heap::empty_string(); |
| 784 primary_[i].value = Builtins::builtin(Builtins::Illegal); | 919 primary_[i].value = Builtins::builtin(Builtins::Illegal); |
| 785 } | 920 } |
| 786 for (int j = 0; j < kSecondaryTableSize; j++) { | 921 for (int j = 0; j < kSecondaryTableSize; j++) { |
| 787 secondary_[j].key = Heap::empty_string(); | 922 secondary_[j].key = Heap::empty_string(); |
| 788 secondary_[j].value = Builtins::builtin(Builtins::Illegal); | 923 secondary_[j].value = Builtins::builtin(Builtins::Illegal); |
| 789 } | 924 } |
| 790 } | 925 } |
| 791 | 926 |
| 792 | 927 |
| 793 // ------------------------------------------------------------------------ | 928 // ------------------------------------------------------------------------ |
| 794 // StubCompiler implementation. | 929 // StubCompiler implementation. |
| 795 | 930 |
| 796 | 931 |
| 797 Object* LoadCallbackProperty(Arguments args) { | 932 MaybeObject* LoadCallbackProperty(Arguments args) { |
| 798 ASSERT(args[0]->IsJSObject()); | 933 ASSERT(args[0]->IsJSObject()); |
| 799 ASSERT(args[1]->IsJSObject()); | 934 ASSERT(args[1]->IsJSObject()); |
| 800 AccessorInfo* callback = AccessorInfo::cast(args[2]); | 935 AccessorInfo* callback = AccessorInfo::cast(args[2]); |
| 801 Address getter_address = v8::ToCData<Address>(callback->getter()); | 936 Address getter_address = v8::ToCData<Address>(callback->getter()); |
| 802 v8::AccessorGetter fun = FUNCTION_CAST<v8::AccessorGetter>(getter_address); | 937 v8::AccessorGetter fun = FUNCTION_CAST<v8::AccessorGetter>(getter_address); |
| 803 ASSERT(fun != NULL); | 938 ASSERT(fun != NULL); |
| 804 CustomArguments custom_args(callback->data(), | 939 CustomArguments custom_args(callback->data(), |
| 805 JSObject::cast(args[0]), | 940 JSObject::cast(args[0]), |
| 806 JSObject::cast(args[1])); | 941 JSObject::cast(args[1])); |
| 807 v8::AccessorInfo info(custom_args.end()); | 942 v8::AccessorInfo info(custom_args.end()); |
| 808 HandleScope scope; | 943 HandleScope scope; |
| 809 v8::Handle<v8::Value> result; | 944 v8::Handle<v8::Value> result; |
| 810 { | 945 { |
| 811 // Leaving JavaScript. | 946 // Leaving JavaScript. |
| 812 VMState state(EXTERNAL); | 947 VMState state(EXTERNAL); |
| 813 #ifdef ENABLE_LOGGING_AND_PROFILING | 948 #ifdef ENABLE_LOGGING_AND_PROFILING |
| 814 state.set_external_callback(getter_address); | 949 state.set_external_callback(getter_address); |
| 815 #endif | 950 #endif |
| 816 result = fun(v8::Utils::ToLocal(args.at<String>(4)), info); | 951 result = fun(v8::Utils::ToLocal(args.at<String>(4)), info); |
| 817 } | 952 } |
| 818 RETURN_IF_SCHEDULED_EXCEPTION(); | 953 RETURN_IF_SCHEDULED_EXCEPTION(); |
| 819 if (result.IsEmpty()) return Heap::undefined_value(); | 954 if (result.IsEmpty()) return Heap::undefined_value(); |
| 820 return *v8::Utils::OpenHandle(*result); | 955 return *v8::Utils::OpenHandle(*result); |
| 821 } | 956 } |
| 822 | 957 |
| 823 | 958 |
| 824 Object* StoreCallbackProperty(Arguments args) { | 959 MaybeObject* StoreCallbackProperty(Arguments args) { |
| 825 JSObject* recv = JSObject::cast(args[0]); | 960 JSObject* recv = JSObject::cast(args[0]); |
| 826 AccessorInfo* callback = AccessorInfo::cast(args[1]); | 961 AccessorInfo* callback = AccessorInfo::cast(args[1]); |
| 827 Address setter_address = v8::ToCData<Address>(callback->setter()); | 962 Address setter_address = v8::ToCData<Address>(callback->setter()); |
| 828 v8::AccessorSetter fun = FUNCTION_CAST<v8::AccessorSetter>(setter_address); | 963 v8::AccessorSetter fun = FUNCTION_CAST<v8::AccessorSetter>(setter_address); |
| 829 ASSERT(fun != NULL); | 964 ASSERT(fun != NULL); |
| 830 Handle<String> name = args.at<String>(2); | 965 Handle<String> name = args.at<String>(2); |
| 831 Handle<Object> value = args.at<Object>(3); | 966 Handle<Object> value = args.at<Object>(3); |
| 832 HandleScope scope; | 967 HandleScope scope; |
| 833 LOG(ApiNamedPropertyAccess("store", recv, *name)); | 968 LOG(ApiNamedPropertyAccess("store", recv, *name)); |
| 834 CustomArguments custom_args(callback->data(), recv, recv); | 969 CustomArguments custom_args(callback->data(), recv, recv); |
| (...skipping 14 matching lines...) Expand all Loading... |
| 849 static const int kAccessorInfoOffsetInInterceptorArgs = 2; | 984 static const int kAccessorInfoOffsetInInterceptorArgs = 2; |
| 850 | 985 |
| 851 | 986 |
| 852 /** | 987 /** |
| 853 * Attempts to load a property with an interceptor (which must be present), | 988 * Attempts to load a property with an interceptor (which must be present), |
| 854 * but doesn't search the prototype chain. | 989 * but doesn't search the prototype chain. |
| 855 * | 990 * |
| 856 * Returns |Heap::no_interceptor_result_sentinel()| if interceptor doesn't | 991 * Returns |Heap::no_interceptor_result_sentinel()| if interceptor doesn't |
| 857 * provide any value for the given name. | 992 * provide any value for the given name. |
| 858 */ | 993 */ |
| 859 Object* LoadPropertyWithInterceptorOnly(Arguments args) { | 994 MaybeObject* LoadPropertyWithInterceptorOnly(Arguments args) { |
| 860 Handle<String> name_handle = args.at<String>(0); | 995 Handle<String> name_handle = args.at<String>(0); |
| 861 Handle<InterceptorInfo> interceptor_info = args.at<InterceptorInfo>(1); | 996 Handle<InterceptorInfo> interceptor_info = args.at<InterceptorInfo>(1); |
| 862 ASSERT(kAccessorInfoOffsetInInterceptorArgs == 2); | 997 ASSERT(kAccessorInfoOffsetInInterceptorArgs == 2); |
| 863 ASSERT(args[2]->IsJSObject()); // Receiver. | 998 ASSERT(args[2]->IsJSObject()); // Receiver. |
| 864 ASSERT(args[3]->IsJSObject()); // Holder. | 999 ASSERT(args[3]->IsJSObject()); // Holder. |
| 865 ASSERT(args.length() == 5); // Last arg is data object. | 1000 ASSERT(args.length() == 5); // Last arg is data object. |
| 866 | 1001 |
| 867 Address getter_address = v8::ToCData<Address>(interceptor_info->getter()); | 1002 Address getter_address = v8::ToCData<Address>(interceptor_info->getter()); |
| 868 v8::NamedPropertyGetter getter = | 1003 v8::NamedPropertyGetter getter = |
| 869 FUNCTION_CAST<v8::NamedPropertyGetter>(getter_address); | 1004 FUNCTION_CAST<v8::NamedPropertyGetter>(getter_address); |
| (...skipping 13 matching lines...) Expand all Loading... |
| 883 RETURN_IF_SCHEDULED_EXCEPTION(); | 1018 RETURN_IF_SCHEDULED_EXCEPTION(); |
| 884 if (!r.IsEmpty()) { | 1019 if (!r.IsEmpty()) { |
| 885 return *v8::Utils::OpenHandle(*r); | 1020 return *v8::Utils::OpenHandle(*r); |
| 886 } | 1021 } |
| 887 } | 1022 } |
| 888 | 1023 |
| 889 return Heap::no_interceptor_result_sentinel(); | 1024 return Heap::no_interceptor_result_sentinel(); |
| 890 } | 1025 } |
| 891 | 1026 |
| 892 | 1027 |
| 893 static Object* ThrowReferenceError(String* name) { | 1028 static MaybeObject* ThrowReferenceError(String* name) { |
| 894 // If the load is non-contextual, just return the undefined result. | 1029 // If the load is non-contextual, just return the undefined result. |
| 895 // Note that both keyed and non-keyed loads may end up here, so we | 1030 // Note that both keyed and non-keyed loads may end up here, so we |
| 896 // can't use either LoadIC or KeyedLoadIC constructors. | 1031 // can't use either LoadIC or KeyedLoadIC constructors. |
| 897 IC ic(IC::NO_EXTRA_FRAME); | 1032 IC ic(IC::NO_EXTRA_FRAME); |
| 898 ASSERT(ic.target()->is_load_stub() || ic.target()->is_keyed_load_stub()); | 1033 ASSERT(ic.target()->is_load_stub() || ic.target()->is_keyed_load_stub()); |
| 899 if (!ic.SlowIsContextual()) return Heap::undefined_value(); | 1034 if (!ic.SlowIsContextual()) return Heap::undefined_value(); |
| 900 | 1035 |
| 901 // Throw a reference error. | 1036 // Throw a reference error. |
| 902 HandleScope scope; | 1037 HandleScope scope; |
| 903 Handle<String> name_handle(name); | 1038 Handle<String> name_handle(name); |
| 904 Handle<Object> error = | 1039 Handle<Object> error = |
| 905 Factory::NewReferenceError("not_defined", | 1040 Factory::NewReferenceError("not_defined", |
| 906 HandleVector(&name_handle, 1)); | 1041 HandleVector(&name_handle, 1)); |
| 907 return Top::Throw(*error); | 1042 return Top::Throw(*error); |
| 908 } | 1043 } |
| 909 | 1044 |
| 910 | 1045 |
| 911 static Object* LoadWithInterceptor(Arguments* args, | 1046 static MaybeObject* LoadWithInterceptor(Arguments* args, |
| 912 PropertyAttributes* attrs) { | 1047 PropertyAttributes* attrs) { |
| 913 Handle<String> name_handle = args->at<String>(0); | 1048 Handle<String> name_handle = args->at<String>(0); |
| 914 Handle<InterceptorInfo> interceptor_info = args->at<InterceptorInfo>(1); | 1049 Handle<InterceptorInfo> interceptor_info = args->at<InterceptorInfo>(1); |
| 915 ASSERT(kAccessorInfoOffsetInInterceptorArgs == 2); | 1050 ASSERT(kAccessorInfoOffsetInInterceptorArgs == 2); |
| 916 Handle<JSObject> receiver_handle = args->at<JSObject>(2); | 1051 Handle<JSObject> receiver_handle = args->at<JSObject>(2); |
| 917 Handle<JSObject> holder_handle = args->at<JSObject>(3); | 1052 Handle<JSObject> holder_handle = args->at<JSObject>(3); |
| 918 ASSERT(args->length() == 5); // Last arg is data object. | 1053 ASSERT(args->length() == 5); // Last arg is data object. |
| 919 | 1054 |
| 920 Address getter_address = v8::ToCData<Address>(interceptor_info->getter()); | 1055 Address getter_address = v8::ToCData<Address>(interceptor_info->getter()); |
| 921 v8::NamedPropertyGetter getter = | 1056 v8::NamedPropertyGetter getter = |
| 922 FUNCTION_CAST<v8::NamedPropertyGetter>(getter_address); | 1057 FUNCTION_CAST<v8::NamedPropertyGetter>(getter_address); |
| (...skipping 10 matching lines...) Expand all Loading... |
| 933 VMState state(EXTERNAL); | 1068 VMState state(EXTERNAL); |
| 934 r = getter(v8::Utils::ToLocal(name_handle), info); | 1069 r = getter(v8::Utils::ToLocal(name_handle), info); |
| 935 } | 1070 } |
| 936 RETURN_IF_SCHEDULED_EXCEPTION(); | 1071 RETURN_IF_SCHEDULED_EXCEPTION(); |
| 937 if (!r.IsEmpty()) { | 1072 if (!r.IsEmpty()) { |
| 938 *attrs = NONE; | 1073 *attrs = NONE; |
| 939 return *v8::Utils::OpenHandle(*r); | 1074 return *v8::Utils::OpenHandle(*r); |
| 940 } | 1075 } |
| 941 } | 1076 } |
| 942 | 1077 |
| 943 Object* result = holder_handle->GetPropertyPostInterceptor( | 1078 MaybeObject* result = holder_handle->GetPropertyPostInterceptor( |
| 944 *receiver_handle, | 1079 *receiver_handle, |
| 945 *name_handle, | 1080 *name_handle, |
| 946 attrs); | 1081 attrs); |
| 947 RETURN_IF_SCHEDULED_EXCEPTION(); | 1082 RETURN_IF_SCHEDULED_EXCEPTION(); |
| 948 return result; | 1083 return result; |
| 949 } | 1084 } |
| 950 | 1085 |
| 951 | 1086 |
| 952 /** | 1087 /** |
| 953 * Loads a property with an interceptor performing post interceptor | 1088 * Loads a property with an interceptor performing post interceptor |
| 954 * lookup if interceptor failed. | 1089 * lookup if interceptor failed. |
| 955 */ | 1090 */ |
| 956 Object* LoadPropertyWithInterceptorForLoad(Arguments args) { | 1091 MaybeObject* LoadPropertyWithInterceptorForLoad(Arguments args) { |
| 957 PropertyAttributes attr = NONE; | 1092 PropertyAttributes attr = NONE; |
| 958 Object* result = LoadWithInterceptor(&args, &attr); | 1093 Object* result; |
| 959 if (result->IsFailure()) return result; | 1094 { MaybeObject* maybe_result = LoadWithInterceptor(&args, &attr); |
| 1095 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 1096 } |
| 960 | 1097 |
| 961 // If the property is present, return it. | 1098 // If the property is present, return it. |
| 962 if (attr != ABSENT) return result; | 1099 if (attr != ABSENT) return result; |
| 963 return ThrowReferenceError(String::cast(args[0])); | 1100 return ThrowReferenceError(String::cast(args[0])); |
| 964 } | 1101 } |
| 965 | 1102 |
| 966 | 1103 |
| 967 Object* LoadPropertyWithInterceptorForCall(Arguments args) { | 1104 MaybeObject* LoadPropertyWithInterceptorForCall(Arguments args) { |
| 968 PropertyAttributes attr; | 1105 PropertyAttributes attr; |
| 969 Object* result = LoadWithInterceptor(&args, &attr); | 1106 MaybeObject* result = LoadWithInterceptor(&args, &attr); |
| 970 RETURN_IF_SCHEDULED_EXCEPTION(); | 1107 RETURN_IF_SCHEDULED_EXCEPTION(); |
| 971 // This is call IC. In this case, we simply return the undefined result which | 1108 // This is call IC. In this case, we simply return the undefined result which |
| 972 // will lead to an exception when trying to invoke the result as a | 1109 // will lead to an exception when trying to invoke the result as a |
| 973 // function. | 1110 // function. |
| 974 return result; | 1111 return result; |
| 975 } | 1112 } |
| 976 | 1113 |
| 977 | 1114 |
| 978 Object* StoreInterceptorProperty(Arguments args) { | 1115 MaybeObject* StoreInterceptorProperty(Arguments args) { |
| 979 JSObject* recv = JSObject::cast(args[0]); | 1116 JSObject* recv = JSObject::cast(args[0]); |
| 980 String* name = String::cast(args[1]); | 1117 String* name = String::cast(args[1]); |
| 981 Object* value = args[2]; | 1118 Object* value = args[2]; |
| 982 ASSERT(recv->HasNamedInterceptor()); | 1119 ASSERT(recv->HasNamedInterceptor()); |
| 983 PropertyAttributes attr = NONE; | 1120 PropertyAttributes attr = NONE; |
| 984 Object* result = recv->SetPropertyWithInterceptor(name, value, attr); | 1121 MaybeObject* result = recv->SetPropertyWithInterceptor(name, value, attr); |
| 985 return result; | 1122 return result; |
| 986 } | 1123 } |
| 987 | 1124 |
| 988 | 1125 |
| 989 Object* KeyedLoadPropertyWithInterceptor(Arguments args) { | 1126 MaybeObject* KeyedLoadPropertyWithInterceptor(Arguments args) { |
| 990 JSObject* receiver = JSObject::cast(args[0]); | 1127 JSObject* receiver = JSObject::cast(args[0]); |
| 991 ASSERT(Smi::cast(args[1])->value() >= 0); | 1128 ASSERT(Smi::cast(args[1])->value() >= 0); |
| 992 uint32_t index = Smi::cast(args[1])->value(); | 1129 uint32_t index = Smi::cast(args[1])->value(); |
| 993 return receiver->GetElementWithInterceptor(receiver, index); | 1130 return receiver->GetElementWithInterceptor(receiver, index); |
| 994 } | 1131 } |
| 995 | 1132 |
| 996 | 1133 |
| 997 Object* StubCompiler::CompileCallInitialize(Code::Flags flags) { | 1134 MaybeObject* StubCompiler::CompileCallInitialize(Code::Flags flags) { |
| 998 HandleScope scope; | 1135 HandleScope scope; |
| 999 int argc = Code::ExtractArgumentsCountFromFlags(flags); | 1136 int argc = Code::ExtractArgumentsCountFromFlags(flags); |
| 1000 Code::Kind kind = Code::ExtractKindFromFlags(flags); | 1137 Code::Kind kind = Code::ExtractKindFromFlags(flags); |
| 1001 if (kind == Code::CALL_IC) { | 1138 if (kind == Code::CALL_IC) { |
| 1002 CallIC::GenerateInitialize(masm(), argc); | 1139 CallIC::GenerateInitialize(masm(), argc); |
| 1003 } else { | 1140 } else { |
| 1004 KeyedCallIC::GenerateInitialize(masm(), argc); | 1141 KeyedCallIC::GenerateInitialize(masm(), argc); |
| 1005 } | 1142 } |
| 1006 Object* result = GetCodeWithFlags(flags, "CompileCallInitialize"); | 1143 Object* result; |
| 1007 if (!result->IsFailure()) { | 1144 { MaybeObject* maybe_result = |
| 1008 Counters::call_initialize_stubs.Increment(); | 1145 GetCodeWithFlags(flags, "CompileCallInitialize"); |
| 1009 Code* code = Code::cast(result); | 1146 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 1010 USE(code); | |
| 1011 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_INITIALIZE_TAG), | |
| 1012 code, code->arguments_count())); | |
| 1013 } | 1147 } |
| 1148 Counters::call_initialize_stubs.Increment(); |
| 1149 Code* code = Code::cast(result); |
| 1150 USE(code); |
| 1151 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_INITIALIZE_TAG), |
| 1152 code, code->arguments_count())); |
| 1014 return result; | 1153 return result; |
| 1015 } | 1154 } |
| 1016 | 1155 |
| 1017 | 1156 |
| 1018 Object* StubCompiler::CompileCallPreMonomorphic(Code::Flags flags) { | 1157 MaybeObject* StubCompiler::CompileCallPreMonomorphic(Code::Flags flags) { |
| 1019 HandleScope scope; | 1158 HandleScope scope; |
| 1020 int argc = Code::ExtractArgumentsCountFromFlags(flags); | 1159 int argc = Code::ExtractArgumentsCountFromFlags(flags); |
| 1021 // The code of the PreMonomorphic stub is the same as the code | 1160 // The code of the PreMonomorphic stub is the same as the code |
| 1022 // of the Initialized stub. They just differ on the code object flags. | 1161 // of the Initialized stub. They just differ on the code object flags. |
| 1023 Code::Kind kind = Code::ExtractKindFromFlags(flags); | 1162 Code::Kind kind = Code::ExtractKindFromFlags(flags); |
| 1024 if (kind == Code::CALL_IC) { | 1163 if (kind == Code::CALL_IC) { |
| 1025 CallIC::GenerateInitialize(masm(), argc); | 1164 CallIC::GenerateInitialize(masm(), argc); |
| 1026 } else { | 1165 } else { |
| 1027 KeyedCallIC::GenerateInitialize(masm(), argc); | 1166 KeyedCallIC::GenerateInitialize(masm(), argc); |
| 1028 } | 1167 } |
| 1029 Object* result = GetCodeWithFlags(flags, "CompileCallPreMonomorphic"); | 1168 Object* result; |
| 1030 if (!result->IsFailure()) { | 1169 { MaybeObject* maybe_result = |
| 1031 Counters::call_premonomorphic_stubs.Increment(); | 1170 GetCodeWithFlags(flags, "CompileCallPreMonomorphic"); |
| 1032 Code* code = Code::cast(result); | 1171 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 1033 USE(code); | |
| 1034 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_PRE_MONOMORPHIC_TAG), | |
| 1035 code, code->arguments_count())); | |
| 1036 } | 1172 } |
| 1173 Counters::call_premonomorphic_stubs.Increment(); |
| 1174 Code* code = Code::cast(result); |
| 1175 USE(code); |
| 1176 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_PRE_MONOMORPHIC_TAG), |
| 1177 code, code->arguments_count())); |
| 1037 return result; | 1178 return result; |
| 1038 } | 1179 } |
| 1039 | 1180 |
| 1040 | 1181 |
| 1041 Object* StubCompiler::CompileCallNormal(Code::Flags flags) { | 1182 MaybeObject* StubCompiler::CompileCallNormal(Code::Flags flags) { |
| 1042 HandleScope scope; | 1183 HandleScope scope; |
| 1043 int argc = Code::ExtractArgumentsCountFromFlags(flags); | 1184 int argc = Code::ExtractArgumentsCountFromFlags(flags); |
| 1044 Code::Kind kind = Code::ExtractKindFromFlags(flags); | 1185 Code::Kind kind = Code::ExtractKindFromFlags(flags); |
| 1045 if (kind == Code::CALL_IC) { | 1186 if (kind == Code::CALL_IC) { |
| 1046 CallIC::GenerateNormal(masm(), argc); | 1187 CallIC::GenerateNormal(masm(), argc); |
| 1047 } else { | 1188 } else { |
| 1048 KeyedCallIC::GenerateNormal(masm(), argc); | 1189 KeyedCallIC::GenerateNormal(masm(), argc); |
| 1049 } | 1190 } |
| 1050 Object* result = GetCodeWithFlags(flags, "CompileCallNormal"); | 1191 Object* result; |
| 1051 if (!result->IsFailure()) { | 1192 { MaybeObject* maybe_result = GetCodeWithFlags(flags, "CompileCallNormal"); |
| 1052 Counters::call_normal_stubs.Increment(); | 1193 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 1053 Code* code = Code::cast(result); | |
| 1054 USE(code); | |
| 1055 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_NORMAL_TAG), | |
| 1056 code, code->arguments_count())); | |
| 1057 } | 1194 } |
| 1195 Counters::call_normal_stubs.Increment(); |
| 1196 Code* code = Code::cast(result); |
| 1197 USE(code); |
| 1198 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_NORMAL_TAG), |
| 1199 code, code->arguments_count())); |
| 1058 return result; | 1200 return result; |
| 1059 } | 1201 } |
| 1060 | 1202 |
| 1061 | 1203 |
| 1062 Object* StubCompiler::CompileCallMegamorphic(Code::Flags flags) { | 1204 MaybeObject* StubCompiler::CompileCallMegamorphic(Code::Flags flags) { |
| 1063 HandleScope scope; | 1205 HandleScope scope; |
| 1064 int argc = Code::ExtractArgumentsCountFromFlags(flags); | 1206 int argc = Code::ExtractArgumentsCountFromFlags(flags); |
| 1065 Code::Kind kind = Code::ExtractKindFromFlags(flags); | 1207 Code::Kind kind = Code::ExtractKindFromFlags(flags); |
| 1066 if (kind == Code::CALL_IC) { | 1208 if (kind == Code::CALL_IC) { |
| 1067 CallIC::GenerateMegamorphic(masm(), argc); | 1209 CallIC::GenerateMegamorphic(masm(), argc); |
| 1068 } else { | 1210 } else { |
| 1069 KeyedCallIC::GenerateMegamorphic(masm(), argc); | 1211 KeyedCallIC::GenerateMegamorphic(masm(), argc); |
| 1070 } | 1212 } |
| 1071 | 1213 |
| 1072 Object* result = GetCodeWithFlags(flags, "CompileCallMegamorphic"); | 1214 Object* result; |
| 1073 if (!result->IsFailure()) { | 1215 { MaybeObject* maybe_result = |
| 1074 Counters::call_megamorphic_stubs.Increment(); | 1216 GetCodeWithFlags(flags, "CompileCallMegamorphic"); |
| 1075 Code* code = Code::cast(result); | 1217 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 1076 USE(code); | |
| 1077 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_MEGAMORPHIC_TAG), | |
| 1078 code, code->arguments_count())); | |
| 1079 } | 1218 } |
| 1219 Counters::call_megamorphic_stubs.Increment(); |
| 1220 Code* code = Code::cast(result); |
| 1221 USE(code); |
| 1222 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_MEGAMORPHIC_TAG), |
| 1223 code, code->arguments_count())); |
| 1080 return result; | 1224 return result; |
| 1081 } | 1225 } |
| 1082 | 1226 |
| 1083 | 1227 |
| 1084 Object* StubCompiler::CompileCallMiss(Code::Flags flags) { | 1228 MaybeObject* StubCompiler::CompileCallMiss(Code::Flags flags) { |
| 1085 HandleScope scope; | 1229 HandleScope scope; |
| 1086 int argc = Code::ExtractArgumentsCountFromFlags(flags); | 1230 int argc = Code::ExtractArgumentsCountFromFlags(flags); |
| 1087 Code::Kind kind = Code::ExtractKindFromFlags(flags); | 1231 Code::Kind kind = Code::ExtractKindFromFlags(flags); |
| 1088 if (kind == Code::CALL_IC) { | 1232 if (kind == Code::CALL_IC) { |
| 1089 CallIC::GenerateMiss(masm(), argc); | 1233 CallIC::GenerateMiss(masm(), argc); |
| 1090 } else { | 1234 } else { |
| 1091 KeyedCallIC::GenerateMiss(masm(), argc); | 1235 KeyedCallIC::GenerateMiss(masm(), argc); |
| 1092 } | 1236 } |
| 1093 Object* result = GetCodeWithFlags(flags, "CompileCallMiss"); | 1237 Object* result; |
| 1094 if (!result->IsFailure()) { | 1238 { MaybeObject* maybe_result = GetCodeWithFlags(flags, "CompileCallMiss"); |
| 1095 Counters::call_megamorphic_stubs.Increment(); | 1239 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 1096 Code* code = Code::cast(result); | |
| 1097 USE(code); | |
| 1098 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_MISS_TAG), | |
| 1099 code, code->arguments_count())); | |
| 1100 } | 1240 } |
| 1241 Counters::call_megamorphic_stubs.Increment(); |
| 1242 Code* code = Code::cast(result); |
| 1243 USE(code); |
| 1244 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_MISS_TAG), |
| 1245 code, code->arguments_count())); |
| 1101 return result; | 1246 return result; |
| 1102 } | 1247 } |
| 1103 | 1248 |
| 1104 | 1249 |
| 1105 #ifdef ENABLE_DEBUGGER_SUPPORT | 1250 #ifdef ENABLE_DEBUGGER_SUPPORT |
| 1106 Object* StubCompiler::CompileCallDebugBreak(Code::Flags flags) { | 1251 MaybeObject* StubCompiler::CompileCallDebugBreak(Code::Flags flags) { |
| 1107 HandleScope scope; | 1252 HandleScope scope; |
| 1108 Debug::GenerateCallICDebugBreak(masm()); | 1253 Debug::GenerateCallICDebugBreak(masm()); |
| 1109 Object* result = GetCodeWithFlags(flags, "CompileCallDebugBreak"); | 1254 Object* result; |
| 1110 if (!result->IsFailure()) { | 1255 { MaybeObject* maybe_result = |
| 1111 Code* code = Code::cast(result); | 1256 GetCodeWithFlags(flags, "CompileCallDebugBreak"); |
| 1112 USE(code); | 1257 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 1113 Code::Kind kind = Code::ExtractKindFromFlags(flags); | |
| 1114 USE(kind); | |
| 1115 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_DEBUG_BREAK_TAG), | |
| 1116 code, code->arguments_count())); | |
| 1117 } | 1258 } |
| 1259 Code* code = Code::cast(result); |
| 1260 USE(code); |
| 1261 Code::Kind kind = Code::ExtractKindFromFlags(flags); |
| 1262 USE(kind); |
| 1263 PROFILE(CodeCreateEvent(CALL_LOGGER_TAG(kind, CALL_DEBUG_BREAK_TAG), |
| 1264 code, code->arguments_count())); |
| 1118 return result; | 1265 return result; |
| 1119 } | 1266 } |
| 1120 | 1267 |
| 1121 | 1268 |
| 1122 Object* StubCompiler::CompileCallDebugPrepareStepIn(Code::Flags flags) { | 1269 MaybeObject* StubCompiler::CompileCallDebugPrepareStepIn(Code::Flags flags) { |
| 1123 HandleScope scope; | 1270 HandleScope scope; |
| 1124 // Use the same code for the the step in preparations as we do for | 1271 // Use the same code for the the step in preparations as we do for |
| 1125 // the miss case. | 1272 // the miss case. |
| 1126 int argc = Code::ExtractArgumentsCountFromFlags(flags); | 1273 int argc = Code::ExtractArgumentsCountFromFlags(flags); |
| 1127 Code::Kind kind = Code::ExtractKindFromFlags(flags); | 1274 Code::Kind kind = Code::ExtractKindFromFlags(flags); |
| 1128 if (kind == Code::CALL_IC) { | 1275 if (kind == Code::CALL_IC) { |
| 1129 CallIC::GenerateMiss(masm(), argc); | 1276 CallIC::GenerateMiss(masm(), argc); |
| 1130 } else { | 1277 } else { |
| 1131 KeyedCallIC::GenerateMiss(masm(), argc); | 1278 KeyedCallIC::GenerateMiss(masm(), argc); |
| 1132 } | 1279 } |
| 1133 Object* result = GetCodeWithFlags(flags, "CompileCallDebugPrepareStepIn"); | 1280 Object* result; |
| 1134 if (!result->IsFailure()) { | 1281 { MaybeObject* maybe_result = |
| 1135 Code* code = Code::cast(result); | 1282 GetCodeWithFlags(flags, "CompileCallDebugPrepareStepIn"); |
| 1136 USE(code); | 1283 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 1137 PROFILE(CodeCreateEvent( | |
| 1138 CALL_LOGGER_TAG(kind, CALL_DEBUG_PREPARE_STEP_IN_TAG), | |
| 1139 code, | |
| 1140 code->arguments_count())); | |
| 1141 } | 1284 } |
| 1285 Code* code = Code::cast(result); |
| 1286 USE(code); |
| 1287 PROFILE(CodeCreateEvent( |
| 1288 CALL_LOGGER_TAG(kind, CALL_DEBUG_PREPARE_STEP_IN_TAG), |
| 1289 code, |
| 1290 code->arguments_count())); |
| 1142 return result; | 1291 return result; |
| 1143 } | 1292 } |
| 1144 #endif | 1293 #endif |
| 1145 | 1294 |
| 1146 #undef CALL_LOGGER_TAG | 1295 #undef CALL_LOGGER_TAG |
| 1147 | 1296 |
| 1148 Object* StubCompiler::GetCodeWithFlags(Code::Flags flags, const char* name) { | 1297 MaybeObject* StubCompiler::GetCodeWithFlags(Code::Flags flags, |
| 1298 const char* name) { |
| 1149 // Check for allocation failures during stub compilation. | 1299 // Check for allocation failures during stub compilation. |
| 1150 if (failure_->IsFailure()) return failure_; | 1300 if (failure_->IsFailure()) return failure_; |
| 1151 | 1301 |
| 1152 // Create code object in the heap. | 1302 // Create code object in the heap. |
| 1153 CodeDesc desc; | 1303 CodeDesc desc; |
| 1154 masm_.GetCode(&desc); | 1304 masm_.GetCode(&desc); |
| 1155 Object* result = Heap::CreateCode(desc, flags, masm_.CodeObject()); | 1305 MaybeObject* result = Heap::CreateCode(desc, flags, masm_.CodeObject()); |
| 1156 #ifdef ENABLE_DISASSEMBLER | 1306 #ifdef ENABLE_DISASSEMBLER |
| 1157 if (FLAG_print_code_stubs && !result->IsFailure()) { | 1307 if (FLAG_print_code_stubs && !result->IsFailure()) { |
| 1158 Code::cast(result)->Disassemble(name); | 1308 Code::cast(result->ToObjectUnchecked())->Disassemble(name); |
| 1159 } | 1309 } |
| 1160 #endif | 1310 #endif |
| 1161 return result; | 1311 return result; |
| 1162 } | 1312 } |
| 1163 | 1313 |
| 1164 | 1314 |
| 1165 Object* StubCompiler::GetCodeWithFlags(Code::Flags flags, String* name) { | 1315 MaybeObject* StubCompiler::GetCodeWithFlags(Code::Flags flags, String* name) { |
| 1166 if (FLAG_print_code_stubs && (name != NULL)) { | 1316 if (FLAG_print_code_stubs && (name != NULL)) { |
| 1167 return GetCodeWithFlags(flags, *name->ToCString()); | 1317 return GetCodeWithFlags(flags, *name->ToCString()); |
| 1168 } | 1318 } |
| 1169 return GetCodeWithFlags(flags, reinterpret_cast<char*>(NULL)); | 1319 return GetCodeWithFlags(flags, reinterpret_cast<char*>(NULL)); |
| 1170 } | 1320 } |
| 1171 | 1321 |
| 1172 | 1322 |
| 1173 void StubCompiler::LookupPostInterceptor(JSObject* holder, | 1323 void StubCompiler::LookupPostInterceptor(JSObject* holder, |
| 1174 String* name, | 1324 String* name, |
| 1175 LookupResult* lookup) { | 1325 LookupResult* lookup) { |
| 1176 holder->LocalLookupRealNamedProperty(name, lookup); | 1326 holder->LocalLookupRealNamedProperty(name, lookup); |
| 1177 if (!lookup->IsProperty()) { | 1327 if (!lookup->IsProperty()) { |
| 1178 lookup->NotFound(); | 1328 lookup->NotFound(); |
| 1179 Object* proto = holder->GetPrototype(); | 1329 Object* proto = holder->GetPrototype(); |
| 1180 if (proto != Heap::null_value()) { | 1330 if (proto != Heap::null_value()) { |
| 1181 proto->Lookup(name, lookup); | 1331 proto->Lookup(name, lookup); |
| 1182 } | 1332 } |
| 1183 } | 1333 } |
| 1184 } | 1334 } |
| 1185 | 1335 |
| 1186 | 1336 |
| 1187 | 1337 |
| 1188 Object* LoadStubCompiler::GetCode(PropertyType type, String* name) { | 1338 MaybeObject* LoadStubCompiler::GetCode(PropertyType type, String* name) { |
| 1189 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, type); | 1339 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::LOAD_IC, type); |
| 1190 Object* result = GetCodeWithFlags(flags, name); | 1340 MaybeObject* result = GetCodeWithFlags(flags, name); |
| 1191 if (!result->IsFailure()) { | 1341 if (!result->IsFailure()) { |
| 1192 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, Code::cast(result), name)); | 1342 PROFILE(CodeCreateEvent(Logger::LOAD_IC_TAG, |
| 1343 Code::cast(result->ToObjectUnchecked()), |
| 1344 name)); |
| 1193 } | 1345 } |
| 1194 return result; | 1346 return result; |
| 1195 } | 1347 } |
| 1196 | 1348 |
| 1197 | 1349 |
| 1198 Object* KeyedLoadStubCompiler::GetCode(PropertyType type, String* name) { | 1350 MaybeObject* KeyedLoadStubCompiler::GetCode(PropertyType type, String* name) { |
| 1199 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, type); | 1351 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, type); |
| 1200 Object* result = GetCodeWithFlags(flags, name); | 1352 MaybeObject* result = GetCodeWithFlags(flags, name); |
| 1201 if (!result->IsFailure()) { | 1353 if (!result->IsFailure()) { |
| 1202 PROFILE( | 1354 PROFILE(CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, |
| 1203 CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, Code::cast(result), name)); | 1355 Code::cast(result->ToObjectUnchecked()), |
| 1356 name)); |
| 1204 } | 1357 } |
| 1205 return result; | 1358 return result; |
| 1206 } | 1359 } |
| 1207 | 1360 |
| 1208 | 1361 |
| 1209 Object* StoreStubCompiler::GetCode(PropertyType type, String* name) { | 1362 MaybeObject* StoreStubCompiler::GetCode(PropertyType type, String* name) { |
| 1210 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::STORE_IC, type); | 1363 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::STORE_IC, type); |
| 1211 Object* result = GetCodeWithFlags(flags, name); | 1364 MaybeObject* result = GetCodeWithFlags(flags, name); |
| 1212 if (!result->IsFailure()) { | 1365 if (!result->IsFailure()) { |
| 1213 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, Code::cast(result), name)); | 1366 PROFILE(CodeCreateEvent(Logger::STORE_IC_TAG, |
| 1367 Code::cast(result->ToObjectUnchecked()), |
| 1368 name)); |
| 1214 } | 1369 } |
| 1215 return result; | 1370 return result; |
| 1216 } | 1371 } |
| 1217 | 1372 |
| 1218 | 1373 |
| 1219 Object* KeyedStoreStubCompiler::GetCode(PropertyType type, String* name) { | 1374 MaybeObject* KeyedStoreStubCompiler::GetCode(PropertyType type, String* name) { |
| 1220 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::KEYED_STORE_IC, type); | 1375 Code::Flags flags = Code::ComputeMonomorphicFlags(Code::KEYED_STORE_IC, type); |
| 1221 Object* result = GetCodeWithFlags(flags, name); | 1376 MaybeObject* result = GetCodeWithFlags(flags, name); |
| 1222 if (!result->IsFailure()) { | 1377 if (!result->IsFailure()) { |
| 1223 PROFILE( | 1378 PROFILE(CodeCreateEvent(Logger::KEYED_STORE_IC_TAG, |
| 1224 CodeCreateEvent(Logger::KEYED_STORE_IC_TAG, Code::cast(result), name)); | 1379 Code::cast(result->ToObjectUnchecked()), |
| 1380 name)); |
| 1225 } | 1381 } |
| 1226 return result; | 1382 return result; |
| 1227 } | 1383 } |
| 1228 | 1384 |
| 1229 | 1385 |
| 1230 CallStubCompiler::CallStubCompiler(int argc, | 1386 CallStubCompiler::CallStubCompiler(int argc, |
| 1231 InLoopFlag in_loop, | 1387 InLoopFlag in_loop, |
| 1232 Code::Kind kind, | 1388 Code::Kind kind, |
| 1233 InlineCacheHolderFlag cache_holder) | 1389 InlineCacheHolderFlag cache_holder) |
| 1234 : arguments_(argc) | 1390 : arguments_(argc) |
| 1235 , in_loop_(in_loop) | 1391 , in_loop_(in_loop) |
| 1236 , kind_(kind) | 1392 , kind_(kind) |
| 1237 , cache_holder_(cache_holder) { | 1393 , cache_holder_(cache_holder) { |
| 1238 } | 1394 } |
| 1239 | 1395 |
| 1240 | 1396 |
| 1241 Object* CallStubCompiler::CompileCustomCall(int generator_id, | 1397 MaybeObject* CallStubCompiler::CompileCustomCall(int generator_id, |
| 1242 Object* object, | 1398 Object* object, |
| 1243 JSObject* holder, | 1399 JSObject* holder, |
| 1244 JSGlobalPropertyCell* cell, | 1400 JSGlobalPropertyCell* cell, |
| 1245 JSFunction* function, | 1401 JSFunction* function, |
| 1246 String* fname) { | 1402 String* fname) { |
| 1247 ASSERT(generator_id >= 0 && generator_id < kNumCallGenerators); | 1403 ASSERT(generator_id >= 0 && generator_id < kNumCallGenerators); |
| 1248 switch (generator_id) { | 1404 switch (generator_id) { |
| 1249 #define CALL_GENERATOR_CASE(ignored1, ignored2, name) \ | 1405 #define CALL_GENERATOR_CASE(ignored1, ignored2, name) \ |
| 1250 case k##name##CallGenerator: \ | 1406 case k##name##CallGenerator: \ |
| 1251 return CallStubCompiler::Compile##name##Call(object, \ | 1407 return CallStubCompiler::Compile##name##Call(object, \ |
| 1252 holder, \ | 1408 holder, \ |
| 1253 cell, \ | 1409 cell, \ |
| 1254 function, \ | 1410 function, \ |
| 1255 fname); | 1411 fname); |
| 1256 CUSTOM_CALL_IC_GENERATORS(CALL_GENERATOR_CASE) | 1412 CUSTOM_CALL_IC_GENERATORS(CALL_GENERATOR_CASE) |
| 1257 #undef CALL_GENERATOR_CASE | 1413 #undef CALL_GENERATOR_CASE |
| 1258 } | 1414 } |
| 1259 UNREACHABLE(); | 1415 UNREACHABLE(); |
| 1260 return Heap::undefined_value(); | 1416 return Heap::undefined_value(); |
| 1261 } | 1417 } |
| 1262 | 1418 |
| 1263 | 1419 |
| 1264 Object* CallStubCompiler::GetCode(PropertyType type, String* name) { | 1420 MaybeObject* CallStubCompiler::GetCode(PropertyType type, String* name) { |
| 1265 int argc = arguments_.immediate(); | 1421 int argc = arguments_.immediate(); |
| 1266 Code::Flags flags = Code::ComputeMonomorphicFlags(kind_, | 1422 Code::Flags flags = Code::ComputeMonomorphicFlags(kind_, |
| 1267 type, | 1423 type, |
| 1268 cache_holder_, | 1424 cache_holder_, |
| 1269 in_loop_, | 1425 in_loop_, |
| 1270 argc); | 1426 argc); |
| 1271 return GetCodeWithFlags(flags, name); | 1427 return GetCodeWithFlags(flags, name); |
| 1272 } | 1428 } |
| 1273 | 1429 |
| 1274 | 1430 |
| 1275 Object* CallStubCompiler::GetCode(JSFunction* function) { | 1431 MaybeObject* CallStubCompiler::GetCode(JSFunction* function) { |
| 1276 String* function_name = NULL; | 1432 String* function_name = NULL; |
| 1277 if (function->shared()->name()->IsString()) { | 1433 if (function->shared()->name()->IsString()) { |
| 1278 function_name = String::cast(function->shared()->name()); | 1434 function_name = String::cast(function->shared()->name()); |
| 1279 } | 1435 } |
| 1280 return GetCode(CONSTANT_FUNCTION, function_name); | 1436 return GetCode(CONSTANT_FUNCTION, function_name); |
| 1281 } | 1437 } |
| 1282 | 1438 |
| 1283 | 1439 |
| 1284 Object* ConstructStubCompiler::GetCode() { | 1440 MaybeObject* ConstructStubCompiler::GetCode() { |
| 1285 Code::Flags flags = Code::ComputeFlags(Code::STUB); | 1441 Code::Flags flags = Code::ComputeFlags(Code::STUB); |
| 1286 Object* result = GetCodeWithFlags(flags, "ConstructStub"); | 1442 Object* result; |
| 1287 if (!result->IsFailure()) { | 1443 { MaybeObject* maybe_result = GetCodeWithFlags(flags, "ConstructStub"); |
| 1288 Code* code = Code::cast(result); | 1444 if (!maybe_result->ToObject(&result)) return maybe_result; |
| 1289 USE(code); | |
| 1290 PROFILE(CodeCreateEvent(Logger::STUB_TAG, code, "ConstructStub")); | |
| 1291 } | 1445 } |
| 1446 Code* code = Code::cast(result); |
| 1447 USE(code); |
| 1448 PROFILE(CodeCreateEvent(Logger::STUB_TAG, code, "ConstructStub")); |
| 1292 return result; | 1449 return result; |
| 1293 } | 1450 } |
| 1294 | 1451 |
| 1295 | 1452 |
| 1296 CallOptimization::CallOptimization(LookupResult* lookup) { | 1453 CallOptimization::CallOptimization(LookupResult* lookup) { |
| 1297 if (!lookup->IsProperty() || !lookup->IsCacheable() || | 1454 if (!lookup->IsProperty() || !lookup->IsCacheable() || |
| 1298 lookup->type() != CONSTANT_FUNCTION) { | 1455 lookup->type() != CONSTANT_FUNCTION) { |
| 1299 Initialize(NULL); | 1456 Initialize(NULL); |
| 1300 } else { | 1457 } else { |
| 1301 // We only optimize constant function calls. | 1458 // We only optimize constant function calls. |
| (...skipping 52 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 1354 expected_receiver_type_ = | 1511 expected_receiver_type_ = |
| 1355 FunctionTemplateInfo::cast(signature->receiver()); | 1512 FunctionTemplateInfo::cast(signature->receiver()); |
| 1356 } | 1513 } |
| 1357 } | 1514 } |
| 1358 | 1515 |
| 1359 is_simple_api_call_ = true; | 1516 is_simple_api_call_ = true; |
| 1360 } | 1517 } |
| 1361 | 1518 |
| 1362 | 1519 |
| 1363 } } // namespace v8::internal | 1520 } } // namespace v8::internal |
| OLD | NEW |