// Copyright 2017 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. #ifndef V8_BUILTINS_BUILTINS_ARRAY_GEN_H_ #define V8_BUILTINS_BUILTINS_ARRAY_GEN_H_ #include "src/codegen/code-stub-assembler.h" namespace v8 { namespace internal { class ArrayBuiltinsAssembler : public CodeStubAssembler { public: explicit ArrayBuiltinsAssembler(compiler::CodeAssemblerState* state); using BuiltinResultGenerator = std::function; using CallResultProcessor = std::function; using PostLoopAction = std::function; void FindResultGenerator(); Node* FindProcessor(Node* k_value, Node* k); void FindIndexResultGenerator(); Node* FindIndexProcessor(Node* k_value, Node* k); void ForEachResultGenerator(); Node* ForEachProcessor(Node* k_value, Node* k); void SomeResultGenerator(); Node* SomeProcessor(Node* k_value, Node* k); void EveryResultGenerator(); Node* EveryProcessor(Node* k_value, Node* k); void ReduceResultGenerator(); Node* ReduceProcessor(Node* k_value, Node* k); void ReducePostLoopAction(); void TypedArrayMapResultGenerator(); Node* SpecCompliantMapProcessor(Node* k_value, Node* k); Node* FastMapProcessor(Node* k_value, Node* k); // See tc39.github.io/ecma262/#sec-%typedarray%.prototype.map. Node* TypedArrayMapProcessor(Node* k_value, Node* k); void NullPostLoopAction(); // Uses memset to effectively initialize the given FixedArray with Smi zeroes. void FillFixedArrayWithSmiZero(TNode array, TNode smi_length); TNode CallJSArrayArrayJoinConcatToSequentialString( TNode fixed_array, TNode length, TNode sep, TNode dest) { TNode func = ExternalConstant( ExternalReference::jsarray_array_join_concat_to_sequential_string()); TNode isolate_ptr = ExternalConstant(ExternalReference::isolate_address(isolate())); return UncheckedCast( CallCFunction(func, MachineType::AnyTagged(), // String std::make_pair(MachineType::Pointer(), isolate_ptr), std::make_pair(MachineType::AnyTagged(), fixed_array), std::make_pair(MachineType::IntPtr(), length), std::make_pair(MachineType::AnyTagged(), sep), std::make_pair(MachineType::AnyTagged(), dest))); } protected: TNode context() { return context_; } TNode receiver() { return receiver_; } TNode argc() { return argc_; } TNode o() { return o_; } TNode len() { return len_; } Node* callbackfn() { return callbackfn_; } Node* this_arg() { return this_arg_; } TNode k() { return CAST(k_.value()); } Node* a() { return a_.value(); } void ReturnFromBuiltin(Node* value); void InitIteratingArrayBuiltinBody(TNode context, TNode receiver, Node* callbackfn, Node* this_arg, TNode argc); void GenerateIteratingTypedArrayBuiltinBody( const char* name, const BuiltinResultGenerator& generator, const CallResultProcessor& processor, const PostLoopAction& action, ForEachDirection direction = ForEachDirection::kForward); void TailCallArrayConstructorStub( const Callable& callable, TNode context, TNode target, TNode allocation_site_or_undefined, TNode argc); void GenerateDispatchToArrayStub( TNode context, TNode target, TNode argc, AllocationSiteOverrideMode mode, TNode allocation_site = TNode()); void CreateArrayDispatchNoArgument( TNode context, TNode target, TNode argc, AllocationSiteOverrideMode mode, TNode allocation_site = TNode()); void CreateArrayDispatchSingleArgument( TNode context, TNode target, TNode argc, AllocationSiteOverrideMode mode, TNode allocation_site = TNode()); void GenerateConstructor(Node* context, Node* array_function, Node* array_map, Node* array_size, Node* allocation_site, ElementsKind elements_kind, AllocationSiteMode mode); void GenerateArrayNoArgumentConstructor(ElementsKind kind, AllocationSiteOverrideMode mode); void GenerateArraySingleArgumentConstructor(ElementsKind kind, AllocationSiteOverrideMode mode); void GenerateArrayNArgumentsConstructor( TNode context, TNode target, TNode new_target, TNode argc, TNode maybe_allocation_site); private: static ElementsKind ElementsKindForInstanceType(InstanceType type); void VisitAllTypedArrayElements(Node* array_buffer, const CallResultProcessor& processor, Label* detached, ForEachDirection direction, TNode typed_array); // Perform ArraySpeciesCreate (ES6 #sec-arrayspeciescreate). // This version is specialized to create a zero length array // of the elements kind of the input array. void GenerateArraySpeciesCreate(); // Perform ArraySpeciesCreate (ES6 #sec-arrayspeciescreate). void GenerateArraySpeciesCreate(TNode len); Node* callbackfn_ = nullptr; TNode o_; Node* this_arg_ = nullptr; TNode len_; TNode context_; TNode receiver_; TNode argc_; Node* fast_typed_array_target_ = nullptr; const char* name_ = nullptr; Variable k_; Variable a_; Variable to_; Label fully_spec_compliant_; ElementsKind source_elements_kind_ = ElementsKind::NO_ELEMENTS; }; } // namespace internal } // namespace v8 #endif // V8_BUILTINS_BUILTINS_ARRAY_GEN_H_