// Copyright 2015 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_COMPILER_JS_NATIVE_CONTEXT_SPECIALIZATION_H_
|
#define V8_COMPILER_JS_NATIVE_CONTEXT_SPECIALIZATION_H_
|
|
#include "src/base/flags.h"
|
#include "src/compiler/graph-reducer.h"
|
#include "src/deoptimize-reason.h"
|
#include "src/objects/map.h"
|
|
namespace v8 {
|
namespace internal {
|
|
// Forward declarations.
|
class Factory;
|
class FeedbackNexus;
|
|
namespace compiler {
|
|
// Forward declarations.
|
enum class AccessMode;
|
class CommonOperatorBuilder;
|
class CompilationDependencies;
|
class ElementAccessInfo;
|
class JSGraph;
|
class JSHeapBroker;
|
class JSOperatorBuilder;
|
class MachineOperatorBuilder;
|
class PropertyAccessInfo;
|
class SimplifiedOperatorBuilder;
|
class TypeCache;
|
|
// Specializes a given JSGraph to a given native context, potentially constant
|
// folding some {LoadGlobal} nodes or strength reducing some {StoreGlobal}
|
// nodes. And also specializes {LoadNamed} and {StoreNamed} nodes according
|
// to type feedback (if available).
|
class JSNativeContextSpecialization final : public AdvancedReducer {
|
public:
|
// Flags that control the mode of operation.
|
enum Flag {
|
kNoFlags = 0u,
|
kAccessorInliningEnabled = 1u << 0,
|
kBailoutOnUninitialized = 1u << 1
|
};
|
typedef base::Flags<Flag> Flags;
|
|
JSNativeContextSpecialization(Editor* editor, JSGraph* jsgraph,
|
JSHeapBroker* js_heap_broker, Flags flags,
|
Handle<Context> native_context,
|
CompilationDependencies* dependencies,
|
Zone* zone);
|
|
const char* reducer_name() const override {
|
return "JSNativeContextSpecialization";
|
}
|
|
Reduction Reduce(Node* node) final;
|
|
private:
|
Reduction ReduceJSAdd(Node* node);
|
Reduction ReduceJSGetSuperConstructor(Node* node);
|
Reduction ReduceJSInstanceOf(Node* node);
|
Reduction ReduceJSHasInPrototypeChain(Node* node);
|
Reduction ReduceJSOrdinaryHasInstance(Node* node);
|
Reduction ReduceJSPromiseResolve(Node* node);
|
Reduction ReduceJSResolvePromise(Node* node);
|
Reduction ReduceJSLoadContext(Node* node);
|
Reduction ReduceJSLoadGlobal(Node* node);
|
Reduction ReduceJSStoreGlobal(Node* node);
|
Reduction ReduceJSLoadNamed(Node* node);
|
Reduction ReduceJSStoreNamed(Node* node);
|
Reduction ReduceJSLoadProperty(Node* node);
|
Reduction ReduceJSStoreProperty(Node* node);
|
Reduction ReduceJSStoreNamedOwn(Node* node);
|
Reduction ReduceJSStoreDataPropertyInLiteral(Node* node);
|
Reduction ReduceJSStoreInArrayLiteral(Node* node);
|
Reduction ReduceJSToObject(Node* node);
|
|
Reduction ReduceElementAccess(Node* node, Node* index, Node* value,
|
MapHandles const& receiver_maps,
|
AccessMode access_mode,
|
KeyedAccessLoadMode load_mode,
|
KeyedAccessStoreMode store_mode);
|
Reduction ReduceKeyedAccess(Node* node, Node* index, Node* value,
|
FeedbackNexus const& nexus,
|
AccessMode access_mode,
|
KeyedAccessLoadMode load_mode,
|
KeyedAccessStoreMode store_mode);
|
Reduction ReduceNamedAccessFromNexus(Node* node, Node* value,
|
FeedbackNexus const& nexus,
|
Handle<Name> name,
|
AccessMode access_mode);
|
Reduction ReduceNamedAccess(Node* node, Node* value,
|
MapHandles const& receiver_maps,
|
Handle<Name> name, AccessMode access_mode,
|
Node* index = nullptr);
|
Reduction ReduceGlobalAccess(Node* node, Node* receiver, Node* value,
|
Handle<Name> name, AccessMode access_mode,
|
Node* index = nullptr);
|
|
Reduction ReduceSoftDeoptimize(Node* node, DeoptimizeReason reason);
|
|
// A triple of nodes that represents a continuation.
|
class ValueEffectControl final {
|
public:
|
ValueEffectControl()
|
: value_(nullptr), effect_(nullptr), control_(nullptr) {}
|
ValueEffectControl(Node* value, Node* effect, Node* control)
|
: value_(value), effect_(effect), control_(control) {}
|
|
Node* value() const { return value_; }
|
Node* effect() const { return effect_; }
|
Node* control() const { return control_; }
|
|
private:
|
Node* value_;
|
Node* effect_;
|
Node* control_;
|
};
|
|
// Construct the appropriate subgraph for property access.
|
ValueEffectControl BuildPropertyAccess(Node* receiver, Node* value,
|
Node* context, Node* frame_state,
|
Node* effect, Node* control,
|
Handle<Name> name,
|
ZoneVector<Node*>* if_exceptions,
|
PropertyAccessInfo const& access_info,
|
AccessMode access_mode);
|
ValueEffectControl BuildPropertyLoad(Node* receiver, Node* context,
|
Node* frame_state, Node* effect,
|
Node* control, Handle<Name> name,
|
ZoneVector<Node*>* if_exceptions,
|
PropertyAccessInfo const& access_info);
|
|
ValueEffectControl BuildPropertyStore(Node* receiver, Node* value,
|
Node* context, Node* frame_state,
|
Node* effect, Node* control,
|
Handle<Name> name,
|
ZoneVector<Node*>* if_exceptions,
|
PropertyAccessInfo const& access_info,
|
AccessMode access_mode);
|
|
// Helpers for accessor inlining.
|
Node* InlinePropertyGetterCall(Node* receiver, Node* context,
|
Node* frame_state, Node** effect,
|
Node** control,
|
ZoneVector<Node*>* if_exceptions,
|
PropertyAccessInfo const& access_info);
|
void InlinePropertySetterCall(Node* receiver, Node* value, Node* context,
|
Node* frame_state, Node** effect,
|
Node** control,
|
ZoneVector<Node*>* if_exceptions,
|
PropertyAccessInfo const& access_info);
|
Node* InlineApiCall(Node* receiver, Node* holder, Node* frame_state,
|
Node* value, Node** effect, Node** control,
|
Handle<SharedFunctionInfo> shared_info,
|
Handle<FunctionTemplateInfo> function_template_info);
|
|
// Construct the appropriate subgraph for element access.
|
ValueEffectControl BuildElementAccess(
|
Node* receiver, Node* index, Node* value, Node* effect, Node* control,
|
ElementAccessInfo const& access_info, AccessMode access_mode,
|
KeyedAccessLoadMode load_mode, KeyedAccessStoreMode store_mode);
|
|
// Construct appropriate subgraph to load from a String.
|
Node* BuildIndexedStringLoad(Node* receiver, Node* index, Node* length,
|
Node** effect, Node** control,
|
KeyedAccessLoadMode load_mode);
|
|
// Construct appropriate subgraph to extend properties backing store.
|
Node* BuildExtendPropertiesBackingStore(Handle<Map> map, Node* properties,
|
Node* effect, Node* control);
|
|
// Construct appropriate subgraph to check that the {value} matches
|
// the previously recorded {name} feedback.
|
Node* BuildCheckEqualsName(Handle<Name> name, Node* value, Node* effect,
|
Node* control);
|
|
// Checks if we can turn the hole into undefined when loading an element
|
// from an object with one of the {receiver_maps}; sets up appropriate
|
// code dependencies and might use the array protector cell.
|
bool CanTreatHoleAsUndefined(MapHandles const& receiver_maps);
|
|
// Extract receiver maps from {nexus} and filter based on {receiver} if
|
// possible.
|
bool ExtractReceiverMaps(Node* receiver, Node* effect,
|
FeedbackNexus const& nexus,
|
MapHandles* receiver_maps);
|
|
// Try to infer maps for the given {receiver} at the current {effect}.
|
// If maps are returned then you can be sure that the {receiver} definitely
|
// has one of the returned maps at this point in the program (identified
|
// by {effect}).
|
bool InferReceiverMaps(Node* receiver, Node* effect,
|
MapHandles* receiver_maps);
|
// Try to infer a root map for the {receiver} independent of the current
|
// program location.
|
MaybeHandle<Map> InferReceiverRootMap(Node* receiver);
|
|
// Checks if we know at compile time that the {receiver} either definitely
|
// has the {prototype} in it's prototype chain, or the {receiver} definitely
|
// doesn't have the {prototype} in it's prototype chain.
|
enum InferHasInPrototypeChainResult {
|
kIsInPrototypeChain,
|
kIsNotInPrototypeChain,
|
kMayBeInPrototypeChain
|
};
|
InferHasInPrototypeChainResult InferHasInPrototypeChain(
|
Node* receiver, Node* effect, Handle<HeapObject> prototype);
|
|
// Script context lookup logic.
|
struct ScriptContextTableLookupResult;
|
bool LookupInScriptContextTable(Handle<Name> name,
|
ScriptContextTableLookupResult* result);
|
|
Graph* graph() const;
|
JSGraph* jsgraph() const { return jsgraph_; }
|
|
JSHeapBroker* js_heap_broker() const { return js_heap_broker_; }
|
Isolate* isolate() const;
|
Factory* factory() const;
|
CommonOperatorBuilder* common() const;
|
JSOperatorBuilder* javascript() const;
|
SimplifiedOperatorBuilder* simplified() const;
|
Flags flags() const { return flags_; }
|
Handle<JSGlobalObject> global_object() const { return global_object_; }
|
Handle<JSGlobalProxy> global_proxy() const { return global_proxy_; }
|
const NativeContextRef& native_context() const { return native_context_; }
|
CompilationDependencies* dependencies() const { return dependencies_; }
|
Zone* zone() const { return zone_; }
|
|
JSGraph* const jsgraph_;
|
JSHeapBroker* const js_heap_broker_;
|
Flags const flags_;
|
Handle<JSGlobalObject> global_object_;
|
Handle<JSGlobalProxy> global_proxy_;
|
NativeContextRef native_context_;
|
CompilationDependencies* const dependencies_;
|
Zone* const zone_;
|
TypeCache const& type_cache_;
|
|
DISALLOW_COPY_AND_ASSIGN(JSNativeContextSpecialization);
|
};
|
|
DEFINE_OPERATORS_FOR_FLAGS(JSNativeContextSpecialization::Flags)
|
|
} // namespace compiler
|
} // namespace internal
|
} // namespace v8
|
|
#endif // V8_COMPILER_JS_NATIVE_CONTEXT_SPECIALIZATION_H_
|