| Index: src/stub-cache.cc
|
| diff --git a/src/stub-cache.cc b/src/stub-cache.cc
|
| index 08954ba618ae7ef761a9e9a8df1fc8289fb9b9d9..73272a250e426109de3eccf3c45cb0a95245c0db 100644
|
| --- a/src/stub-cache.cc
|
| +++ b/src/stub-cache.cc
|
| @@ -371,69 +371,48 @@ Handle<Code> StubCache::ComputeStoreField(Handle<String> name,
|
| }
|
|
|
|
|
| -Handle<Code> StubCache::ComputeKeyedLoadOrStoreElement(
|
| +Handle<Code> StubCache::ComputeKeyedLoadElement(Handle<Map> receiver_map) {
|
| + Code::Flags flags =
|
| + Code::ComputeMonomorphicFlags(Code::KEYED_LOAD_IC, Code::NORMAL);
|
| + Handle<String> name =
|
| + isolate()->factory()->KeyedLoadElementMonomorphic_symbol();
|
| +
|
| + Handle<Object> probe(receiver_map->FindInCodeCache(*name, flags), isolate_);
|
| + if (probe->IsCode()) return Handle<Code>::cast(probe);
|
| +
|
| + KeyedLoadStubCompiler compiler(isolate());
|
| + Handle<Code> code = compiler.CompileLoadElement(receiver_map);
|
| +
|
| + PROFILE(isolate(), CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, *code, 0));
|
| + Map::UpdateCodeCache(receiver_map, name, code);
|
| + return code;
|
| +}
|
| +
|
| +
|
| +Handle<Code> StubCache::ComputeKeyedStoreElement(
|
| Handle<Map> receiver_map,
|
| - KeyedIC::StubKind stub_kind,
|
| - StrictModeFlag strict_mode) {
|
| - KeyedAccessGrowMode grow_mode =
|
| - KeyedIC::GetGrowModeFromStubKind(stub_kind);
|
| + KeyedStoreIC::StubKind stub_kind,
|
| + StrictModeFlag strict_mode,
|
| + KeyedAccessGrowMode grow_mode) {
|
| Code::ExtraICState extra_state =
|
| Code::ComputeExtraICState(grow_mode, strict_mode);
|
| - Code::Flags flags =
|
| - Code::ComputeMonomorphicFlags(
|
| - stub_kind == KeyedIC::LOAD ? Code::KEYED_LOAD_IC
|
| - : Code::KEYED_STORE_IC,
|
| - Code::NORMAL,
|
| - extra_state);
|
| - Handle<String> name;
|
| - switch (stub_kind) {
|
| - case KeyedIC::LOAD:
|
| - name = isolate()->factory()->KeyedLoadElementMonomorphic_symbol();
|
| - break;
|
| - case KeyedIC::STORE_NO_TRANSITION:
|
| - name = isolate()->factory()->KeyedStoreElementMonomorphic_symbol();
|
| - break;
|
| - case KeyedIC::STORE_AND_GROW_NO_TRANSITION:
|
| - name = isolate()->factory()->KeyedStoreAndGrowElementMonomorphic_symbol();
|
| - break;
|
| - default:
|
| - UNREACHABLE();
|
| - break;
|
| - }
|
| + Code::Flags flags = Code::ComputeMonomorphicFlags(
|
| + Code::KEYED_STORE_IC, Code::NORMAL, extra_state);
|
| +
|
| + ASSERT(stub_kind == KeyedStoreIC::STORE_NO_TRANSITION ||
|
| + stub_kind == KeyedStoreIC::STORE_AND_GROW_NO_TRANSITION);
|
| +
|
| + Handle<String> name = stub_kind == KeyedStoreIC::STORE_NO_TRANSITION
|
| + ? isolate()->factory()->KeyedStoreElementMonomorphic_symbol()
|
| + : isolate()->factory()->KeyedStoreAndGrowElementMonomorphic_symbol();
|
| +
|
| Handle<Object> probe(receiver_map->FindInCodeCache(*name, flags), isolate_);
|
| if (probe->IsCode()) return Handle<Code>::cast(probe);
|
|
|
| - Handle<Code> code;
|
| - switch (stub_kind) {
|
| - case KeyedIC::LOAD: {
|
| - KeyedLoadStubCompiler compiler(isolate_);
|
| - code = compiler.CompileLoadElement(receiver_map);
|
| - break;
|
| - }
|
| - case KeyedIC::STORE_AND_GROW_NO_TRANSITION: {
|
| - KeyedStoreStubCompiler compiler(isolate_, strict_mode,
|
| - ALLOW_JSARRAY_GROWTH);
|
| - code = compiler.CompileStoreElement(receiver_map);
|
| - break;
|
| - }
|
| - case KeyedIC::STORE_NO_TRANSITION: {
|
| - KeyedStoreStubCompiler compiler(isolate_, strict_mode,
|
| - DO_NOT_ALLOW_JSARRAY_GROWTH);
|
| - code = compiler.CompileStoreElement(receiver_map);
|
| - break;
|
| - }
|
| - default:
|
| - UNREACHABLE();
|
| - break;
|
| - }
|
| -
|
| - ASSERT(!code.is_null());
|
| + KeyedStoreStubCompiler compiler(isolate(), strict_mode, grow_mode);
|
| + Handle<Code> code = compiler.CompileStoreElement(receiver_map);
|
|
|
| - if (stub_kind == KeyedIC::LOAD) {
|
| - PROFILE(isolate_, CodeCreateEvent(Logger::KEYED_LOAD_IC_TAG, *code, 0));
|
| - } else {
|
| - PROFILE(isolate_, CodeCreateEvent(Logger::KEYED_STORE_IC_TAG, *code, 0));
|
| - }
|
| + PROFILE(isolate(), CodeCreateEvent(Logger::KEYED_STORE_IC_TAG, *code, 0));
|
| Map::UpdateCodeCache(receiver_map, name, code);
|
| return code;
|
| }
|
| @@ -851,6 +830,41 @@ Handle<Code> StubCache::ComputeCallMiss(int argc,
|
| }
|
|
|
|
|
| +Handle<Code> StubCache::ComputeLoadElementPolymorphic(
|
| + MapHandleList* receiver_maps) {
|
| + Code::Flags flags = Code::ComputeFlags(Code::KEYED_LOAD_IC, POLYMORPHIC);
|
| + Handle<PolymorphicCodeCache> cache =
|
| + isolate_->factory()->polymorphic_code_cache();
|
| + Handle<Object> probe = cache->Lookup(receiver_maps, flags);
|
| + if (probe->IsCode()) return Handle<Code>::cast(probe);
|
| +
|
| + KeyedLoadStubCompiler compiler(isolate_);
|
| + Handle<Code> code = compiler.CompileLoadElementPolymorphic(receiver_maps);
|
| + PolymorphicCodeCache::Update(cache, receiver_maps, flags, code);
|
| + return code;
|
| +}
|
| +
|
| +
|
| +Handle<Code> StubCache::ComputeStoreElementPolymorphic(
|
| + MapHandleList* receiver_maps,
|
| + KeyedAccessGrowMode grow_mode,
|
| + StrictModeFlag strict_mode) {
|
| + Handle<PolymorphicCodeCache> cache =
|
| + isolate_->factory()->polymorphic_code_cache();
|
| + Code::ExtraICState extra_state = Code::ComputeExtraICState(grow_mode,
|
| + strict_mode);
|
| + Code::Flags flags =
|
| + Code::ComputeFlags(Code::KEYED_STORE_IC, POLYMORPHIC, extra_state);
|
| + Handle<Object> probe = cache->Lookup(receiver_maps, flags);
|
| + if (probe->IsCode()) return Handle<Code>::cast(probe);
|
| +
|
| + KeyedStoreStubCompiler compiler(isolate_, strict_mode, grow_mode);
|
| + Handle<Code> code = compiler.CompileStoreElementPolymorphic(receiver_maps);
|
| + PolymorphicCodeCache::Update(cache, receiver_maps, flags, code);
|
| + return code;
|
| +}
|
| +
|
| +
|
| #ifdef ENABLE_DEBUGGER_SUPPORT
|
| Handle<Code> StubCache::ComputeCallDebugBreak(int argc,
|
| Code::Kind kind) {
|
| @@ -1366,6 +1380,40 @@ Handle<Code> KeyedLoadStubCompiler::GetCode(Code::StubType type,
|
| }
|
|
|
|
|
| +Handle<Code> KeyedLoadStubCompiler::CompileLoadElementPolymorphic(
|
| + MapHandleList* receiver_maps) {
|
| + CodeHandleList handler_ics(receiver_maps->length());
|
| + for (int i = 0; i < receiver_maps->length(); ++i) {
|
| + Handle<Map> receiver_map = receiver_maps->at(i);
|
| + Handle<Code> cached_stub;
|
| +
|
| + if ((receiver_map->instance_type() & kNotStringTag) == 0) {
|
| + cached_stub = isolate()->builtins()->KeyedLoadIC_String();
|
| + } else {
|
| + bool is_js_array = receiver_map->instance_type() == JS_ARRAY_TYPE;
|
| + ElementsKind elements_kind = receiver_map->elements_kind();
|
| +
|
| + if (IsFastElementsKind(elements_kind) ||
|
| + IsExternalArrayElementsKind(elements_kind)) {
|
| + cached_stub =
|
| + KeyedLoadFastElementStub(is_js_array, elements_kind).GetCode();
|
| + } else {
|
| + ASSERT(elements_kind == DICTIONARY_ELEMENTS);
|
| + cached_stub = KeyedLoadDictionaryElementStub().GetCode();
|
| + }
|
| + }
|
| +
|
| + handler_ics.Add(cached_stub);
|
| + }
|
| + Handle<Code> code = CompileLoadPolymorphic(receiver_maps, &handler_ics);
|
| + isolate()->counters()->keyed_load_polymorphic_stubs()->Increment();
|
| + PROFILE(isolate(),
|
| + CodeCreateEvent(Logger::KEYED_LOAD_POLYMORPHIC_IC_TAG, *code, 0));
|
| + return code;
|
| +}
|
| +
|
| +
|
| +
|
| Handle<Code> StoreStubCompiler::GetCode(Code::StubType type,
|
| Handle<String> name) {
|
| Code::Flags flags =
|
| @@ -1391,6 +1439,50 @@ Handle<Code> KeyedStoreStubCompiler::GetCode(Code::StubType type,
|
| }
|
|
|
|
|
| +Handle<Code> KeyedStoreStubCompiler::CompileStoreElementPolymorphic(
|
| + MapHandleList* receiver_maps) {
|
| + // Collect MONOMORPHIC stubs for all |receiver_maps|.
|
| + CodeHandleList handler_ics(receiver_maps->length());
|
| + MapHandleList transitioned_maps(receiver_maps->length());
|
| + for (int i = 0; i < receiver_maps->length(); ++i) {
|
| + Handle<Map> receiver_map(receiver_maps->at(i));
|
| + Handle<Code> cached_stub;
|
| + Handle<Map> transitioned_map =
|
| + receiver_map->FindTransitionedMap(receiver_maps);
|
| +
|
| + // TODO(mvstanton): The code below is doing pessimistic elements
|
| + // transitions. I would like to stop doing that and rely on Allocation Site
|
| + // Tracking to do a better job of ensuring the data types are what they need
|
| + // to be. Not all the elements are in place yet, pessimistic elements
|
| + // transitions are still important for performance.
|
| + bool is_js_array = receiver_map->instance_type() == JS_ARRAY_TYPE;
|
| + ElementsKind elements_kind = receiver_map->elements_kind();
|
| + if (!transitioned_map.is_null()) {
|
| + cached_stub = ElementsTransitionAndStoreStub(
|
| + elements_kind,
|
| + transitioned_map->elements_kind(),
|
| + is_js_array,
|
| + strict_mode_,
|
| + grow_mode_).GetCode();
|
| + } else {
|
| + cached_stub = KeyedStoreElementStub(
|
| + is_js_array,
|
| + elements_kind,
|
| + grow_mode_).GetCode();
|
| + }
|
| + ASSERT(!cached_stub.is_null());
|
| + handler_ics.Add(cached_stub);
|
| + transitioned_maps.Add(transitioned_map);
|
| + }
|
| + Handle<Code> code =
|
| + CompileStorePolymorphic(receiver_maps, &handler_ics, &transitioned_maps);
|
| + isolate()->counters()->keyed_store_polymorphic_stubs()->Increment();
|
| + PROFILE(isolate(),
|
| + CodeCreateEvent(Logger::KEYED_STORE_POLYMORPHIC_IC_TAG, *code, 0));
|
| + return code;
|
| +}
|
| +
|
| +
|
| void KeyedStoreStubCompiler::GenerateStoreDictionaryElement(
|
| MacroAssembler* masm) {
|
| KeyedStoreIC::GenerateSlow(masm);
|
|
|