// Copyright 2016 the V8 project authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "src/builtins/builtins-utils.h" #include "src/builtins/builtins.h" #include "src/code-stub-assembler.h" #include "src/ic/handler-compiler.h" #include "src/ic/ic.h" #include "src/ic/keyed-store-generic.h" #include "src/objects-inl.h" namespace v8 { namespace internal { TF_BUILTIN(KeyedLoadIC_IndexedString, CodeStubAssembler) { typedef LoadWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* index = Parameter(Descriptor::kName); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); Label miss(this); Node* index_intptr = TryToIntptr(index, &miss); Node* length = SmiUntag(LoadStringLength(receiver)); GotoIf(UintPtrGreaterThanOrEqual(index_intptr, length), &miss); Node* code = StringCharCodeAt(receiver, index_intptr, INTPTR_PARAMETERS); Node* result = StringFromCharCode(code); Return(result); Bind(&miss); TailCallRuntime(Runtime::kKeyedLoadIC_Miss, context, receiver, index, slot, vector); } TF_BUILTIN(KeyedLoadIC_Miss, CodeStubAssembler) { typedef LoadWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); TailCallRuntime(Runtime::kKeyedLoadIC_Miss, context, receiver, name, slot, vector); } TF_BUILTIN(KeyedLoadIC_Slow, CodeStubAssembler) { typedef LoadWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* context = Parameter(Descriptor::kContext); TailCallRuntime(Runtime::kKeyedGetProperty, context, receiver, name); } void Builtins::Generate_KeyedStoreIC_Megamorphic( compiler::CodeAssemblerState* state) { KeyedStoreGenericGenerator::Generate(state, SLOPPY); } void Builtins::Generate_KeyedStoreIC_Megamorphic_Strict( compiler::CodeAssemblerState* state) { KeyedStoreGenericGenerator::Generate(state, STRICT); } TF_BUILTIN(KeyedStoreIC_Miss, CodeStubAssembler) { typedef StoreWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* value = Parameter(Descriptor::kValue); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); TailCallRuntime(Runtime::kKeyedStoreIC_Miss, context, value, slot, vector, receiver, name); } TF_BUILTIN(KeyedStoreIC_Slow, CodeStubAssembler) { typedef StoreWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* value = Parameter(Descriptor::kValue); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); // The slow case calls into the runtime to complete the store without causing // an IC miss that would otherwise cause a transition to the generic stub. TailCallRuntime(Runtime::kKeyedStoreIC_Slow, context, value, slot, vector, receiver, name); } TF_BUILTIN(LoadGlobalIC_Miss, CodeStubAssembler) { typedef LoadGlobalWithVectorDescriptor Descriptor; Node* name = Parameter(Descriptor::kName); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); TailCallRuntime(Runtime::kLoadGlobalIC_Miss, context, name, slot, vector); } TF_BUILTIN(LoadGlobalIC_Slow, CodeStubAssembler) { typedef LoadGlobalWithVectorDescriptor Descriptor; Node* name = Parameter(Descriptor::kName); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); TailCallRuntime(Runtime::kLoadGlobalIC_Slow, context, name, slot, vector); } void Builtins::Generate_LoadIC_Getter_ForDeopt(MacroAssembler* masm) { NamedLoadHandlerCompiler::GenerateLoadViaGetterForDeopt(masm); } TF_BUILTIN(LoadIC_FunctionPrototype, CodeStubAssembler) { typedef LoadWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); Label miss(this); Node* proto_or_map = LoadObjectField(receiver, JSFunction::kPrototypeOrInitialMapOffset); GotoIf(IsTheHole(proto_or_map), &miss); Variable var_result(this, MachineRepresentation::kTagged, proto_or_map); Label done(this, &var_result); GotoIfNot(IsMap(proto_or_map), &done); var_result.Bind(LoadMapPrototype(proto_or_map)); Goto(&done); Bind(&done); Return(var_result.value()); Bind(&miss); TailCallRuntime(Runtime::kLoadIC_Miss, context, receiver, name, slot, vector); } TF_BUILTIN(LoadIC_Miss, CodeStubAssembler) { typedef LoadWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); TailCallRuntime(Runtime::kLoadIC_Miss, context, receiver, name, slot, vector); } TF_BUILTIN(LoadIC_Normal, CodeStubAssembler) { typedef LoadWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* context = Parameter(Descriptor::kContext); Label slow(this); { Node* properties = LoadProperties(receiver); Variable var_name_index(this, MachineType::PointerRepresentation()); Label found(this, &var_name_index); NameDictionaryLookup<NameDictionary>(properties, name, &found, &var_name_index, &slow); Bind(&found); { Variable var_details(this, MachineRepresentation::kWord32); Variable var_value(this, MachineRepresentation::kTagged); LoadPropertyFromNameDictionary(properties, var_name_index.value(), &var_details, &var_value); Node* value = CallGetterIfAccessor(var_value.value(), var_details.value(), context, receiver, &slow); Return(value); } } Bind(&slow); TailCallRuntime(Runtime::kGetProperty, context, receiver, name); } TF_BUILTIN(LoadIC_Slow, CodeStubAssembler) { typedef LoadWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* context = Parameter(Descriptor::kContext); TailCallRuntime(Runtime::kGetProperty, context, receiver, name); } TF_BUILTIN(StoreIC_Miss, CodeStubAssembler) { typedef StoreWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* value = Parameter(Descriptor::kValue); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); TailCallRuntime(Runtime::kStoreIC_Miss, context, value, slot, vector, receiver, name); } TF_BUILTIN(StoreIC_Normal, CodeStubAssembler) { typedef StoreWithVectorDescriptor Descriptor; Node* receiver = Parameter(Descriptor::kReceiver); Node* name = Parameter(Descriptor::kName); Node* value = Parameter(Descriptor::kValue); Node* slot = Parameter(Descriptor::kSlot); Node* vector = Parameter(Descriptor::kVector); Node* context = Parameter(Descriptor::kContext); Label slow(this); { Node* properties = LoadProperties(receiver); Variable var_name_index(this, MachineType::PointerRepresentation()); Label found(this, &var_name_index); NameDictionaryLookup<NameDictionary>(properties, name, &found, &var_name_index, &slow); Bind(&found); { Node* details = LoadDetailsByKeyIndex<NameDictionary>( properties, var_name_index.value()); // Check that the property is a writable data property (no accessor). const int kTypeAndReadOnlyMask = PropertyDetails::KindField::kMask | PropertyDetails::kAttributesReadOnlyMask; STATIC_ASSERT(kData == 0); GotoIf(IsSetWord32(details, kTypeAndReadOnlyMask), &slow); StoreValueByKeyIndex<NameDictionary>(properties, var_name_index.value(), value); Return(value); } } Bind(&slow); TailCallRuntime(Runtime::kStoreIC_Miss, context, value, slot, vector, receiver, name); } void Builtins::Generate_StoreIC_Setter_ForDeopt(MacroAssembler* masm) { NamedStoreHandlerCompiler::GenerateStoreViaSetterForDeopt(masm); } } // namespace internal } // namespace v8