// Copyright 2018 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_HEAP_BROKER_H_ #define V8_COMPILER_JS_HEAP_BROKER_H_ #include "src/base/compiler-specific.h" #include "src/base/optional.h" #include "src/globals.h" #include "src/objects.h" #include "src/zone/zone-containers.h" namespace v8 { namespace internal { namespace compiler { enum class OddballType : uint8_t { kNone, // Not an Oddball. kBoolean, // True or False. kUndefined, kNull, kHole, kUninitialized, kOther // Oddball, but none of the above. }; // TODO(neis): Get rid of the HeapObjectType class. class HeapObjectType { public: enum Flag : uint8_t { kUndetectable = 1 << 0, kCallable = 1 << 1 }; typedef base::Flags<Flag> Flags; HeapObjectType(InstanceType instance_type, Flags flags, OddballType oddball_type) : instance_type_(instance_type), oddball_type_(oddball_type), flags_(flags) { DCHECK_EQ(instance_type == ODDBALL_TYPE, oddball_type != OddballType::kNone); } OddballType oddball_type() const { return oddball_type_; } InstanceType instance_type() const { return instance_type_; } Flags flags() const { return flags_; } bool is_callable() const { return flags_ & kCallable; } bool is_undetectable() const { return flags_ & kUndetectable; } private: InstanceType const instance_type_; OddballType const oddball_type_; Flags const flags_; }; // This list is sorted such that subtypes appear before their supertypes. // DO NOT VIOLATE THIS PROPERTY! #define HEAP_BROKER_OBJECT_LIST(V) \ /* Subtypes of JSObject */ \ V(JSArray) \ V(JSFunction) \ V(JSGlobalProxy) \ V(JSRegExp) \ /* Subtypes of Context */ \ V(NativeContext) \ /* Subtypes of FixedArrayBase */ \ V(BytecodeArray) \ V(FixedArray) \ V(FixedDoubleArray) \ /* Subtypes of Name */ \ V(InternalizedString) \ V(String) \ /* Subtypes of HeapObject */ \ V(AllocationSite) \ V(Cell) \ V(Code) \ V(FeedbackVector) \ V(Map) \ V(Module) \ V(ScopeInfo) \ V(ScriptContextTable) \ V(SharedFunctionInfo) \ V(Context) \ V(FixedArrayBase) \ V(HeapNumber) \ V(JSObject) \ V(MutableHeapNumber) \ V(Name) \ V(PropertyCell) \ /* Subtypes of Object */ \ V(HeapObject) class CompilationDependencies; class JSHeapBroker; class ObjectData; #define FORWARD_DECL(Name) class Name##Ref; HEAP_BROKER_OBJECT_LIST(FORWARD_DECL) #undef FORWARD_DECL class ObjectRef { public: ObjectRef(JSHeapBroker* broker, Handle<Object> object); explicit ObjectRef(ObjectData* data) : data_(data) { CHECK_NOT_NULL(data_); } bool equals(const ObjectRef& other) const; Handle<Object> object() const; // TODO(neis): Remove eventually. template <typename T> Handle<T> object() const { AllowHandleDereference handle_dereference; return Handle<T>::cast(object()); } OddballType oddball_type() const; bool IsSmi() const; int AsSmi() const; #define HEAP_IS_METHOD_DECL(Name) bool Is##Name() const; HEAP_BROKER_OBJECT_LIST(HEAP_IS_METHOD_DECL) #undef HEAP_IS_METHOD_DECL #define HEAP_AS_METHOD_DECL(Name) Name##Ref As##Name() const; HEAP_BROKER_OBJECT_LIST(HEAP_AS_METHOD_DECL) #undef HEAP_AS_METHOD_DECL StringRef TypeOf() const; bool BooleanValue(); double OddballToNumber() const; Isolate* isolate() const; protected: JSHeapBroker* broker() const; ObjectData* data() const; private: ObjectData* data_; }; class HeapObjectRef : public ObjectRef { public: using ObjectRef::ObjectRef; HeapObjectType type() const; MapRef map() const; base::Optional<MapRef> TryGetObjectCreateMap() const; bool IsSeqString() const; bool IsExternalString() const; }; class PropertyCellRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; ObjectRef value() const; PropertyDetails property_details() const; }; class JSObjectRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; bool IsUnboxedDoubleField(FieldIndex index) const; double RawFastDoublePropertyAt(FieldIndex index) const; ObjectRef RawFastPropertyAt(FieldIndex index) const; FixedArrayBaseRef elements() const; void EnsureElementsTenured(); ElementsKind GetElementsKind() const; }; class JSFunctionRef : public JSObjectRef { public: using JSObjectRef::JSObjectRef; bool IsConstructor() const; bool has_initial_map() const; MapRef initial_map() const; bool has_prototype() const; ObjectRef prototype() const; bool PrototypeRequiresRuntimeLookup() const; JSGlobalProxyRef global_proxy() const; int InitialMapInstanceSizeWithMinSlack() const; SharedFunctionInfoRef shared() const; }; class JSRegExpRef : public JSObjectRef { public: using JSObjectRef::JSObjectRef; ObjectRef raw_properties_or_hash() const; ObjectRef data() const; ObjectRef source() const; ObjectRef flags() const; ObjectRef last_index() const; }; class HeapNumberRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; double value() const; }; class MutableHeapNumberRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; double value() const; }; class ContextRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; base::Optional<ContextRef> previous() const; ObjectRef get(int index) const; }; #define BROKER_NATIVE_CONTEXT_FIELDS(V) \ V(JSFunction, array_function) \ V(JSFunction, object_function) \ V(JSFunction, promise_function) \ V(Map, fast_aliased_arguments_map) \ V(Map, initial_array_iterator_map) \ V(Map, iterator_result_map) \ V(Map, js_array_holey_double_elements_map) \ V(Map, js_array_holey_elements_map) \ V(Map, js_array_holey_smi_elements_map) \ V(Map, js_array_packed_double_elements_map) \ V(Map, js_array_packed_elements_map) \ V(Map, js_array_packed_smi_elements_map) \ V(Map, map_key_iterator_map) \ V(Map, map_key_value_iterator_map) \ V(Map, map_value_iterator_map) \ V(Map, set_key_value_iterator_map) \ V(Map, set_value_iterator_map) \ V(Map, sloppy_arguments_map) \ V(Map, strict_arguments_map) \ V(Map, string_iterator_map) \ V(ScriptContextTable, script_context_table) class NativeContextRef : public ContextRef { public: using ContextRef::ContextRef; #define DECL_ACCESSOR(type, name) type##Ref name() const; BROKER_NATIVE_CONTEXT_FIELDS(DECL_ACCESSOR) #undef DECL_ACCESSOR MapRef GetFunctionMapFromIndex(int index) const; MapRef GetInitialJSArrayMap(ElementsKind kind) const; }; class NameRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; }; class ScriptContextTableRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; struct LookupResult { ContextRef context; bool immutable; int index; }; base::Optional<LookupResult> lookup(const NameRef& name) const; }; class FeedbackVectorRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; ObjectRef get(FeedbackSlot slot) const; }; class AllocationSiteRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; bool PointsToLiteral() const; PretenureFlag GetPretenureMode() const; ObjectRef nested_site() const; // {IsFastLiteral} determines whether the given array or object literal // boilerplate satisfies all limits to be considered for fast deep-copying // and computes the total size of all objects that are part of the graph. // // If PointsToLiteral() is false, then IsFastLiteral() is also false. bool IsFastLiteral() const; // We only serialize boilerplate if IsFastLiteral is true. base::Optional<JSObjectRef> boilerplate() const; ElementsKind GetElementsKind() const; bool CanInlineCall() const; }; class MapRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; int instance_size() const; InstanceType instance_type() const; int GetInObjectProperties() const; int GetInObjectPropertiesStartInWords() const; int NumberOfOwnDescriptors() const; int GetInObjectPropertyOffset(int index) const; ElementsKind elements_kind() const; bool is_stable() const; bool has_prototype_slot() const; bool is_deprecated() const; bool CanBeDeprecated() const; bool CanTransition() const; bool IsInobjectSlackTrackingInProgress() const; bool is_dictionary_map() const; bool IsJSArrayMap() const; bool IsFixedCowArrayMap() const; ObjectRef constructor_or_backpointer() const; base::Optional<MapRef> AsElementsKind(ElementsKind kind) const; // Concerning the underlying instance_descriptors: MapRef FindFieldOwner(int descriptor) const; PropertyDetails GetPropertyDetails(int i) const; NameRef GetPropertyKey(int i) const; FieldIndex GetFieldIndexFor(int i) const; ObjectRef GetFieldType(int descriptor) const; }; class FixedArrayBaseRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; int length() const; }; class FixedArrayRef : public FixedArrayBaseRef { public: using FixedArrayBaseRef::FixedArrayBaseRef; ObjectRef get(int i) const; bool is_the_hole(int i) const; }; class FixedDoubleArrayRef : public FixedArrayBaseRef { public: using FixedArrayBaseRef::FixedArrayBaseRef; double get_scalar(int i) const; bool is_the_hole(int i) const; }; class BytecodeArrayRef : public FixedArrayBaseRef { public: using FixedArrayBaseRef::FixedArrayBaseRef; int register_count() const; }; class JSArrayRef : public JSObjectRef { public: using JSObjectRef::JSObjectRef; ObjectRef length() const; }; class ScopeInfoRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; int ContextLength() const; }; #define BROKER_SFI_FIELDS(V) \ V(int, internal_formal_parameter_count) \ V(bool, has_duplicate_parameters) \ V(int, function_map_index) \ V(FunctionKind, kind) \ V(LanguageMode, language_mode) \ V(bool, native) \ V(bool, HasBreakInfo) \ V(bool, HasBuiltinFunctionId) \ V(bool, HasBuiltinId) \ V(BuiltinFunctionId, builtin_function_id) \ V(bool, construct_as_builtin) \ V(bool, HasBytecodeArray) class SharedFunctionInfoRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; int builtin_id() const; BytecodeArrayRef GetBytecodeArray() const; #define DECL_ACCESSOR(type, name) type name() const; BROKER_SFI_FIELDS(DECL_ACCESSOR) #undef DECL_ACCSESOR }; class StringRef : public NameRef { public: using NameRef::NameRef; int length() const; uint16_t GetFirstChar(); base::Optional<double> ToNumber(); }; class ModuleRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; CellRef GetCell(int cell_index); }; class CellRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; }; class JSGlobalProxyRef : public JSObjectRef { public: using JSObjectRef::JSObjectRef; }; class CodeRef : public HeapObjectRef { public: using HeapObjectRef::HeapObjectRef; }; class InternalizedStringRef : public StringRef { public: using StringRef::StringRef; }; class V8_EXPORT_PRIVATE JSHeapBroker : public NON_EXPORTED_BASE(ZoneObject) { public: JSHeapBroker(Isolate* isolate, Zone* zone); void SerializeStandardObjects(); HeapObjectType HeapObjectTypeFromMap(Handle<Map> map) const { AllowHandleDereference handle_dereference; return HeapObjectTypeFromMap(*map); } Isolate* isolate() const { return isolate_; } Zone* zone() const { return zone_; } enum BrokerMode { kDisabled, kSerializing, kSerialized }; BrokerMode mode() const { return mode_; } void StopSerializing() { CHECK_EQ(mode_, kSerializing); mode_ = kSerialized; } bool SerializingAllowed() const; // Returns nullptr iff handle unknown. ObjectData* GetData(Handle<Object>) const; // Never returns nullptr. ObjectData* GetOrCreateData(Handle<Object>); void Trace(const char* format, ...) const; private: friend class HeapObjectRef; friend class ObjectRef; friend class ObjectData; // TODO(neis): Remove eventually. HeapObjectType HeapObjectTypeFromMap(Map* map) const; void AddData(Handle<Object> object, ObjectData* data); Isolate* const isolate_; Zone* const zone_; ZoneUnorderedMap<Address, ObjectData*> refs_; BrokerMode mode_; }; #define ASSIGN_RETURN_NO_CHANGE_IF_DATA_MISSING(something_var, \ optionally_something) \ auto optionally_something_ = optionally_something; \ if (!optionally_something_) \ return NoChangeBecauseOfMissingData(js_heap_broker(), __FUNCTION__, \ __LINE__); \ something_var = *optionally_something_; class Reduction; Reduction NoChangeBecauseOfMissingData(JSHeapBroker* broker, const char* function, int line); } // namespace compiler } // namespace internal } // namespace v8 #endif // V8_COMPILER_JS_HEAP_BROKER_H_