|  | // Copyright (c) 2012, the Dart project authors.  Please see the AUTHORS file | 
|  | // for details. All rights reserved. Use of this source code is governed by a | 
|  | // BSD-style license that can be found in the LICENSE file. | 
|  |  | 
|  | #ifndef RUNTIME_VM_OBJECT_H_ | 
|  | #define RUNTIME_VM_OBJECT_H_ | 
|  |  | 
|  | #if defined(SHOULD_NOT_INCLUDE_RUNTIME) | 
|  | #error "Should not include runtime" | 
|  | #endif | 
|  |  | 
|  | #include <limits> | 
|  | #include <tuple> | 
|  | #include <utility> | 
|  |  | 
|  | #include "include/dart_api.h" | 
|  | #include "platform/assert.h" | 
|  | #include "platform/atomic.h" | 
|  | #include "platform/thread_sanitizer.h" | 
|  | #include "platform/utils.h" | 
|  | #include "vm/bitmap.h" | 
|  | #include "vm/code_comments.h" | 
|  | #include "vm/code_entry_kind.h" | 
|  | #include "vm/compiler/assembler/object_pool_builder.h" | 
|  | #include "vm/compiler/method_recognizer.h" | 
|  | #include "vm/compiler/runtime_api.h" | 
|  | #include "vm/dart.h" | 
|  | #include "vm/flags.h" | 
|  | #include "vm/globals.h" | 
|  | #include "vm/growable_array.h" | 
|  | #include "vm/handles.h" | 
|  | #include "vm/heap/heap.h" | 
|  | #include "vm/isolate.h" | 
|  | #include "vm/json_stream.h" | 
|  | #include "vm/os.h" | 
|  | #include "vm/raw_object.h" | 
|  | #include "vm/report.h" | 
|  | #include "vm/static_type_exactness_state.h" | 
|  | #include "vm/thread.h" | 
|  | #include "vm/token_position.h" | 
|  |  | 
|  | namespace dart { | 
|  |  | 
|  | // Forward declarations. | 
|  | namespace compiler { | 
|  | class Assembler; | 
|  | } | 
|  |  | 
|  | namespace kernel { | 
|  | class Program; | 
|  | class TreeNode; | 
|  | }  // namespace kernel | 
|  |  | 
|  | #define DEFINE_FORWARD_DECLARATION(clazz) class clazz; | 
|  | CLASS_LIST(DEFINE_FORWARD_DECLARATION) | 
|  | #undef DEFINE_FORWARD_DECLARATION | 
|  | class Api; | 
|  | class ArgumentsDescriptor; | 
|  | class BitVector; | 
|  | class Closure; | 
|  | class Code; | 
|  | class DeoptInstr; | 
|  | class DisassemblyFormatter; | 
|  | class FinalizablePersistentHandle; | 
|  | class FlowGraphCompiler; | 
|  | class HierarchyInfo; | 
|  | class LocalScope; | 
|  | class CallSiteResetter; | 
|  | class CodeStatistics; | 
|  | class IsolateGroupReloadContext; | 
|  | class ObjectGraphCopier; | 
|  | class FunctionTypeMapping; | 
|  | class NativeArguments; | 
|  |  | 
|  | #define REUSABLE_FORWARD_DECLARATION(name) class Reusable##name##HandleScope; | 
|  | REUSABLE_HANDLE_LIST(REUSABLE_FORWARD_DECLARATION) | 
|  | #undef REUSABLE_FORWARD_DECLARATION | 
|  |  | 
|  | class Symbols; | 
|  | class BaseTextBuffer; | 
|  |  | 
|  | #if defined(DEBUG) | 
|  | #define CHECK_HANDLE() CheckHandle(); | 
|  | #else | 
|  | #define CHECK_HANDLE() | 
|  | #endif | 
|  |  | 
|  | // For AllStatic classes like OneByteString. Checks that | 
|  | // ContainsCompressedPointers() returns the same value for AllStatic class and | 
|  | // class used for handles. | 
|  | #define ALLSTATIC_CONTAINS_COMPRESSED_IMPLEMENTATION(object, handle)           \ | 
|  | public: /* NOLINT */                                                          \ | 
|  | using UntaggedObjectType = dart::Untagged##object;                           \ | 
|  | using ObjectPtrType = dart::object##Ptr;                                     \ | 
|  | static_assert(std::is_base_of<dart::handle##Ptr, ObjectPtrType>::value,      \ | 
|  | #object "Ptr must be a subtype of " #handle "Ptr");            \ | 
|  | static_assert(dart::handle::ContainsCompressedPointers() ==                  \ | 
|  | UntaggedObjectType::kContainsCompressedPointers,           \ | 
|  | "Pointer compression in Untagged" #object                      \ | 
|  | " must match pointer compression in Untagged" #handle);        \ | 
|  | static constexpr bool ContainsCompressedPointers() {                         \ | 
|  | return UntaggedObjectType::kContainsCompressedPointers;                    \ | 
|  | }                                                                            \ | 
|  | \ | 
|  | private: /* NOLINT */ | 
|  |  | 
|  | #define BASE_OBJECT_IMPLEMENTATION(object, super)                              \ | 
|  | public: /* NOLINT */                                                          \ | 
|  | using UntaggedObjectType = dart::Untagged##object;                           \ | 
|  | using ObjectPtrType = dart::object##Ptr;                                     \ | 
|  | static_assert(!dart::super::ContainsCompressedPointers() ||                  \ | 
|  | UntaggedObjectType::kContainsCompressedPointers,           \ | 
|  | "Untagged" #object                                             \ | 
|  | " must have compressed pointers, as supertype Untagged" #super \ | 
|  | " has compressed pointers");                                   \ | 
|  | static constexpr bool ContainsCompressedPointers() {                         \ | 
|  | return UntaggedObjectType::kContainsCompressedPointers;                    \ | 
|  | }                                                                            \ | 
|  | object##Ptr ptr() const {                                                    \ | 
|  | return static_cast<object##Ptr>(ptr_);                                     \ | 
|  | }                                                                            \ | 
|  | intptr_t HandleTag() const {                                                 \ | 
|  | return k##object##Cid;                                                     \ | 
|  | }                                                                            \ | 
|  | bool Is##object() const {                                                    \ | 
|  | return true;                                                               \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& Handle() {                                      \ | 
|  | return static_cast<object&>(                                               \ | 
|  | HandleImpl(Thread::Current()->zone(), object::null(), kClassId));      \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& Handle(Zone* zone) {                            \ | 
|  | return static_cast<object&>(HandleImpl(zone, object::null(), kClassId));   \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& Handle(object##Ptr ptr) {                       \ | 
|  | return static_cast<object&>(                                               \ | 
|  | HandleImpl(Thread::Current()->zone(), ptr, kClassId));                 \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& Handle(Zone* zone, object##Ptr ptr) {           \ | 
|  | return static_cast<object&>(HandleImpl(zone, ptr, kClassId));              \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& ZoneHandle() {                                  \ | 
|  | return static_cast<object&>(                                               \ | 
|  | ZoneHandleImpl(Thread::Current()->zone(), object::null(), kClassId));  \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& ZoneHandle(Zone* zone) {                        \ | 
|  | return static_cast<object&>(                                               \ | 
|  | ZoneHandleImpl(zone, object::null(), kClassId));                       \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& ZoneHandle(object##Ptr ptr) {                   \ | 
|  | return static_cast<object&>(                                               \ | 
|  | ZoneHandleImpl(Thread::Current()->zone(), ptr, kClassId));             \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& ZoneHandle(Zone* zone, object##Ptr ptr) {       \ | 
|  | return static_cast<object&>(ZoneHandleImpl(zone, ptr, kClassId));          \ | 
|  | }                                                                            \ | 
|  | static object* ReadOnlyHandle() {                                            \ | 
|  | return static_cast<object*>(ReadOnlyHandleImpl(kClassId));                 \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& CheckedHandle(Zone* zone, ObjectPtr ptr) {      \ | 
|  | object* obj = reinterpret_cast<object*>(VMHandles::AllocateHandle(zone));  \ | 
|  | initializeHandle(obj, ptr);                                                \ | 
|  | if (!obj->Is##object()) {                                                  \ | 
|  | FATAL("Handle check failed: saw %s expected %s", obj->ToCString(),       \ | 
|  | #object);                                                          \ | 
|  | }                                                                          \ | 
|  | return *obj;                                                               \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& CheckedZoneHandle(Zone* zone, ObjectPtr ptr) {  \ | 
|  | object* obj =                                                              \ | 
|  | reinterpret_cast<object*>(VMHandles::AllocateZoneHandle(zone));        \ | 
|  | initializeHandle(obj, ptr);                                                \ | 
|  | if (!obj->Is##object()) {                                                  \ | 
|  | FATAL("Handle check failed: saw %s expected %s", obj->ToCString(),       \ | 
|  | #object);                                                          \ | 
|  | }                                                                          \ | 
|  | return *obj;                                                               \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE static object& CheckedZoneHandle(ObjectPtr ptr) {              \ | 
|  | return CheckedZoneHandle(Thread::Current()->zone(), ptr);                  \ | 
|  | }                                                                            \ | 
|  | /* T::Cast cannot be applied to a null Object, because the object vtable */  \ | 
|  | /* is not setup for type T, although some methods are supposed to work   */  \ | 
|  | /* with null, for example Instance::Equals().                            */  \ | 
|  | static const object& Cast(const Object& obj) {                               \ | 
|  | ASSERT(obj.Is##object());                                                  \ | 
|  | return reinterpret_cast<const object&>(obj);                               \ | 
|  | }                                                                            \ | 
|  | static object##Ptr RawCast(ObjectPtr raw) {                                  \ | 
|  | ASSERT(Is##object##NoHandle(raw));                                         \ | 
|  | return static_cast<object##Ptr>(raw);                                      \ | 
|  | }                                                                            \ | 
|  | static object##Ptr null() {                                                  \ | 
|  | return static_cast<object##Ptr>(Object::null());                           \ | 
|  | }                                                                            \ | 
|  | virtual const char* ToCString() const;                                       \ | 
|  | static const ClassId kClassId = k##object##Cid;                              \ | 
|  | \ | 
|  | private: /* NOLINT */                                                         \ | 
|  | /* Initialize the handle based on the ptr in the presence of null. */        \ | 
|  | static void initializeHandle(object* obj, ObjectPtr ptr) {                   \ | 
|  | obj->setPtr(ptr, kClassId);                                                \ | 
|  | }                                                                            \ | 
|  | /* Disallow allocation, copy constructors and override super assignment. */  \ | 
|  | public: /* NOLINT */                                                          \ | 
|  | void operator delete(void* pointer) {                                        \ | 
|  | UNREACHABLE();                                                             \ | 
|  | }                                                                            \ | 
|  | \ | 
|  | private: /* NOLINT */                                                         \ | 
|  | void* operator new(size_t size);                                             \ | 
|  | object(const object& value) = delete;                                        \ | 
|  | void operator=(super##Ptr value) = delete;                                   \ | 
|  | void operator=(const object& value) = delete;                                \ | 
|  | void operator=(const super& value) = delete; | 
|  |  | 
|  | // Conditionally include object_service.cc functionality in the vtable to avoid | 
|  | // link errors like the following: | 
|  | // | 
|  | // object.o:(.rodata._ZTVN4....E[_ZTVN4...E]+0x278): | 
|  | // undefined reference to | 
|  | // `dart::Instance::PrintSharedInstanceJSON(dart::JSONObject*, bool) const'. | 
|  | // | 
|  | #ifndef PRODUCT | 
|  | #define OBJECT_SERVICE_SUPPORT(object)                                         \ | 
|  | protected: /* NOLINT */                                                       \ | 
|  | /* Object is printed as JSON into stream. If ref is true only a header */    \ | 
|  | /* with an object id is printed. If ref is false the object is fully   */    \ | 
|  | /* printed.                                                            */    \ | 
|  | virtual void PrintJSONImpl(JSONStream* stream, bool ref) const;              \ | 
|  | /* Prints JSON objects that describe the implementation-level fields of */   \ | 
|  | /* the current Object to |jsarr_fields|.                                */   \ | 
|  | virtual void PrintImplementationFieldsImpl(const JSONArray& jsarr_fields)    \ | 
|  | const;                                                                   \ | 
|  | virtual const char* JSONType() const {                                       \ | 
|  | return "" #object;                                                         \ | 
|  | } | 
|  | #else | 
|  | #define OBJECT_SERVICE_SUPPORT(object) protected: /* NOLINT */ | 
|  | #endif                                            // !PRODUCT | 
|  |  | 
|  | #define SNAPSHOT_SUPPORT(object)                                               \ | 
|  | friend class object##MessageSerializationCluster;                            \ | 
|  | friend class object##MessageDeserializationCluster; | 
|  |  | 
|  | #define OBJECT_IMPLEMENTATION(object, super)                                   \ | 
|  | public: /* NOLINT */                                                          \ | 
|  | DART_NOINLINE void operator=(object##Ptr value) {                            \ | 
|  | initializeHandle(this, value);                                             \ | 
|  | }                                                                            \ | 
|  | DART_NOINLINE void operator^=(ObjectPtr value) {                             \ | 
|  | initializeHandle(this, value);                                             \ | 
|  | ASSERT(IsNull() || Is##object());                                          \ | 
|  | }                                                                            \ | 
|  | \ | 
|  | protected: /* NOLINT */                                                       \ | 
|  | object() : super() {}                                                        \ | 
|  | BASE_OBJECT_IMPLEMENTATION(object, super)                                    \ | 
|  | OBJECT_SERVICE_SUPPORT(object)                                               \ | 
|  | friend class Object; | 
|  |  | 
|  | extern "C" void DFLRT_ExitSafepoint(NativeArguments __unusable_); | 
|  |  | 
|  | #define HEAP_OBJECT_IMPLEMENTATION(object, super)                              \ | 
|  | OBJECT_IMPLEMENTATION(object, super);                                        \ | 
|  | Untagged##object* untag() const {                                            \ | 
|  | ASSERT(ptr() != null());                                                   \ | 
|  | return const_cast<Untagged##object*>(ptr()->untag());                      \ | 
|  | }                                                                            \ | 
|  | SNAPSHOT_SUPPORT(object)                                                     \ | 
|  | friend class StackFrame;                                                     \ | 
|  | friend class Thread;                                                         \ | 
|  | friend void DFLRT_ExitSafepoint(NativeArguments __unusable_); | 
|  |  | 
|  | // This macro is used to denote types that do not have a sub-type. | 
|  | #define FINAL_HEAP_OBJECT_IMPLEMENTATION_HELPER(object, rettype, super)        \ | 
|  | public: /* NOLINT */                                                          \ | 
|  | void operator=(object##Ptr value) {                                          \ | 
|  | ptr_ = value;                                                              \ | 
|  | CHECK_HANDLE();                                                            \ | 
|  | }                                                                            \ | 
|  | void operator^=(ObjectPtr value) {                                           \ | 
|  | ptr_ = value;                                                              \ | 
|  | CHECK_HANDLE();                                                            \ | 
|  | }                                                                            \ | 
|  | \ | 
|  | private: /* NOLINT */                                                         \ | 
|  | object() : super() {}                                                        \ | 
|  | BASE_OBJECT_IMPLEMENTATION(object, super)                                    \ | 
|  | OBJECT_SERVICE_SUPPORT(object)                                               \ | 
|  | Untagged##object* untag() const {                                            \ | 
|  | ASSERT(ptr() != null());                                                   \ | 
|  | return const_cast<Untagged##object*>(ptr()->untag());                      \ | 
|  | }                                                                            \ | 
|  | static intptr_t NextFieldOffset() {                                          \ | 
|  | return -kWordSize;                                                         \ | 
|  | }                                                                            \ | 
|  | SNAPSHOT_SUPPORT(rettype)                                                    \ | 
|  | friend class Object;                                                         \ | 
|  | friend class StackFrame;                                                     \ | 
|  | friend class Thread;                                                         \ | 
|  | friend void DFLRT_ExitSafepoint(NativeArguments __unusable_); | 
|  |  | 
|  | #define FINAL_HEAP_OBJECT_IMPLEMENTATION(object, super)                        \ | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION_HELPER(object, object, super) | 
|  |  | 
|  | #define MINT_OBJECT_IMPLEMENTATION(object, rettype, super)                     \ | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION_HELPER(object, rettype, super) | 
|  |  | 
|  | // Different kinds of type visibility. | 
|  | enum class TypeVisibility { | 
|  | // Internal types used by the VM at runtime. | 
|  | // Internal implementation classes (like _Smi and _OneByteString) | 
|  | // are preserved as well. | 
|  | kInternalType, | 
|  |  | 
|  | // User visible types. | 
|  | // Internal VM types (such as _Smi, _OneByteString and _GrowableList) are | 
|  | // replaced with public user-visible types (e.g. int, String and List). | 
|  | kUserVisibleType | 
|  | }; | 
|  |  | 
|  | // In precompiled runtime, there is no access to runtime_api.cc since host | 
|  | // and target are the same. In those cases, the namespace dart is used to refer | 
|  | // to the target namespace | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | namespace RTN = dart; | 
|  | #else | 
|  | namespace RTN = dart::compiler::target; | 
|  | #endif  //  defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | class Object { | 
|  | public: | 
|  | using UntaggedObjectType = UntaggedObject; | 
|  | using ObjectPtrType = ObjectPtr; | 
|  |  | 
|  | // We use 30 bits for the hash code so hashes in a snapshot taken on a | 
|  | // 64-bit architecture stay in Smi range when loaded on a 32-bit | 
|  | // architecture. | 
|  | static constexpr intptr_t kHashBits = 30; | 
|  |  | 
|  | static ObjectPtr RawCast(ObjectPtr obj) { return obj; } | 
|  |  | 
|  | virtual ~Object() {} | 
|  |  | 
|  | static constexpr bool ContainsCompressedPointers() { | 
|  | return UntaggedObject::kContainsCompressedPointers; | 
|  | } | 
|  | ObjectPtr ptr() const { return ptr_; } | 
|  | void operator=(ObjectPtr value) { initializeHandle(this, value); } | 
|  |  | 
|  | bool IsCanonical() const { return ptr()->untag()->IsCanonical(); } | 
|  | void SetCanonical() const { ptr()->untag()->SetCanonical(); } | 
|  | void ClearCanonical() const { ptr()->untag()->ClearCanonical(); } | 
|  | bool IsImmutable() const { return ptr()->untag()->IsImmutable(); } | 
|  | void SetImmutable() const { ptr()->untag()->SetImmutable(); } | 
|  | void ClearImmutable() const { ptr()->untag()->ClearImmutable(); } | 
|  | intptr_t GetClassId() const { return ptr()->GetClassId(); } | 
|  | inline ClassPtr clazz() const; | 
|  | static intptr_t tags_offset() { return OFFSET_OF(UntaggedObject, tags_); } | 
|  |  | 
|  | virtual intptr_t HandleTag() const { return kObjectCid; } | 
|  |  | 
|  | #define DEFINE_LEAF_CHECK(clazz)                                               \ | 
|  | bool Is##clazz() const { return HandleTag() == k##clazz##Cid; } | 
|  | LEAF_HANDLE_LIST(DEFINE_LEAF_CHECK) | 
|  | #undef DEFINE_LEAF_CHECK | 
|  | bool IsTypedDataBase() const { | 
|  | switch (HandleTag()) { | 
|  | case kTypedDataBaseCid: | 
|  | case kTypedDataCid: | 
|  | case kExternalTypedDataCid: | 
|  | case kTypedDataViewCid: | 
|  | return true; | 
|  | default: | 
|  | return false; | 
|  | } | 
|  | } | 
|  | bool IsFinalizerBase() const { | 
|  | switch (HandleTag()) { | 
|  | case kFinalizerBaseCid: | 
|  | case kFinalizerCid: | 
|  | case kNativeFinalizerCid: | 
|  | return true; | 
|  | default: | 
|  | return false; | 
|  | } | 
|  | } | 
|  | bool IsCallSiteData() const { return IsCallSiteDataClassId(HandleTag()); } | 
|  | bool IsError() const { return IsErrorClassId(HandleTag()); } | 
|  | bool IsNumber() const { return IsNumberClassId(HandleTag()); } | 
|  | bool IsInteger() const { return IsIntegerClassId(HandleTag()); } | 
|  | bool IsInstance() const { return HandleTag() >= kInstanceCid; } | 
|  | bool IsAbstractType() const { return IsAbstractTypeClassId(HandleTag()); } | 
|  |  | 
|  | // Class testers. | 
|  | #define DEFINE_CLASS_TESTER(clazz)                                             \ | 
|  | static bool Is##clazz##NoHandle(ObjectPtr ptr) {                             \ | 
|  | /* Use a stack handle to make RawCast safe in contexts where handles   */  \ | 
|  | /* should not be allocated, such as GC or runtime transitions. Not     */  \ | 
|  | /* using Object's constructor to avoid Is##clazz being de-virtualized. */  \ | 
|  | char buf[sizeof(Object)];                                                  \ | 
|  | Object* obj = reinterpret_cast<Object*>(&buf);                             \ | 
|  | initializeHandle(obj, ptr);                                                \ | 
|  | return obj->IsNull() || obj->Is##clazz();                                  \ | 
|  | } | 
|  | CLASS_LIST_FOR_HANDLES(DEFINE_CLASS_TESTER); | 
|  | #undef DEFINE_CLASS_TESTER | 
|  |  | 
|  | bool IsNull() const { return ptr_ == null_; } | 
|  |  | 
|  | // Matches Object.toString on instances (except String::ToCString, bug 20583). | 
|  | virtual const char* ToCString() const { | 
|  | if (IsNull()) { | 
|  | return "null"; | 
|  | } else { | 
|  | return "Object"; | 
|  | } | 
|  | } | 
|  |  | 
|  | #ifndef PRODUCT | 
|  | void PrintJSON(JSONStream* stream, bool ref = true) const; | 
|  | virtual void PrintJSONImpl(JSONStream* stream, bool ref) const; | 
|  | void PrintImplementationFields(JSONStream* stream) const; | 
|  | virtual void PrintImplementationFieldsImpl( | 
|  | const JSONArray& jsarr_fields) const; | 
|  | virtual const char* JSONType() const { return IsNull() ? "null" : "Object"; } | 
|  | #endif | 
|  |  | 
|  | // Returns the name that is used to identify an object in the | 
|  | // namespace dictionary. | 
|  | // Object::DictionaryName() returns String::null(). Only subclasses | 
|  | // of Object that need to be entered in the library and library prefix | 
|  | // namespaces need to provide an implementation. | 
|  | virtual StringPtr DictionaryName() const; | 
|  |  | 
|  | bool IsNew() const { return ptr()->IsNewObject(); } | 
|  | bool IsOld() const { return ptr()->IsOldObject(); } | 
|  | #if defined(DEBUG) | 
|  | bool InVMIsolateHeap() const; | 
|  | #else | 
|  | bool InVMIsolateHeap() const { return ptr()->untag()->InVMIsolateHeap(); } | 
|  | #endif  // DEBUG | 
|  |  | 
|  | // Print the object on stdout for debugging. | 
|  | void Print() const; | 
|  |  | 
|  | #if defined(DEBUG) | 
|  | bool IsZoneHandle() const; | 
|  | bool IsReadOnlyHandle() const; | 
|  | bool IsNotTemporaryScopedHandle() const; | 
|  | #endif | 
|  |  | 
|  | static Object& Handle() { | 
|  | return HandleImpl(Thread::Current()->zone(), null_, kObjectCid); | 
|  | } | 
|  | static Object& Handle(Zone* zone) { | 
|  | return HandleImpl(zone, null_, kObjectCid); | 
|  | } | 
|  | static Object& Handle(ObjectPtr ptr) { | 
|  | return HandleImpl(Thread::Current()->zone(), ptr, kObjectCid); | 
|  | } | 
|  | static Object& Handle(Zone* zone, ObjectPtr ptr) { | 
|  | return HandleImpl(zone, ptr, kObjectCid); | 
|  | } | 
|  | static Object& ZoneHandle() { | 
|  | return ZoneHandleImpl(Thread::Current()->zone(), null_, kObjectCid); | 
|  | } | 
|  | static Object& ZoneHandle(Zone* zone) { | 
|  | return ZoneHandleImpl(zone, null_, kObjectCid); | 
|  | } | 
|  | static Object& ZoneHandle(ObjectPtr ptr) { | 
|  | return ZoneHandleImpl(Thread::Current()->zone(), ptr, kObjectCid); | 
|  | } | 
|  | static Object& ZoneHandle(Zone* zone, ObjectPtr ptr) { | 
|  | return ZoneHandleImpl(zone, ptr, kObjectCid); | 
|  | } | 
|  | static Object* ReadOnlyHandle() { return ReadOnlyHandleImpl(kObjectCid); } | 
|  |  | 
|  | static ObjectPtr null() { return null_; } | 
|  |  | 
|  | #if defined(HASH_IN_OBJECT_HEADER) | 
|  | static uint32_t GetCachedHash(const ObjectPtr obj) { | 
|  | return obj->untag()->GetHeaderHash(); | 
|  | } | 
|  |  | 
|  | static uint32_t SetCachedHashIfNotSet(ObjectPtr obj, uint32_t hash) { | 
|  | return obj->untag()->SetHeaderHashIfNotSet(hash); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | // The list below enumerates read-only handles for singleton | 
|  | // objects that are shared between the different isolates. | 
|  | // | 
|  | // - sentinel is a value that cannot be produced by Dart code. It can be used | 
|  | // to mark special values, for example to distinguish "uninitialized" fields. | 
|  | // - unknown_constant and non_constant are optimizing compiler's constant | 
|  | // propagation constants. | 
|  | // - optimized_out results from deopt environment pruning or failure to | 
|  | // capture variables in a closure's context | 
|  | #define SHARED_READONLY_HANDLES_LIST(V)                                        \ | 
|  | V(Object, null_object)                                                       \ | 
|  | V(Class, null_class)                                                         \ | 
|  | V(Array, null_array)                                                         \ | 
|  | V(String, null_string)                                                       \ | 
|  | V(Instance, null_instance)                                                   \ | 
|  | V(Function, null_function)                                                   \ | 
|  | V(FunctionType, null_function_type)                                          \ | 
|  | V(RecordType, null_record_type)                                              \ | 
|  | V(TypeArguments, null_type_arguments)                                        \ | 
|  | V(CompressedStackMaps, null_compressed_stackmaps)                            \ | 
|  | V(Closure, null_closure)                                                     \ | 
|  | V(TypeArguments, empty_type_arguments)                                       \ | 
|  | V(Array, empty_array)                                                        \ | 
|  | V(Array, empty_instantiations_cache_array)                                   \ | 
|  | V(Array, empty_subtype_test_cache_array)                                     \ | 
|  | V(ContextScope, empty_context_scope)                                         \ | 
|  | V(ObjectPool, empty_object_pool)                                             \ | 
|  | V(CompressedStackMaps, empty_compressed_stackmaps)                           \ | 
|  | V(PcDescriptors, empty_descriptors)                                          \ | 
|  | V(LocalVarDescriptors, empty_var_descriptors)                                \ | 
|  | V(ExceptionHandlers, empty_exception_handlers)                               \ | 
|  | V(ExceptionHandlers, empty_async_exception_handlers)                         \ | 
|  | V(Array, synthetic_getter_parameter_types)                                   \ | 
|  | V(Array, synthetic_getter_parameter_names)                                   \ | 
|  | V(Bytecode, implicit_getter_bytecode)                                        \ | 
|  | V(Bytecode, implicit_setter_bytecode)                                        \ | 
|  | V(Bytecode, implicit_static_getter_bytecode)                                 \ | 
|  | V(Bytecode, implicit_static_setter_bytecode)                                 \ | 
|  | V(Bytecode, method_extractor_bytecode)                                       \ | 
|  | V(Bytecode, invoke_closure_bytecode)                                         \ | 
|  | V(Bytecode, invoke_field_bytecode)                                           \ | 
|  | V(Bytecode, nsm_dispatcher_bytecode)                                         \ | 
|  | V(Bytecode, dynamic_invocation_forwarder_bytecode)                           \ | 
|  | V(Bytecode, implicit_static_closure_bytecode)                                \ | 
|  | V(Bytecode, implicit_instance_closure_bytecode)                              \ | 
|  | V(Bytecode, implicit_constructor_closure_bytecode)                           \ | 
|  | V(Sentinel, sentinel)                                                        \ | 
|  | V(Sentinel, unknown_constant)                                                \ | 
|  | V(Sentinel, non_constant)                                                    \ | 
|  | V(Sentinel, optimized_out)                                                   \ | 
|  | V(Bool, bool_true)                                                           \ | 
|  | V(Bool, bool_false)                                                          \ | 
|  | V(Smi, smi_illegal_cid)                                                      \ | 
|  | V(Smi, smi_zero)                                                             \ | 
|  | V(ApiError, no_callbacks_error)                                              \ | 
|  | V(UnwindError, unwind_error)                                                 \ | 
|  | V(UnwindError, unwind_in_progress_error)                                     \ | 
|  | V(LanguageError, snapshot_writer_error)                                      \ | 
|  | V(LanguageError, branch_offset_error)                                        \ | 
|  | V(LanguageError, background_compilation_error)                               \ | 
|  | V(LanguageError, no_debuggable_code_error)                                   \ | 
|  | V(LanguageError, out_of_memory_error)                                        \ | 
|  | V(UnhandledException, unhandled_oom_exception)                               \ | 
|  | V(Array, vm_isolate_snapshot_object_table)                                   \ | 
|  | V(Type, dynamic_type)                                                        \ | 
|  | V(Type, void_type)                                                           \ | 
|  | V(AbstractType, null_abstract_type)                                          \ | 
|  | V(TypedData, uninitialized_index)                                            \ | 
|  | V(Array, uninitialized_data) | 
|  |  | 
|  | #define DEFINE_SHARED_READONLY_HANDLE_GETTER(Type, name)                       \ | 
|  | static const Type& name() {                                                  \ | 
|  | ASSERT(name##_ != nullptr);                                                \ | 
|  | return *name##_;                                                           \ | 
|  | } | 
|  | SHARED_READONLY_HANDLES_LIST(DEFINE_SHARED_READONLY_HANDLE_GETTER) | 
|  | #undef DEFINE_SHARED_READONLY_HANDLE_GETTER | 
|  |  | 
|  | static void set_vm_isolate_snapshot_object_table(const Array& table); | 
|  |  | 
|  | static ClassPtr class_class() { return class_class_; } | 
|  | static ClassPtr dynamic_class() { return dynamic_class_; } | 
|  | static ClassPtr void_class() { return void_class_; } | 
|  | static ClassPtr type_parameters_class() { return type_parameters_class_; } | 
|  | static ClassPtr type_arguments_class() { return type_arguments_class_; } | 
|  | static ClassPtr patch_class_class() { return patch_class_class_; } | 
|  | static ClassPtr function_class() { return function_class_; } | 
|  | static ClassPtr closure_data_class() { return closure_data_class_; } | 
|  | static ClassPtr ffi_trampoline_data_class() { | 
|  | return ffi_trampoline_data_class_; | 
|  | } | 
|  | static ClassPtr field_class() { return field_class_; } | 
|  | static ClassPtr script_class() { return script_class_; } | 
|  | static ClassPtr library_class() { return library_class_; } | 
|  | static ClassPtr namespace_class() { return namespace_class_; } | 
|  | static ClassPtr kernel_program_info_class() { | 
|  | return kernel_program_info_class_; | 
|  | } | 
|  | static ClassPtr code_class() { return code_class_; } | 
|  | static ClassPtr instructions_class() { return instructions_class_; } | 
|  | static ClassPtr instructions_section_class() { | 
|  | return instructions_section_class_; | 
|  | } | 
|  | static ClassPtr instructions_table_class() { | 
|  | return instructions_table_class_; | 
|  | } | 
|  | static ClassPtr object_pool_class() { return object_pool_class_; } | 
|  | static ClassPtr pc_descriptors_class() { return pc_descriptors_class_; } | 
|  | static ClassPtr code_source_map_class() { return code_source_map_class_; } | 
|  | static ClassPtr compressed_stackmaps_class() { | 
|  | return compressed_stackmaps_class_; | 
|  | } | 
|  | static ClassPtr var_descriptors_class() { return var_descriptors_class_; } | 
|  | static ClassPtr exception_handlers_class() { | 
|  | return exception_handlers_class_; | 
|  | } | 
|  | static ClassPtr context_class() { return context_class_; } | 
|  | static ClassPtr context_scope_class() { return context_scope_class_; } | 
|  | static ClassPtr bytecode_class() { return bytecode_class_; } | 
|  | static ClassPtr sentinel_class() { return sentinel_class_; } | 
|  | static ClassPtr api_error_class() { return api_error_class_; } | 
|  | static ClassPtr language_error_class() { return language_error_class_; } | 
|  | static ClassPtr unhandled_exception_class() { | 
|  | return unhandled_exception_class_; | 
|  | } | 
|  | static ClassPtr unwind_error_class() { return unwind_error_class_; } | 
|  | static ClassPtr singletargetcache_class() { return singletargetcache_class_; } | 
|  | static ClassPtr unlinkedcall_class() { return unlinkedcall_class_; } | 
|  | static ClassPtr monomorphicsmiablecall_class() { | 
|  | return monomorphicsmiablecall_class_; | 
|  | } | 
|  | static ClassPtr icdata_class() { return icdata_class_; } | 
|  | static ClassPtr megamorphic_cache_class() { return megamorphic_cache_class_; } | 
|  | static ClassPtr subtypetestcache_class() { return subtypetestcache_class_; } | 
|  | static ClassPtr loadingunit_class() { return loadingunit_class_; } | 
|  | static ClassPtr weak_serialization_reference_class() { | 
|  | return weak_serialization_reference_class_; | 
|  | } | 
|  | static ClassPtr weak_array_class() { return weak_array_class_; } | 
|  |  | 
|  | // Initialize the VM isolate. | 
|  | static void InitNullAndBool(IsolateGroup* isolate_group); | 
|  | static void Init(IsolateGroup* isolate_group); | 
|  | static void InitVtables(); | 
|  | static void FinishInit(IsolateGroup* isolate_group); | 
|  | static void FinalizeVMIsolate(IsolateGroup* isolate_group); | 
|  | static void FinalizeReadOnlyObject(ObjectPtr object); | 
|  |  | 
|  | static void Cleanup(); | 
|  |  | 
|  | // Initialize a new isolate either from a Kernel IR, from source, or from a | 
|  | // snapshot. | 
|  | static ErrorPtr Init(IsolateGroup* isolate_group, | 
|  | const uint8_t* kernel_buffer, | 
|  | intptr_t kernel_buffer_size); | 
|  |  | 
|  | static void MakeUnusedSpaceTraversable(const Object& obj, | 
|  | intptr_t original_size, | 
|  | intptr_t used_size); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedObject)); | 
|  | } | 
|  |  | 
|  | template <class FakeObject> | 
|  | static void VerifyBuiltinVtable(intptr_t cid) { | 
|  | FakeObject fake; | 
|  | if (cid >= kNumPredefinedCids) { | 
|  | cid = kInstanceCid; | 
|  | } | 
|  | ASSERT(builtin_vtables_[cid] == fake.vtable()); | 
|  | } | 
|  | static void VerifyBuiltinVtables(); | 
|  |  | 
|  | static const ClassId kClassId = kObjectCid; | 
|  |  | 
|  | // Different kinds of name visibility. | 
|  | enum NameVisibility { | 
|  | // Internal names are the true names of classes, fields, | 
|  | // etc. inside the vm.  These names include privacy suffixes, | 
|  | // getter prefixes, and trailing dots on unnamed constructors. | 
|  | // | 
|  | // The names of core implementation classes (like _OneByteString) | 
|  | // are preserved as well. | 
|  | // | 
|  | // e.g. | 
|  | //   private getter             -> get:foo@6be832b | 
|  | //   private constructor        -> _MyClass@6b3832b. | 
|  | //   private named constructor  -> _MyClass@6b3832b.named | 
|  | //   core impl class name shown -> _OneByteString | 
|  | kInternalName = 0, | 
|  |  | 
|  | // Scrubbed names drop privacy suffixes, getter prefixes, and | 
|  | // trailing dots on unnamed constructors.  These names are used in | 
|  | // the vm service. | 
|  | // | 
|  | // e.g. | 
|  | //   get:foo@6be832b        -> foo | 
|  | //   _MyClass@6b3832b.      -> _MyClass | 
|  | //   _MyClass@6b3832b.named -> _MyClass.named | 
|  | //   _OneByteString         -> _OneByteString (not remapped) | 
|  | kScrubbedName, | 
|  |  | 
|  | // User visible names are appropriate for reporting type errors | 
|  | // directly to programmers.  The names have been scrubbed and | 
|  | // the names of core implementation classes are remapped to their | 
|  | // public interface names. | 
|  | // | 
|  | // e.g. | 
|  | //   get:foo@6be832b        -> foo | 
|  | //   _MyClass@6b3832b.      -> _MyClass | 
|  | //   _MyClass@6b3832b.named -> _MyClass.named | 
|  | //   _OneByteString         -> String (remapped) | 
|  | kUserVisibleName | 
|  | }; | 
|  |  | 
|  | // Sometimes simple formating might produce the same name for two different | 
|  | // entities, for example we might inject a synthetic forwarder into the | 
|  | // class which has the same name as an already existing function, or | 
|  | // two different types can be formatted as X<T> because T has different | 
|  | // meaning (refers to a different type parameter) in these two types. | 
|  | // Such ambiguity might be acceptable in some contexts but not in others, so | 
|  | // some formatting methods have two modes - one which tries to be more | 
|  | // user friendly, and another one which tries to avoid name conflicts by | 
|  | // emitting longer and less user friendly names. | 
|  | enum class NameDisambiguation { | 
|  | kYes, | 
|  | kNo, | 
|  | }; | 
|  |  | 
|  | static bool ShouldHaveImmutabilityBitSet(classid_t class_id); | 
|  |  | 
|  | protected: | 
|  | friend ObjectPtr AllocateObject(intptr_t, intptr_t, intptr_t); | 
|  |  | 
|  | // Used for extracting the C++ vtable during bringup. | 
|  | Object() : ptr_(null_) {} | 
|  |  | 
|  | uword raw_value() const { return static_cast<uword>(ptr()); } | 
|  |  | 
|  | inline void setPtr(ObjectPtr value, intptr_t default_cid); | 
|  | void CheckHandle() const; | 
|  | DART_NOINLINE static Object& HandleImpl(Zone* zone, | 
|  | ObjectPtr ptr, | 
|  | intptr_t default_cid) { | 
|  | Object* obj = reinterpret_cast<Object*>(VMHandles::AllocateHandle(zone)); | 
|  | obj->setPtr(ptr, default_cid); | 
|  | return *obj; | 
|  | } | 
|  | DART_NOINLINE static Object& ZoneHandleImpl(Zone* zone, | 
|  | ObjectPtr ptr, | 
|  | intptr_t default_cid) { | 
|  | Object* obj = | 
|  | reinterpret_cast<Object*>(VMHandles::AllocateZoneHandle(zone)); | 
|  | obj->setPtr(ptr, default_cid); | 
|  | return *obj; | 
|  | } | 
|  | DART_NOINLINE static Object* ReadOnlyHandleImpl(intptr_t cid) { | 
|  | Object* obj = reinterpret_cast<Object*>(Dart::AllocateReadOnlyHandle()); | 
|  | obj->setPtr(Object::null(), cid); | 
|  | return obj; | 
|  | } | 
|  |  | 
|  | // Memcpy to account for the strict aliasing rule. | 
|  | // Explicit cast to silence -Wdynamic-class-memaccess. | 
|  | // This is still undefined behavior because we're messing with the internal | 
|  | // representation of C++ objects, but works okay in practice with | 
|  | // -fno-strict-vtable-pointers. | 
|  | cpp_vtable vtable() const { | 
|  | cpp_vtable result; | 
|  | memcpy(&result, reinterpret_cast<const void*>(this),  // NOLINT | 
|  | sizeof(result)); | 
|  | return result; | 
|  | } | 
|  | void set_vtable(cpp_vtable value) { | 
|  | memcpy(reinterpret_cast<void*>(this), &value,  // NOLINT | 
|  | sizeof(cpp_vtable)); | 
|  | } | 
|  |  | 
|  | static ObjectPtr Allocate(intptr_t cls_id, | 
|  | intptr_t size, | 
|  | Heap::Space space, | 
|  | bool compressed, | 
|  | uword ptr_field_start_offset, | 
|  | uword ptr_field_end_offset); | 
|  |  | 
|  | // Templates of Allocate that retrieve the appropriate values to pass from | 
|  | // the class. | 
|  |  | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static typename T::ObjectPtrType Allocate( | 
|  | Heap::Space space) { | 
|  | return static_cast<typename T::ObjectPtrType>(Allocate( | 
|  | T::kClassId, T::InstanceSize(), space, T::ContainsCompressedPointers(), | 
|  | Object::from_offset<T>(), Object::to_offset<T>())); | 
|  | } | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static typename T::ObjectPtrType Allocate( | 
|  | Heap::Space space, | 
|  | intptr_t elements) { | 
|  | return static_cast<typename T::ObjectPtrType>( | 
|  | Allocate(T::kClassId, T::InstanceSize(elements), space, | 
|  | T::ContainsCompressedPointers(), Object::from_offset<T>(), | 
|  | Object::to_offset<T>(elements))); | 
|  | } | 
|  |  | 
|  | // Additional versions that also take a class_id for types like Array, Map, | 
|  | // and Set that have more than one possible class id. | 
|  |  | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static typename T::ObjectPtrType AllocateVariant( | 
|  | intptr_t class_id, | 
|  | Heap::Space space) { | 
|  | return static_cast<typename T::ObjectPtrType>(Allocate( | 
|  | class_id, T::InstanceSize(), space, T::ContainsCompressedPointers(), | 
|  | Object::from_offset<T>(), Object::to_offset<T>())); | 
|  | } | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static typename T::ObjectPtrType | 
|  | AllocateVariant(intptr_t class_id, Heap::Space space, intptr_t elements) { | 
|  | return static_cast<typename T::ObjectPtrType>( | 
|  | Allocate(class_id, T::InstanceSize(elements), space, | 
|  | T::ContainsCompressedPointers(), Object::from_offset<T>(), | 
|  | Object::to_offset<T>(elements))); | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t RoundedAllocationSize(intptr_t size) { | 
|  | return Utils::RoundUp(size, kObjectAlignment); | 
|  | } | 
|  |  | 
|  | bool Contains(uword addr) const { return ptr()->untag()->Contains(addr); } | 
|  |  | 
|  | // Start of field mutator guards. | 
|  | // | 
|  | // All writes to heap objects should ultimately pass through one of the | 
|  | // methods below or their counterparts in UntaggedObject, to ensure that the | 
|  | // write barrier is correctly applied. | 
|  |  | 
|  | template <typename type, std::memory_order order = std::memory_order_relaxed> | 
|  | type LoadPointer(type const* addr) const { | 
|  | return ptr()->untag()->LoadPointer<type, order>(addr); | 
|  | } | 
|  |  | 
|  | template <typename type, std::memory_order order = std::memory_order_relaxed> | 
|  | void StorePointer(type const* addr, type value) const { | 
|  | ptr()->untag()->StorePointer<type, order>(addr, value); | 
|  | } | 
|  | template <typename type, | 
|  | typename compressed_type, | 
|  | std::memory_order order = std::memory_order_relaxed> | 
|  | void StoreCompressedPointer(compressed_type const* addr, type value) const { | 
|  | ptr()->untag()->StoreCompressedPointer<type, compressed_type, order>(addr, | 
|  | value); | 
|  | } | 
|  | template <typename type> | 
|  | void StorePointerUnaligned(type const* addr, | 
|  | type value, | 
|  | Thread* thread) const { | 
|  | ptr()->untag()->StorePointerUnaligned<type>(addr, value, thread); | 
|  | } | 
|  |  | 
|  | // Use for storing into an explicitly Smi-typed field of an object | 
|  | // (i.e., both the previous and new value are Smis). | 
|  | void StoreSmi(SmiPtr const* addr, SmiPtr value) const { | 
|  | ptr()->untag()->StoreSmi(addr, value); | 
|  | } | 
|  |  | 
|  | template <typename FieldType> | 
|  | void StoreSimd128(const FieldType* addr, simd128_value_t value) const { | 
|  | ASSERT(Contains(reinterpret_cast<uword>(addr))); | 
|  | value.writeTo(const_cast<FieldType*>(addr)); | 
|  | } | 
|  |  | 
|  | template <typename FieldType> | 
|  | FieldType LoadNonPointer(const FieldType* addr) const { | 
|  | return *const_cast<FieldType*>(addr); | 
|  | } | 
|  |  | 
|  | template <typename FieldType, std::memory_order order> | 
|  | FieldType LoadNonPointer(const FieldType* addr) const { | 
|  | return reinterpret_cast<std::atomic<FieldType>*>( | 
|  | const_cast<FieldType*>(addr)) | 
|  | ->load(order); | 
|  | } | 
|  |  | 
|  | // Needs two template arguments to allow assigning enums to fixed-size ints. | 
|  | template <typename FieldType, typename ValueType> | 
|  | void StoreNonPointer(const FieldType* addr, ValueType value) const { | 
|  | // Can't use Contains, as it uses tags_, which is set through this method. | 
|  | ASSERT(reinterpret_cast<uword>(addr) >= UntaggedObject::ToAddr(ptr())); | 
|  | *const_cast<FieldType*>(addr) = value; | 
|  | } | 
|  |  | 
|  | template <typename FieldType, typename ValueType, std::memory_order order> | 
|  | void StoreNonPointer(const FieldType* addr, ValueType value) const { | 
|  | // Can't use Contains, as it uses tags_, which is set through this method. | 
|  | ASSERT(reinterpret_cast<uword>(addr) >= UntaggedObject::ToAddr(ptr())); | 
|  | reinterpret_cast<std::atomic<FieldType>*>(const_cast<FieldType*>(addr)) | 
|  | ->store(value, order); | 
|  | } | 
|  |  | 
|  | // Provides non-const access to non-pointer fields within the object. Such | 
|  | // access does not need a write barrier, but it is *not* GC-safe, since the | 
|  | // object might move, hence must be fully contained within a NoSafepointScope. | 
|  | template <typename FieldType> | 
|  | FieldType* UnsafeMutableNonPointer(const FieldType* addr) const { | 
|  | // Allow pointers at the end of variable-length data, and disallow pointers | 
|  | // within the header word. | 
|  | ASSERT(Contains(reinterpret_cast<uword>(addr) - 1) && | 
|  | Contains(reinterpret_cast<uword>(addr) - kWordSize)); | 
|  | // At least check that there is a NoSafepointScope and hope it's big enough. | 
|  | ASSERT(Thread::Current()->no_safepoint_scope_depth() > 0); | 
|  | return const_cast<FieldType*>(addr); | 
|  | } | 
|  |  | 
|  | // Fail at link time if StoreNonPointer or UnsafeMutableNonPointer is | 
|  | // instantiated with an object pointer type. | 
|  | #define STORE_NON_POINTER_ILLEGAL_TYPE(type)                                   \ | 
|  | template <typename ValueType>                                                \ | 
|  | void StoreNonPointer(type##Ptr const* addr, ValueType value) const {         \ | 
|  | UnimplementedMethod();                                                     \ | 
|  | }                                                                            \ | 
|  | type##Ptr* UnsafeMutableNonPointer(type##Ptr const* addr) const {            \ | 
|  | UnimplementedMethod();                                                     \ | 
|  | return nullptr;                                                            \ | 
|  | } | 
|  |  | 
|  | CLASS_LIST(STORE_NON_POINTER_ILLEGAL_TYPE); | 
|  | void UnimplementedMethod() const; | 
|  | #undef STORE_NON_POINTER_ILLEGAL_TYPE | 
|  |  | 
|  | // Allocate an object and copy the body of 'orig'. | 
|  | static ObjectPtr Clone(const Object& orig, | 
|  | Heap::Space space, | 
|  | bool load_with_relaxed_atomics = false); | 
|  |  | 
|  | // End of field mutator guards. | 
|  |  | 
|  | ObjectPtr ptr_;  // The raw object reference. | 
|  |  | 
|  | protected: | 
|  | // The first offset in an allocated object of the given type that contains a | 
|  | // (possibly compressed) object pointer. Used to initialize object pointer | 
|  | // fields to Object::null() instead of 0. | 
|  | // | 
|  | // Always returns an offset after the object header tags. | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static uword from_offset() { | 
|  | return UntaggedObject::from_offset<typename T::UntaggedObjectType>(); | 
|  | } | 
|  |  | 
|  | // The last offset in an allocated object of the given type that contains a | 
|  | // (possibly compressed) object pointer. Used to initialize object pointer | 
|  | // fields to Object::null() instead of 0. | 
|  | // | 
|  | // Takes an optional argument that is the number of elements in the payload, | 
|  | // which is ignored if the object never contains a payload. | 
|  | // | 
|  | // If there are no pointer fields in the object, then | 
|  | // to_offset<T>() < from_offset<T>(). | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static uword to_offset(intptr_t length = 0) { | 
|  | return UntaggedObject::to_offset<typename T::UntaggedObjectType>(length); | 
|  | } | 
|  |  | 
|  | void AddCommonObjectProperties(JSONObject* jsobj, | 
|  | const char* protocol_type, | 
|  | bool ref) const; | 
|  |  | 
|  | private: | 
|  | static intptr_t NextFieldOffset() { | 
|  | // Indicates this class cannot be extended by dart code. | 
|  | return -kWordSize; | 
|  | } | 
|  |  | 
|  | static void InitializeObject(uword address, | 
|  | intptr_t id, | 
|  | intptr_t size, | 
|  | bool compressed, | 
|  | uword ptr_field_start_offset, | 
|  | uword ptr_field_end_offset); | 
|  |  | 
|  | // Templates of InitializeObject that retrieve the appropriate values to pass | 
|  | // from the class. | 
|  |  | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static void InitializeObject(uword address) { | 
|  | return InitializeObject(address, T::kClassId, T::InstanceSize(), | 
|  | T::ContainsCompressedPointers(), | 
|  | Object::from_offset<T>(), Object::to_offset<T>()); | 
|  | } | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static void InitializeObject(uword address, | 
|  | intptr_t elements) { | 
|  | return InitializeObject(address, T::kClassId, T::InstanceSize(elements), | 
|  | T::ContainsCompressedPointers(), | 
|  | Object::from_offset<T>(), | 
|  | Object::to_offset<T>(elements)); | 
|  | } | 
|  |  | 
|  | // Additional versions that also take a class_id for types like Array, Map, | 
|  | // and Set that have more than one possible class id. | 
|  |  | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static void InitializeObjectVariant(uword address, | 
|  | intptr_t class_id) { | 
|  | return InitializeObject(address, class_id, T::InstanceSize(), | 
|  | T::ContainsCompressedPointers(), | 
|  | Object::from_offset<T>(), Object::to_offset<T>()); | 
|  | } | 
|  | template <typename T> | 
|  | DART_FORCE_INLINE static void InitializeObjectVariant(uword address, | 
|  | intptr_t class_id, | 
|  | intptr_t elements) { | 
|  | return InitializeObject(address, class_id, T::InstanceSize(elements), | 
|  | T::ContainsCompressedPointers(), | 
|  | Object::from_offset<T>(), | 
|  | Object::to_offset<T>(elements)); | 
|  | } | 
|  |  | 
|  | static void RegisterClass(const Class& cls, | 
|  | const String& name, | 
|  | const Library& lib); | 
|  | static void RegisterPrivateClass(const Class& cls, | 
|  | const String& name, | 
|  | const Library& lib); | 
|  |  | 
|  | /* Initialize the handle based on the ptr in the presence of null. */ | 
|  | static void initializeHandle(Object* obj, ObjectPtr ptr) { | 
|  | obj->setPtr(ptr, kObjectCid); | 
|  | } | 
|  |  | 
|  | static cpp_vtable builtin_vtables_[kNumPredefinedCids]; | 
|  |  | 
|  | // The static values below are singletons shared between the different | 
|  | // isolates. They are all allocated in the non-GC'd Dart::vm_isolate_. | 
|  | static ObjectPtr null_; | 
|  | static BoolPtr true_; | 
|  | static BoolPtr false_; | 
|  |  | 
|  | static ClassPtr class_class_; | 
|  | static ClassPtr dynamic_class_; | 
|  | static ClassPtr void_class_; | 
|  | static ClassPtr type_parameters_class_; | 
|  | static ClassPtr type_arguments_class_; | 
|  | static ClassPtr patch_class_class_; | 
|  | static ClassPtr function_class_; | 
|  | static ClassPtr closure_data_class_; | 
|  | static ClassPtr ffi_trampoline_data_class_; | 
|  | static ClassPtr field_class_; | 
|  | static ClassPtr script_class_; | 
|  | static ClassPtr library_class_; | 
|  | static ClassPtr namespace_class_; | 
|  | static ClassPtr kernel_program_info_class_; | 
|  | static ClassPtr code_class_; | 
|  | static ClassPtr instructions_class_; | 
|  | static ClassPtr instructions_section_class_; | 
|  | static ClassPtr instructions_table_class_; | 
|  | static ClassPtr object_pool_class_; | 
|  | static ClassPtr pc_descriptors_class_; | 
|  | static ClassPtr code_source_map_class_; | 
|  | static ClassPtr compressed_stackmaps_class_; | 
|  | static ClassPtr var_descriptors_class_; | 
|  | static ClassPtr exception_handlers_class_; | 
|  | static ClassPtr context_class_; | 
|  | static ClassPtr context_scope_class_; | 
|  | static ClassPtr bytecode_class_; | 
|  | static ClassPtr sentinel_class_; | 
|  | static ClassPtr singletargetcache_class_; | 
|  | static ClassPtr unlinkedcall_class_; | 
|  | static ClassPtr monomorphicsmiablecall_class_; | 
|  | static ClassPtr icdata_class_; | 
|  | static ClassPtr megamorphic_cache_class_; | 
|  | static ClassPtr subtypetestcache_class_; | 
|  | static ClassPtr loadingunit_class_; | 
|  | static ClassPtr api_error_class_; | 
|  | static ClassPtr language_error_class_; | 
|  | static ClassPtr unhandled_exception_class_; | 
|  | static ClassPtr unwind_error_class_; | 
|  | static ClassPtr weak_serialization_reference_class_; | 
|  | static ClassPtr weak_array_class_; | 
|  |  | 
|  | #define DECLARE_SHARED_READONLY_HANDLE(Type, name) static Type* name##_; | 
|  | SHARED_READONLY_HANDLES_LIST(DECLARE_SHARED_READONLY_HANDLE) | 
|  | #undef DECLARE_SHARED_READONLY_HANDLE | 
|  |  | 
|  | friend void ClassTable::Register(const Class& cls); | 
|  | friend void UntaggedObject::Validate(IsolateGroup* isolate_group) const; | 
|  | friend class Closure; | 
|  | friend class InstanceDeserializationCluster; | 
|  | friend class Interpreter; | 
|  | friend class ObjectGraphCopier;  // For Object::InitializeObject | 
|  | friend class Simd128MessageDeserializationCluster; | 
|  | friend class OneByteString; | 
|  | friend class TwoByteString; | 
|  | friend class Thread; | 
|  |  | 
|  | #define REUSABLE_FRIEND_DECLARATION(name)                                      \ | 
|  | friend class Reusable##name##HandleScope; | 
|  | REUSABLE_HANDLE_LIST(REUSABLE_FRIEND_DECLARATION) | 
|  | #undef REUSABLE_FRIEND_DECLARATION | 
|  |  | 
|  | DISALLOW_ALLOCATION(); | 
|  | DISALLOW_COPY_AND_ASSIGN(Object); | 
|  | }; | 
|  |  | 
|  | // Used to declare setters and getters for untagged object fields that are | 
|  | // defined with the WSR_COMPRESSED_POINTER_FIELD macro. | 
|  | // | 
|  | // In the precompiler, the getter transparently unwraps the | 
|  | // WeakSerializationReference, if present, to get the wrapped value of the | 
|  | // appropriate type, since a WeakSerializationReference object should be | 
|  | // transparent to the parts of the precompiler that are not the serializer. | 
|  | // Meanwhile, the setter takes an Object to allow the precompiler to set the | 
|  | // field to a WeakSerializationReference. | 
|  | // | 
|  | // Since WeakSerializationReferences are only used during precompilation, | 
|  | // this macro creates the normally expected getter and setter otherwise. | 
|  | #if defined(DART_PRECOMPILER) | 
|  | #define PRECOMPILER_WSR_FIELD_DECLARATION(Type, Name)                          \ | 
|  | Type##Ptr Name() const;                                                      \ | 
|  | void set_##Name(const Object& value) const {                                 \ | 
|  | untag()->set_##Name(value.ptr());                                          \ | 
|  | } | 
|  | #else | 
|  | #define PRECOMPILER_WSR_FIELD_DECLARATION(Type, Name)                          \ | 
|  | Type##Ptr Name() const {                                                     \ | 
|  | return untag()->Name();                                                    \ | 
|  | }                                                                            \ | 
|  | void set_##Name(const Type& value) const; | 
|  | #endif | 
|  |  | 
|  | class PassiveObject : public Object { | 
|  | public: | 
|  | void operator=(ObjectPtr value) { ptr_ = value; } | 
|  | void operator^=(ObjectPtr value) { ptr_ = value; } | 
|  |  | 
|  | static PassiveObject& Handle(Zone* zone, ObjectPtr ptr) { | 
|  | PassiveObject* obj = | 
|  | reinterpret_cast<PassiveObject*>(VMHandles::AllocateHandle(zone)); | 
|  | obj->ptr_ = ptr; | 
|  | obj->set_vtable(0); | 
|  | return *obj; | 
|  | } | 
|  | static PassiveObject& Handle(ObjectPtr ptr) { | 
|  | return Handle(Thread::Current()->zone(), ptr); | 
|  | } | 
|  | static PassiveObject& Handle() { | 
|  | return Handle(Thread::Current()->zone(), Object::null()); | 
|  | } | 
|  | static PassiveObject& Handle(Zone* zone) { | 
|  | return Handle(zone, Object::null()); | 
|  | } | 
|  | static PassiveObject& ZoneHandle(Zone* zone, ObjectPtr ptr) { | 
|  | PassiveObject* obj = | 
|  | reinterpret_cast<PassiveObject*>(VMHandles::AllocateZoneHandle(zone)); | 
|  | obj->ptr_ = ptr; | 
|  | obj->set_vtable(0); | 
|  | return *obj; | 
|  | } | 
|  | static PassiveObject& ZoneHandle(ObjectPtr ptr) { | 
|  | return ZoneHandle(Thread::Current()->zone(), ptr); | 
|  | } | 
|  | static PassiveObject& ZoneHandle() { | 
|  | return ZoneHandle(Thread::Current()->zone(), Object::null()); | 
|  | } | 
|  | static PassiveObject& ZoneHandle(Zone* zone) { | 
|  | return ZoneHandle(zone, Object::null()); | 
|  | } | 
|  |  | 
|  | private: | 
|  | PassiveObject() : Object() {} | 
|  | DISALLOW_ALLOCATION(); | 
|  | DISALLOW_COPY_AND_ASSIGN(PassiveObject); | 
|  | }; | 
|  |  | 
|  | // A URIs array contains triplets of strings. | 
|  | // The first string in the triplet is a type name (usually a class). | 
|  | // The second string in the triplet is the URI of the type. | 
|  | // The third string in the triplet is "print" if the triplet should be printed. | 
|  | typedef ZoneGrowableHandlePtrArray<const String> URIs; | 
|  |  | 
|  | enum class Nullability : uint8_t { | 
|  | kNullable = 0, | 
|  | kNonNullable = 1, | 
|  | }; | 
|  |  | 
|  | // Equality kind between types. | 
|  | enum class TypeEquality { | 
|  | kCanonical = 0, | 
|  | kSyntactical = 1, | 
|  | kInSubtypeTest = 2, | 
|  | }; | 
|  |  | 
|  | class Class : public Object { | 
|  | public: | 
|  | enum InvocationDispatcherEntry { | 
|  | kInvocationDispatcherName, | 
|  | kInvocationDispatcherArgsDesc, | 
|  | kInvocationDispatcherFunction, | 
|  | kInvocationDispatcherEntrySize, | 
|  | }; | 
|  |  | 
|  | bool HasCompressedPointers() const; | 
|  | intptr_t host_instance_size() const { | 
|  | ASSERT(is_finalized() || is_prefinalized()); | 
|  | return (untag()->host_instance_size_in_words_ * kCompressedWordSize); | 
|  | } | 
|  | intptr_t target_instance_size() const { | 
|  | ASSERT(is_finalized() || is_prefinalized()); | 
|  | #if defined(DART_PRECOMPILER) | 
|  | return (untag()->target_instance_size_in_words_ * | 
|  | compiler::target::kCompressedWordSize); | 
|  | #else | 
|  | return host_instance_size(); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  | static intptr_t host_instance_size(ClassPtr clazz) { | 
|  | return (clazz->untag()->host_instance_size_in_words_ * kCompressedWordSize); | 
|  | } | 
|  | static intptr_t target_instance_size(ClassPtr clazz) { | 
|  | #if defined(DART_PRECOMPILER) | 
|  | return (clazz->untag()->target_instance_size_in_words_ * | 
|  | compiler::target::kCompressedWordSize); | 
|  | #else | 
|  | return host_instance_size(clazz); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  | void set_instance_size(intptr_t host_value_in_bytes, | 
|  | intptr_t target_value_in_bytes) const { | 
|  | ASSERT(kCompressedWordSize != 0); | 
|  | set_instance_size_in_words( | 
|  | host_value_in_bytes / kCompressedWordSize, | 
|  | target_value_in_bytes / compiler::target::kCompressedWordSize); | 
|  | } | 
|  | void set_instance_size_in_words(intptr_t host_value, | 
|  | intptr_t target_value) const { | 
|  | ASSERT( | 
|  | Utils::IsAligned((host_value * kCompressedWordSize), kObjectAlignment)); | 
|  | StoreNonPointer(&untag()->host_instance_size_in_words_, host_value); | 
|  | #if defined(DART_PRECOMPILER) | 
|  | ASSERT( | 
|  | Utils::IsAligned((target_value * compiler::target::kCompressedWordSize), | 
|  | compiler::target::kObjectAlignment)); | 
|  | StoreNonPointer(&untag()->target_instance_size_in_words_, target_value); | 
|  | #else | 
|  | // Could be different only during cross-compilation. | 
|  | ASSERT_EQUAL(host_value, target_value); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  |  | 
|  | intptr_t host_next_field_offset() const { | 
|  | return untag()->host_next_field_offset_in_words_ * kCompressedWordSize; | 
|  | } | 
|  | intptr_t target_next_field_offset() const { | 
|  | #if defined(DART_PRECOMPILER) | 
|  | return untag()->target_next_field_offset_in_words_ * | 
|  | compiler::target::kCompressedWordSize; | 
|  | #else | 
|  | return host_next_field_offset(); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  | void set_next_field_offset(intptr_t host_value_in_bytes, | 
|  | intptr_t target_value_in_bytes) const { | 
|  | set_next_field_offset_in_words( | 
|  | host_value_in_bytes / kCompressedWordSize, | 
|  | target_value_in_bytes / compiler::target::kCompressedWordSize); | 
|  | } | 
|  | void set_next_field_offset_in_words(intptr_t host_value, | 
|  | intptr_t target_value) const { | 
|  | // Assert that the next field offset is either negative (ie, this object | 
|  | // can't be extended by dart code), or rounds up to the kObjectAligned | 
|  | // instance size. | 
|  | ASSERT((host_value < 0) || | 
|  | ((host_value <= untag()->host_instance_size_in_words_) && | 
|  | (host_value + (kObjectAlignment / kCompressedWordSize) > | 
|  | untag()->host_instance_size_in_words_))); | 
|  | StoreNonPointer(&untag()->host_next_field_offset_in_words_, host_value); | 
|  | #if defined(DART_PRECOMPILER) | 
|  | ASSERT((target_value < 0) || | 
|  | ((target_value <= untag()->target_instance_size_in_words_) && | 
|  | (target_value + (compiler::target::kObjectAlignment / | 
|  | compiler::target::kCompressedWordSize) > | 
|  | untag()->target_instance_size_in_words_))); | 
|  | StoreNonPointer(&untag()->target_next_field_offset_in_words_, target_value); | 
|  | #else | 
|  | // Could be different only during cross-compilation. | 
|  | ASSERT_EQUAL(host_value, target_value); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  |  | 
|  | static bool is_valid_id(intptr_t value) { | 
|  | return UntaggedObject::ClassIdTag::is_valid(value); | 
|  | } | 
|  | intptr_t id() const { return untag()->id_; } | 
|  | void set_id(intptr_t value) const { | 
|  | ASSERT(value >= 0 && value < std::numeric_limits<classid_t>::max()); | 
|  | StoreNonPointer(&untag()->id_, value); | 
|  | } | 
|  | static intptr_t id_offset() { return OFFSET_OF(UntaggedClass, id_); } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | // If the interface of this class has a single concrete implementation, either | 
|  | // via `extends` or by `implements`, returns its CID. | 
|  | // If it has no implementation, returns kIllegalCid. | 
|  | // If it has more than one implementation, returns kDynamicCid. | 
|  | intptr_t implementor_cid() const { | 
|  | // Classes in VM isolate use kVoidCid instead of kDynamicCid | 
|  | // so that we could distinguish them. | 
|  | intptr_t cid = untag()->implementor_cid_; | 
|  | return cid == kVoidCid ? static_cast<intptr_t>(kDynamicCid) : cid; | 
|  | } | 
|  |  | 
|  | // Returns true if the implementor tracking state changes and so must be | 
|  | // propagated to this class's superclass and interfaces. | 
|  | bool NoteImplementor(const Class& implementor) const; | 
|  |  | 
|  | // Used by hot reload to reset the state. | 
|  | void ClearImplementor() const; | 
|  | #endif | 
|  |  | 
|  | static intptr_t num_type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedClass, num_type_arguments_); | 
|  | } | 
|  |  | 
|  | StringPtr Name() const; | 
|  | StringPtr ScrubbedName() const; | 
|  | const char* ScrubbedNameCString() const; | 
|  | StringPtr UserVisibleName() const; | 
|  | const char* UserVisibleNameCString() const; | 
|  |  | 
|  | const char* NameCString(NameVisibility name_visibility) const; | 
|  |  | 
|  | // The mixin for this class if one exists. Otherwise, returns a raw pointer | 
|  | // to this class. | 
|  | ClassPtr Mixin() const; | 
|  |  | 
|  | bool IsInFullSnapshot() const; | 
|  |  | 
|  | virtual StringPtr DictionaryName() const { return Name(); } | 
|  |  | 
|  | ScriptPtr script() const { return untag()->script(); } | 
|  | void set_script(const Script& value) const; | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | KernelProgramInfoPtr KernelProgramInfo() const; | 
|  | #endif | 
|  |  | 
|  | TokenPosition token_pos() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return TokenPosition::kNoSource; | 
|  | #else | 
|  | return untag()->token_pos_; | 
|  | #endif  // defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void set_token_pos(TokenPosition value) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | TokenPosition end_token_pos() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return TokenPosition::kNoSource; | 
|  | #else | 
|  | return untag()->end_token_pos_; | 
|  | #endif  // defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void set_end_token_pos(TokenPosition value) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | uint32_t Hash() const; | 
|  | static uint32_t Hash(ClassPtr); | 
|  |  | 
|  | int32_t SourceFingerprint() const; | 
|  |  | 
|  | // Return the Type with type arguments instantiated to bounds. | 
|  | TypePtr RareType() const; | 
|  |  | 
|  | // Return the non-nullable Type whose arguments are the type parameters | 
|  | // declared by this class. | 
|  | TypePtr DeclarationType() const; | 
|  |  | 
|  | static intptr_t declaration_type_offset() { | 
|  | return OFFSET_OF(UntaggedClass, declaration_type_); | 
|  | } | 
|  |  | 
|  | // Returns flattened instance type arguments vector for | 
|  | // instance of this class, parameterized with declared | 
|  | // type parameters of this class. | 
|  | TypeArgumentsPtr GetDeclarationInstanceTypeArguments() const; | 
|  |  | 
|  | // Returns flattened instance type arguments vector for | 
|  | // instance of this type, parameterized with given type arguments. | 
|  | // | 
|  | // Length of [type_arguments] should match number of type parameters | 
|  | // returned by [NumTypeParameters]. | 
|  | TypeArgumentsPtr GetInstanceTypeArguments(Thread* thread, | 
|  | const TypeArguments& type_arguments, | 
|  | bool canonicalize = true) const; | 
|  |  | 
|  | LibraryPtr library() const { return untag()->library(); } | 
|  | void set_library(const Library& value) const; | 
|  |  | 
|  | // The formal type parameters and their bounds (no defaults), are specified as | 
|  | // an object of type TypeParameters. | 
|  | TypeParametersPtr type_parameters() const { | 
|  | ASSERT(is_declaration_loaded()); | 
|  | return untag()->type_parameters(); | 
|  | } | 
|  | void set_type_parameters(const TypeParameters& value) const; | 
|  | intptr_t NumTypeParameters(Thread* thread) const; | 
|  | intptr_t NumTypeParameters() const { | 
|  | return NumTypeParameters(Thread::Current()); | 
|  | } | 
|  |  | 
|  | // Return the type parameter declared at index. | 
|  | TypeParameterPtr TypeParameterAt( | 
|  | intptr_t index, | 
|  | Nullability nullability = Nullability::kNonNullable) const; | 
|  |  | 
|  | // Length of the flattened instance type arguments vector. | 
|  | // Includes type arguments of the super class. | 
|  | intptr_t NumTypeArguments() const; | 
|  |  | 
|  | // Return true if this class declares type parameters. | 
|  | bool IsGeneric() const { | 
|  | // If the declaration is not loaded, fall back onto NumTypeParameters. | 
|  | if (!is_declaration_loaded()) { | 
|  | return NumTypeParameters(Thread::Current()) > 0; | 
|  | } | 
|  | return type_parameters() != Object::null(); | 
|  | } | 
|  |  | 
|  | // Returns a canonicalized vector of the type parameters instantiated | 
|  | // to bounds (e.g., the type arguments used if no TAV is provided for class | 
|  | // instantiation). | 
|  | // | 
|  | // If non-generic, the empty type arguments vector is returned. | 
|  | TypeArgumentsPtr DefaultTypeArguments(Zone* zone) const; | 
|  |  | 
|  | // If this class is parameterized, each instance has a type_arguments field. | 
|  | static constexpr intptr_t kNoTypeArguments = -1; | 
|  | intptr_t host_type_arguments_field_offset() const { | 
|  | ASSERT(is_type_finalized() || is_prefinalized()); | 
|  | if (untag()->host_type_arguments_field_offset_in_words_ == | 
|  | kNoTypeArguments) { | 
|  | return kNoTypeArguments; | 
|  | } | 
|  | return untag()->host_type_arguments_field_offset_in_words_ * | 
|  | kCompressedWordSize; | 
|  | } | 
|  | intptr_t target_type_arguments_field_offset() const { | 
|  | #if defined(DART_PRECOMPILER) | 
|  | ASSERT(is_type_finalized() || is_prefinalized()); | 
|  | if (untag()->target_type_arguments_field_offset_in_words_ == | 
|  | compiler::target::Class::kNoTypeArguments) { | 
|  | return compiler::target::Class::kNoTypeArguments; | 
|  | } | 
|  | return untag()->target_type_arguments_field_offset_in_words_ * | 
|  | compiler::target::kCompressedWordSize; | 
|  | #else | 
|  | return host_type_arguments_field_offset(); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  | void set_type_arguments_field_offset(intptr_t host_value_in_bytes, | 
|  | intptr_t target_value_in_bytes) const { | 
|  | intptr_t host_value, target_value; | 
|  | if (host_value_in_bytes == kNoTypeArguments || | 
|  | target_value_in_bytes == RTN::Class::kNoTypeArguments) { | 
|  | ASSERT(host_value_in_bytes == kNoTypeArguments && | 
|  | target_value_in_bytes == RTN::Class::kNoTypeArguments); | 
|  | host_value = kNoTypeArguments; | 
|  | target_value = RTN::Class::kNoTypeArguments; | 
|  | } else { | 
|  | ASSERT(kCompressedWordSize != 0 && compiler::target::kCompressedWordSize); | 
|  | host_value = host_value_in_bytes / kCompressedWordSize; | 
|  | target_value = | 
|  | target_value_in_bytes / compiler::target::kCompressedWordSize; | 
|  | } | 
|  | set_type_arguments_field_offset_in_words(host_value, target_value); | 
|  | } | 
|  | void set_type_arguments_field_offset_in_words(intptr_t host_value, | 
|  | intptr_t target_value) const { | 
|  | StoreNonPointer(&untag()->host_type_arguments_field_offset_in_words_, | 
|  | host_value); | 
|  | #if defined(DART_PRECOMPILER) | 
|  | StoreNonPointer(&untag()->target_type_arguments_field_offset_in_words_, | 
|  | target_value); | 
|  | #else | 
|  | // Could be different only during cross-compilation. | 
|  | ASSERT_EQUAL(host_value, target_value); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  | static intptr_t host_type_arguments_field_offset_in_words_offset() { | 
|  | return OFFSET_OF(UntaggedClass, host_type_arguments_field_offset_in_words_); | 
|  | } | 
|  |  | 
|  | // The super type of this class, Object type if not explicitly specified. | 
|  | TypePtr super_type() const { | 
|  | ASSERT(is_declaration_loaded()); | 
|  | return untag()->super_type(); | 
|  | } | 
|  | void set_super_type(const Type& value) const; | 
|  | static intptr_t super_type_offset() { | 
|  | return OFFSET_OF(UntaggedClass, super_type_); | 
|  | } | 
|  |  | 
|  | // Asserts that the class of the super type has been resolved. | 
|  | // If |class_table| is provided it will be used to resolve class id to the | 
|  | // actual class object, instead of using current class table on the isolate | 
|  | // group. | 
|  | ClassPtr SuperClass(ClassTable* class_table = nullptr) const; | 
|  |  | 
|  | // Interfaces is an array of Types. | 
|  | ArrayPtr interfaces() const { | 
|  | ASSERT(is_declaration_loaded()); | 
|  | return untag()->interfaces(); | 
|  | } | 
|  | void set_interfaces(const Array& value) const; | 
|  |  | 
|  | // Returns whether a path from [this] to [cls] can be found, where the first | 
|  | // element is a direct supertype of [this], each following element is a direct | 
|  | // supertype of the previous element and the final element has [cls] as its | 
|  | // type class. If [this] and [cls] are the same class, then the path is empty. | 
|  | // | 
|  | // If [path] is not nullptr, then the elements of the path are added to it. | 
|  | // This path can then be used to compute type arguments of [cls] given type | 
|  | // arguments for an instance of [this]. | 
|  | // | 
|  | // Note: There may be multiple paths to [cls], but the result of applying each | 
|  | // path must be equal to the other results. | 
|  | bool FindInstantiationOf(Zone* zone, | 
|  | const Class& cls, | 
|  | GrowableArray<const Type*>* path, | 
|  | bool consider_only_super_classes = false) const; | 
|  | bool FindInstantiationOf(Zone* zone, | 
|  | const Class& cls, | 
|  | bool consider_only_super_classes = false) const { | 
|  | return FindInstantiationOf(zone, cls, /*path=*/nullptr, | 
|  | consider_only_super_classes); | 
|  | } | 
|  |  | 
|  | // Returns whether a path from [this] to [type] can be found, where the first | 
|  | // element is a direct supertype of [this], each following element is a direct | 
|  | // supertype of the previous element and the final element has the same type | 
|  | // class as [type]. If [this] is the type class of [type], then the path is | 
|  | // empty. | 
|  | // | 
|  | // If [path] is not nullptr, then the elements of the path are added to it. | 
|  | // This path can then be used to compute type arguments of [type]'s type | 
|  | // class given type arguments for an instance of [this]. | 
|  | // | 
|  | // Note: There may be multiple paths to [type]'s type class, but the result of | 
|  | // applying each path must be equal to the other results. | 
|  | bool FindInstantiationOf(Zone* zone, | 
|  | const Type& type, | 
|  | GrowableArray<const Type*>* path, | 
|  | bool consider_only_super_classes = false) const; | 
|  | bool FindInstantiationOf(Zone* zone, | 
|  | const Type& type, | 
|  | bool consider_only_super_classes = false) const { | 
|  | return FindInstantiationOf(zone, type, /*path=*/nullptr, | 
|  | consider_only_super_classes); | 
|  | } | 
|  |  | 
|  | // If [this] is a subtype of a type with type class [cls], then this | 
|  | // returns [cls]<X_0, ..., X_n>, where n is the number of type arguments for | 
|  | // [cls] and where each type argument X_k is either instantiated or has free | 
|  | // class type parameters corresponding to the type parameters of [this]. | 
|  | // Thus, given an instance of [this], the result can be instantiated | 
|  | // with the instance type arguments to get the type of the instance. | 
|  | // | 
|  | // If [this] is not a subtype of a type with type class [cls], returns null. | 
|  | TypePtr GetInstantiationOf(Zone* zone, const Class& cls) const; | 
|  |  | 
|  | // If [this] is a subtype of [type], then this returns [cls]<X_0, ..., X_n>, | 
|  | // where [cls] is the type class of [type], n is the number of type arguments | 
|  | // for [cls], and where each type argument X_k is either instantiated or has | 
|  | // free class type parameters corresponding to the type parameters of [this]. | 
|  | // Thus, given an instance of [this], the result can be instantiated with the | 
|  | // instance type arguments to get the type of the instance. | 
|  | // | 
|  | // If [this] is not a subtype of a type with type class [cls], returns null. | 
|  | TypePtr GetInstantiationOf(Zone* zone, const Type& type) const; | 
|  |  | 
|  | #if !defined(PRODUCT) || !defined(DART_PRECOMPILED_RUNTIME) | 
|  | // Returns the list of classes directly implementing this class. | 
|  | GrowableObjectArrayPtr direct_implementors() const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadReader()); | 
|  | return untag()->direct_implementors(); | 
|  | } | 
|  | GrowableObjectArrayPtr direct_implementors_unsafe() const { | 
|  | return untag()->direct_implementors(); | 
|  | } | 
|  | #endif  // !defined(PRODUCT) || !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void set_direct_implementors(const GrowableObjectArray& implementors) const; | 
|  | void AddDirectImplementor(const Class& subclass, bool is_mixin) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | #if !defined(PRODUCT) || !defined(DART_PRECOMPILED_RUNTIME) | 
|  | // Returns the list of classes having this class as direct superclass. | 
|  | GrowableObjectArrayPtr direct_subclasses() const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadReader()); | 
|  | return direct_subclasses_unsafe(); | 
|  | } | 
|  | GrowableObjectArrayPtr direct_subclasses_unsafe() const { | 
|  | return untag()->direct_subclasses(); | 
|  | } | 
|  | #endif  // !defined(PRODUCT) || !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void set_direct_subclasses(const GrowableObjectArray& subclasses) const; | 
|  | void AddDirectSubclass(const Class& subclass) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | // Check if this class represents the class of null. | 
|  | bool IsNullClass() const { return id() == kNullCid; } | 
|  |  | 
|  | // Check if this class represents the 'dynamic' class. | 
|  | bool IsDynamicClass() const { return id() == kDynamicCid; } | 
|  |  | 
|  | // Check if this class represents the 'void' class. | 
|  | bool IsVoidClass() const { return id() == kVoidCid; } | 
|  |  | 
|  | // Check if this class represents the 'Never' class. | 
|  | bool IsNeverClass() const { return id() == kNeverCid; } | 
|  |  | 
|  | // Check if this class represents the 'Object' class. | 
|  | bool IsObjectClass() const { return id() == kInstanceCid; } | 
|  |  | 
|  | // Check if this class represents the 'Function' class. | 
|  | bool IsDartFunctionClass() const; | 
|  |  | 
|  | // Check if this class represents the 'Future' class. | 
|  | bool IsFutureClass() const; | 
|  |  | 
|  | // Check if this class represents the 'FutureOr' class. | 
|  | bool IsFutureOrClass() const { return id() == kFutureOrCid; } | 
|  |  | 
|  | // Check if this class represents the 'Closure' class. | 
|  | bool IsClosureClass() const { return id() == kClosureCid; } | 
|  | static bool IsClosureClass(ClassPtr cls) { | 
|  | return GetClassId(cls) == kClosureCid; | 
|  | } | 
|  |  | 
|  | // Check if this class represents the 'Record' class. | 
|  | bool IsRecordClass() const { return id() == kRecordCid; } | 
|  |  | 
|  | static bool IsInFullSnapshot(ClassPtr cls) { | 
|  | NoSafepointScope no_safepoint; | 
|  | return UntaggedLibrary::InFullSnapshotBit::decode( | 
|  | cls->untag()->library()->untag()->flags_); | 
|  | } | 
|  |  | 
|  | static intptr_t GetClassId(ClassPtr cls) { | 
|  | NoSafepointScope no_safepoint; | 
|  | return cls->untag()->id_; | 
|  | } | 
|  |  | 
|  | // Returns true if the type specified by cls, type_arguments, and nullability | 
|  | // is a subtype of the other type. | 
|  | static bool IsSubtypeOf( | 
|  | const Class& cls, | 
|  | const TypeArguments& type_arguments, | 
|  | Nullability nullability, | 
|  | const AbstractType& other, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr); | 
|  |  | 
|  | // Check if this is the top level class. | 
|  | bool IsTopLevel() const; | 
|  |  | 
|  | bool IsPrivate() const; | 
|  |  | 
|  | DART_WARN_UNUSED_RESULT | 
|  | ErrorPtr VerifyEntryPoint() const; | 
|  |  | 
|  | // Returns an array of instance and static fields defined by this class. | 
|  | ArrayPtr fields() const { | 
|  | // We rely on the fact that any loads from the array are dependent loads | 
|  | // and avoid the load-acquire barrier here. | 
|  | return untag()->fields(); | 
|  | } | 
|  | void SetFields(const Array& value) const; | 
|  | void AddField(const Field& field) const; | 
|  | void AddFields(const GrowableArray<const Field*>& fields) const; | 
|  |  | 
|  | intptr_t FindFieldIndex(const Field& needle) const; | 
|  | FieldPtr FieldFromIndex(intptr_t idx) const; | 
|  |  | 
|  | // If this is a dart:internal.ClassID class, then inject our own const | 
|  | // fields. Returns true if synthetic fields are injected and regular | 
|  | // field declarations should be ignored. | 
|  | bool InjectCIDFields() const; | 
|  |  | 
|  | // Returns an array of all instance fields of this class and its superclasses | 
|  | // indexed by offset in words. | 
|  | // If |class_table| is provided it will be used to resolve super classes by | 
|  | // class id, instead of the current class_table stored in the isolate. | 
|  | ArrayPtr OffsetToFieldMap(ClassTable* class_table = nullptr) const; | 
|  |  | 
|  | // Returns true if non-static fields are defined. | 
|  | bool HasInstanceFields() const; | 
|  |  | 
|  | ArrayPtr current_functions() const { | 
|  | // We rely on the fact that any loads from the array are dependent loads | 
|  | // and avoid the load-acquire barrier here. | 
|  | return untag()->functions(); | 
|  | } | 
|  | ArrayPtr functions() const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadReader()); | 
|  | return current_functions(); | 
|  | } | 
|  | void SetFunctions(const Array& value) const; | 
|  | void AddFunction(const Function& function) const; | 
|  | intptr_t FindFunctionIndex(const Function& needle) const; | 
|  | FunctionPtr FunctionFromIndex(intptr_t idx) const; | 
|  | intptr_t FindImplicitClosureFunctionIndex(const Function& needle) const; | 
|  | FunctionPtr ImplicitClosureFunctionFromIndex(intptr_t idx) const; | 
|  |  | 
|  | FunctionPtr LookupFunctionReadLocked(const String& name) const; | 
|  | FunctionPtr LookupDynamicFunctionUnsafe(const String& name) const; | 
|  |  | 
|  | FunctionPtr LookupDynamicFunctionAllowPrivate(const String& name) const; | 
|  | FunctionPtr LookupStaticFunction(const String& name) const; | 
|  | FunctionPtr LookupStaticFunctionAllowPrivate(const String& name) const; | 
|  | FunctionPtr LookupConstructor(const String& name) const; | 
|  | FunctionPtr LookupConstructorAllowPrivate(const String& name) const; | 
|  | FunctionPtr LookupFactory(const String& name) const; | 
|  | FunctionPtr LookupFactoryAllowPrivate(const String& name) const; | 
|  | FunctionPtr LookupFunctionAllowPrivate(const String& name) const; | 
|  | FunctionPtr LookupGetterFunction(const String& name) const; | 
|  | FunctionPtr LookupSetterFunction(const String& name) const; | 
|  | FieldPtr LookupInstanceField(const String& name) const; | 
|  | FieldPtr LookupStaticField(const String& name) const; | 
|  | FieldPtr LookupField(const String& name) const; | 
|  | FieldPtr LookupFieldAllowPrivate(const String& name, | 
|  | bool instance_only = false) const; | 
|  | FieldPtr LookupInstanceFieldAllowPrivate(const String& name) const; | 
|  | FieldPtr LookupStaticFieldAllowPrivate(const String& name) const; | 
|  |  | 
|  | // The methods above are more efficient than this generic one. | 
|  | InstancePtr LookupCanonicalInstance(Zone* zone, const Instance& value) const; | 
|  |  | 
|  | InstancePtr InsertCanonicalConstant(Zone* zone, | 
|  | const Instance& constant) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedClass)); | 
|  | } | 
|  |  | 
|  | // Returns true if any class implements this interface via `implements`. | 
|  | // Returns false if all possible implementations of this interface must be | 
|  | // instances of this class or its subclasses. | 
|  | bool is_implemented() const { return ImplementedBit::decode(state_bits()); } | 
|  | void set_is_implemented(bool value) const; | 
|  | void set_is_implemented_unsafe(bool value) const; | 
|  |  | 
|  | bool is_abstract() const { return AbstractBit::decode(state_bits()); } | 
|  | void set_is_abstract() const; | 
|  |  | 
|  | UntaggedClass::ClassLoadingState class_loading_state() const { | 
|  | return ClassLoadingBits::decode(state_bits()); | 
|  | } | 
|  |  | 
|  | bool is_declaration_loaded() const { | 
|  | return class_loading_state() >= UntaggedClass::kDeclarationLoaded; | 
|  | } | 
|  | void set_is_declaration_loaded() const; | 
|  | void set_is_declaration_loaded_unsafe() const; | 
|  |  | 
|  | bool is_type_finalized() const { | 
|  | return class_loading_state() >= UntaggedClass::kTypeFinalized; | 
|  | } | 
|  | void set_is_type_finalized() const; | 
|  |  | 
|  | bool is_synthesized_class() const { | 
|  | return SynthesizedClassBit::decode(state_bits()); | 
|  | } | 
|  | void set_is_synthesized_class() const; | 
|  | void set_is_synthesized_class_unsafe() const; | 
|  |  | 
|  | bool is_enum_class() const { return EnumBit::decode(state_bits()); } | 
|  | void set_is_enum_class() const; | 
|  |  | 
|  | bool is_finalized() const { | 
|  | return ClassFinalizedBits::decode(state_bits()) == | 
|  | UntaggedClass::kFinalized || | 
|  | ClassFinalizedBits::decode(state_bits()) == | 
|  | UntaggedClass::kAllocateFinalized; | 
|  | } | 
|  | void set_is_finalized() const; | 
|  | void set_is_finalized_unsafe() const; | 
|  |  | 
|  | bool is_allocate_finalized() const { | 
|  | return ClassFinalizedBits::decode(state_bits()) == | 
|  | UntaggedClass::kAllocateFinalized; | 
|  | } | 
|  | void set_is_allocate_finalized() const; | 
|  |  | 
|  | bool is_prefinalized() const { | 
|  | return ClassFinalizedBits::decode(state_bits()) == | 
|  | UntaggedClass::kPreFinalized; | 
|  | } | 
|  |  | 
|  | void set_is_prefinalized() const; | 
|  |  | 
|  | bool is_const() const { return ConstBit::decode(state_bits()); } | 
|  | void set_is_const() const; | 
|  |  | 
|  | // Tests if this is a mixin application class which was desugared | 
|  | // to a normal class by kernel mixin transformation | 
|  | // (pkg/kernel/lib/transformations/mixin_full_resolution.dart). | 
|  | // | 
|  | // In such case, its mixed-in type was pulled into the end of | 
|  | // interfaces list. | 
|  | bool is_transformed_mixin_application() const { | 
|  | return TransformedMixinApplicationBit::decode(state_bits()); | 
|  | } | 
|  | void set_is_transformed_mixin_application() const; | 
|  |  | 
|  | bool is_sealed() const { return SealedBit::decode(state_bits()); } | 
|  | void set_is_sealed() const; | 
|  |  | 
|  | bool is_mixin_class() const { return MixinClassBit::decode(state_bits()); } | 
|  | void set_is_mixin_class() const; | 
|  |  | 
|  | bool is_base_class() const { return BaseClassBit::decode(state_bits()); } | 
|  | void set_is_base_class() const; | 
|  |  | 
|  | bool is_interface_class() const { | 
|  | return InterfaceClassBit::decode(state_bits()); | 
|  | } | 
|  | void set_is_interface_class() const; | 
|  |  | 
|  | bool is_final() const { return FinalBit::decode(state_bits()); } | 
|  | void set_is_final() const; | 
|  |  | 
|  | bool is_fields_marked_nullable() const { | 
|  | return FieldsMarkedNullableBit::decode(state_bits()); | 
|  | } | 
|  | void set_is_fields_marked_nullable() const; | 
|  |  | 
|  | bool is_allocated() const { return IsAllocatedBit::decode(state_bits()); } | 
|  | void set_is_allocated(bool value) const; | 
|  | void set_is_allocated_unsafe(bool value) const; | 
|  |  | 
|  | bool is_loaded() const { return IsLoadedBit::decode(state_bits()); } | 
|  | void set_is_loaded(bool value) const; | 
|  |  | 
|  | #if defined(DART_DYNAMIC_MODULES) | 
|  | bool is_declared_in_bytecode() const { | 
|  | return IsDeclaredInBytecodeBit::decode(state_bits()); | 
|  | } | 
|  | void set_is_declared_in_bytecode(bool value) const; | 
|  | #else | 
|  | bool is_declared_in_bytecode() const { return false; } | 
|  | #endif  // defined(DART_DYNAMIC_MODULES) | 
|  |  | 
|  | uint16_t num_native_fields() const { return untag()->num_native_fields_; } | 
|  | void set_num_native_fields(uint16_t value) const { | 
|  | StoreNonPointer(&untag()->num_native_fields_, value); | 
|  | } | 
|  | static uint16_t NumNativeFieldsOf(ClassPtr clazz) { | 
|  | return clazz->untag()->num_native_fields_; | 
|  | } | 
|  | static bool IsIsolateUnsendable(ClassPtr clazz) { | 
|  | return IsIsolateUnsendableBit::decode(clazz->untag()->state_bits_); | 
|  | } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | CodePtr allocation_stub() const { return untag()->allocation_stub(); } | 
|  | void set_allocation_stub(const Code& value) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | intptr_t kernel_offset() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return 0; | 
|  | #else | 
|  | return untag()->kernel_offset_; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void set_kernel_offset(intptr_t value) const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | #else | 
|  | ASSERT(value >= 0); | 
|  | StoreNonPointer(&untag()->kernel_offset_, value); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void DisableAllocationStub() const; | 
|  |  | 
|  | ArrayPtr constants() const; | 
|  | void set_constants(const Array& value) const; | 
|  |  | 
|  | intptr_t FindInvocationDispatcherFunctionIndex(const Function& needle) const; | 
|  | FunctionPtr InvocationDispatcherFunctionFromIndex(intptr_t idx) const; | 
|  |  | 
|  | FunctionPtr GetInvocationDispatcher(const String& target_name, | 
|  | const Array& args_desc, | 
|  | UntaggedFunction::Kind kind, | 
|  | bool create_if_absent) const; | 
|  |  | 
|  | FunctionPtr GetRecordFieldGetter(const String& getter_name) const; | 
|  |  | 
|  | void Finalize() const; | 
|  |  | 
|  | ObjectPtr Invoke(const String& selector, | 
|  | const Array& arguments, | 
|  | const Array& argument_names, | 
|  | bool check_is_entrypoint = true, | 
|  | bool respect_reflectable = true) const; | 
|  | ObjectPtr InvokeGetter(const String& selector, | 
|  | bool check_is_entrypoint = true, | 
|  | bool respect_reflectable = true, | 
|  | bool for_invocation = false) const; | 
|  | ObjectPtr InvokeSetter(const String& selector, | 
|  | const Instance& argument, | 
|  | bool check_is_entrypoint = true, | 
|  | bool respect_reflectable = true) const; | 
|  |  | 
|  | // Evaluate the given expression as if it appeared in a static method of this | 
|  | // class and return the resulting value, or an error object if evaluating the | 
|  | // expression fails. The method has the formal (type) parameters given in | 
|  | // (type_)param_names, and is invoked with the (type)argument values given in | 
|  | // (type_)param_values. | 
|  | ObjectPtr EvaluateCompiledExpression( | 
|  | const ExternalTypedData& kernel_buffer, | 
|  | const Array& type_definitions, | 
|  | const Array& param_values, | 
|  | const TypeArguments& type_param_values) const; | 
|  |  | 
|  | // Load class declaration (super type, interfaces, type parameters and | 
|  | // number of type arguments) if it is not loaded yet. | 
|  | void EnsureDeclarationLoaded() const; | 
|  |  | 
|  | ErrorPtr EnsureIsFinalized(Thread* thread) const; | 
|  | ErrorPtr EnsureIsAllocateFinalized(Thread* thread) const; | 
|  |  | 
|  | // Allocate a class used for VM internal objects. | 
|  | template <class FakeObject, class TargetFakeObject> | 
|  | static ClassPtr New(IsolateGroup* isolate_group, bool register_class = true); | 
|  |  | 
|  | // Allocate instance classes. | 
|  | static ClassPtr New(const Library& lib, | 
|  | const String& name, | 
|  | const Script& script, | 
|  | TokenPosition token_pos, | 
|  | bool register_class = true); | 
|  | static ClassPtr NewNativeWrapper(const Library& library, | 
|  | const String& name, | 
|  | int num_fields); | 
|  |  | 
|  | // Allocate the raw string classes. | 
|  | static ClassPtr NewStringClass(intptr_t class_id, | 
|  | IsolateGroup* isolate_group); | 
|  |  | 
|  | // Allocate the raw TypedData classes. | 
|  | static ClassPtr NewTypedDataClass(intptr_t class_id, | 
|  | IsolateGroup* isolate_group); | 
|  |  | 
|  | // Allocate the raw TypedDataView/ByteDataView classes. | 
|  | static ClassPtr NewTypedDataViewClass(intptr_t class_id, | 
|  | IsolateGroup* isolate_group); | 
|  | static ClassPtr NewUnmodifiableTypedDataViewClass( | 
|  | intptr_t class_id, | 
|  | IsolateGroup* isolate_group); | 
|  |  | 
|  | // Allocate the raw ExternalTypedData classes. | 
|  | static ClassPtr NewExternalTypedDataClass(intptr_t class_id, | 
|  | IsolateGroup* isolate); | 
|  |  | 
|  | // Allocate the raw Pointer classes. | 
|  | static ClassPtr NewPointerClass(intptr_t class_id, | 
|  | IsolateGroup* isolate_group); | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | // Register code that has used CHA for optimization. | 
|  | // TODO(srdjan): Also register kind of CHA optimization (e.g.: leaf class, | 
|  | // leaf method, ...). | 
|  | void RegisterCHACode(const Code& code); | 
|  |  | 
|  | void DisableCHAOptimizedCode(const Class& subclass); | 
|  |  | 
|  | void DisableAllCHAOptimizedCode(); | 
|  |  | 
|  | void DisableCHAImplementorUsers() { DisableAllCHAOptimizedCode(); } | 
|  |  | 
|  | // Return the list of code objects that were compiled using CHA of this class. | 
|  | // These code objects will be invalidated if new subclasses of this class | 
|  | // are finalized. | 
|  | WeakArrayPtr dependent_code() const; | 
|  | void set_dependent_code(const WeakArray& array) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | bool TraceAllocation(IsolateGroup* isolate_group) const; | 
|  | void SetTraceAllocation(bool trace_allocation) const; | 
|  |  | 
|  | void CopyStaticFieldValues(ProgramReloadContext* reload_context, | 
|  | const Class& old_cls) const; | 
|  | void PatchFieldsAndFunctions() const; | 
|  | void MigrateImplicitStaticClosures(ProgramReloadContext* context, | 
|  | const Class& new_cls) const; | 
|  | void CopyCanonicalConstants(const Class& old_cls) const; | 
|  | void CopyDeclarationType(const Class& old_cls) const; | 
|  | void CheckReload(const Class& replacement, | 
|  | ProgramReloadContext* context) const; | 
|  |  | 
|  | void AddInvocationDispatcher(const String& target_name, | 
|  | const Array& args_desc, | 
|  | const Function& dispatcher) const; | 
|  |  | 
|  | static int32_t host_instance_size_in_words(const ClassPtr cls) { | 
|  | return cls->untag()->host_instance_size_in_words_; | 
|  | } | 
|  |  | 
|  | static int32_t target_instance_size_in_words(const ClassPtr cls) { | 
|  | #if defined(DART_PRECOMPILER) | 
|  | return cls->untag()->target_instance_size_in_words_; | 
|  | #else | 
|  | return host_instance_size_in_words(cls); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  |  | 
|  | static int32_t host_next_field_offset_in_words(const ClassPtr cls) { | 
|  | return cls->untag()->host_next_field_offset_in_words_; | 
|  | } | 
|  |  | 
|  | static int32_t target_next_field_offset_in_words(const ClassPtr cls) { | 
|  | #if defined(DART_PRECOMPILER) | 
|  | return cls->untag()->target_next_field_offset_in_words_; | 
|  | #else | 
|  | return host_next_field_offset_in_words(cls); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  |  | 
|  | static int32_t host_type_arguments_field_offset_in_words(const ClassPtr cls) { | 
|  | return cls->untag()->host_type_arguments_field_offset_in_words_; | 
|  | } | 
|  |  | 
|  | static int32_t target_type_arguments_field_offset_in_words( | 
|  | const ClassPtr cls) { | 
|  | #if defined(DART_PRECOMPILER) | 
|  | return cls->untag()->target_type_arguments_field_offset_in_words_; | 
|  | #else | 
|  | return host_type_arguments_field_offset_in_words(cls); | 
|  | #endif  // defined(DART_PRECOMPILER) | 
|  | } | 
|  |  | 
|  | static intptr_t UnboxedFieldSizeInBytesByCid(intptr_t cid); | 
|  | void MarkFieldBoxedDuringReload(ClassTable* class_table, | 
|  | const Field& field) const; | 
|  |  | 
|  | #if !defined(PRODUCT) || defined(FORCE_INCLUDE_SAMPLING_HEAP_PROFILER) | 
|  | void SetUserVisibleNameInClassTable(); | 
|  | #endif  // !defined(PRODUCT) || defined(FORCE_INCLUDE_SAMPLING_HEAP_PROFILER) | 
|  |  | 
|  | private: | 
|  | TypePtr declaration_type() const { | 
|  | return untag()->declaration_type<std::memory_order_acquire>(); | 
|  | } | 
|  |  | 
|  | // Caches the declaration type of this class. | 
|  | void set_declaration_type(const Type& type) const; | 
|  |  | 
|  | TypeArgumentsPtr declaration_instance_type_arguments() const { | 
|  | return untag() | 
|  | ->declaration_instance_type_arguments<std::memory_order_acquire>(); | 
|  | } | 
|  | void set_declaration_instance_type_arguments( | 
|  | const TypeArguments& value) const; | 
|  |  | 
|  | bool CanReloadFinalized(const Class& replacement, | 
|  | ProgramReloadContext* context) const; | 
|  | bool CanReloadPreFinalized(const Class& replacement, | 
|  | ProgramReloadContext* context) const; | 
|  |  | 
|  | // Tells whether instances need morphing for reload. | 
|  | bool RequiresInstanceMorphing(ClassTable* class_table, | 
|  | const Class& replacement) const; | 
|  |  | 
|  | template <class FakeInstance, class TargetFakeInstance> | 
|  | static ClassPtr NewCommon(intptr_t index); | 
|  |  | 
|  | enum MemberKind { | 
|  | kAny = 0, | 
|  | kStatic, | 
|  | kInstance, | 
|  | kInstanceAllowAbstract, | 
|  | kConstructor, | 
|  | kFactory, | 
|  | }; | 
|  | using ConstBit = BitField<uint32_t, bool>; | 
|  | using ImplementedBit = BitField<uint32_t, bool, ConstBit::kNextBit>; | 
|  | using ClassFinalizedBits = BitField<uint32_t, | 
|  | UntaggedClass::ClassFinalizedState, | 
|  | ImplementedBit::kNextBit, | 
|  | 2>; | 
|  | using ClassLoadingBits = BitField<uint32_t, | 
|  | UntaggedClass::ClassLoadingState, | 
|  | ClassFinalizedBits::kNextBit, | 
|  | 2>; | 
|  | using AbstractBit = BitField<uint32_t, bool, ClassLoadingBits::kNextBit>; | 
|  | using SynthesizedClassBit = BitField<uint32_t, bool, AbstractBit::kNextBit>; | 
|  | using FieldsMarkedNullableBit = | 
|  | BitField<uint32_t, bool, SynthesizedClassBit::kNextBit>; | 
|  | using EnumBit = BitField<uint32_t, bool, FieldsMarkedNullableBit::kNextBit>; | 
|  | using TransformedMixinApplicationBit = | 
|  | BitField<uint32_t, bool, EnumBit::kNextBit>; | 
|  | using IsAllocatedBit = | 
|  | BitField<uint32_t, bool, TransformedMixinApplicationBit::kNextBit>; | 
|  | using IsLoadedBit = BitField<uint32_t, bool, IsAllocatedBit::kNextBit>; | 
|  | using HasPragmaBit = BitField<uint32_t, bool, IsLoadedBit::kNextBit>; | 
|  | using SealedBit = BitField<uint32_t, bool, HasPragmaBit::kNextBit>; | 
|  | using MixinClassBit = BitField<uint32_t, bool, SealedBit::kNextBit>; | 
|  | using BaseClassBit = BitField<uint32_t, bool, MixinClassBit::kNextBit>; | 
|  | using InterfaceClassBit = BitField<uint32_t, bool, BaseClassBit::kNextBit>; | 
|  | using FinalBit = BitField<uint32_t, bool, InterfaceClassBit::kNextBit>; | 
|  | // Whether instances of the class cannot be sent across ports. | 
|  | // | 
|  | // Will be true iff | 
|  | //    - class is marked with `@pragma('vm:isolate-unsendable') | 
|  | //    - super class / super interface classes are marked as unsendable. | 
|  | //    - class has native fields. | 
|  | using IsIsolateUnsendableBit = BitField<uint32_t, bool, FinalBit::kNextBit>; | 
|  | // True if this class has `@pragma('vm:isolate-unsendable')` annotation or | 
|  | // base class or implemented interfaces has this bit. | 
|  | using IsIsolateUnsendableDueToPragmaBit = | 
|  | BitField<uint32_t, bool, IsIsolateUnsendableBit::kNextBit>; | 
|  | // Will be set to 1 for the following classes: | 
|  | // | 
|  | // 1. Deeply immutable class. | 
|  | //    a. Statically guaranteed deeply immutable classes. | 
|  | //       `@pragma('vm:deeply-immutable')`. | 
|  | //    b. VM recognized deeply immutable classes. | 
|  | //       `IsDeeplyImmutableCid(intptr_t predefined_cid)`. | 
|  | // | 
|  | // See also ImmutableBit in raw_object.h. | 
|  | using IsDeeplyImmutableBit = | 
|  | BitField<uint32_t, bool, IsIsolateUnsendableDueToPragmaBit::kNextBit>; | 
|  | // This class is a subtype of Future. | 
|  | using IsFutureSubtypeBit = | 
|  | BitField<uint32_t, bool, IsDeeplyImmutableBit::kNextBit>; | 
|  | // This class has a non-abstract subtype which is a subtype of Future. | 
|  | // It means that variable of static type based on this class may hold | 
|  | // a Future instance. | 
|  | using CanBeFutureBit = BitField<uint32_t, bool, IsFutureSubtypeBit::kNextBit>; | 
|  | // This class has a dynamically extendable subtype. | 
|  | using HasDynamicallyExtendableSubtypesBit = | 
|  | BitField<uint32_t, bool, CanBeFutureBit::kNextBit>; | 
|  | // This class was loaded from bytecode at runtime. | 
|  | using IsDeclaredInBytecodeBit = | 
|  | BitField<uint32_t, bool, HasDynamicallyExtendableSubtypesBit::kNextBit>; | 
|  |  | 
|  | void set_name(const String& value) const; | 
|  | void set_user_name(const String& value) const; | 
|  | const char* GenerateUserVisibleName() const; | 
|  | void set_state_bits(intptr_t bits) const; | 
|  | void set_implementor_cid(intptr_t value) const; | 
|  |  | 
|  | FunctionPtr CreateInvocationDispatcher(const String& target_name, | 
|  | const Array& args_desc, | 
|  | UntaggedFunction::Kind kind) const; | 
|  |  | 
|  | FunctionPtr CreateRecordFieldGetter(const String& getter_name) const; | 
|  |  | 
|  | // Returns the bitmap of unboxed fields | 
|  | UnboxedFieldBitmap CalculateFieldOffsets() const; | 
|  |  | 
|  | // functions_hash_table is in use iff there are at least this many functions. | 
|  | static constexpr intptr_t kFunctionLookupHashThreshold = 16; | 
|  |  | 
|  | // Initial value for the cached number of type arguments. | 
|  | static constexpr intptr_t kUnknownNumTypeArguments = -1; | 
|  |  | 
|  | int16_t num_type_arguments() const { | 
|  | return LoadNonPointer<int16_t, std::memory_order_relaxed>( | 
|  | &untag()->num_type_arguments_); | 
|  | } | 
|  |  | 
|  | uint32_t state_bits() const { | 
|  | // Ensure any following load instructions do not get performed before this | 
|  | // one. | 
|  | return LoadNonPointer<uint32_t, std::memory_order_acquire>( | 
|  | &untag()->state_bits_); | 
|  | } | 
|  |  | 
|  | public: | 
|  | void set_num_type_arguments(intptr_t value) const; | 
|  | void set_num_type_arguments_unsafe(intptr_t value) const; | 
|  |  | 
|  | bool has_pragma() const { return HasPragmaBit::decode(state_bits()); } | 
|  | void set_has_pragma(bool value) const; | 
|  |  | 
|  | void set_is_isolate_unsendable(bool value) const; | 
|  | bool is_isolate_unsendable() const { | 
|  | ASSERT(is_finalized());  // This bit is initialized in class finalizer. | 
|  | return IsIsolateUnsendableBit::decode(state_bits()); | 
|  | } | 
|  |  | 
|  | void set_is_isolate_unsendable_due_to_pragma(bool value) const; | 
|  | bool is_isolate_unsendable_due_to_pragma() const { | 
|  | return IsIsolateUnsendableDueToPragmaBit::decode(state_bits()); | 
|  | } | 
|  |  | 
|  | void set_is_deeply_immutable(bool value) const; | 
|  | bool is_deeply_immutable() const { | 
|  | return IsDeeplyImmutableBit::decode(state_bits()); | 
|  | } | 
|  | static bool IsDeeplyImmutable(ClassPtr clazz) { | 
|  | return IsDeeplyImmutableBit::decode(clazz->untag()->state_bits_); | 
|  | } | 
|  |  | 
|  | void set_is_future_subtype(bool value) const; | 
|  | bool is_future_subtype() const { | 
|  | ASSERT(is_type_finalized()); | 
|  | return IsFutureSubtypeBit::decode(state_bits()); | 
|  | } | 
|  |  | 
|  | void set_can_be_future(bool value) const; | 
|  | bool can_be_future() const { return CanBeFutureBit::decode(state_bits()); } | 
|  |  | 
|  | void set_has_dynamically_extendable_subtypes(bool value) const; | 
|  | bool has_dynamically_extendable_subtypes() const { | 
|  | return HasDynamicallyExtendableSubtypesBit::decode(state_bits()); | 
|  | } | 
|  |  | 
|  | private: | 
|  | void set_functions(const Array& value) const; | 
|  | void set_fields(const Array& value) const; | 
|  | void set_invocation_dispatcher_cache(const Array& cache) const; | 
|  |  | 
|  | ArrayPtr invocation_dispatcher_cache() const; | 
|  |  | 
|  | // Calculates number of type arguments of this class. | 
|  | // This includes type arguments of a superclass and takes overlapping | 
|  | // of type arguments into account. | 
|  | intptr_t ComputeNumTypeArguments() const; | 
|  |  | 
|  | // Assigns empty array to all raw class array fields. | 
|  | void InitEmptyFields() const; | 
|  |  | 
|  | static FunctionPtr CheckFunctionType(const Function& func, MemberKind kind); | 
|  | FunctionPtr LookupFunctionReadLocked(const String& name, | 
|  | MemberKind kind) const; | 
|  | FunctionPtr LookupFunctionAllowPrivate(const String& name, | 
|  | MemberKind kind) const; | 
|  | FieldPtr LookupField(const String& name, MemberKind kind) const; | 
|  |  | 
|  | FunctionPtr LookupAccessorFunction(const char* prefix, | 
|  | intptr_t prefix_length, | 
|  | const String& name) const; | 
|  |  | 
|  | // Allocate an instance class which has a VM implementation. | 
|  | template <class FakeInstance, class TargetFakeInstance> | 
|  | static ClassPtr New(intptr_t id, | 
|  | IsolateGroup* isolate_group, | 
|  | bool register_class = true, | 
|  | bool is_abstract = false); | 
|  |  | 
|  | // Helper that calls 'Class::New<Instance>(kIllegalCid)'. | 
|  | static ClassPtr NewInstanceClass(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Class, Object); | 
|  | friend class AbstractType; | 
|  | friend class Instance; | 
|  | friend class Object; | 
|  | friend class Type; | 
|  | friend class InterpreterHelpers; | 
|  | friend class Intrinsifier; | 
|  | friend class ProgramWalker; | 
|  | friend class Precompiler; | 
|  | friend class ClassFinalizer; | 
|  | }; | 
|  |  | 
|  | // Classification of type genericity according to type parameter owners. | 
|  | enum Genericity { | 
|  | kAny,           // Consider type params of current class and functions. | 
|  | kCurrentClass,  // Consider type params of current class only. | 
|  | kFunctions,     // Consider type params of current and parent functions. | 
|  | }; | 
|  |  | 
|  | // Wrapper of a [Class] with different [Script] and kernel binary. | 
|  | // | 
|  | // We use this as owner of [Field]/[Function] objects that were from a different | 
|  | // script/kernel than the actual class object. | 
|  | // | 
|  | //  * used for corelib patches that live in different .dart files than the | 
|  | //    library itself. | 
|  | // | 
|  | //  * used for library parts that live in different .dart files than the library | 
|  | //    itself. | 
|  | // | 
|  | //  * used in reload to make old [Function]/[Field] objects have the old script | 
|  | //    kernel data. | 
|  | // | 
|  | class PatchClass : public Object { | 
|  | public: | 
|  | ClassPtr wrapped_class() const { return untag()->wrapped_class(); } | 
|  | ScriptPtr script() const { return untag()->script(); } | 
|  |  | 
|  | intptr_t kernel_library_index() const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | return untag()->kernel_library_index_; | 
|  | #else | 
|  | return -1; | 
|  | #endif | 
|  | } | 
|  | void set_kernel_library_index(intptr_t index) const { | 
|  | NOT_IN_PRECOMPILED(StoreNonPointer(&untag()->kernel_library_index_, index)); | 
|  | } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | KernelProgramInfoPtr kernel_program_info() const { | 
|  | return untag()->kernel_program_info(); | 
|  | } | 
|  | void set_kernel_program_info(const KernelProgramInfo& info) const; | 
|  | #endif | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedPatchClass)); | 
|  | } | 
|  | static bool IsInFullSnapshot(PatchClassPtr cls) { | 
|  | NoSafepointScope no_safepoint; | 
|  | return Class::IsInFullSnapshot(cls->untag()->wrapped_class()); | 
|  | } | 
|  |  | 
|  | static PatchClassPtr New(const Class& wrapped_class, | 
|  | const KernelProgramInfo& info, | 
|  | const Script& source); | 
|  |  | 
|  | private: | 
|  | void set_wrapped_class(const Class& value) const; | 
|  | void set_script(const Script& value) const; | 
|  |  | 
|  | static PatchClassPtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(PatchClass, Object); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class SingleTargetCache : public Object { | 
|  | public: | 
|  | CodePtr target() const { return untag()->target(); } | 
|  | void set_target(const Code& target) const; | 
|  | static intptr_t target_offset() { | 
|  | return OFFSET_OF(UntaggedSingleTargetCache, target_); | 
|  | } | 
|  |  | 
|  | #define DEFINE_NON_POINTER_FIELD_ACCESSORS(type, name)                         \ | 
|  | type name() const { return untag()->name##_; }                               \ | 
|  | void set_##name(type value) const {                                          \ | 
|  | StoreNonPointer(&untag()->name##_, value);                                 \ | 
|  | }                                                                            \ | 
|  | static intptr_t name##_offset() {                                            \ | 
|  | return OFFSET_OF(UntaggedSingleTargetCache, name##_);                      \ | 
|  | } | 
|  |  | 
|  | DEFINE_NON_POINTER_FIELD_ACCESSORS(uword, entry_point); | 
|  | DEFINE_NON_POINTER_FIELD_ACCESSORS(intptr_t, lower_limit); | 
|  | DEFINE_NON_POINTER_FIELD_ACCESSORS(intptr_t, upper_limit); | 
|  | #undef DEFINE_NON_POINTER_FIELD_ACCESSORS | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedSingleTargetCache)); | 
|  | } | 
|  |  | 
|  | static SingleTargetCachePtr New(); | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(SingleTargetCache, Object); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class MonomorphicSmiableCall : public Object { | 
|  | public: | 
|  | classid_t expected_cid() const { return untag()->expected_cid_; } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedMonomorphicSmiableCall)); | 
|  | } | 
|  |  | 
|  | static MonomorphicSmiableCallPtr New(classid_t expected_cid, | 
|  | const Code& target); | 
|  |  | 
|  | static intptr_t expected_cid_offset() { | 
|  | return OFFSET_OF(UntaggedMonomorphicSmiableCall, expected_cid_); | 
|  | } | 
|  |  | 
|  | static intptr_t entrypoint_offset() { | 
|  | return OFFSET_OF(UntaggedMonomorphicSmiableCall, entrypoint_); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(MonomorphicSmiableCall, Object); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class CallSiteData : public Object { | 
|  | public: | 
|  | StringPtr target_name() const { return untag()->target_name(); } | 
|  | ArrayPtr arguments_descriptor() const { return untag()->args_descriptor(); } | 
|  |  | 
|  | intptr_t TypeArgsLen() const; | 
|  |  | 
|  | intptr_t CountWithTypeArgs() const; | 
|  |  | 
|  | intptr_t CountWithoutTypeArgs() const; | 
|  |  | 
|  | intptr_t SizeWithoutTypeArgs() const; | 
|  |  | 
|  | intptr_t SizeWithTypeArgs() const; | 
|  |  | 
|  | static intptr_t target_name_offset() { | 
|  | return OFFSET_OF(UntaggedCallSiteData, target_name_); | 
|  | } | 
|  |  | 
|  | static intptr_t arguments_descriptor_offset() { | 
|  | return OFFSET_OF(UntaggedCallSiteData, args_descriptor_); | 
|  | } | 
|  |  | 
|  | private: | 
|  | void set_target_name(const String& value) const; | 
|  | void set_arguments_descriptor(const Array& value) const; | 
|  |  | 
|  | HEAP_OBJECT_IMPLEMENTATION(CallSiteData, Object) | 
|  |  | 
|  | friend class ICData; | 
|  | friend class MegamorphicCache; | 
|  | }; | 
|  |  | 
|  | class UnlinkedCall : public CallSiteData { | 
|  | public: | 
|  | bool can_patch_to_monomorphic() const { | 
|  | return untag()->can_patch_to_monomorphic_; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedUnlinkedCall)); | 
|  | } | 
|  |  | 
|  | uword Hash() const; | 
|  | bool Equals(const UnlinkedCall& other) const; | 
|  |  | 
|  | static UnlinkedCallPtr New(); | 
|  |  | 
|  | private: | 
|  | friend class ICData;  // For set_*() methods. | 
|  |  | 
|  | void set_can_patch_to_monomorphic(bool value) const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(UnlinkedCall, CallSiteData); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Object holding information about an IC: test classes and their | 
|  | // corresponding targets. The owner of the ICData can be either the function | 
|  | // or the original ICData object. In case of background compilation we | 
|  | // copy the ICData in a child object, thus freezing it during background | 
|  | // compilation. Code may contain only original ICData objects. | 
|  | // | 
|  | // ICData's backing store is an array that logically contains several valid | 
|  | // entries followed by a sentinel entry. | 
|  | // | 
|  | //   [<entry-0>, <...>, <entry-N>, <sentinel>] | 
|  | // | 
|  | // Each entry has the following form: | 
|  | // | 
|  | //   [arg0?, arg1?, argN?, count, target-function/code, exactness?] | 
|  | // | 
|  | // The <entry-X> need to contain valid type feedback. | 
|  | // The <sentinel> entry and must have kIllegalCid value for all | 
|  | // members of the entry except for the last one (`exactness` if | 
|  | // present, otherwise `target-function/code`) - which we use as a backref: | 
|  | // | 
|  | //   * For empty ICData we use a cached/shared backing store. So there is no | 
|  | //     unique backref, we use kIllegalCid instead. | 
|  | //   * For non-empty ICData the backref in the backing store array will point to | 
|  | //     the ICData object. | 
|  | // | 
|  | // Updating the ICData happens under a lock to avoid phantom-reads. The backing | 
|  | // is treated as an immutable Copy-on-Write data structure: Adding to the ICData | 
|  | // makes a copy with length+1 which will be store-release'd so any reader can | 
|  | // see it (and doesn't need to hold a lock). | 
|  | class ICData : public CallSiteData { | 
|  | public: | 
|  | FunctionPtr Owner() const; | 
|  |  | 
|  | ICDataPtr Original() const; | 
|  |  | 
|  | void SetOriginal(const ICData& value) const; | 
|  |  | 
|  | bool IsOriginal() const { return Original() == this->ptr(); } | 
|  |  | 
|  | intptr_t NumArgsTested() const; | 
|  |  | 
|  | intptr_t deopt_id() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | return -1; | 
|  | #else | 
|  | return untag()->deopt_id_; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | bool IsImmutable() const; | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | AbstractTypePtr receivers_static_type() const { | 
|  | return untag()->receivers_static_type(); | 
|  | } | 
|  | bool is_tracking_exactness() const { | 
|  | return untag()->state_bits_.Read<TrackingExactnessBit>(); | 
|  | } | 
|  | #else | 
|  | bool is_tracking_exactness() const { return false; } | 
|  | #endif | 
|  |  | 
|  | // Note: only deopts with reasons before Unknown in this list are recorded in | 
|  | // the ICData. All other reasons are used purely for informational messages | 
|  | // printed during deoptimization itself. | 
|  | #define DEOPT_REASONS(V)                                                       \ | 
|  | V(BinarySmiOp)                                                               \ | 
|  | V(BinaryInt64Op)                                                             \ | 
|  | V(DoubleToSmi)                                                               \ | 
|  | V(CheckSmi)                                                                  \ | 
|  | V(CheckClass)                                                                \ | 
|  | V(Unknown)                                                                   \ | 
|  | V(PolymorphicInstanceCallTestFail)                                           \ | 
|  | V(UnaryInt64Op)                                                              \ | 
|  | V(BinaryDoubleOp)                                                            \ | 
|  | V(UnaryOp)                                                                   \ | 
|  | V(UnboxInteger)                                                              \ | 
|  | V(Unbox)                                                                     \ | 
|  | V(CheckArrayBound)                                                           \ | 
|  | V(AtCall)                                                                    \ | 
|  | V(GuardField)                                                                \ | 
|  | V(TestCids)                                                                  \ | 
|  | V(NumReasons) | 
|  |  | 
|  | enum DeoptReasonId { | 
|  | #define DEFINE_ENUM_LIST(name) kDeopt##name, | 
|  | DEOPT_REASONS(DEFINE_ENUM_LIST) | 
|  | #undef DEFINE_ENUM_LIST | 
|  | }; | 
|  |  | 
|  | static constexpr intptr_t kLastRecordedDeoptReason = kDeoptUnknown - 1; | 
|  |  | 
|  | enum DeoptFlags { | 
|  | // Deoptimization is caused by an optimistically hoisted instruction. | 
|  | kHoisted = 1 << 0, | 
|  |  | 
|  | // Deoptimization is caused by an optimistically generalized bounds check. | 
|  | kGeneralized = 1 << 1 | 
|  | }; | 
|  |  | 
|  | bool HasDeoptReasons() const { return DeoptReasons() != 0; } | 
|  | uint32_t DeoptReasons() const; | 
|  | void SetDeoptReasons(uint32_t reasons) const; | 
|  |  | 
|  | bool HasDeoptReason(ICData::DeoptReasonId reason) const; | 
|  | void AddDeoptReason(ICData::DeoptReasonId reason) const; | 
|  |  | 
|  | // Call site classification that is helpful for hot-reload. Call sites with | 
|  | // different `RebindRule` have to be rebound differently. | 
|  | #define FOR_EACH_REBIND_RULE(V)                                                \ | 
|  | V(Instance)                                                                  \ | 
|  | V(NoRebind)                                                                  \ | 
|  | V(NSMDispatch)                                                               \ | 
|  | V(Optimized)                                                                 \ | 
|  | V(Static)                                                                    \ | 
|  | V(Super) | 
|  |  | 
|  | enum RebindRule { | 
|  | #define REBIND_ENUM_DEF(name) k##name, | 
|  | FOR_EACH_REBIND_RULE(REBIND_ENUM_DEF) | 
|  | #undef REBIND_ENUM_DEF | 
|  | kNumRebindRules, | 
|  | }; | 
|  | static const char* RebindRuleToCString(RebindRule r); | 
|  | static bool ParseRebindRule(const char* str, RebindRule* out); | 
|  | RebindRule rebind_rule() const; | 
|  |  | 
|  | void set_is_megamorphic(bool value) const { | 
|  | untag()->state_bits_.UpdateBool<MegamorphicBit, std::memory_order_release>( | 
|  | value); | 
|  | } | 
|  |  | 
|  | // The length of the array. This includes all sentinel entries including | 
|  | // the final one. | 
|  | intptr_t Length() const; | 
|  |  | 
|  | intptr_t NumberOfChecks() const; | 
|  |  | 
|  | // Discounts any checks with usage of zero. | 
|  | // Takes O(result)) time! | 
|  | intptr_t NumberOfUsedChecks() const; | 
|  |  | 
|  | bool NumberOfChecksIs(intptr_t n) const; | 
|  |  | 
|  | bool IsValidEntryIndex(intptr_t index) const { | 
|  | return 0 <= index && index < NumberOfChecks(); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedICData)); | 
|  | } | 
|  |  | 
|  | static intptr_t state_bits_offset() { | 
|  | return OFFSET_OF(UntaggedICData, state_bits_); | 
|  | } | 
|  |  | 
|  | static intptr_t NumArgsTestedShift() { return NumArgsTestedBits::shift(); } | 
|  |  | 
|  | static intptr_t NumArgsTestedMask() { | 
|  | return NumArgsTestedBits::mask_in_place(); | 
|  | } | 
|  |  | 
|  | static intptr_t entries_offset() { | 
|  | return OFFSET_OF(UntaggedICData, entries_); | 
|  | } | 
|  |  | 
|  | static intptr_t owner_offset() { return OFFSET_OF(UntaggedICData, owner_); } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | static intptr_t receivers_static_type_offset() { | 
|  | return OFFSET_OF(UntaggedICData, receivers_static_type_); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | // NOTE: Can only be called during reload. | 
|  | void Clear(const CallSiteResetter& proof_of_reload) const { | 
|  | TruncateTo(0, proof_of_reload); | 
|  | } | 
|  |  | 
|  | // NOTE: Can only be called during reload. | 
|  | void TruncateTo(intptr_t num_checks, | 
|  | const CallSiteResetter& proof_of_reload) const; | 
|  |  | 
|  | // Clears the count for entry |index|. | 
|  | // NOTE: Can only be called during reload. | 
|  | void ClearCountAt(intptr_t index, | 
|  | const CallSiteResetter& proof_of_reload) const; | 
|  |  | 
|  | // Clear all entries with the sentinel value and reset the first entry | 
|  | // with the dummy target entry. | 
|  | // NOTE: Can only be called during reload. | 
|  | void ClearAndSetStaticTarget(const Function& func, | 
|  | const CallSiteResetter& proof_of_reload) const; | 
|  |  | 
|  | void DebugDump() const; | 
|  |  | 
|  | // Adding checks. | 
|  |  | 
|  | // Ensures there is a check for [class_ids]. | 
|  | // | 
|  | // Calls [AddCheck] iff there is no existing check. Ensures test (and | 
|  | // potential update) will be performed under exclusive lock to guard against | 
|  | // multiple threads trying to add the same check. | 
|  | void EnsureHasCheck(const GrowableArray<intptr_t>& class_ids, | 
|  | const Function& target, | 
|  | intptr_t count = 1) const; | 
|  |  | 
|  | // Adds one more class test to ICData. Length of 'classes' must be equal to | 
|  | // the number of arguments tested. Use only for num_args_tested > 1. | 
|  | void AddCheck(const GrowableArray<intptr_t>& class_ids, | 
|  | const Function& target, | 
|  | intptr_t count = 1) const; | 
|  |  | 
|  | StaticTypeExactnessState GetExactnessAt(intptr_t count) const; | 
|  |  | 
|  | // Ensures there is a receiver check for [receiver_class_id]. | 
|  | // | 
|  | // Calls [AddCheckReceiverCheck] iff there is no existing check. Ensures | 
|  | // test (and potential update) will be performed under exclusive lock to | 
|  | // guard against multiple threads trying to add the same check. | 
|  | void EnsureHasReceiverCheck( | 
|  | intptr_t receiver_class_id, | 
|  | const Function& target, | 
|  | intptr_t count = 1, | 
|  | StaticTypeExactnessState exactness = | 
|  | StaticTypeExactnessState::NotTracking()) const; | 
|  |  | 
|  | // Adds sorted so that Smi is the first class-id. Use only for | 
|  | // num_args_tested == 1. | 
|  | void AddReceiverCheck(intptr_t receiver_class_id, | 
|  | const Function& target, | 
|  | intptr_t count = 1, | 
|  | StaticTypeExactnessState exactness = | 
|  | StaticTypeExactnessState::NotTracking()) const; | 
|  |  | 
|  | // Retrieving checks. | 
|  |  | 
|  | void GetCheckAt(intptr_t index, | 
|  | GrowableArray<intptr_t>* class_ids, | 
|  | Function* target) const; | 
|  | void GetClassIdsAt(intptr_t index, GrowableArray<intptr_t>* class_ids) const; | 
|  |  | 
|  | // Only for 'num_args_checked == 1'. | 
|  | void GetOneClassCheckAt(intptr_t index, | 
|  | intptr_t* class_id, | 
|  | Function* target) const; | 
|  | // Only for 'num_args_checked == 1'. | 
|  | intptr_t GetCidAt(intptr_t index) const; | 
|  |  | 
|  | intptr_t GetReceiverClassIdAt(intptr_t index) const; | 
|  | intptr_t GetClassIdAt(intptr_t index, intptr_t arg_nr) const; | 
|  |  | 
|  | FunctionPtr GetTargetAt(intptr_t index) const; | 
|  |  | 
|  | void IncrementCountAt(intptr_t index, intptr_t value) const; | 
|  | void SetCountAt(intptr_t index, intptr_t value) const; | 
|  | intptr_t GetCountAt(intptr_t index) const; | 
|  | intptr_t AggregateCount() const; | 
|  |  | 
|  | // Returns this->untag() if num_args_tested == 1 and arg_nr == 1, otherwise | 
|  | // returns a new ICData object containing only unique arg_nr checks. | 
|  | // Returns only used entries. | 
|  | ICDataPtr AsUnaryClassChecksForArgNr(intptr_t arg_nr) const; | 
|  | ICDataPtr AsUnaryClassChecks() const { return AsUnaryClassChecksForArgNr(0); } | 
|  |  | 
|  | // Returns ICData with aggregated receiver count, sorted by highest count. | 
|  | // Smi not first!! (the convention for ICData used in code generation is that | 
|  | // Smi check is first) | 
|  | // Used for printing and optimizations. | 
|  | ICDataPtr AsUnaryClassChecksSortedByCount() const; | 
|  |  | 
|  | UnlinkedCallPtr AsUnlinkedCall() const; | 
|  |  | 
|  | bool HasReceiverClassId(intptr_t class_id) const; | 
|  |  | 
|  | // Note: passing non-null receiver_type enables exactness tracking for | 
|  | // the receiver type. Receiver type is expected to be a fully | 
|  | // instantiated generic (but not a FutureOr). | 
|  | // See StaticTypeExactnessState for more information. | 
|  | static ICDataPtr New( | 
|  | const Function& owner, | 
|  | const String& target_name, | 
|  | const Array& arguments_descriptor, | 
|  | intptr_t deopt_id, | 
|  | intptr_t num_args_tested, | 
|  | RebindRule rebind_rule, | 
|  | const AbstractType& receiver_type = Object::null_abstract_type()); | 
|  |  | 
|  | // Similar to [New] makes the ICData have an initial (cids, target) entry. | 
|  | static ICDataPtr NewWithCheck( | 
|  | const Function& owner, | 
|  | const String& target_name, | 
|  | const Array& arguments_descriptor, | 
|  | intptr_t deopt_id, | 
|  | intptr_t num_args_tested, | 
|  | RebindRule rebind_rule, | 
|  | GrowableArray<intptr_t>* cids, | 
|  | const Function& target, | 
|  | const AbstractType& receiver_type = Object::null_abstract_type()); | 
|  |  | 
|  | static ICDataPtr NewForStaticCall(const Function& owner, | 
|  | const Function& target, | 
|  | const Array& arguments_descriptor, | 
|  | intptr_t deopt_id, | 
|  | intptr_t num_args_tested, | 
|  | RebindRule rebind_rule); | 
|  |  | 
|  | static ICDataPtr NewFrom(const ICData& from, intptr_t num_args_tested); | 
|  |  | 
|  | // Generates a new ICData with descriptor and data array copied (deep clone). | 
|  | static ICDataPtr Clone(const ICData& from); | 
|  |  | 
|  | // Gets the [ICData] from the [ICData::entries_] array (which stores a back | 
|  | // ref). | 
|  | // | 
|  | // May return `null` if the [ICData] is empty. | 
|  | static ICDataPtr ICDataOfEntriesArray(const Array& array); | 
|  |  | 
|  | static intptr_t TestEntryLengthFor(intptr_t num_args, | 
|  | bool tracking_exactness); | 
|  |  | 
|  | static intptr_t CountIndexFor(intptr_t num_args) { return num_args; } | 
|  | static intptr_t EntryPointIndexFor(intptr_t num_args) { return num_args; } | 
|  |  | 
|  | static intptr_t TargetIndexFor(intptr_t num_args) { return num_args + 1; } | 
|  | static intptr_t CodeIndexFor(intptr_t num_args) { return num_args + 1; } | 
|  |  | 
|  | static intptr_t ExactnessIndexFor(intptr_t num_args) { return num_args + 2; } | 
|  |  | 
|  | bool IsUsedAt(intptr_t i) const; | 
|  |  | 
|  | void PrintToJSONArray(const JSONArray& jsarray, | 
|  | TokenPosition token_pos) const; | 
|  |  | 
|  | // Initialize the preallocated empty ICData entry arrays. | 
|  | static void Init(); | 
|  |  | 
|  | // Clear the preallocated empty ICData entry arrays. | 
|  | static void Cleanup(); | 
|  |  | 
|  | // We cache ICData with 0, 1, 2 arguments tested without exactness | 
|  | // tracking and with 1 argument tested with exactness tracking. | 
|  | enum { | 
|  | kCachedICDataZeroArgTestedWithoutExactnessTrackingIdx = 0, | 
|  | kCachedICDataMaxArgsTestedWithoutExactnessTracking = 2, | 
|  | kCachedICDataOneArgWithExactnessTrackingIdx = | 
|  | kCachedICDataZeroArgTestedWithoutExactnessTrackingIdx + | 
|  | kCachedICDataMaxArgsTestedWithoutExactnessTracking + 1, | 
|  | kCachedICDataArrayCount = kCachedICDataOneArgWithExactnessTrackingIdx + 1, | 
|  | }; | 
|  |  | 
|  | bool is_static_call() const; | 
|  |  | 
|  | intptr_t FindCheck(const GrowableArray<intptr_t>& cids) const; | 
|  |  | 
|  | ArrayPtr entries() const { | 
|  | return untag()->entries<std::memory_order_acquire>(); | 
|  | } | 
|  |  | 
|  | bool receiver_cannot_be_smi() const { | 
|  | return untag()->state_bits_.Read<ReceiverCannotBeSmiBit>(); | 
|  | } | 
|  |  | 
|  | void set_receiver_cannot_be_smi(bool value) const { | 
|  | untag()->state_bits_.UpdateBool<ReceiverCannotBeSmiBit>(value); | 
|  | } | 
|  |  | 
|  | uword Hash() const; | 
|  |  | 
|  | private: | 
|  | static ICDataPtr New(); | 
|  |  | 
|  | // Grows the array and also sets the argument to the index that should be used | 
|  | // for the new entry. | 
|  | ArrayPtr Grow(intptr_t* index) const; | 
|  |  | 
|  | void set_deopt_id(intptr_t value) const; | 
|  | void set_entries(const Array& value) const; | 
|  | void set_owner(const Function& value) const; | 
|  | void set_rebind_rule(uint32_t rebind_rule) const; | 
|  | void clear_state_bits() const; | 
|  | void set_tracking_exactness(bool value) const { | 
|  | untag()->state_bits_.UpdateBool<TrackingExactnessBit>(value); | 
|  | } | 
|  |  | 
|  | // Does entry |index| contain the sentinel value? | 
|  | void SetNumArgsTested(intptr_t value) const; | 
|  | void SetReceiversStaticType(const AbstractType& type) const; | 
|  | DEBUG_ONLY(void AssertInvariantsAreSatisfied() const;) | 
|  |  | 
|  | static void SetTargetAtPos(const Array& data, | 
|  | intptr_t data_pos, | 
|  | intptr_t num_args_tested, | 
|  | const Function& target); | 
|  | void AddCheckInternal(const GrowableArray<intptr_t>& class_ids, | 
|  | const Function& target, | 
|  | intptr_t count) const; | 
|  | void AddReceiverCheckInternal(intptr_t receiver_class_id, | 
|  | const Function& target, | 
|  | intptr_t count, | 
|  | StaticTypeExactnessState exactness) const; | 
|  |  | 
|  | // This bit is set when a call site becomes megamorphic and starts using a | 
|  | // MegamorphicCache instead of ICData. It means that the entries in the | 
|  | // ICData are incomplete and the MegamorphicCache needs to also be consulted | 
|  | // to list the call site's observed receiver classes and targets. | 
|  | // In the compiler, this should only be read once by CallTargets to avoid the | 
|  | // compiler seeing an unstable set of feedback. | 
|  | bool is_megamorphic() const { | 
|  | // Ensure any following load instructions do not get performed before this | 
|  | // one. | 
|  | return untag() | 
|  | ->state_bits_.Read<MegamorphicBit, std::memory_order_acquire>(); | 
|  | } | 
|  |  | 
|  | bool ValidateInterceptor(const Function& target) const; | 
|  |  | 
|  | using NumArgsTestedBits = | 
|  | BitField<decltype(UntaggedICData::state_bits_), uint32_t, 0, 2>; | 
|  | using TrackingExactnessBit = BitField<decltype(UntaggedICData::state_bits_), | 
|  | bool, | 
|  | NumArgsTestedBits::kNextBit>; | 
|  | using DeoptReasonBits = BitField<decltype(UntaggedICData::state_bits_), | 
|  | uint32_t, | 
|  | TrackingExactnessBit::kNextBit, | 
|  | kLastRecordedDeoptReason + 1>; | 
|  | using RebindRuleBits = BitField<decltype(UntaggedICData::state_bits_), | 
|  | uint32_t, | 
|  | DeoptReasonBits::kNextBit, | 
|  | Utils::BitLength(kNumRebindRules - 1)>; | 
|  | using MegamorphicBit = BitField<decltype(UntaggedICData::state_bits_), | 
|  | bool, | 
|  | RebindRuleBits::kNextBit>; | 
|  | using ReceiverCannotBeSmiBit = BitField<decltype(UntaggedICData::state_bits_), | 
|  | bool, | 
|  | MegamorphicBit::kNextBit>; | 
|  |  | 
|  | #if defined(DEBUG) | 
|  | // Used in asserts to verify that a check is not added twice. | 
|  | bool HasCheck(const GrowableArray<intptr_t>& cids) const; | 
|  | #endif  // DEBUG | 
|  |  | 
|  | intptr_t TestEntryLength() const; | 
|  | static ArrayPtr NewNonCachedEmptyICDataArray(intptr_t num_args_tested, | 
|  | bool tracking_exactness); | 
|  | static ArrayPtr CachedEmptyICDataArray(intptr_t num_args_tested, | 
|  | bool tracking_exactness); | 
|  | static bool IsCachedEmptyEntry(const Array& array); | 
|  | static ICDataPtr NewDescriptor(Zone* zone, | 
|  | const Function& owner, | 
|  | const String& target_name, | 
|  | const Array& arguments_descriptor, | 
|  | intptr_t deopt_id, | 
|  | intptr_t num_args_tested, | 
|  | RebindRule rebind_rule, | 
|  | const AbstractType& receiver_type); | 
|  |  | 
|  | static void WriteSentinel(const Array& data, | 
|  | intptr_t test_entry_length, | 
|  | const Object& back_ref); | 
|  |  | 
|  | // A cache of VM heap allocated preinitialized empty ic data entry arrays. | 
|  | static ArrayPtr cached_icdata_arrays_[kCachedICDataArrayCount]; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(ICData, CallSiteData); | 
|  | friend class CallSiteResetter; | 
|  | friend class CallTargets; | 
|  | friend class Class; | 
|  | friend class VMDeserializationRoots; | 
|  | friend class ICDataTestTask; | 
|  | friend class Interpreter; | 
|  | friend class VMSerializationRoots; | 
|  | }; | 
|  |  | 
|  | // Often used constants for number of free function type parameters. | 
|  | enum { | 
|  | kNoneFree = 0, | 
|  |  | 
|  | // 'kCurrentAndEnclosingFree' is used when partially applying a signature | 
|  | // function to a set of type arguments. It indicates that the set of type | 
|  | // parameters declared by the current function and enclosing functions should | 
|  | // be considered free, and the current function type parameters should be | 
|  | // substituted as well. | 
|  | // | 
|  | // For instance, if the signature "<T>(T, R) => T" is instantiated with | 
|  | // function type arguments [int, String] and kCurrentAndEnclosingFree is | 
|  | // supplied, the result of the instantiation will be "(String, int) => int". | 
|  | kCurrentAndEnclosingFree = kMaxInt32 - 1, | 
|  |  | 
|  | // Only parameters declared by enclosing functions are free. | 
|  | kAllFree = kMaxInt32, | 
|  | }; | 
|  |  | 
|  | // Formatting configuration for Function::PrintName. | 
|  | struct NameFormattingParams { | 
|  | Object::NameVisibility name_visibility; | 
|  | bool disambiguate_names; | 
|  |  | 
|  | // By default function name includes the name of the enclosing class if any. | 
|  | // However in some contexts this information is redundant and class name | 
|  | // is already known. In this case setting |include_class_name| to false | 
|  | // allows you to exclude this information from the formatted name. | 
|  | bool include_class_name = true; | 
|  |  | 
|  | // By default function name includes the name of the enclosing function if | 
|  | // any. However in some contexts this information is redundant and | 
|  | // the name of the enclosing function is already known. In this case | 
|  | // setting |include_parent_name| to false allows to exclude this information | 
|  | // from the formatted name. | 
|  | bool include_parent_name = true; | 
|  |  | 
|  | NameFormattingParams(Object::NameVisibility visibility, | 
|  | Object::NameDisambiguation name_disambiguation = | 
|  | Object::NameDisambiguation::kNo) | 
|  | : name_visibility(visibility), | 
|  | disambiguate_names(name_disambiguation == | 
|  | Object::NameDisambiguation::kYes) {} | 
|  |  | 
|  | static NameFormattingParams DisambiguatedWithoutClassName( | 
|  | Object::NameVisibility visibility) { | 
|  | NameFormattingParams params(visibility, Object::NameDisambiguation::kYes); | 
|  | params.include_class_name = false; | 
|  | return params; | 
|  | } | 
|  |  | 
|  | static NameFormattingParams DisambiguatedUnqualified( | 
|  | Object::NameVisibility visibility) { | 
|  | NameFormattingParams params(visibility, Object::NameDisambiguation::kYes); | 
|  | params.include_class_name = false; | 
|  | params.include_parent_name = false; | 
|  | return params; | 
|  | } | 
|  | }; | 
|  |  | 
|  | enum class FfiCallbackKind : uint8_t { | 
|  | kIsolateLocalStaticCallback, | 
|  | kIsolateGroupSharedStaticCallback, | 
|  | kIsolateLocalClosureCallback, | 
|  | kIsolateGroupSharedClosureCallback, | 
|  | kAsyncCallback, | 
|  | }; | 
|  |  | 
|  | enum class EntryPointPragma { | 
|  | kAlways, | 
|  | kNever, | 
|  | kGetterOnly, | 
|  | kSetterOnly, | 
|  | kCallOnly | 
|  | }; | 
|  |  | 
|  | struct EntryPointPragmaUtils : public AllStatic { | 
|  | static constexpr bool AllowsCall(EntryPointPragma pragma) { | 
|  | return pragma == EntryPointPragma::kAlways || | 
|  | pragma == EntryPointPragma::kCallOnly; | 
|  | } | 
|  |  | 
|  | static constexpr bool AllowsGet(EntryPointPragma pragma) { | 
|  | return pragma == EntryPointPragma::kAlways || | 
|  | pragma == EntryPointPragma::kGetterOnly; | 
|  | } | 
|  |  | 
|  | static constexpr bool AllowsSet(EntryPointPragma pragma) { | 
|  | return pragma == EntryPointPragma::kAlways || | 
|  | pragma == EntryPointPragma::kSetterOnly; | 
|  | } | 
|  |  | 
|  | static constexpr bool AllowsAccess(EntryPointPragma pragma) { | 
|  | // The CFE should ensure that non-kAlways annotations are appropriate | 
|  | // for the given member. | 
|  | return pragma != EntryPointPragma::kNever; | 
|  | } | 
|  | }; | 
|  |  | 
|  | class Function : public Object { | 
|  | public: | 
|  | StringPtr name() const { return untag()->name(); } | 
|  | StringPtr UserVisibleName() const;  // Same as scrubbed name. | 
|  | const char* UserVisibleNameCString() const; | 
|  |  | 
|  | const char* NameCString(NameVisibility name_visibility) const; | 
|  |  | 
|  | void PrintName(const NameFormattingParams& params, | 
|  | BaseTextBuffer* printer) const; | 
|  | StringPtr QualifiedScrubbedName() const; | 
|  | const char* QualifiedScrubbedNameCString() const; | 
|  | StringPtr QualifiedUserVisibleName() const; | 
|  | const char* QualifiedUserVisibleNameCString() const; | 
|  |  | 
|  | virtual StringPtr DictionaryName() const { return name(); } | 
|  |  | 
|  | StringPtr GetSource() const; | 
|  |  | 
|  | // Set the "C signature" for an FFI trampoline. | 
|  | // Can only be used on FFI trampolines. | 
|  | void SetFfiCSignature(const FunctionType& sig) const; | 
|  |  | 
|  | // Retrieves the "C signature" for an FFI trampoline or FFI native. | 
|  | FunctionTypePtr FfiCSignature() const; | 
|  |  | 
|  | bool FfiCSignatureContainsHandles() const; | 
|  | bool FfiCSignatureReturnsStruct() const; | 
|  | bool FfiCSignatureReturnsHandle() const; | 
|  |  | 
|  | // Can only be called on FFI trampolines. | 
|  | int32_t FfiCallbackId() const; | 
|  |  | 
|  | // Should be called when ffi trampoline function object is created. | 
|  | void AssignFfiCallbackId(int32_t callback_id) const; | 
|  |  | 
|  | // Can only be called on FFI natives and FFI call closures. | 
|  | bool FfiIsLeaf() const; | 
|  |  | 
|  | // Can only be called on FFI trampolines. | 
|  | FunctionPtr FfiCallbackTarget() const; | 
|  |  | 
|  | // Can only be called on FFI trampolines. | 
|  | void SetFfiCallbackTarget(const Function& target) const; | 
|  |  | 
|  | // Can only be called on FFI trampolines. | 
|  | InstancePtr FfiCallbackExceptionalReturn() const; | 
|  |  | 
|  | // Can only be called on FFI trampolines. | 
|  | void SetFfiCallbackExceptionalReturn(const Instance& value) const; | 
|  |  | 
|  | // Can only be called on FFI trampolines. | 
|  | FfiCallbackKind GetFfiCallbackKind() const; | 
|  |  | 
|  | // Can only be called on FFI trampolines. | 
|  | void SetFfiCallbackKind(FfiCallbackKind value) const; | 
|  |  | 
|  | // Return the signature of this function. | 
|  | PRECOMPILER_WSR_FIELD_DECLARATION(FunctionType, signature); | 
|  | void SetSignature(const FunctionType& value) const; | 
|  | static intptr_t signature_offset() { | 
|  | return OFFSET_OF(UntaggedFunction, signature_); | 
|  | } | 
|  |  | 
|  | // Build a string of the form '<T>(T, {B b, C c}) => R' representing the | 
|  | // internal signature of the given function. In this example, T is a type | 
|  | // parameter of this function and R is a type parameter of class C, the owner | 
|  | // of the function. B and C are not type parameters. | 
|  | StringPtr InternalSignature() const; | 
|  |  | 
|  | // Build a string of the form '<T>(T, {B b, C c}) => R' representing the | 
|  | // user visible signature of the given function. In this example, T is a type | 
|  | // parameter of this function and R is a type parameter of class C, the owner | 
|  | // of the function. B and C are not type parameters. | 
|  | // Implicit parameters are hidden. | 
|  | StringPtr UserVisibleSignature() const; | 
|  |  | 
|  | // Returns true if the signature of this function is instantiated, i.e. if it | 
|  | // does not involve generic parameter types or generic result type. | 
|  | // Note that function type parameters declared by this function do not make | 
|  | // its signature uninstantiated, only type parameters declared by parent | 
|  | // generic functions or class type parameters. | 
|  | bool HasInstantiatedSignature( | 
|  | Genericity genericity = kAny, | 
|  | intptr_t num_free_fun_type_params = kAllFree) const; | 
|  |  | 
|  | bool IsPrivate() const; | 
|  |  | 
|  | ClassPtr Owner() const { return Owner(ptr()); } | 
|  | static ClassPtr Owner(FunctionPtr function); | 
|  | void set_owner(const Object& value) const; | 
|  | ScriptPtr script() const; | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | KernelProgramInfoPtr KernelProgramInfo() const; | 
|  | #endif | 
|  | ObjectPtr RawOwner() const { return untag()->owner(); } | 
|  |  | 
|  | RegExpPtr regexp() const; | 
|  | intptr_t string_specialization_cid() const; | 
|  | bool is_sticky_specialization() const; | 
|  | void SetRegExpData(const RegExp& regexp, | 
|  | intptr_t string_specialization_cid, | 
|  | bool sticky) const; | 
|  |  | 
|  | StringPtr native_name() const; | 
|  | void set_native_name(const String& name) const; | 
|  |  | 
|  | InstancePtr GetNativeAnnotation() const; | 
|  | bool is_ffi_native() const; | 
|  | bool is_old_native() const; | 
|  |  | 
|  | AbstractTypePtr result_type() const { | 
|  | return signature()->untag()->result_type(); | 
|  | } | 
|  |  | 
|  | // The parameters, starting with NumImplicitParameters() parameters which are | 
|  | // only visible to the VM, but not to Dart users. | 
|  | // Note that type checks exclude implicit parameters. | 
|  | AbstractTypePtr ParameterTypeAt(intptr_t index) const; | 
|  | ArrayPtr parameter_types() const { | 
|  | return signature()->untag()->parameter_types(); | 
|  | } | 
|  |  | 
|  | // Outside of the AOT runtime, functions store the names for their positional | 
|  | // parameters, and delegate storage of the names for named parameters to | 
|  | // their signature. These methods handle fetching the name from and | 
|  | // setting the name to the correct location. | 
|  | StringPtr ParameterNameAt(intptr_t index) const; | 
|  | // Only valid for positional parameter indexes, as this should be called | 
|  | // explicitly on the signature for named parameters. | 
|  | void SetParameterNameAt(intptr_t index, const String& value) const; | 
|  | // Creates an appropriately sized array in the function to hold positional | 
|  | // parameter names, using the positional parameter count in the signature. | 
|  | // Uses same default space as Function::New. | 
|  | void CreateNameArray(Heap::Space space = Heap::kOld) const; | 
|  |  | 
|  | // Delegates to the signature, which stores the named parameter flags. | 
|  | bool IsRequiredAt(intptr_t index) const; | 
|  |  | 
|  | // The formal type parameters, their bounds, and defaults, are specified as an | 
|  | // object of type TypeParameters stored in the signature. | 
|  | TypeParametersPtr type_parameters() const { | 
|  | return signature()->untag()->type_parameters(); | 
|  | } | 
|  |  | 
|  | // Returns the number of local type arguments for this function. | 
|  | intptr_t NumTypeParameters() const; | 
|  | // Return the cumulative number of type arguments in all parent functions. | 
|  | intptr_t NumParentTypeArguments() const; | 
|  | // Return the cumulative number of type arguments for this function, including | 
|  | // type arguments for all parent functions. | 
|  | intptr_t NumTypeArguments() const; | 
|  | // Return whether this function declares local type arguments. | 
|  | bool IsGeneric() const; | 
|  | // Returns whether any parent function of this function is generic. | 
|  | bool HasGenericParent() const { return NumParentTypeArguments() > 0; } | 
|  |  | 
|  | // Return the type parameter declared at index. | 
|  | TypeParameterPtr TypeParameterAt( | 
|  | intptr_t index, | 
|  | Nullability nullability = Nullability::kNonNullable) const; | 
|  |  | 
|  | // Not thread-safe; must be called in the main thread. | 
|  | // Sets function's code and code's function. | 
|  | void InstallOptimizedCode(const Code& code) const; | 
|  | void AttachCode(const Code& value) const; | 
|  | void SetInstructions(const Code& value) const; | 
|  | void SetInstructionsSafe(const Code& value) const; | 
|  | void ClearCode() const; | 
|  | void ClearCodeSafe() const; | 
|  |  | 
|  | // Disables optimized code and switches to unoptimized code. | 
|  | void SwitchToUnoptimizedCode() const; | 
|  |  | 
|  | // Ensures that the function has code. If there is no code it compiles the | 
|  | // unoptimized version of the code.  If the code contains errors, it calls | 
|  | // Exceptions::PropagateError and does not return.  Normally returns the | 
|  | // current code, whether it is optimized or unoptimized. | 
|  | CodePtr EnsureHasCode() const; | 
|  |  | 
|  | // Ensures that the function has code. If there is no code, this method | 
|  | // compiles the unoptimized version of the code. If an error occurs during | 
|  | // compilation, the error is returned. Normally returns the function's code, | 
|  | // whether optimized or unoptimized. | 
|  | ObjectPtr EnsureHasCodeNoThrow() const; | 
|  |  | 
|  | // Disables optimized code and switches to unoptimized code (or the lazy | 
|  | // compilation stub). | 
|  | void SwitchToLazyCompiledUnoptimizedCode() const; | 
|  |  | 
|  | // Compiles unoptimized code (if necessary) and attaches it to the function. | 
|  | // If an error occurs during compilation, |Exceptions::PropagateError| will be | 
|  | // called to propagate it. | 
|  | void EnsureHasCompiledUnoptimizedCode() const; | 
|  |  | 
|  | // Compiles unoptimized code (if necessary) and attaches it to the function. | 
|  | // If an error occurs during compilation, the error is returned. Otherwise, | 
|  | // |Error::null()| is returned. | 
|  | ErrorPtr EnsureHasCompiledUnoptimizedCodeNoThrow() const; | 
|  |  | 
|  | // Return the most recently compiled and installed code for this function. | 
|  | // It is not the only Code object that points to this function. | 
|  | CodePtr CurrentCode() const { return CurrentCodeOf(ptr()); } | 
|  |  | 
|  | bool SafeToClosurize() const; | 
|  |  | 
|  | static CodePtr CurrentCodeOf(const FunctionPtr function) { | 
|  | return function->untag()->code<std::memory_order_acquire>(); | 
|  | } | 
|  |  | 
|  | CodePtr unoptimized_code() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return static_cast<CodePtr>(Object::null()); | 
|  | #else | 
|  | return untag()->unoptimized_code(); | 
|  | #endif | 
|  | } | 
|  | void set_unoptimized_code(const Code& value) const; | 
|  | bool HasCode() const; | 
|  | static bool HasCode(FunctionPtr function); | 
|  |  | 
|  | static intptr_t code_offset() { return OFFSET_OF(UntaggedFunction, code_); } | 
|  |  | 
|  | uword entry_point() const { return EntryPointOf(ptr()); } | 
|  | static uword EntryPointOf(const FunctionPtr function) { | 
|  | return function->untag()->entry_point_; | 
|  | } | 
|  |  | 
|  | static intptr_t entry_point_offset( | 
|  | CodeEntryKind entry_kind = CodeEntryKind::kNormal) { | 
|  | switch (entry_kind) { | 
|  | case CodeEntryKind::kNormal: | 
|  | return OFFSET_OF(UntaggedFunction, entry_point_); | 
|  | case CodeEntryKind::kUnchecked: | 
|  | return OFFSET_OF(UntaggedFunction, unchecked_entry_point_); | 
|  | default: | 
|  | UNREACHABLE(); | 
|  | } | 
|  | } | 
|  |  | 
|  | static intptr_t unchecked_entry_point_offset() { | 
|  | return OFFSET_OF(UntaggedFunction, unchecked_entry_point_); | 
|  | } | 
|  |  | 
|  | #if defined(DART_DYNAMIC_MODULES) | 
|  | void AttachBytecode(const Bytecode& bytecode) const; | 
|  | void ClearBytecode() const; | 
|  | inline BytecodePtr GetBytecode() const; | 
|  | static inline BytecodePtr GetBytecode(FunctionPtr function); | 
|  | inline bool HasBytecode() const; | 
|  | static inline bool HasBytecode(FunctionPtr function); | 
|  | #else | 
|  | inline bool HasBytecode() const { return false; } | 
|  | #endif | 
|  |  | 
|  | virtual uword Hash() const; | 
|  |  | 
|  | // Returns true if there is at least one debugger breakpoint | 
|  | // set in this function. | 
|  | bool HasBreakpoint() const; | 
|  |  | 
|  | ContextScopePtr context_scope() const; | 
|  | void set_context_scope(const ContextScope& value) const; | 
|  |  | 
|  | struct AwaiterLink { | 
|  | // Context depth at which the `@pragma('vm:awaiter-link')` variable | 
|  | // is located. | 
|  | uint8_t depth = UntaggedClosureData::kNoAwaiterLinkDepth; | 
|  | // Context index at which the `@pragma('vm:awaiter-link')` variable | 
|  | // is located. | 
|  | uint8_t index = static_cast<uint8_t>(-1); | 
|  | }; | 
|  |  | 
|  | AwaiterLink awaiter_link() const; | 
|  | void set_awaiter_link(AwaiterLink link) const; | 
|  | bool HasAwaiterLink() const { | 
|  | return IsClosureFunction() && | 
|  | (awaiter_link().depth != UntaggedClosureData::kNoAwaiterLinkDepth); | 
|  | } | 
|  |  | 
|  | // Enclosing function of this local function. | 
|  | FunctionPtr parent_function() const; | 
|  |  | 
|  | // Returns a canonicalized vector of the type parameters instantiated | 
|  | // to bounds (e.g., the local type arguments that are used if no TAV is | 
|  | // provided when the function is invoked). | 
|  | // | 
|  | // If the function is owned by a generic class or has any generic parent | 
|  | // functions, then the returned vector may require instantiation, see | 
|  | // default_type_arguments_instantiation_mode() for Closure functions | 
|  | // or TypeArguments::GetInstantiationMode() otherwise. | 
|  | // | 
|  | // If non-generic, the empty type arguments vector is returned. | 
|  | TypeArgumentsPtr DefaultTypeArguments(Zone* zone) const; | 
|  |  | 
|  | // Only usable for closure functions. | 
|  | InstantiationMode default_type_arguments_instantiation_mode() const; | 
|  | void set_default_type_arguments_instantiation_mode( | 
|  | InstantiationMode value) const; | 
|  |  | 
|  | // Enclosing outermost function of this local function. | 
|  | FunctionPtr GetOutermostFunction() const; | 
|  |  | 
|  | void set_extracted_method_closure(const Function& function) const; | 
|  | FunctionPtr extracted_method_closure() const; | 
|  |  | 
|  | void set_saved_args_desc(const Array& array) const; | 
|  | ArrayPtr saved_args_desc() const; | 
|  |  | 
|  | bool HasSavedArgumentsDescriptor() const { | 
|  | return IsInvokeFieldDispatcher() || IsNoSuchMethodDispatcher(); | 
|  | } | 
|  |  | 
|  | void set_accessor_field(const Field& value) const; | 
|  | FieldPtr accessor_field() const; | 
|  |  | 
|  | bool IsRegularFunction() const { | 
|  | return kind() == UntaggedFunction::kRegularFunction; | 
|  | } | 
|  |  | 
|  | bool IsMethodExtractor() const { | 
|  | return kind() == UntaggedFunction::kMethodExtractor; | 
|  | } | 
|  |  | 
|  | bool IsNoSuchMethodDispatcher() const { | 
|  | return kind() == UntaggedFunction::kNoSuchMethodDispatcher; | 
|  | } | 
|  |  | 
|  | bool IsRecordFieldGetter() const { | 
|  | return kind() == UntaggedFunction::kRecordFieldGetter; | 
|  | } | 
|  |  | 
|  | bool IsInvokeFieldDispatcher() const { | 
|  | return kind() == UntaggedFunction::kInvokeFieldDispatcher; | 
|  | } | 
|  |  | 
|  | bool IsDynamicInvokeFieldDispatcher() const { | 
|  | return IsInvokeFieldDispatcher() && | 
|  | IsDynamicInvocationForwarderName(name()); | 
|  | } | 
|  |  | 
|  | // Returns true if this function is _Closure.dyn:call, which implements | 
|  | // dynamically checked closure calls. | 
|  | bool IsDynamicClosureCallDispatcher() const; | 
|  |  | 
|  | // Returns true if this function is _Closure.call, which implements the | 
|  | // Function interface for closures. | 
|  | bool IsClosureCallDispatcher() const; | 
|  |  | 
|  | bool IsDynamicInvocationForwarder() const { | 
|  | return kind() == UntaggedFunction::kDynamicInvocationForwarder; | 
|  | } | 
|  |  | 
|  | bool IsImplicitGetterOrSetter() const { | 
|  | return kind() == UntaggedFunction::kImplicitGetter || | 
|  | kind() == UntaggedFunction::kImplicitSetter || | 
|  | kind() == UntaggedFunction::kImplicitStaticGetter; | 
|  | } | 
|  |  | 
|  | // Returns true iff an implicit closure function has been created | 
|  | // for this function. | 
|  | bool HasImplicitClosureFunction() const { | 
|  | return implicit_closure_function() != null(); | 
|  | } | 
|  |  | 
|  | // Returns the closure function implicitly created for this function.  If none | 
|  | // exists yet, create one and remember it.  Implicit closure functions are | 
|  | // used in VM Closure instances that represent results of tear-off operations. | 
|  | FunctionPtr ImplicitClosureFunction() const; | 
|  | void DropUncompiledImplicitClosureFunction() const; | 
|  |  | 
|  | // Return the closure implicitly created for this function. | 
|  | // If none exists yet, create one and remember it. | 
|  | ClosurePtr ImplicitStaticClosure() const; | 
|  |  | 
|  | ClosurePtr ImplicitInstanceClosure(const Instance& receiver) const; | 
|  |  | 
|  | // Returns the target of the implicit closure or null if the target is now | 
|  | // invalid (e.g., mismatched argument shapes after a reload). | 
|  | FunctionPtr ImplicitClosureTarget(Zone* zone) const; | 
|  |  | 
|  | FunctionPtr ForwardingTarget() const; | 
|  | void SetForwardingTarget(const Function& target) const; | 
|  |  | 
|  | UntaggedFunction::Kind kind() const { return KindOf(ptr()); } | 
|  | static UntaggedFunction::Kind KindOf(FunctionPtr func) { | 
|  | return func->untag()->kind_tag_.Read<KindBits>(); | 
|  | } | 
|  |  | 
|  | UntaggedFunction::AsyncModifier modifier() const { | 
|  | return untag()->kind_tag_.Read<ModifierBits>(); | 
|  | } | 
|  |  | 
|  | static const char* KindToCString(UntaggedFunction::Kind kind); | 
|  |  | 
|  | bool IsConstructor() const { | 
|  | return kind() == UntaggedFunction::kConstructor; | 
|  | } | 
|  | bool IsGenerativeConstructor() const { | 
|  | return IsConstructor() && !is_static(); | 
|  | } | 
|  | bool IsImplicitConstructor() const; | 
|  | bool IsFactory() const { return IsConstructor() && is_static(); } | 
|  |  | 
|  | bool HasThisParameter() const { | 
|  | return IsDynamicFunction(/*allow_abstract=*/true) || | 
|  | IsGenerativeConstructor() || (IsFieldInitializer() && !is_static()); | 
|  | } | 
|  |  | 
|  | bool IsDynamicFunction(bool allow_abstract = false) const { | 
|  | if (is_static() || (!allow_abstract && is_abstract())) { | 
|  | return false; | 
|  | } | 
|  | switch (kind()) { | 
|  | case UntaggedFunction::kRegularFunction: | 
|  | case UntaggedFunction::kGetterFunction: | 
|  | case UntaggedFunction::kSetterFunction: | 
|  | case UntaggedFunction::kImplicitGetter: | 
|  | case UntaggedFunction::kImplicitSetter: | 
|  | case UntaggedFunction::kMethodExtractor: | 
|  | case UntaggedFunction::kNoSuchMethodDispatcher: | 
|  | case UntaggedFunction::kInvokeFieldDispatcher: | 
|  | case UntaggedFunction::kDynamicInvocationForwarder: | 
|  | case UntaggedFunction::kRecordFieldGetter: | 
|  | return true; | 
|  | case UntaggedFunction::kClosureFunction: | 
|  | case UntaggedFunction::kImplicitClosureFunction: | 
|  | case UntaggedFunction::kConstructor: | 
|  | case UntaggedFunction::kImplicitStaticGetter: | 
|  | case UntaggedFunction::kFieldInitializer: | 
|  | case UntaggedFunction::kIrregexpFunction: | 
|  | return false; | 
|  | default: | 
|  | UNREACHABLE(); | 
|  | return false; | 
|  | } | 
|  | } | 
|  | bool IsStaticFunction() const { | 
|  | if (!is_static()) { | 
|  | return false; | 
|  | } | 
|  | switch (kind()) { | 
|  | case UntaggedFunction::kRegularFunction: | 
|  | case UntaggedFunction::kGetterFunction: | 
|  | case UntaggedFunction::kSetterFunction: | 
|  | case UntaggedFunction::kImplicitGetter: | 
|  | case UntaggedFunction::kImplicitSetter: | 
|  | case UntaggedFunction::kImplicitStaticGetter: | 
|  | case UntaggedFunction::kFieldInitializer: | 
|  | case UntaggedFunction::kIrregexpFunction: | 
|  | return true; | 
|  | case UntaggedFunction::kClosureFunction: | 
|  | case UntaggedFunction::kImplicitClosureFunction: | 
|  | case UntaggedFunction::kConstructor: | 
|  | case UntaggedFunction::kMethodExtractor: | 
|  | case UntaggedFunction::kNoSuchMethodDispatcher: | 
|  | case UntaggedFunction::kInvokeFieldDispatcher: | 
|  | case UntaggedFunction::kDynamicInvocationForwarder: | 
|  | case UntaggedFunction::kFfiTrampoline: | 
|  | case UntaggedFunction::kRecordFieldGetter: | 
|  | return false; | 
|  | default: | 
|  | UNREACHABLE(); | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool NeedsTypeArgumentTypeChecks() const { | 
|  | return !(is_static() || (kind() == UntaggedFunction::kConstructor)); | 
|  | } | 
|  |  | 
|  | bool NeedsArgumentTypeChecks() const { | 
|  | return !(is_static() || (kind() == UntaggedFunction::kConstructor)); | 
|  | } | 
|  |  | 
|  | bool NeedsMonomorphicCheckedEntry(Zone* zone) const; | 
|  | bool HasDynamicCallers(Zone* zone) const; | 
|  | bool PrologueNeedsArgumentsDescriptor() const; | 
|  |  | 
|  | bool MayHaveUncheckedEntryPoint() const; | 
|  |  | 
|  | TokenPosition token_pos() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return TokenPosition::kNoSource; | 
|  | #else | 
|  | return untag()->token_pos_; | 
|  | #endif | 
|  | } | 
|  | void set_token_pos(TokenPosition value) const; | 
|  |  | 
|  | TokenPosition end_token_pos() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return TokenPosition::kNoSource; | 
|  | #else | 
|  | return untag()->end_token_pos_; | 
|  | #endif | 
|  | } | 
|  | void set_end_token_pos(TokenPosition value) const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | #else | 
|  | StoreNonPointer(&untag()->end_token_pos_, value); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | #if !defined(PRODUCT) &&                                                       \ | 
|  | (defined(DART_PRECOMPILER) || defined(DART_PRECOMPILED_RUNTIME)) | 
|  | int32_t line() const { return untag()->token_pos_.Serialize(); } | 
|  |  | 
|  | void set_line(int32_t line) const { | 
|  | StoreNonPointer(&untag()->token_pos_, TokenPosition::Deserialize(line)); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | // Returns the size of the source for this function. | 
|  | intptr_t SourceSize() const; | 
|  |  | 
|  | // Returns the number of required positional parameters. | 
|  | intptr_t num_fixed_parameters() const; | 
|  | // Returns the number of optional parameters, whether positional or named. | 
|  | bool HasOptionalParameters() const; | 
|  | // Returns whether the function has optional named parameters. | 
|  | bool HasOptionalNamedParameters() const; | 
|  | // Returns whether the function has required named parameters. | 
|  | bool HasRequiredNamedParameters() const; | 
|  | // Returns whether the function has optional positional parameters. | 
|  | bool HasOptionalPositionalParameters() const; | 
|  | // Returns the number of optional parameters, or 0 if none. | 
|  | intptr_t NumOptionalParameters() const; | 
|  | // Returns the number of optional positional parameters, or 0 if none. | 
|  | intptr_t NumOptionalPositionalParameters() const; | 
|  | // Returns the number of optional named parameters, or 0 if none. | 
|  | intptr_t NumOptionalNamedParameters() const; | 
|  | // Returns the total number of both required and optional parameters. | 
|  | intptr_t NumParameters() const; | 
|  | // Returns the number of implicit parameters, e.g., this for instance methods. | 
|  | intptr_t NumImplicitParameters() const; | 
|  |  | 
|  | // Returns true if parameters of this function are copied into the frame | 
|  | // in the function prologue. | 
|  | bool MakesCopyOfParameters() const { | 
|  | return HasOptionalParameters() || IsSuspendableFunction(); | 
|  | } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | intptr_t MaxNumberOfParametersInRegisters(Zone* zone) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | #define DEFINE_GETTERS_AND_SETTERS(return_type, type, name)                    \ | 
|  | static intptr_t name##_offset() {                                            \ | 
|  | UNREACHABLE();                                                             \ | 
|  | return 0;                                                                  \ | 
|  | }                                                                            \ | 
|  | return_type name() const { return 0; }                                       \ | 
|  | \ | 
|  | void set_##name(type value) const { UNREACHABLE(); } | 
|  | #else | 
|  | #define DEFINE_GETTERS_AND_SETTERS(return_type, type, name)                    \ | 
|  | static intptr_t name##_offset() {                                            \ | 
|  | return OFFSET_OF(UntaggedFunction, name##_);                               \ | 
|  | }                                                                            \ | 
|  | return_type name() const {                                                   \ | 
|  | return LoadNonPointer<type, std::memory_order_relaxed>(&untag()->name##_); \ | 
|  | }                                                                            \ | 
|  | \ | 
|  | void set_##name(type value) const {                                          \ | 
|  | StoreNonPointer<type, type, std::memory_order_relaxed>(&untag()->name##_,  \ | 
|  | value);             \ | 
|  | } | 
|  | #endif | 
|  |  | 
|  | JIT_FUNCTION_COUNTERS(DEFINE_GETTERS_AND_SETTERS) | 
|  |  | 
|  | #undef DEFINE_GETTERS_AND_SETTERS | 
|  |  | 
|  | #if defined(DART_DYNAMIC_MODULES) | 
|  | bool is_declared_in_bytecode() const; | 
|  | #else | 
|  | bool is_declared_in_bytecode() const { return false; } | 
|  | #endif  // defined(DART_DYNAMIC_MODULES) | 
|  |  | 
|  | intptr_t kernel_offset() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return 0; | 
|  | #else | 
|  | return untag()->kernel_offset_; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void set_kernel_offset(intptr_t value) const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | #else | 
|  | ASSERT(value >= 0); | 
|  | StoreNonPointer(&untag()->kernel_offset_, value); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void InheritKernelOffsetFrom(const Function& src) const; | 
|  | void InheritKernelOffsetFrom(const Field& src) const; | 
|  |  | 
|  | static constexpr intptr_t kMaxInstructionCount = (1 << 16) - 1; | 
|  |  | 
|  | void SetOptimizedInstructionCountClamped(uintptr_t value) const { | 
|  | if (value > kMaxInstructionCount) value = kMaxInstructionCount; | 
|  | set_optimized_instruction_count(value); | 
|  | } | 
|  |  | 
|  | void SetOptimizedCallSiteCountClamped(uintptr_t value) const { | 
|  | if (value > kMaxInstructionCount) value = kMaxInstructionCount; | 
|  | set_optimized_call_site_count(value); | 
|  | } | 
|  |  | 
|  | void SetKernelLibraryAndEvalScript( | 
|  | const Script& script, | 
|  | const class KernelProgramInfo& kernel_program_info, | 
|  | intptr_t index) const; | 
|  |  | 
|  | intptr_t KernelLibraryOffset() const; | 
|  | intptr_t KernelLibraryIndex() const; | 
|  |  | 
|  | TypedDataViewPtr KernelLibrary() const; | 
|  |  | 
|  | bool IsOptimizable() const; | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void SetIsOptimizable(bool value) const { | 
|  | ASSERT(!is_native()); | 
|  | set_is_optimizable(value); | 
|  | if (!value) { | 
|  | set_is_inlinable(false); | 
|  | set_usage_counter(INT32_MIN); | 
|  | } | 
|  | } | 
|  | #endif | 
|  |  | 
|  | // Whether this function must be optimized immediately and cannot be compiled | 
|  | // with the unoptimizing compiler. Such a function must be sure to not | 
|  | // deoptimize, since we won't generate deoptimization info or register | 
|  | // dependencies. It will be compiled into optimized code immediately when it's | 
|  | // run. | 
|  | bool ForceOptimize() const; | 
|  |  | 
|  | // Whether this function should be inlined if at all possible. | 
|  | bool IsPreferInline() const; | 
|  |  | 
|  | // Whether this function is idempotent (i.e. calling it twice has the same | 
|  | // effect as calling it once - no visible side effects). | 
|  | // | 
|  | // If a function is idempotent VM may decide to abort halfway through one call | 
|  | // and retry it again. | 
|  | bool IsIdempotent() const; | 
|  |  | 
|  | bool IsCachableIdempotent() const; | 
|  |  | 
|  | // Whether this function's |recognized_kind| requires optimization. | 
|  | bool RecognizedKindForceOptimize() const; | 
|  |  | 
|  | bool CanBeInlined() const; | 
|  |  | 
|  | MethodRecognizer::Kind recognized_kind() const { | 
|  | return untag()->kind_tag_.Read<RecognizedBits>(); | 
|  | } | 
|  | void set_recognized_kind(MethodRecognizer::Kind value) const; | 
|  |  | 
|  | bool IsRecognized() const { | 
|  | return recognized_kind() != MethodRecognizer::kUnknown; | 
|  | } | 
|  |  | 
|  | bool HasOptimizedCode() const; | 
|  |  | 
|  | // Returns true if the argument counts are valid for calling this function. | 
|  | // Otherwise, it returns false and the reason (if error_message is not | 
|  | // nullptr). | 
|  | bool AreValidArgumentCounts(intptr_t num_type_arguments, | 
|  | intptr_t num_arguments, | 
|  | intptr_t num_named_arguments, | 
|  | String* error_message) const; | 
|  |  | 
|  | // Returns a TypeError if the provided arguments don't match the function | 
|  | // parameter types, null otherwise. Assumes AreValidArguments is called first. | 
|  | // | 
|  | // If the function has a non-null receiver in the arguments, the instantiator | 
|  | // type arguments are retrieved from the receiver, otherwise the null type | 
|  | // arguments vector is used. | 
|  | // | 
|  | // If the function is generic, the appropriate function type arguments are | 
|  | // retrieved either from the arguments array or the receiver (if a closure). | 
|  | // If no function type arguments are available in either location, the bounds | 
|  | // of the function type parameters are instantiated and used as the function | 
|  | // type arguments. | 
|  | // | 
|  | // The local function type arguments (_not_ parent function type arguments) | 
|  | // are also checked against the bounds of the corresponding parameters to | 
|  | // ensure they are appropriate subtypes if the function is generic. | 
|  | ObjectPtr DoArgumentTypesMatch(const Array& args, | 
|  | const ArgumentsDescriptor& arg_names) const; | 
|  |  | 
|  | // Returns a TypeError if the provided arguments don't match the function | 
|  | // parameter types, null otherwise. Assumes AreValidArguments is called first. | 
|  | // | 
|  | // If the function is generic, the appropriate function type arguments are | 
|  | // retrieved either from the arguments array or the receiver (if a closure). | 
|  | // If no function type arguments are available in either location, the bounds | 
|  | // of the function type parameters are instantiated and used as the function | 
|  | // type arguments. | 
|  | // | 
|  | // The local function type arguments (_not_ parent function type arguments) | 
|  | // are also checked against the bounds of the corresponding parameters to | 
|  | // ensure they are appropriate subtypes if the function is generic. | 
|  | ObjectPtr DoArgumentTypesMatch( | 
|  | const Array& args, | 
|  | const ArgumentsDescriptor& arg_names, | 
|  | const TypeArguments& instantiator_type_args) const; | 
|  |  | 
|  | // Returns a TypeError if the provided arguments don't match the function | 
|  | // parameter types, null otherwise. Assumes AreValidArguments is called first. | 
|  | // | 
|  | // The local function type arguments (_not_ parent function type arguments) | 
|  | // are also checked against the bounds of the corresponding parameters to | 
|  | // ensure they are appropriate subtypes if the function is generic. | 
|  | ObjectPtr DoArgumentTypesMatch(const Array& args, | 
|  | const ArgumentsDescriptor& arg_names, | 
|  | const TypeArguments& instantiator_type_args, | 
|  | const TypeArguments& function_type_args) const; | 
|  |  | 
|  | // Returns true if the type argument count, total argument count and the names | 
|  | // of optional arguments are valid for calling this function. | 
|  | // Otherwise, it returns false and the reason (if error_message is not | 
|  | // nullptr). | 
|  | bool AreValidArguments(intptr_t num_type_arguments, | 
|  | intptr_t num_arguments, | 
|  | const Array& argument_names, | 
|  | String* error_message) const; | 
|  | bool AreValidArguments(const ArgumentsDescriptor& args_desc, | 
|  | String* error_message) const; | 
|  |  | 
|  | // Fully qualified name uniquely identifying the function under gdb and during | 
|  | // ast printing. The special ':' character, if present, is replaced by '_'. | 
|  | const char* ToFullyQualifiedCString() const; | 
|  |  | 
|  | const char* ToLibNamePrefixedQualifiedCString() const; | 
|  |  | 
|  | const char* ToQualifiedCString() const; | 
|  |  | 
|  | static constexpr intptr_t maximum_unboxed_parameter_count() { | 
|  | // Subtracts one that represents the return value | 
|  | return UntaggedFunction::UnboxedParameterBitmap::kCapacity - 1; | 
|  | } | 
|  |  | 
|  | void reset_unboxed_parameters_and_return() const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | StoreNonPointer(&untag()->unboxed_parameters_info_, | 
|  | UntaggedFunction::UnboxedParameterBitmap()); | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | void set_unboxed_integer_parameter_at(intptr_t index) const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | ASSERT(index >= 0 && index < maximum_unboxed_parameter_count()); | 
|  | index++;  // position 0 is reserved for the return value | 
|  | const_cast<UntaggedFunction::UnboxedParameterBitmap*>( | 
|  | &untag()->unboxed_parameters_info_) | 
|  | ->SetUnboxedInteger(index); | 
|  | #else | 
|  | UNREACHABLE(); | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | void set_unboxed_double_parameter_at(intptr_t index) const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | ASSERT(index >= 0 && index < maximum_unboxed_parameter_count()); | 
|  | index++;  // position 0 is reserved for the return value | 
|  | const_cast<UntaggedFunction::UnboxedParameterBitmap*>( | 
|  | &untag()->unboxed_parameters_info_) | 
|  | ->SetUnboxedDouble(index); | 
|  |  | 
|  | #else | 
|  | UNREACHABLE(); | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | void set_unboxed_integer_return() const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | const_cast<UntaggedFunction::UnboxedParameterBitmap*>( | 
|  | &untag()->unboxed_parameters_info_) | 
|  | ->SetUnboxedInteger(0); | 
|  | #else | 
|  | UNREACHABLE(); | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | void set_unboxed_double_return() const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | const_cast<UntaggedFunction::UnboxedParameterBitmap*>( | 
|  | &untag()->unboxed_parameters_info_) | 
|  | ->SetUnboxedDouble(0); | 
|  |  | 
|  | #else | 
|  | UNREACHABLE(); | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | void set_unboxed_record_return() const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | const_cast<UntaggedFunction::UnboxedParameterBitmap*>( | 
|  | &untag()->unboxed_parameters_info_) | 
|  | ->SetUnboxedRecord(0); | 
|  |  | 
|  | #else | 
|  | UNREACHABLE(); | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | bool is_unboxed_parameter_at(intptr_t index) const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | ASSERT(index >= 0); | 
|  | index++;  // position 0 is reserved for the return value | 
|  | return untag()->unboxed_parameters_info_.IsUnboxed(index); | 
|  | #else | 
|  | return false; | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | bool is_unboxed_integer_parameter_at(intptr_t index) const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | ASSERT(index >= 0); | 
|  | index++;  // position 0 is reserved for the return value | 
|  | return untag()->unboxed_parameters_info_.IsUnboxedInteger(index); | 
|  | #else | 
|  | return false; | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | bool is_unboxed_double_parameter_at(intptr_t index) const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | ASSERT(index >= 0); | 
|  | index++;  // position 0 is reserved for the return value | 
|  | return untag()->unboxed_parameters_info_.IsUnboxedDouble(index); | 
|  | #else | 
|  | return false; | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | bool has_unboxed_return() const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | return untag()->unboxed_parameters_info_.IsUnboxed(0); | 
|  | #else | 
|  | return false; | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | bool has_unboxed_integer_return() const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | return untag()->unboxed_parameters_info_.IsUnboxedInteger(0); | 
|  | #else | 
|  | return false; | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | bool has_unboxed_double_return() const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | return untag()->unboxed_parameters_info_.IsUnboxedDouble(0); | 
|  | #else | 
|  | return false; | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | bool has_unboxed_record_return() const { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | return untag()->unboxed_parameters_info_.IsUnboxedRecord(0); | 
|  | #else | 
|  | return false; | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | bool HasUnboxedParameters() const { | 
|  | return untag()->unboxed_parameters_info_.HasUnboxedParameters(); | 
|  | } | 
|  | bool HasUnboxedReturnValue() const { return has_unboxed_return(); } | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | bool IsDispatcherOrImplicitAccessor() const { | 
|  | switch (kind()) { | 
|  | case UntaggedFunction::kImplicitGetter: | 
|  | case UntaggedFunction::kImplicitSetter: | 
|  | case UntaggedFunction::kImplicitStaticGetter: | 
|  | case UntaggedFunction::kNoSuchMethodDispatcher: | 
|  | case UntaggedFunction::kInvokeFieldDispatcher: | 
|  | case UntaggedFunction::kDynamicInvocationForwarder: | 
|  | return true; | 
|  | default: | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents an explicit getter function. | 
|  | bool IsGetterFunction() const { | 
|  | return kind() == UntaggedFunction::kGetterFunction; | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents an implicit getter function. | 
|  | bool IsImplicitGetterFunction() const { | 
|  | return kind() == UntaggedFunction::kImplicitGetter; | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents an implicit static getter | 
|  | // function. | 
|  | bool IsImplicitStaticGetterFunction() const { | 
|  | return kind() == UntaggedFunction::kImplicitStaticGetter; | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents an explicit setter function. | 
|  | bool IsSetterFunction() const { | 
|  | return kind() == UntaggedFunction::kSetterFunction; | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents an implicit setter function. | 
|  | bool IsImplicitSetterFunction() const { | 
|  | return kind() == UntaggedFunction::kImplicitSetter; | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents an initializer for a static or | 
|  | // instance field. The function returns the initial value and the caller is | 
|  | // responsible for setting the field. | 
|  | bool IsFieldInitializer() const { | 
|  | return kind() == UntaggedFunction::kFieldInitializer; | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents a (possibly implicit) closure | 
|  | // function. | 
|  | bool IsClosureFunction() const { | 
|  | UntaggedFunction::Kind k = kind(); | 
|  | return (k == UntaggedFunction::kClosureFunction) || | 
|  | (k == UntaggedFunction::kImplicitClosureFunction); | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents a generated irregexp function. | 
|  | bool IsIrregexpFunction() const { | 
|  | return kind() == UntaggedFunction::kIrregexpFunction; | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents an implicit closure function. | 
|  | bool IsImplicitClosureFunction() const { | 
|  | return kind() == UntaggedFunction::kImplicitClosureFunction; | 
|  | } | 
|  | static bool IsImplicitClosureFunction(FunctionPtr func) { | 
|  | return KindOf(func) == UntaggedFunction::kImplicitClosureFunction; | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents a non implicit closure function. | 
|  | bool IsNonImplicitClosureFunction() const { | 
|  | return IsClosureFunction() && !IsImplicitClosureFunction(); | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents an implicit static closure | 
|  | // function. | 
|  | bool IsImplicitStaticClosureFunction() const { | 
|  | return IsImplicitClosureFunction() && is_static(); | 
|  | } | 
|  | static bool IsImplicitStaticClosureFunction(FunctionPtr func); | 
|  |  | 
|  | // Returns true if this function represents an implicit instance closure | 
|  | // function. | 
|  | bool IsImplicitInstanceClosureFunction() const { | 
|  | return IsImplicitClosureFunction() && !is_static(); | 
|  | } | 
|  | static bool IsImplicitInstanceClosureFunction(FunctionPtr func); | 
|  |  | 
|  | // Returns true if this function has a parent function. | 
|  | bool HasParent() const { return parent_function() != Function::null(); } | 
|  |  | 
|  | // Returns true if this function is a local function. | 
|  | bool IsLocalFunction() const { | 
|  | return !IsImplicitClosureFunction() && HasParent(); | 
|  | } | 
|  |  | 
|  | // Returns true if this function represents an ffi trampoline. | 
|  | bool IsFfiCallbackTrampoline() const { | 
|  | return kind() == UntaggedFunction::kFfiTrampoline; | 
|  | } | 
|  | static bool IsFfiCallbackTrampoline(FunctionPtr function) { | 
|  | NoSafepointScope no_safepoint; | 
|  | return function->untag()->kind_tag_.Read<KindBits>() == | 
|  | UntaggedFunction::kFfiTrampoline; | 
|  | } | 
|  |  | 
|  | // Returns true if this function is a closure function | 
|  | // used to represent ffi call. | 
|  | bool IsFfiCallClosure() const; | 
|  |  | 
|  | // Returns value of vm:ffi:call-closure pragma. | 
|  | InstancePtr GetFfiCallClosurePragmaValue() const; | 
|  |  | 
|  | // Returns true for functions which execution can be suspended | 
|  | // using Suspend/Resume stubs. Such functions have an artificial | 
|  | // :suspend_state local variable at the fixed location of the frame. | 
|  | bool IsSuspendableFunction() const { | 
|  | return modifier() != UntaggedFunction::kNoModifier; | 
|  | } | 
|  |  | 
|  | // Returns true if this function is marked with 'async' modifier. | 
|  | bool IsAsyncFunction() const { | 
|  | return modifier() == UntaggedFunction::kAsync; | 
|  | } | 
|  |  | 
|  | // Returns true if this function is marked with 'sync*' modifier. | 
|  | bool IsSyncGenerator() const { | 
|  | return modifier() == UntaggedFunction::kSyncGen; | 
|  | } | 
|  |  | 
|  | // Returns true if this function is marked with 'async*' modifier. | 
|  | bool IsAsyncGenerator() const { | 
|  | return modifier() == UntaggedFunction::kAsyncGen; | 
|  | } | 
|  |  | 
|  | bool IsTypedDataViewFactory() const; | 
|  | bool IsUnmodifiableTypedDataViewFactory() const; | 
|  |  | 
|  | DART_WARN_UNUSED_RESULT | 
|  | ErrorPtr VerifyEntryPoint(EntryPointPragma pragma) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedFunction)); | 
|  | } | 
|  |  | 
|  | static FunctionPtr New(const FunctionType& signature, | 
|  | const String& name, | 
|  | UntaggedFunction::Kind kind, | 
|  | bool is_static, | 
|  | bool is_const, | 
|  | bool is_abstract, | 
|  | bool is_external, | 
|  | bool is_native, | 
|  | const Object& owner, | 
|  | TokenPosition token_pos, | 
|  | Heap::Space space = Heap::kOld); | 
|  |  | 
|  | // Allocates a new Function object representing a closure function | 
|  | // with given kind - kClosureFunction or kImplicitClosureFunction. | 
|  | static FunctionPtr NewClosureFunctionWithKind(UntaggedFunction::Kind kind, | 
|  | const String& name, | 
|  | const Function& parent, | 
|  | bool is_static, | 
|  | TokenPosition token_pos, | 
|  | const Object& owner); | 
|  |  | 
|  | // Allocates a new Function object representing a closure function. | 
|  | static FunctionPtr NewClosureFunction(const String& name, | 
|  | const Function& parent, | 
|  | TokenPosition token_pos); | 
|  |  | 
|  | // Allocates a new Function object representing an implicit closure function. | 
|  | static FunctionPtr NewImplicitClosureFunction(const String& name, | 
|  | const Function& parent, | 
|  | TokenPosition token_pos); | 
|  |  | 
|  | FunctionPtr CreateMethodExtractor(const String& getter_name) const; | 
|  | FunctionPtr GetMethodExtractor(const String& getter_name) const; | 
|  |  | 
|  | static bool IsDynamicInvocationForwarderName(const String& name); | 
|  | static bool IsDynamicInvocationForwarderName(StringPtr name); | 
|  |  | 
|  | static StringPtr DemangleDynamicInvocationForwarderName(const String& name); | 
|  | static const String& DropImplicitCallPrefix(const String& name); | 
|  |  | 
|  | static StringPtr CreateDynamicInvocationForwarderName(const String& name); | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) || defined(DART_DYNAMIC_MODULES) | 
|  | FunctionPtr CreateDynamicInvocationForwarder( | 
|  | const String& mangled_name) const; | 
|  |  | 
|  | FunctionPtr GetDynamicInvocationForwarder(const String& mangled_name) const; | 
|  |  | 
|  | // Returns true if this function needs dynamic invocation forwarder: | 
|  | // that is if any of the arguments require checking on the dynamic | 
|  | // call-site: if function has no parameters or has only covariant parameters | 
|  | // as such function already checks all of its parameters. | 
|  | bool NeedsDynamicInvocationForwarder() const; | 
|  |  | 
|  | // Fills in [is_covariant] and [is_generic_covariant_impl] vectors | 
|  | // according to covariance attributes of function parameters. | 
|  | // | 
|  | // [is_covariant] and [is_generic_covariant_impl] should contain bitvectors | 
|  | // of function.NumParameters() length. | 
|  | void ReadParameterCovariance(BitVector* is_covariant, | 
|  | BitVector* is_generic_covariant_impl) const; | 
|  | #endif | 
|  |  | 
|  | // Slow function, use in asserts to track changes in important library | 
|  | // functions. | 
|  | int32_t SourceFingerprint() const; | 
|  |  | 
|  | // Return false and report an error if the fingerprint does not match. | 
|  | bool CheckSourceFingerprint(int32_t fp, const char* kind = nullptr) const; | 
|  |  | 
|  | // Works with map [deopt-id] -> ICData. | 
|  | void SaveICDataMap( | 
|  | const ZoneGrowableArray<const ICData*>& deopt_id_to_ic_data, | 
|  | const Array& edge_counters_array, | 
|  | const Array& coverage_array) const; | 
|  | // Uses 'ic_data_array' to populate the table 'deopt_id_to_ic_data'. Clone | 
|  | // ic_data (array and descriptor) if 'clone_ic_data' is true. | 
|  | void RestoreICDataMap(ZoneGrowableArray<const ICData*>* deopt_id_to_ic_data, | 
|  | bool clone_ic_data) const; | 
|  |  | 
|  | // ic_data_array attached to the function stores edge counters in the | 
|  | // first element, coverage data array in the second element and the rest | 
|  | // are ICData objects. | 
|  | struct ICDataArrayIndices { | 
|  | static constexpr intptr_t kEdgeCounters = 0; | 
|  | static constexpr intptr_t kCoverageData = 1; | 
|  | static constexpr intptr_t kFirstICData = 2; | 
|  | }; | 
|  |  | 
|  | ArrayPtr ic_data_array() const; | 
|  | void ClearICDataArray() const; | 
|  | ICDataPtr FindICData(intptr_t deopt_id) const; | 
|  |  | 
|  | // Coverage data array is a list of pairs: | 
|  | //   element 2 * i + 0 is token position | 
|  | //   element 2 * i + 1 is coverage hit (zero meaning code was not hit) | 
|  | ArrayPtr GetCoverageArray() const; | 
|  |  | 
|  | // Outputs this function's service ID to the provided JSON object. | 
|  | void AddFunctionServiceId(const JSONObject& obj) const; | 
|  |  | 
|  | // Sets deopt reason in all ICData-s with given deopt_id. | 
|  | void SetDeoptReasonForAll(intptr_t deopt_id, ICData::DeoptReasonId reason); | 
|  |  | 
|  | void set_modifier(UntaggedFunction::AsyncModifier value) const; | 
|  |  | 
|  | // 'WasCompiled' is true if the function was compiled once in this | 
|  | // VM instantiation. It is independent from presence of type feedback | 
|  | // (ic_data_array) and code, which may be loaded from a snapshot. | 
|  | // 'WasExecuted' is true if the usage counter has ever been positive. | 
|  | // 'ProhibitsInstructionHoisting' is true if this function deoptimized before on | 
|  | // a hoisted instruction. | 
|  | // 'ProhibitsBoundsCheckGeneralization' is true if this function deoptimized | 
|  | // before on a generalized bounds check. | 
|  | // IsDynamicallyOverridden: This function can be overridden in a dynamically | 
|  | //                          loaded class. | 
|  | #define STATE_BITS_LIST(V)                                                     \ | 
|  | V(WasCompiled)                                                               \ | 
|  | V(WasExecutedBit)                                                            \ | 
|  | V(ProhibitsInstructionHoisting)                                              \ | 
|  | V(ProhibitsBoundsCheckGeneralization)                                        \ | 
|  | V(IsDynamicallyOverridden) | 
|  |  | 
|  | enum StateBits { | 
|  | #define DECLARE_FLAG_POS(Name) k##Name##Pos, | 
|  | STATE_BITS_LIST(DECLARE_FLAG_POS) | 
|  | #undef DECLARE_FLAG_POS | 
|  | }; | 
|  | #define DEFINE_FLAG_BIT(Name)                                                  \ | 
|  | using Name##Bit = BitField<uint8_t, bool, k##Name##Pos>; | 
|  | STATE_BITS_LIST(DEFINE_FLAG_BIT) | 
|  | #undef DEFINE_FLAG_BIT | 
|  |  | 
|  | #define DEFINE_FLAG_ACCESSORS(Name)                                            \ | 
|  | void Set##Name(bool value) const {                                           \ | 
|  | set_state_bits(Name##Bit::update(value, state_bits()));                    \ | 
|  | }                                                                            \ | 
|  | bool Name() const { return Name##Bit::decode(state_bits()); } | 
|  | STATE_BITS_LIST(DEFINE_FLAG_ACCESSORS) | 
|  | #undef DEFINE_FLAG_ACCESSORS | 
|  |  | 
|  | void SetUsageCounter(intptr_t value) const { | 
|  | if (usage_counter() > 0) { | 
|  | SetWasExecuted(true); | 
|  | } | 
|  | set_usage_counter(value); | 
|  | } | 
|  |  | 
|  | bool WasExecuted() const { return (usage_counter() > 0) || WasExecutedBit(); } | 
|  |  | 
|  | void SetWasExecuted(bool value) const { SetWasExecutedBit(value); } | 
|  |  | 
|  | static intptr_t data_offset() { return OFFSET_OF(UntaggedFunction, data_); } | 
|  |  | 
|  | static intptr_t kind_tag_offset() { | 
|  | return OFFSET_OF(UntaggedFunction, kind_tag_); | 
|  | } | 
|  |  | 
|  | // static: Considered during class-side or top-level resolution rather than | 
|  | //         instance-side resolution. | 
|  | // const: Valid target of a const constructor call. | 
|  | // abstract: Skipped during instance-side resolution. | 
|  | // reflectable: Enumerated by mirrors, invocable by mirrors. False for private | 
|  | //              functions of dart: libraries. | 
|  | // debuggable: Valid location of a breakpoint. Synthetic code is not | 
|  | //             debuggable. | 
|  | // visible: Frame is included in stack traces. Synthetic code such as | 
|  | //          dispatchers is not visible. Synthetic code that can trigger | 
|  | //          exceptions such as the outer async functions that create Futures | 
|  | //          is visible. | 
|  | // intrinsic: Has a hand-written assembly prologue. | 
|  | // inlinable: Candidate for inlining. False for functions with features we | 
|  | //            don't support during inlining (e.g., optional parameters), | 
|  | //            functions which are too big, etc. | 
|  | // native: Bridge to C/C++ code. | 
|  | // external: Just a declaration that expects to be defined in another patch | 
|  | //           file. | 
|  | // polymorphic_target: A polymorphic method. | 
|  | // has_pragma: Has a @pragma decoration. | 
|  | // no_such_method_forwarder: A stub method that just calls noSuchMethod. | 
|  |  | 
|  | // Bits that are set when function is created, don't have to worry about | 
|  | // concurrent updates. | 
|  | #define FOR_EACH_FUNCTION_KIND_BIT(V)                                          \ | 
|  | V(Static, is_static)                                                         \ | 
|  | V(Const, is_const)                                                           \ | 
|  | V(Abstract, is_abstract)                                                     \ | 
|  | V(Reflectable, is_reflectable)                                               \ | 
|  | V(Visible, is_visible)                                                       \ | 
|  | V(Debuggable, is_debuggable)                                                 \ | 
|  | V(Intrinsic, is_intrinsic)                                                   \ | 
|  | V(Native, is_native)                                                         \ | 
|  | V(External, is_external)                                                     \ | 
|  | V(PolymorphicTarget, is_polymorphic_target)                                  \ | 
|  | V(HasPragma, has_pragma)                                                     \ | 
|  | V(IsSynthetic, is_synthetic)                                                 \ | 
|  | V(IsExtensionMember, is_extension_member)                                    \ | 
|  | V(IsExtensionTypeMember, is_extension_type_member)                           \ | 
|  | V(IsRedirectingFactory, is_redirecting_factory) | 
|  | // Bit that is updated after function is constructed, has to be updated in | 
|  | // concurrent-safe manner. | 
|  | #define FOR_EACH_FUNCTION_VOLATILE_KIND_BIT(V) V(Inlinable, is_inlinable) | 
|  |  | 
|  | #define DEFINE_ACCESSORS(name, accessor_name)                                  \ | 
|  | void set_##accessor_name(bool value) const {                                 \ | 
|  | untag()->kind_tag_.UpdateUnsynchronized<name##Bit>(value);                 \ | 
|  | }                                                                            \ | 
|  | bool accessor_name() const { return untag()->kind_tag_.Read<name##Bit>(); } | 
|  | FOR_EACH_FUNCTION_KIND_BIT(DEFINE_ACCESSORS) | 
|  | #undef DEFINE_ACCESSORS | 
|  |  | 
|  | static bool is_visible(FunctionPtr f) { | 
|  | return f.untag()->kind_tag_.Read<VisibleBit>(); | 
|  | } | 
|  |  | 
|  | #define DEFINE_ACCESSORS(name, accessor_name)                                  \ | 
|  | void set_##accessor_name(bool value) const {                                 \ | 
|  | untag()->kind_tag_.UpdateBool<name##Bit>(value);                           \ | 
|  | }                                                                            \ | 
|  | bool accessor_name() const { return untag()->kind_tag_.Read<name##Bit>(); } | 
|  | FOR_EACH_FUNCTION_VOLATILE_KIND_BIT(DEFINE_ACCESSORS) | 
|  | #undef DEFINE_ACCESSORS | 
|  |  | 
|  | // optimizable: Candidate for going through the optimizing compiler. False for | 
|  | //              some functions known to be execute infrequently and functions | 
|  | //              which have been de-optimized too many times. | 
|  | bool is_optimizable() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return false; | 
|  | #else | 
|  | return untag()->is_optimizable_.load(std::memory_order_relaxed); | 
|  | #endif | 
|  | } | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void set_is_optimizable(bool value) const { | 
|  | untag()->is_optimizable_.store(value, std::memory_order_relaxed); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | using KindBits = BitField<decltype(UntaggedFunction::kind_tag_), | 
|  | UntaggedFunction::Kind, | 
|  | 0, | 
|  | UntaggedFunction::kKindBitSize>; | 
|  | using RecognizedBits = BitField<decltype(UntaggedFunction::kind_tag_), | 
|  | MethodRecognizer::Kind, | 
|  | KindBits::kNextBit, | 
|  | MethodRecognizer::kKindBitSize>; | 
|  | using ModifierBits = BitField<decltype(UntaggedFunction::kind_tag_), | 
|  | UntaggedFunction::AsyncModifier, | 
|  | RecognizedBits::kNextBit, | 
|  | UntaggedFunction::kAsyncModifierBitSize>; | 
|  |  | 
|  | enum KindTagBits { | 
|  | // Single bit sized fields start here. | 
|  | #define DECLARE_BIT(name, _) k##name##Bit, | 
|  | FOR_EACH_FUNCTION_KIND_BIT(DECLARE_BIT) | 
|  | FOR_EACH_FUNCTION_VOLATILE_KIND_BIT(DECLARE_BIT) | 
|  | #undef DECLARE_BIT | 
|  | }; | 
|  |  | 
|  | #define DEFINE_BIT(name, _)                                                    \ | 
|  | using name##Bit = BitField<decltype(UntaggedFunction::kind_tag_), bool,      \ | 
|  | ModifierBits::kNextBit + k##name##Bit>; | 
|  | FOR_EACH_FUNCTION_KIND_BIT(DEFINE_BIT) | 
|  | FOR_EACH_FUNCTION_VOLATILE_KIND_BIT(DEFINE_BIT) | 
|  | #undef DEFINE_BIT | 
|  |  | 
|  | private: | 
|  | enum class EvalFunctionData { | 
|  | kScript, | 
|  | kKernelProgramInfo, | 
|  | kKernelLibraryIndex, | 
|  | kLength, | 
|  | }; | 
|  | enum NativeFunctionData { | 
|  | kNativeName, | 
|  | kTearOff, | 
|  | kLength, | 
|  | }; | 
|  |  | 
|  | void set_ic_data_array(const Array& value) const; | 
|  | void set_name(const String& value) const; | 
|  | void set_kind(UntaggedFunction::Kind value) const; | 
|  | void set_parent_function(const Function& value) const; | 
|  | FunctionPtr implicit_closure_function() const; | 
|  | void set_implicit_closure_function(const Function& value) const; | 
|  | ClosurePtr implicit_static_closure() const; | 
|  | void set_implicit_static_closure(const Closure& closure) const; | 
|  | ScriptPtr eval_script() const; | 
|  | void set_eval_script(const Script& value) const; | 
|  | void set_num_optional_parameters(intptr_t value) const;  // Encoded value. | 
|  | void set_kind_tag(uint32_t value) const; | 
|  | bool is_eval_function() const; | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | ArrayPtr positional_parameter_names() const { | 
|  | return untag()->positional_parameter_names(); | 
|  | } | 
|  | void set_positional_parameter_names(const Array& value) const; | 
|  | #endif | 
|  |  | 
|  | ObjectPtr data() const { return untag()->data<std::memory_order_acquire>(); } | 
|  | void set_data(const Object& value) const; | 
|  |  | 
|  | static FunctionPtr New(Heap::Space space = Heap::kOld); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Function, Object); | 
|  | friend class Class; | 
|  | friend class Parser;  // For set_eval_script. | 
|  | // UntaggedFunction::VisitFunctionPointers accesses the private constructor of | 
|  | // Function. | 
|  | friend class UntaggedFunction; | 
|  | friend class ClassFinalizer;  // To reset parent_function. | 
|  | friend class Type;            // To adjust parent_function. | 
|  | friend class Precompiler;     // To access closure data. | 
|  | friend class ProgramVisitor;  // For set_parameter_types/names. | 
|  | }; | 
|  |  | 
|  | class ClosureData : public Object { | 
|  | public: | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedClosureData)); | 
|  | } | 
|  |  | 
|  | static intptr_t packed_fields_offset() { | 
|  | return OFFSET_OF(UntaggedClosureData, packed_fields_); | 
|  | } | 
|  |  | 
|  | using PackedInstantiationMode = UntaggedClosureData::PackedInstantiationMode; | 
|  |  | 
|  | static constexpr uint8_t kNoAwaiterLinkDepth = | 
|  | UntaggedClosureData::kNoAwaiterLinkDepth; | 
|  |  | 
|  | private: | 
|  | ContextScopePtr context_scope() const { return untag()->context_scope(); } | 
|  | void set_context_scope(const ContextScope& value) const; | 
|  |  | 
|  | void set_packed_fields(uint32_t value) const { | 
|  | untag()->packed_fields_ = value; | 
|  | } | 
|  |  | 
|  | Function::AwaiterLink awaiter_link() const; | 
|  | void set_awaiter_link(Function::AwaiterLink link) const; | 
|  |  | 
|  | // Enclosing function of this local function. | 
|  | PRECOMPILER_WSR_FIELD_DECLARATION(Function, parent_function) | 
|  |  | 
|  | ClosurePtr implicit_static_closure() const { | 
|  | return untag()->closure<std::memory_order_acquire>(); | 
|  | } | 
|  | void set_implicit_static_closure(const Closure& closure) const; | 
|  |  | 
|  | static InstantiationMode DefaultTypeArgumentsInstantiationMode( | 
|  | ClosureDataPtr ptr) { | 
|  | return ptr->untag()->packed_fields_.Read<PackedInstantiationMode>(); | 
|  | } | 
|  | InstantiationMode default_type_arguments_instantiation_mode() const { | 
|  | return DefaultTypeArgumentsInstantiationMode(ptr()); | 
|  | } | 
|  | void set_default_type_arguments_instantiation_mode( | 
|  | InstantiationMode value) const; | 
|  |  | 
|  | static ClosureDataPtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(ClosureData, Object); | 
|  | friend class Class; | 
|  | friend class Function; | 
|  | friend class Precompiler;  // To wrap parent functions in WSRs. | 
|  | }; | 
|  |  | 
|  | class FfiTrampolineData : public Object { | 
|  | public: | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedFfiTrampolineData)); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FunctionTypePtr c_signature() const { return untag()->c_signature(); } | 
|  | void set_c_signature(const FunctionType& value) const; | 
|  |  | 
|  | FunctionPtr callback_target() const { return untag()->callback_target(); } | 
|  | void set_callback_target(const Function& value) const; | 
|  |  | 
|  | InstancePtr callback_exceptional_return() const { | 
|  | return untag()->callback_exceptional_return(); | 
|  | } | 
|  | void set_callback_exceptional_return(const Instance& value) const; | 
|  |  | 
|  | FfiCallbackKind ffi_function_kind() const { | 
|  | return static_cast<FfiCallbackKind>(untag()->ffi_function_kind_); | 
|  | } | 
|  | void set_ffi_function_kind(FfiCallbackKind kind) const; | 
|  |  | 
|  | int32_t callback_id() const { return untag()->callback_id_; } | 
|  | void set_callback_id(int32_t value) const; | 
|  |  | 
|  | static FfiTrampolineDataPtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(FfiTrampolineData, Object); | 
|  | friend class Class; | 
|  | friend class Function; | 
|  | }; | 
|  |  | 
|  | class Field : public Object { | 
|  | public: | 
|  | // The field that this field was cloned from, or this field itself if it isn't | 
|  | // a clone. The purpose of cloning is that the fields the background compiler | 
|  | // sees are consistent. | 
|  | FieldPtr Original() const; | 
|  |  | 
|  | // Set the original field that this field was cloned from. | 
|  | void SetOriginal(const Field& value) const; | 
|  |  | 
|  | // Returns whether this field is an original or a clone. | 
|  | bool IsOriginal() const { | 
|  | if (IsNull()) { | 
|  | return true; | 
|  | } | 
|  | NoSafepointScope no_safepoint; | 
|  | return !untag()->owner()->IsField(); | 
|  | } | 
|  |  | 
|  | // Returns a field cloned from 'this'. 'this' is set as the | 
|  | // original field of result. | 
|  | FieldPtr CloneFromOriginal() const; | 
|  |  | 
|  | StringPtr name() const { return untag()->name(); } | 
|  | StringPtr UserVisibleName() const;  // Same as scrubbed name. | 
|  | const char* UserVisibleNameCString() const; | 
|  | virtual StringPtr DictionaryName() const { return name(); } | 
|  |  | 
|  | bool is_static() const { return untag()->kind_bits_.Read<StaticBit>(); } | 
|  | bool is_instance() const { return !is_static(); } | 
|  | bool is_final() const { return untag()->kind_bits_.Read<FinalBit>(); } | 
|  | bool is_const() const { return untag()->kind_bits_.Read<ConstBit>(); } | 
|  | bool is_late() const { return untag()->kind_bits_.Read<IsLateBit>(); } | 
|  | bool is_extension_member() const { | 
|  | return untag()->kind_bits_.Read<IsExtensionMemberBit>(); | 
|  | } | 
|  | bool is_extension_type_member() const { | 
|  | return untag()->kind_bits_.Read<IsExtensionTypeMemberBit>(); | 
|  | } | 
|  | bool needs_load_guard() const { | 
|  | return untag()->kind_bits_.Read<NeedsLoadGuardBit>(); | 
|  | } | 
|  | bool is_reflectable() const { | 
|  | return untag()->kind_bits_.Read<ReflectableBit>(); | 
|  | } | 
|  | void set_is_reflectable(bool value) const { | 
|  | ASSERT(IsOriginal()); | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<ReflectableBit>(value); | 
|  | } | 
|  |  | 
|  | bool initializer_changed_after_initialization() const { | 
|  | return untag()->kind_bits_.Read<InitializerChangedAfterInitializationBit>(); | 
|  | } | 
|  | void set_initializer_changed_after_initialization(bool value) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<InitializerChangedAfterInitializationBit>( | 
|  | value); | 
|  | } | 
|  |  | 
|  | bool has_pragma() const { return untag()->kind_bits_.Read<HasPragmaBit>(); } | 
|  | void set_has_pragma(bool value) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<HasPragmaBit>(value); | 
|  | } | 
|  |  | 
|  | bool is_covariant() const { return untag()->kind_bits_.Read<CovariantBit>(); } | 
|  | void set_is_covariant(bool value) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<CovariantBit>(value); | 
|  | } | 
|  |  | 
|  | bool is_generic_covariant_impl() const { | 
|  | return untag()->kind_bits_.Read<GenericCovariantImplBit>(); | 
|  | } | 
|  | void set_is_generic_covariant_impl(bool value) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<GenericCovariantImplBit>(value); | 
|  | } | 
|  |  | 
|  | void set_is_shared(bool value) const { | 
|  | untag()->kind_bits_.UpdateBool<SharedBit>(value); | 
|  | } | 
|  | bool is_shared() const { return untag()->kind_bits_.Read<SharedBit>(); } | 
|  |  | 
|  | #if defined(DART_DYNAMIC_MODULES) | 
|  | bool is_declared_in_bytecode() const; | 
|  | #else | 
|  | bool is_declared_in_bytecode() const { return false; } | 
|  | #endif  // defined(DART_DYNAMIC_MODULES) | 
|  |  | 
|  | intptr_t kernel_offset() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return 0; | 
|  | #else | 
|  | return untag()->kernel_offset_; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void set_kernel_offset(intptr_t value) const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | #else | 
|  | ASSERT(value >= 0); | 
|  | StoreNonPointer(&untag()->kernel_offset_, value); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void InheritKernelOffsetFrom(const Field& src) const; | 
|  |  | 
|  | TypedDataViewPtr KernelLibrary() const; | 
|  | intptr_t KernelLibraryOffset() const; | 
|  | intptr_t KernelLibraryIndex() const; | 
|  |  | 
|  | // Called during class finalization. | 
|  | inline void SetOffset(intptr_t host_offset_in_bytes, | 
|  | intptr_t target_offset_in_bytes) const; | 
|  |  | 
|  | inline intptr_t HostOffset() const; | 
|  | static intptr_t host_offset_or_field_id_offset() { | 
|  | return OFFSET_OF(UntaggedField, host_offset_or_field_id_); | 
|  | } | 
|  |  | 
|  | inline intptr_t TargetOffset() const; | 
|  | static inline intptr_t TargetOffsetOf(FieldPtr field); | 
|  |  | 
|  | ObjectPtr StaticConstFieldValue() const; | 
|  | void SetStaticConstFieldValue(const Instance& value, | 
|  | bool assert_initializing_store = true) const; | 
|  |  | 
|  | inline ObjectPtr StaticValue() const; | 
|  | void SetStaticValue(const Object& value) const; | 
|  |  | 
|  | inline intptr_t field_id() const; | 
|  | inline void set_field_id(intptr_t field_id) const; | 
|  | inline void set_field_id_unsafe(intptr_t field_id) const; | 
|  |  | 
|  | ClassPtr Owner() const; | 
|  | ScriptPtr Script() const; | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | KernelProgramInfoPtr KernelProgramInfo() const; | 
|  | #endif | 
|  | ObjectPtr RawOwner() const; | 
|  |  | 
|  | uint32_t Hash() const; | 
|  |  | 
|  | AbstractTypePtr type() const { return untag()->type(); } | 
|  | // Used by class finalizer, otherwise initialized in constructor. | 
|  | void SetFieldType(const AbstractType& value) const; | 
|  | void SetFieldTypeSafe(const AbstractType& value) const; | 
|  |  | 
|  | DART_WARN_UNUSED_RESULT | 
|  | ErrorPtr VerifyEntryPoint(EntryPointPragma kind) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedField)); | 
|  | } | 
|  |  | 
|  | static FieldPtr New(const String& name, | 
|  | bool is_static, | 
|  | bool is_final, | 
|  | bool is_const, | 
|  | bool is_reflectable, | 
|  | bool is_late, | 
|  | const Object& owner, | 
|  | const AbstractType& type, | 
|  | TokenPosition token_pos, | 
|  | TokenPosition end_token_pos); | 
|  |  | 
|  | static FieldPtr NewTopLevel(const String& name, | 
|  | bool is_final, | 
|  | bool is_const, | 
|  | bool is_late, | 
|  | const Object& owner, | 
|  | TokenPosition token_pos, | 
|  | TokenPosition end_token_pos); | 
|  |  | 
|  | // Allocate new field object, clone values from this field. The | 
|  | // original is specified. | 
|  | FieldPtr Clone(const Field& original) const; | 
|  |  | 
|  | static intptr_t kind_bits_offset() { | 
|  | return OFFSET_OF(UntaggedField, kind_bits_); | 
|  | } | 
|  |  | 
|  | TokenPosition token_pos() const { return untag()->token_pos_; } | 
|  | TokenPosition end_token_pos() const { return untag()->end_token_pos_; } | 
|  |  | 
|  | int32_t SourceFingerprint() const; | 
|  |  | 
|  | StringPtr InitializingExpression() const; | 
|  |  | 
|  | bool has_nontrivial_initializer() const { | 
|  | return untag()->kind_bits_.Read<HasNontrivialInitializerBit>(); | 
|  | } | 
|  | // Called by parser after allocating field. | 
|  | void set_has_nontrivial_initializer_unsafe( | 
|  | bool has_nontrivial_initializer) const { | 
|  | ASSERT(IsOriginal()); | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<HasNontrivialInitializerBit>( | 
|  | has_nontrivial_initializer); | 
|  | } | 
|  | void set_has_nontrivial_initializer(bool has_nontrivial_initializer) const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadWriter()); | 
|  | set_has_nontrivial_initializer_unsafe(has_nontrivial_initializer); | 
|  | } | 
|  |  | 
|  | bool has_initializer() const { | 
|  | return untag()->kind_bits_.Read<HasInitializerBit>(); | 
|  | } | 
|  | // Called by parser after allocating field. | 
|  | void set_has_initializer_unsafe(bool has_initializer) const { | 
|  | ASSERT(IsOriginal()); | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<HasInitializerBit>(has_initializer); | 
|  | } | 
|  | void set_has_initializer(bool has_initializer) const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadWriter()); | 
|  | set_has_initializer_unsafe(has_initializer); | 
|  | } | 
|  |  | 
|  | bool has_trivial_initializer() const { | 
|  | return has_initializer() && !has_nontrivial_initializer(); | 
|  | } | 
|  |  | 
|  | StaticTypeExactnessState static_type_exactness_state() const { | 
|  | return StaticTypeExactnessState::Decode( | 
|  | LoadNonPointer<int8_t, std::memory_order_relaxed>( | 
|  | &untag()->static_type_exactness_state_)); | 
|  | } | 
|  |  | 
|  | void set_static_type_exactness_state(StaticTypeExactnessState state) const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadWriter()); | 
|  | set_static_type_exactness_state_unsafe(state); | 
|  | } | 
|  |  | 
|  | void set_static_type_exactness_state_unsafe( | 
|  | StaticTypeExactnessState state) const { | 
|  | StoreNonPointer<int8_t, int8_t, std::memory_order_relaxed>( | 
|  | &untag()->static_type_exactness_state_, state.Encode()); | 
|  | } | 
|  |  | 
|  | static intptr_t static_type_exactness_state_offset() { | 
|  | return OFFSET_OF(UntaggedField, static_type_exactness_state_); | 
|  | } | 
|  |  | 
|  | // Return class id that any non-null value read from this field is guaranteed | 
|  | // to have or kDynamicCid if such class id is not known. | 
|  | // Stores to this field must update this information hence the name. | 
|  | intptr_t guarded_cid() const; | 
|  |  | 
|  | void set_guarded_cid(intptr_t cid) const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadWriter()); | 
|  | set_guarded_cid_unsafe(cid); | 
|  | } | 
|  | void set_guarded_cid_unsafe(intptr_t cid) const { | 
|  | StoreNonPointer<ClassIdTagType, ClassIdTagType, std::memory_order_relaxed>( | 
|  | &untag()->guarded_cid_, cid); | 
|  | } | 
|  | static intptr_t guarded_cid_offset() { | 
|  | return OFFSET_OF(UntaggedField, guarded_cid_); | 
|  | } | 
|  | // Return the list length that any list stored in this field is guaranteed | 
|  | // to have. If length is kUnknownFixedLength the length has not | 
|  | // been determined. If length is kNoFixedLength this field has multiple | 
|  | // list lengths associated with it and cannot be predicted. | 
|  | intptr_t guarded_list_length() const; | 
|  | void set_guarded_list_length_unsafe(intptr_t list_length) const; | 
|  | void set_guarded_list_length(intptr_t list_length) const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadWriter()); | 
|  | set_guarded_list_length_unsafe(list_length); | 
|  | } | 
|  | static intptr_t guarded_list_length_offset() { | 
|  | return OFFSET_OF(UntaggedField, guarded_list_length_); | 
|  | } | 
|  | intptr_t guarded_list_length_in_object_offset() const; | 
|  | void set_guarded_list_length_in_object_offset_unsafe(intptr_t offset) const; | 
|  | void set_guarded_list_length_in_object_offset(intptr_t offset) const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadWriter()); | 
|  | set_guarded_list_length_in_object_offset_unsafe(offset); | 
|  | } | 
|  | static intptr_t guarded_list_length_in_object_offset_offset() { | 
|  | return OFFSET_OF(UntaggedField, guarded_list_length_in_object_offset_); | 
|  | } | 
|  |  | 
|  | bool needs_length_check() const { | 
|  | const bool r = guarded_list_length() >= Field::kUnknownFixedLength; | 
|  | ASSERT(!r || is_final()); | 
|  | return r; | 
|  | } | 
|  |  | 
|  | bool NeedsSetter() const; | 
|  | bool NeedsGetter() const; | 
|  |  | 
|  | bool NeedsInitializationCheckOnLoad() const { | 
|  | return needs_load_guard() || (is_late() && !has_trivial_initializer()); | 
|  | } | 
|  |  | 
|  | const char* GuardedPropertiesAsCString() const; | 
|  |  | 
|  | bool is_unboxed() const { return untag()->kind_bits_.Read<UnboxedBit>(); } | 
|  |  | 
|  | // Field unboxing decisions are based either on static types (JIT) or | 
|  | // inferred types (AOT). See the callers of this function. | 
|  | void set_is_unboxed_unsafe(bool b) const { | 
|  | untag()->kind_bits_.UpdateBool<UnboxedBit>(b); | 
|  | } | 
|  |  | 
|  | void set_is_unboxed(bool b) const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadWriter()); | 
|  | set_is_unboxed_unsafe(b); | 
|  | } | 
|  |  | 
|  | enum { | 
|  | kUnknownLengthOffset = -1, | 
|  | kUnknownFixedLength = -1, | 
|  | kNoFixedLength = -2, | 
|  | }; | 
|  | void set_is_late(bool value) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<IsLateBit>(value); | 
|  | } | 
|  | void set_is_extension_member(bool value) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<IsExtensionMemberBit>(value); | 
|  | } | 
|  | void set_is_extension_type_member(bool value) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<IsExtensionTypeMemberBit>(value); | 
|  | } | 
|  | void set_needs_load_guard(bool value) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<NeedsLoadGuardBit>(value); | 
|  | } | 
|  | // Returns false if any value read from this field is guaranteed to be | 
|  | // not null. | 
|  | // Internally we is_nullable_ field contains either kNullCid (nullable) or | 
|  | // kIllegalCid (non-nullable) instead of boolean. This is done to simplify | 
|  | // guarding sequence in the generated code. | 
|  | bool is_nullable() const; | 
|  | void set_is_nullable(bool val) const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadWriter()); | 
|  | set_is_nullable_unsafe(val); | 
|  | } | 
|  | bool is_nullable_unsafe() const { | 
|  | return LoadNonPointer<ClassIdTagType, std::memory_order_relaxed>( | 
|  | &untag()->is_nullable_) == kNullCid; | 
|  | } | 
|  | void set_is_nullable_unsafe(bool val) const { | 
|  | StoreNonPointer<ClassIdTagType, ClassIdTagType, std::memory_order_relaxed>( | 
|  | &untag()->is_nullable_, val ? kNullCid : kIllegalCid); | 
|  | } | 
|  | static intptr_t is_nullable_offset() { | 
|  | return OFFSET_OF(UntaggedField, is_nullable_); | 
|  | } | 
|  |  | 
|  | // Record store of the given value into this field. May trigger | 
|  | // deoptimization of dependent optimized code. | 
|  | void RecordStore(const Object& value) const; | 
|  |  | 
|  | void InitializeGuardedListLengthInObjectOffset(bool unsafe = false) const; | 
|  |  | 
|  | // Return the list of optimized code objects that were optimized under | 
|  | // assumptions about guarded class id and nullability of this field. | 
|  | // These code objects must be deoptimized when field's properties change. | 
|  | // Code objects are held weakly via an indirection through WeakProperty. | 
|  | WeakArrayPtr dependent_code() const; | 
|  | void set_dependent_code(const WeakArray& array) const; | 
|  |  | 
|  | // Add the given code object to the list of dependent ones. | 
|  | void RegisterDependentCode(const Code& code) const; | 
|  |  | 
|  | // Deoptimize all dependent code objects. | 
|  | void DeoptimizeDependentCode(bool are_mutators_stopped = false) const; | 
|  |  | 
|  | // Used by background compiler to check consistency of field copy with its | 
|  | // original. | 
|  | bool IsConsistentWith(const Field& field) const; | 
|  |  | 
|  | bool IsUninitialized() const; | 
|  |  | 
|  | // Run initializer and set field value. | 
|  | DART_WARN_UNUSED_RESULT ErrorPtr | 
|  | InitializeInstance(const Instance& instance) const; | 
|  | DART_WARN_UNUSED_RESULT ErrorPtr InitializeStatic() const; | 
|  |  | 
|  | // Run initializer only. | 
|  | DART_WARN_UNUSED_RESULT ObjectPtr EvaluateInitializer() const; | 
|  |  | 
|  | FunctionPtr EnsureInitializerFunction() const; | 
|  | FunctionPtr InitializerFunction() const { | 
|  | return untag()->initializer_function<std::memory_order_acquire>(); | 
|  | } | 
|  | bool HasInitializerFunction() const; | 
|  | static intptr_t initializer_function_offset() { | 
|  | return OFFSET_OF(UntaggedField, initializer_function_); | 
|  | } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) || defined(DART_DYNAMIC_MODULES) | 
|  | FunctionPtr CreateFieldInitializerFunction(Thread* thread) const; | 
|  | void SetInitializerFunction(const Function& initializer) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) || defined(DART_DYNAMIC_MODULES) | 
|  |  | 
|  | // For static fields only. Constructs a closure that gets/sets the | 
|  | // field value. | 
|  | InstancePtr GetterClosure() const; | 
|  | InstancePtr SetterClosure() const; | 
|  | InstancePtr AccessorClosure(bool make_setter) const; | 
|  |  | 
|  | // Constructs getter and setter names for fields and vice versa. | 
|  | static StringPtr GetterName(const String& field_name); | 
|  | static StringPtr GetterSymbol(const String& field_name); | 
|  | // Returns String::null() if getter symbol does not exist. | 
|  | static StringPtr LookupGetterSymbol(const String& field_name); | 
|  | static StringPtr SetterName(const String& field_name); | 
|  | static StringPtr SetterSymbol(const String& field_name); | 
|  | // Returns String::null() if setter symbol does not exist. | 
|  | static StringPtr LookupSetterSymbol(const String& field_name); | 
|  | static StringPtr NameFromGetter(const String& getter_name); | 
|  | static StringPtr NameFromSetter(const String& setter_name); | 
|  | static StringPtr NameFromInit(const String& init_name); | 
|  | static bool IsGetterName(const String& function_name); | 
|  | static bool IsSetterName(const String& function_name); | 
|  | static bool IsInitName(const String& function_name); | 
|  |  | 
|  | private: | 
|  | static void InitializeNew(const Field& result, | 
|  | const String& name, | 
|  | bool is_static, | 
|  | bool is_final, | 
|  | bool is_const, | 
|  | bool is_reflectable, | 
|  | bool is_late, | 
|  | const Object& owner, | 
|  | TokenPosition token_pos, | 
|  | TokenPosition end_token_pos); | 
|  | friend class Interpreter;      // Access to bit field. | 
|  | friend class StoreFieldInstr;  // Generated code access to bit field. | 
|  |  | 
|  | using ConstBit = BitField<decltype(UntaggedField::kind_bits_), bool>; | 
|  | using StaticBit = | 
|  | BitField<decltype(UntaggedField::kind_bits_), bool, ConstBit::kNextBit>; | 
|  | using FinalBit = | 
|  | BitField<decltype(UntaggedField::kind_bits_), bool, StaticBit::kNextBit>; | 
|  | using HasNontrivialInitializerBit = | 
|  | BitField<decltype(UntaggedField::kind_bits_), bool, FinalBit::kNextBit>; | 
|  | using UnboxedBit = BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | HasNontrivialInitializerBit::kNextBit>; | 
|  | using ReflectableBit = | 
|  | BitField<decltype(UntaggedField::kind_bits_), bool, UnboxedBit::kNextBit>; | 
|  | using InitializerChangedAfterInitializationBit = | 
|  | BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | ReflectableBit::kNextBit>; | 
|  | using HasPragmaBit = | 
|  | BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | InitializerChangedAfterInitializationBit::kNextBit>; | 
|  | using CovariantBit = BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | HasPragmaBit::kNextBit>; | 
|  | using GenericCovariantImplBit = BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | CovariantBit::kNextBit>; | 
|  | using IsLateBit = BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | GenericCovariantImplBit::kNextBit>; | 
|  | using IsExtensionMemberBit = | 
|  | BitField<decltype(UntaggedField::kind_bits_), bool, IsLateBit::kNextBit>; | 
|  | using IsExtensionTypeMemberBit = BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | IsExtensionMemberBit::kNextBit>; | 
|  | using NeedsLoadGuardBit = BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | IsExtensionTypeMemberBit::kNextBit>; | 
|  | using HasInitializerBit = BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | NeedsLoadGuardBit::kNextBit>; | 
|  | using SharedBit = BitField<decltype(UntaggedField::kind_bits_), | 
|  | bool, | 
|  | HasInitializerBit::kNextBit>; | 
|  |  | 
|  | // Force this field's guard to be dynamic and deoptimize dependent code. | 
|  | void ForceDynamicGuardedCidAndLength() const; | 
|  |  | 
|  | void set_name(const String& value) const; | 
|  | void set_is_static(bool is_static) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<StaticBit>(is_static); | 
|  | } | 
|  | void set_is_final(bool is_final) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<FinalBit>(is_final); | 
|  | } | 
|  | void set_is_const(bool value) const { | 
|  | // TODO(36097): Once concurrent access is possible ensure updates are safe. | 
|  | untag()->kind_bits_.UpdateBool<ConstBit>(value); | 
|  | } | 
|  | void set_owner(const Object& value) const { untag()->set_owner(value.ptr()); } | 
|  | void set_token_pos(TokenPosition token_pos) const { | 
|  | StoreNonPointer(&untag()->token_pos_, token_pos); | 
|  | } | 
|  | void set_end_token_pos(TokenPosition token_pos) const { | 
|  | StoreNonPointer(&untag()->end_token_pos_, token_pos); | 
|  | } | 
|  |  | 
|  | static FieldPtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Field, Object); | 
|  | friend class Class; | 
|  | friend class UntaggedField; | 
|  | friend class FieldSerializationCluster; | 
|  | friend class FieldDeserializationCluster; | 
|  | }; | 
|  |  | 
|  | class Script : public Object { | 
|  | public: | 
|  | StringPtr url() const { return untag()->url(); } | 
|  | void set_url(const String& value) const; | 
|  |  | 
|  | // The actual url which was loaded from disk, if provided by the embedder. | 
|  | StringPtr resolved_url() const; | 
|  | bool HasSource() const; | 
|  | StringPtr Source() const; | 
|  | bool IsPartOfDartColonLibrary() const; | 
|  |  | 
|  | GrowableObjectArrayPtr GenerateLineNumberArray() const; | 
|  |  | 
|  | intptr_t line_offset() const { return 0; } | 
|  | intptr_t col_offset() const { return 0; } | 
|  | // Returns the max real token position for this script, or kNoSource | 
|  | // if there is no line starts information. | 
|  | TokenPosition MaxPosition() const; | 
|  |  | 
|  | // The load time in milliseconds since epoch. | 
|  | int64_t load_timestamp() const { return untag()->load_timestamp_; } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | // Initializes thie script object from a kernel file. | 
|  | void InitializeFromKernel(const KernelProgramInfo& info, | 
|  | intptr_t script_index, | 
|  | const TypedData& line_starts, | 
|  | const TypedDataView& constant_coverage) const; | 
|  | #endif | 
|  |  | 
|  | // The index of this script into the [KernelProgramInfo] object's source | 
|  | // table. | 
|  | intptr_t kernel_script_index() const { return untag()->kernel_script_index_; } | 
|  |  | 
|  | static intptr_t line_starts_offset() { | 
|  | return OFFSET_OF(UntaggedScript, line_starts_); | 
|  | } | 
|  |  | 
|  | TypedDataPtr line_starts() const; | 
|  |  | 
|  | #if !defined(PRODUCT) && !defined(DART_PRECOMPILED_RUNTIME) | 
|  | TypedDataViewPtr constant_coverage() const; | 
|  | #endif  // !defined(PRODUCT) && !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | LibraryPtr FindLibrary() const; | 
|  | StringPtr GetLine(intptr_t line_number, Heap::Space space = Heap::kNew) const; | 
|  | StringPtr GetSnippet(intptr_t from_line, | 
|  | intptr_t from_column, | 
|  | intptr_t to_line, | 
|  | intptr_t to_column) const; | 
|  |  | 
|  | // For real token positions when line starts are available, returns whether or | 
|  | // not a GetTokenLocation call would succeed. Returns true for non-real token | 
|  | // positions or if there is no line starts information. | 
|  | bool IsValidTokenPosition(TokenPosition token_pos) const; | 
|  |  | 
|  | // Returns whether a line and column could be computed for the given token | 
|  | // position and, if so, sets *line and *column (if not nullptr). | 
|  | bool GetTokenLocation(const TokenPosition& token_pos, | 
|  | intptr_t* line, | 
|  | intptr_t* column = nullptr) const; | 
|  |  | 
|  | // Returns the length of the token at the given position. If the length cannot | 
|  | // be determined, returns a negative value. | 
|  | intptr_t GetTokenLength(const TokenPosition& token_pos) const; | 
|  |  | 
|  | // Returns whether any tokens were found for the given line. When found, | 
|  | // *first_token_index and *last_token_index are set to the first and | 
|  | // last token on the line, respectively. | 
|  | bool TokenRangeAtLine(intptr_t line_number, | 
|  | TokenPosition* first_token_index, | 
|  | TokenPosition* last_token_index) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedScript)); | 
|  | } | 
|  |  | 
|  | static ScriptPtr New(const String& url, const String& source); | 
|  |  | 
|  | static ScriptPtr New(const String& url, | 
|  | const String& resolved_url, | 
|  | const String& source); | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void LoadSourceFromKernel(const uint8_t* kernel_buffer, | 
|  | intptr_t kernel_buffer_len) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | void CollectTokenPositionsFor() const; | 
|  | ArrayPtr CollectConstConstructorCoverageFrom() const; | 
|  |  | 
|  | private: | 
|  | KernelProgramInfoPtr kernel_program_info() const { | 
|  | return untag()->kernel_program_info(); | 
|  | } | 
|  |  | 
|  | void set_debug_positions(const Array& value) const; | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | bool HasCachedMaxPosition() const; | 
|  |  | 
|  | void SetHasCachedMaxPosition(bool value) const; | 
|  | void SetCachedMaxPosition(intptr_t value) const; | 
|  | #endif  // !defined(DART_PRECOMPILED_RUNTIME) | 
|  |  | 
|  | void set_resolved_url(const String& value) const; | 
|  | void set_source(const String& value) const; | 
|  | void set_load_timestamp(int64_t value) const; | 
|  | ArrayPtr debug_positions() const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Script, Object); | 
|  | friend class Class; | 
|  | friend class Precompiler; | 
|  | }; | 
|  |  | 
|  | class DictionaryIterator : public ValueObject { | 
|  | public: | 
|  | explicit DictionaryIterator(const Library& library); | 
|  |  | 
|  | bool HasNext() const { return next_ix_ < size_; } | 
|  |  | 
|  | // Returns next non-null raw object. | 
|  | ObjectPtr GetNext(); | 
|  |  | 
|  | private: | 
|  | void MoveToNextObject(); | 
|  |  | 
|  | const Array& array_; | 
|  | const int size_;  // Number of elements to iterate over. | 
|  | int next_ix_;     // Index of next element. | 
|  |  | 
|  | friend class ClassDictionaryIterator; | 
|  | DISALLOW_COPY_AND_ASSIGN(DictionaryIterator); | 
|  | }; | 
|  |  | 
|  | class ClassDictionaryIterator : public DictionaryIterator { | 
|  | public: | 
|  | enum IterationKind { | 
|  | // TODO(hausner): fix call sites that use kIteratePrivate. There is only | 
|  | // one top-level class per library left, not an array to iterate over. | 
|  | kIteratePrivate, | 
|  | kNoIteratePrivate | 
|  | }; | 
|  |  | 
|  | ClassDictionaryIterator(const Library& library, | 
|  | IterationKind kind = kNoIteratePrivate); | 
|  |  | 
|  | bool HasNext() const { | 
|  | return (next_ix_ < size_) || !toplevel_class_.IsNull(); | 
|  | } | 
|  |  | 
|  | // Returns a non-null raw class. | 
|  | ClassPtr GetNextClass(); | 
|  |  | 
|  | private: | 
|  | void MoveToNextClass(); | 
|  |  | 
|  | Class& toplevel_class_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(ClassDictionaryIterator); | 
|  | }; | 
|  |  | 
|  | class Library : public Object { | 
|  | public: | 
|  | StringPtr name() const { return untag()->name(); } | 
|  | void SetName(const String& name) const; | 
|  |  | 
|  | StringPtr url() const { return untag()->url(); } | 
|  | static StringPtr UrlOf(LibraryPtr lib) { return lib->untag()->url(); } | 
|  | StringPtr private_key() const { return untag()->private_key(); } | 
|  | bool LoadNotStarted() const { | 
|  | return untag()->load_state_ == UntaggedLibrary::kAllocated; | 
|  | } | 
|  | bool LoadRequested() const { | 
|  | return untag()->load_state_ == UntaggedLibrary::kLoadRequested; | 
|  | } | 
|  | bool LoadInProgress() const { | 
|  | return untag()->load_state_ == UntaggedLibrary::kLoadInProgress; | 
|  | } | 
|  | void SetLoadRequested() const; | 
|  | void SetLoadInProgress() const; | 
|  | bool Loaded() const { | 
|  | return untag()->load_state_ == UntaggedLibrary::kLoaded; | 
|  | } | 
|  | void SetLoaded() const; | 
|  |  | 
|  | LoadingUnitPtr loading_unit() const { return untag()->loading_unit(); } | 
|  | void set_loading_unit(const LoadingUnit& value) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedLibrary)); | 
|  | } | 
|  |  | 
|  | static LibraryPtr New(const String& url); | 
|  |  | 
|  | ObjectPtr Invoke(const String& selector, | 
|  | const Array& arguments, | 
|  | const Array& argument_names, | 
|  | bool check_is_entrypoint = true, | 
|  | bool respect_reflectable = true) const; | 
|  | ObjectPtr InvokeGetter(const String& selector, | 
|  | bool check_is_entrypoint = true, | 
|  | bool respect_reflectable = true, | 
|  | bool for_invocation = false) const; | 
|  | ObjectPtr InvokeSetter(const String& selector, | 
|  | const Instance& argument, | 
|  | bool check_is_entrypoint = true, | 
|  | bool respect_reflectable = true) const; | 
|  |  | 
|  | // Evaluate the given expression as if it appeared in an top-level method of | 
|  | // this library and return the resulting value, or an error object if | 
|  | // evaluating the expression fails. The method has the formal (type) | 
|  | // parameters given in (type_)param_names, and is invoked with the (type) | 
|  | // argument values given in (type_)param_values. | 
|  | ObjectPtr EvaluateCompiledExpression( | 
|  | const ExternalTypedData& kernel_buffer, | 
|  | const Array& type_definitions, | 
|  | const Array& param_values, | 
|  | const TypeArguments& type_param_values) const; | 
|  |  | 
|  | // Library scope name dictionary. | 
|  | // | 
|  | // TODO(turnidge): The Lookup functions are not consistent in how | 
|  | // they deal with private names.  Go through and make them a bit | 
|  | // more regular. | 
|  | void AddClass(const Class& cls) const; | 
|  | void AddObject(const Object& obj, const String& name) const; | 
|  | ObjectPtr LookupReExport( | 
|  | const String& name, | 
|  | ZoneGrowableArray<intptr_t>* visited = nullptr) const; | 
|  | ObjectPtr LookupLocalOrReExportObject(const String& name) const; | 
|  | LibraryPrefixPtr LookupLocalLibraryPrefix(const String& name) const; | 
|  |  | 
|  | // These lookups are local within the library. | 
|  | ClassPtr LookupClass(const String& name) const; | 
|  | ClassPtr LookupClassAllowPrivate(const String& name) const; | 
|  | FieldPtr LookupFieldAllowPrivate(const String& name) const; | 
|  | FunctionPtr LookupFunctionAllowPrivate(const String& name) const; | 
|  |  | 
|  | // Look up a Script based on a url. If 'useResolvedUri' is not provided or is | 
|  | // false, 'url' should have a 'dart:' scheme for Dart core libraries, | 
|  | // a 'package:' scheme for packages, and 'file:' scheme otherwise. | 
|  | // | 
|  | // If 'useResolvedUri' is true, 'url' should have a 'org-dartlang-sdk:' scheme | 
|  | // for Dart core libraries and a 'file:' scheme otherwise. | 
|  | ScriptPtr LookupScript(const String& url, bool useResolvedUri = false) const; | 
|  | ArrayPtr LoadedScripts() const; | 
|  |  | 
|  | void AddExport(const Namespace& ns) const; | 
|  |  | 
|  | void AddMetadata(const Object& declaration, intptr_t kernel_offset) const; | 
|  | ObjectPtr GetMetadata(const Object& declaration) const; | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void EvaluatePragmas(); | 
|  | void CopyPragmas(const Library& old_lib); | 
|  | #endif | 
|  |  | 
|  | // Tries to finds a @pragma annotation on [object]. | 
|  | // | 
|  | // If successful returns `true`. If an error happens during constant | 
|  | // evaluation, returns `false. | 
|  | // | 
|  | // If [only_core] is true, then the annotations on the object will only | 
|  | // be inspected if it is part of a core library. | 
|  | // | 
|  | // If [multiple] is true, then sets [options] to an GrowableObjectArray | 
|  | // containing all results and [options] may not be nullptr. | 
|  | // | 
|  | // WARNING: If the isolate received an [UnwindError] this function will not | 
|  | // return and rather unwinds until the enclosing setjmp() handler. | 
|  | static bool FindPragma(Thread* T, | 
|  | bool only_core, | 
|  | const Object& object, | 
|  | const String& pragma_name, | 
|  | bool multiple = false, | 
|  | Object* options = nullptr); | 
|  |  | 
|  | ClassPtr toplevel_class() const { return untag()->toplevel_class(); } | 
|  | void set_toplevel_class(const Class& value) const; | 
|  |  | 
|  | GrowableObjectArrayPtr used_scripts() const { | 
|  | return untag()->used_scripts(); | 
|  | } | 
|  |  | 
|  | // Library imports. | 
|  | ArrayPtr imports() const { return untag()->imports(); } | 
|  | ArrayPtr exports() const { return untag()->exports(); } | 
|  | void AddImport(const Namespace& ns) const; | 
|  | intptr_t num_imports() const { return untag()->num_imports_; } | 
|  | NamespacePtr ImportAt(intptr_t index) const; | 
|  | LibraryPtr ImportLibraryAt(intptr_t index) const; | 
|  |  | 
|  | ArrayPtr dependencies() const { return untag()->dependencies(); } | 
|  | void set_dependencies(const Array& deps) const; | 
|  |  | 
|  | void DropDependenciesAndCaches() const; | 
|  |  | 
|  | // Resolving native methods for script loaded in the library. | 
|  | Dart_NativeEntryResolver native_entry_resolver() const { | 
|  | return LoadNonPointer<Dart_NativeEntryResolver, std::memory_order_relaxed>( | 
|  | &untag()->native_entry_resolver_); | 
|  | } | 
|  | void set_native_entry_resolver(Dart_NativeEntryResolver value) const { | 
|  | StoreNonPointer<Dart_NativeEntryResolver, Dart_NativeEntryResolver, | 
|  | std::memory_order_relaxed>(&untag()->native_entry_resolver_, | 
|  | value); | 
|  | } | 
|  | Dart_NativeEntrySymbol native_entry_symbol_resolver() const { | 
|  | return LoadNonPointer<Dart_NativeEntrySymbol, std::memory_order_relaxed>( | 
|  | &untag()->native_entry_symbol_resolver_); | 
|  | } | 
|  | void set_native_entry_symbol_resolver( | 
|  | Dart_NativeEntrySymbol native_symbol_resolver) const { | 
|  | StoreNonPointer<Dart_NativeEntrySymbol, Dart_NativeEntrySymbol, | 
|  | std::memory_order_relaxed>( | 
|  | &untag()->native_entry_symbol_resolver_, native_symbol_resolver); | 
|  | } | 
|  |  | 
|  | // Resolver for FFI native function pointers. | 
|  | Dart_FfiNativeResolver ffi_native_resolver() const { | 
|  | return LoadNonPointer<Dart_FfiNativeResolver, std::memory_order_relaxed>( | 
|  | &untag()->ffi_native_resolver_); | 
|  | } | 
|  | void set_ffi_native_resolver(Dart_FfiNativeResolver value) const { | 
|  | StoreNonPointer<Dart_FfiNativeResolver, Dart_FfiNativeResolver, | 
|  | std::memory_order_relaxed>(&untag()->ffi_native_resolver_, | 
|  | value); | 
|  | } | 
|  |  | 
|  | bool is_in_fullsnapshot() const { | 
|  | return UntaggedLibrary::InFullSnapshotBit::decode(untag()->flags_); | 
|  | } | 
|  | void set_is_in_fullsnapshot(bool value) const { | 
|  | set_flags( | 
|  | UntaggedLibrary::InFullSnapshotBit::update(value, untag()->flags_)); | 
|  | } | 
|  |  | 
|  | StringPtr PrivateName(const String& name) const; | 
|  |  | 
|  | intptr_t index() const { return untag()->index_; } | 
|  | void set_index(intptr_t value) const { | 
|  | ASSERT((value == -1) || | 
|  | ((value >= 0) && (value < std::numeric_limits<classid_t>::max()))); | 
|  | StoreNonPointer(&untag()->index_, value); | 
|  | } | 
|  |  | 
|  | void Register(Thread* thread) const; | 
|  | static void RegisterLibraries(Thread* thread, | 
|  | const GrowableObjectArray& libs); | 
|  |  | 
|  | bool IsDebuggable() const { | 
|  | return UntaggedLibrary::DebuggableBit::decode(untag()->flags_); | 
|  | } | 
|  | void set_debuggable(bool value) const { | 
|  | set_flags(UntaggedLibrary::DebuggableBit::update(value, untag()->flags_)); | 
|  | } | 
|  |  | 
|  | bool is_dart_scheme() const { | 
|  | return UntaggedLibrary::DartSchemeBit::decode(untag()->flags_); | 
|  | } | 
|  | void set_is_dart_scheme(bool value) const { | 
|  | set_flags(UntaggedLibrary::DartSchemeBit::update(value, untag()->flags_)); | 
|  | } | 
|  |  | 
|  | // Includes 'dart:async', 'dart:typed_data', etc. | 
|  | bool IsAnyCoreLibrary() const; | 
|  |  | 
|  | inline intptr_t UrlHash() const; | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | KernelProgramInfoPtr kernel_program_info() const { | 
|  | return untag()->kernel_program_info(); | 
|  | } | 
|  | void set_kernel_program_info(const KernelProgramInfo& info) const; | 
|  | TypedDataViewPtr KernelLibrary() const; | 
|  | intptr_t KernelLibraryOffset() const; | 
|  | #endif | 
|  |  | 
|  | intptr_t kernel_library_index() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return 0; | 
|  | #else | 
|  | return untag()->kernel_library_index_; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void set_kernel_library_index(intptr_t value) const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | #else | 
|  | ASSERT(value >= 0); | 
|  | StoreNonPointer(&untag()->kernel_library_index_, value); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static LibraryPtr LookupLibrary(Thread* thread, const String& url); | 
|  | static LibraryPtr GetLibrary(intptr_t index); | 
|  |  | 
|  | static void InitCoreLibrary(IsolateGroup* isolate_group); | 
|  | static void InitNativeWrappersLibrary(IsolateGroup* isolate_group, | 
|  | bool is_kernel_file); | 
|  |  | 
|  | static LibraryPtr AsyncLibrary(); | 
|  | static LibraryPtr ConcurrentLibrary(); | 
|  | static LibraryPtr ConvertLibrary(); | 
|  | static LibraryPtr CoreLibrary(); | 
|  | static LibraryPtr CollectionLibrary(); | 
|  | static LibraryPtr CompactHashLibrary(); | 
|  | static LibraryPtr DeveloperLibrary(); | 
|  | static LibraryPtr FfiLibrary(); | 
|  | static LibraryPtr InternalLibrary(); | 
|  | static LibraryPtr IsolateLibrary(); | 
|  | static LibraryPtr MathLibrary(); | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | static LibraryPtr MirrorsLibrary(); | 
|  | #endif | 
|  | static LibraryPtr NativeWrappersLibrary(); | 
|  | static LibraryPtr TypedDataLibrary(); | 
|  | static LibraryPtr VMServiceLibrary(); | 
|  |  | 
|  | // Eagerly compile all classes and functions in the library. | 
|  | static ErrorPtr CompileAll(bool ignore_error = false); | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | // Finalize all classes in all libraries. | 
|  | static ErrorPtr FinalizeAllClasses(); | 
|  | #endif | 
|  |  | 
|  | #if defined(DEBUG) && !defined(DART_PRECOMPILED_RUNTIME) | 
|  | // Checks function fingerprints. Prints mismatches and aborts if | 
|  | // mismatch found. | 
|  | static void CheckFunctionFingerprints(); | 
|  | #endif  // defined(DEBUG) && !defined(DART_PRECOMPILED_RUNTIME). | 
|  |  | 
|  | static bool IsPrivate(const String& name); | 
|  |  | 
|  | // Construct the full name of a corelib member. | 
|  | static const String& PrivateCoreLibName(const String& member); | 
|  |  | 
|  | // Returns true if [name] matches full name of corelib [member]. | 
|  | static bool IsPrivateCoreLibName(const String& name, const String& member); | 
|  |  | 
|  | // Lookup class in the core lib which also contains various VM | 
|  | // helper methods and classes. Allow look up of private classes. | 
|  | static ClassPtr LookupCoreClass(const String& class_name); | 
|  |  | 
|  | // Return Function::null() if function does not exist in lib. | 
|  | static FunctionPtr GetFunction(const Library& lib, | 
|  | const char* class_name, | 
|  | const char* function_name); | 
|  |  | 
|  | // Character used to indicate a private identifier. | 
|  | static const char kPrivateIdentifierStart = '_'; | 
|  |  | 
|  | // Character used to separate private identifiers from | 
|  | // the library-specific key. | 
|  | static const char kPrivateKeySeparator = '@'; | 
|  |  | 
|  | void CheckReload(const Library& replacement, | 
|  | ProgramReloadContext* context) const; | 
|  |  | 
|  | // Ensures that all top-level functions and variables (fields) are loaded. | 
|  | void EnsureTopLevelClassIsFinalized() const; | 
|  |  | 
|  | private: | 
|  | static constexpr int kInitialImportsCapacity = 4; | 
|  | static constexpr int kImportsCapacityIncrement = 8; | 
|  |  | 
|  | static LibraryPtr New(); | 
|  |  | 
|  | // These methods are only used by the Precompiler to obfuscate | 
|  | // the name and url. | 
|  | void set_name(const String& name) const; | 
|  | void set_url(const String& url) const; | 
|  | void set_private_key(const String& key) const; | 
|  |  | 
|  | void set_num_imports(intptr_t value) const; | 
|  | void set_flags(uint8_t flags) const; | 
|  | bool HasExports() const; | 
|  | ArrayPtr loaded_scripts() const { return untag()->loaded_scripts(); } | 
|  | ArrayPtr metadata() const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadReader()); | 
|  | return untag()->metadata(); | 
|  | } | 
|  | void set_metadata(const Array& value) const; | 
|  | ArrayPtr dictionary() const { return untag()->dictionary(); } | 
|  | void InitClassDictionary() const; | 
|  |  | 
|  | void InitImportList() const; | 
|  | void RehashDictionary(const Array& old_dict, intptr_t new_dict_size) const; | 
|  | static LibraryPtr NewLibraryHelper(const String& url, bool import_core_lib); | 
|  | ObjectPtr LookupEntry(const String& name, intptr_t* index) const; | 
|  | ObjectPtr LookupLocalObject(const String& name) const; | 
|  | ObjectPtr LookupLocalObjectAllowPrivate(const String& name) const; | 
|  |  | 
|  | void AllocatePrivateKey() const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Library, Object); | 
|  |  | 
|  | friend class Bootstrap; | 
|  | friend class Class; | 
|  | friend class Debugger; | 
|  | friend class DictionaryIterator; | 
|  | friend class Isolate; | 
|  | friend class LibraryDeserializationCluster; | 
|  | friend class Namespace; | 
|  | friend class Object; | 
|  | friend class Precompiler; | 
|  | }; | 
|  |  | 
|  | // A Namespace contains the names in a library dictionary, filtered by | 
|  | // the show/hide combinators. | 
|  | class Namespace : public Object { | 
|  | public: | 
|  | LibraryPtr target() const { return untag()->target(); } | 
|  | ArrayPtr show_names() const { return untag()->show_names(); } | 
|  | ArrayPtr hide_names() const { return untag()->hide_names(); } | 
|  | LibraryPtr owner() const { return untag()->owner(); } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedNamespace)); | 
|  | } | 
|  |  | 
|  | bool HidesName(const String& name) const; | 
|  | ObjectPtr Lookup(const String& name, | 
|  | ZoneGrowableArray<intptr_t>* trail = nullptr) const; | 
|  |  | 
|  | static NamespacePtr New(const Library& library, | 
|  | const Array& show_names, | 
|  | const Array& hide_names, | 
|  | const Library& owner); | 
|  |  | 
|  | private: | 
|  | static NamespacePtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Namespace, Object); | 
|  | friend class Class; | 
|  | friend class Precompiler; | 
|  | }; | 
|  |  | 
|  | class KernelProgramInfo : public Object { | 
|  | public: | 
|  | static KernelProgramInfoPtr New(const TypedDataBase& kernel_component, | 
|  | const TypedDataView& string_data, | 
|  | const TypedDataView& metadata_payload, | 
|  | const TypedDataView& metadata_mappings, | 
|  | const TypedDataView& constants_table, | 
|  | const TypedData& string_offsets, | 
|  | const TypedData& canonical_names, | 
|  | const Array& scripts, | 
|  | const Array& libraries_cache, | 
|  | const Array& classes_cache); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedKernelProgramInfo)); | 
|  | } | 
|  |  | 
|  | TypedDataPtr string_offsets() const { return untag()->string_offsets(); } | 
|  |  | 
|  | TypedDataBasePtr kernel_component() const { | 
|  | return untag()->kernel_component(); | 
|  | } | 
|  | TypedDataViewPtr string_data() const { return untag()->string_data(); } | 
|  |  | 
|  | TypedDataPtr canonical_names() const { return untag()->canonical_names(); } | 
|  |  | 
|  | TypedDataViewPtr metadata_payloads() const { | 
|  | return untag()->metadata_payloads(); | 
|  | } | 
|  |  | 
|  | TypedDataViewPtr metadata_mappings() const { | 
|  | return untag()->metadata_mappings(); | 
|  | } | 
|  |  | 
|  | intptr_t KernelLibraryStartOffset(intptr_t library_index) const; | 
|  | intptr_t KernelLibraryEndOffset(intptr_t library_index) const; | 
|  | TypedDataViewPtr KernelLibrary(intptr_t library_index) const; | 
|  |  | 
|  | TypedDataViewPtr constants_table() const { | 
|  | return untag()->constants_table(); | 
|  | } | 
|  |  | 
|  | void set_constants_table(const TypedDataView& value) const; | 
|  |  | 
|  | ArrayPtr scripts() const { return untag()->scripts(); } | 
|  | void set_scripts(const Array& scripts) const; | 
|  |  | 
|  | ArrayPtr constants() const { return untag()->constants(); } | 
|  | void set_constants(const Array& constants) const; | 
|  |  | 
|  | ScriptPtr ScriptAt(intptr_t index) const; | 
|  |  | 
|  | ArrayPtr libraries_cache() const { return untag()->libraries_cache(); } | 
|  | void set_libraries_cache(const Array& cache) const; | 
|  | LibraryPtr LookupLibrary(Thread* thread, const Smi& name_index) const; | 
|  | LibraryPtr InsertLibrary(Thread* thread, | 
|  | const Smi& name_index, | 
|  | const Library& lib) const; | 
|  |  | 
|  | ArrayPtr classes_cache() const { return untag()->classes_cache(); } | 
|  | void set_classes_cache(const Array& cache) const; | 
|  | ClassPtr LookupClass(Thread* thread, const Smi& name_index) const; | 
|  | ClassPtr InsertClass(Thread* thread, | 
|  | const Smi& name_index, | 
|  | const Class& klass) const; | 
|  |  | 
|  | private: | 
|  | static KernelProgramInfoPtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(KernelProgramInfo, Object); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // ObjectPool contains constants, immediates and addresses referenced by | 
|  | // generated code and deoptimization infos. Each entry has an type associated | 
|  | // with it which is stored in-inline after all the entries. | 
|  | class ObjectPool : public Object { | 
|  | public: | 
|  | using EntryType = compiler::ObjectPoolBuilderEntry::EntryType; | 
|  | using Patchability = compiler::ObjectPoolBuilderEntry::Patchability; | 
|  | using SnapshotBehavior = compiler::ObjectPoolBuilderEntry::SnapshotBehavior; | 
|  | using TypeBits = compiler::ObjectPoolBuilderEntry::TypeBits; | 
|  | using PatchableBit = compiler::ObjectPoolBuilderEntry::PatchableBit; | 
|  | using SnapshotBehaviorBits = | 
|  | compiler::ObjectPoolBuilderEntry::SnapshotBehaviorBits; | 
|  |  | 
|  | struct Entry { | 
|  | Entry() : raw_value_(), type_() {} | 
|  | explicit Entry(const Object* obj) | 
|  | : obj_(obj), type_(EntryType::kTaggedObject) {} | 
|  | Entry(uword value, EntryType info) : raw_value_(value), type_(info) {} | 
|  | union { | 
|  | const Object* obj_; | 
|  | uword raw_value_; | 
|  | }; | 
|  | EntryType type_; | 
|  | }; | 
|  |  | 
|  | intptr_t Length() const { return untag()->length_; } | 
|  | void SetLength(intptr_t value) const { | 
|  | StoreNonPointer(&untag()->length_, value); | 
|  | } | 
|  |  | 
|  | static intptr_t length_offset() { | 
|  | return OFFSET_OF(UntaggedObjectPool, length_); | 
|  | } | 
|  | static intptr_t data_offset() { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedObjectPool, data); | 
|  | } | 
|  | static intptr_t element_offset(intptr_t index) { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedObjectPool, data) + | 
|  | sizeof(UntaggedObjectPool::Entry) * index; | 
|  | } | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { | 
|  | return ObjectPool::data_offset(); | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t kElementSize = sizeof(UntaggedObjectPool::Entry); | 
|  | }; | 
|  |  | 
|  | EntryType TypeAt(intptr_t index) const { | 
|  | ASSERT((index >= 0) && (index <= Length())); | 
|  | return TypeBits::decode(untag()->entry_bits()[index]); | 
|  | } | 
|  |  | 
|  | Patchability PatchableAt(intptr_t index) const { | 
|  | ASSERT((index >= 0) && (index <= Length())); | 
|  | return PatchableBit::decode(untag()->entry_bits()[index]); | 
|  | } | 
|  |  | 
|  | SnapshotBehavior SnapshotBehaviorAt(intptr_t index) const { | 
|  | ASSERT((index >= 0) && (index <= Length())); | 
|  | return SnapshotBehaviorBits::decode(untag()->entry_bits()[index]); | 
|  | } | 
|  |  | 
|  | static uint8_t EncodeBits(EntryType type, | 
|  | Patchability patchable, | 
|  | SnapshotBehavior snapshot_behavior) { | 
|  | return PatchableBit::encode(patchable) | TypeBits::encode(type) | | 
|  | SnapshotBehaviorBits::encode(snapshot_behavior); | 
|  | } | 
|  |  | 
|  | void SetTypeAt(intptr_t index, | 
|  | EntryType type, | 
|  | Patchability patchable, | 
|  | SnapshotBehavior snapshot_behavior) const { | 
|  | ASSERT(index >= 0 && index <= Length()); | 
|  | const uint8_t bits = EncodeBits(type, patchable, snapshot_behavior); | 
|  | StoreNonPointer(&untag()->entry_bits()[index], bits); | 
|  | } | 
|  |  | 
|  | template <std::memory_order order = std::memory_order_relaxed> | 
|  | ObjectPtr ObjectAt(intptr_t index) const { | 
|  | ASSERT(TypeAt(index) == EntryType::kTaggedObject); | 
|  | return LoadPointer<ObjectPtr, order>(&(EntryAddr(index)->raw_obj_)); | 
|  | } | 
|  |  | 
|  | template <std::memory_order order = std::memory_order_relaxed> | 
|  | void SetObjectAt(intptr_t index, const Object& obj) const { | 
|  | ASSERT((TypeAt(index) == EntryType::kTaggedObject) || | 
|  | (TypeAt(index) == EntryType::kImmediate && obj.IsSmi())); | 
|  | StorePointer<ObjectPtr, order>(&EntryAddr(index)->raw_obj_, obj.ptr()); | 
|  | } | 
|  |  | 
|  | uword RawValueAt(intptr_t index) const { | 
|  | ASSERT(TypeAt(index) != EntryType::kTaggedObject); | 
|  | return EntryAddr(index)->raw_value_; | 
|  | } | 
|  | void SetRawValueAt(intptr_t index, uword raw_value) const { | 
|  | ASSERT(TypeAt(index) != EntryType::kTaggedObject); | 
|  | StoreNonPointer(&EntryAddr(index)->raw_value_, raw_value); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedObjectPool) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedObjectPool, data)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t kBytesPerElement = | 
|  | sizeof(UntaggedObjectPool::Entry) + sizeof(uint8_t); | 
|  | static constexpr intptr_t kMaxElements = kSmiMax / kBytesPerElement; | 
|  |  | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | // Ensure that variable length data is not adding to the object length. | 
|  | ASSERT(sizeof(UntaggedObjectPool) == | 
|  | (sizeof(UntaggedObject) + (1 * kWordSize))); | 
|  | ASSERT(0 <= len && len <= kMaxElements); | 
|  | return RoundedAllocationSize(sizeof(UntaggedObjectPool) + | 
|  | (len * kBytesPerElement)); | 
|  | } | 
|  |  | 
|  | static ObjectPoolPtr NewFromBuilder( | 
|  | const compiler::ObjectPoolBuilder& builder); | 
|  | static ObjectPoolPtr New(intptr_t len); | 
|  |  | 
|  | void CopyInto(compiler::ObjectPoolBuilder* builder) const; | 
|  |  | 
|  | // Returns the pool index from the offset relative to a tagged ObjectPoolPtr, | 
|  | // adjusting for the tag-bit. | 
|  | static intptr_t IndexFromOffset(intptr_t offset) { | 
|  | ASSERT( | 
|  | Utils::IsAligned(offset + kHeapObjectTag, compiler::target::kWordSize)); | 
|  | #if defined(DART_PRECOMPILER) | 
|  | return (offset + kHeapObjectTag - | 
|  | compiler::target::ObjectPool::element_offset(0)) / | 
|  | compiler::target::kWordSize; | 
|  | #else | 
|  | return (offset + kHeapObjectTag - element_offset(0)) / kWordSize; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static intptr_t OffsetFromIndex(intptr_t index) { | 
|  | return element_offset(index) - kHeapObjectTag; | 
|  | } | 
|  |  | 
|  | void DebugPrint() const; | 
|  |  | 
|  | private: | 
|  | UntaggedObjectPool::Entry const* EntryAddr(intptr_t index) const { | 
|  | ASSERT((index >= 0) && (index < Length())); | 
|  | return &untag()->data()[index]; | 
|  | } | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(ObjectPool, Object); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | friend class UntaggedObjectPool; | 
|  | }; | 
|  |  | 
|  | class Instructions : public Object { | 
|  | public: | 
|  | #define INSTRUCTIONS_FLAGS_LIST(V)                                             \ | 
|  | V(HasMonomorphicEntry)                                                       \ | 
|  | V(ShouldBeAligned) | 
|  |  | 
|  | enum { | 
|  | #define DEFINE_INSTRUCTIONS_FLAG(Name) k##Name##Index, | 
|  | INSTRUCTIONS_FLAGS_LIST(DEFINE_INSTRUCTIONS_FLAG) | 
|  | #undef DEFINE_INSTRUCTIONS_FLAG | 
|  | }; | 
|  |  | 
|  | #define DEFINE_INSTRUCTIONS_FLAG_HANDLING(Name)                                \ | 
|  | using Name##Bit = BitField<decltype(UntaggedInstructions::size_and_flags_),  \ | 
|  | bool, k##Name##Index>;                            \ | 
|  | bool Name() const { return Name##Bit::decode(untag()->size_and_flags_); }    \ | 
|  | static bool Name(const InstructionsPtr instr) {                              \ | 
|  | return Name##Bit::decode(instr->untag()->size_and_flags_);                 \ | 
|  | } | 
|  |  | 
|  | INSTRUCTIONS_FLAGS_LIST(DEFINE_INSTRUCTIONS_FLAG_HANDLING) | 
|  |  | 
|  | #undef DEFINE_INSTRUCTIONS_FLAG_HANDLING | 
|  |  | 
|  | using SizeBits = BitField<decltype(UntaggedInstructions::size_and_flags_), | 
|  | uint32_t, | 
|  | ShouldBeAlignedBit::kNextBit>; | 
|  |  | 
|  | // Excludes HeaderSize(). | 
|  | intptr_t Size() const { return SizeBits::decode(untag()->size_and_flags_); } | 
|  | static intptr_t Size(const InstructionsPtr instr) { | 
|  | return SizeBits::decode(instr->untag()->size_and_flags_); | 
|  | } | 
|  |  | 
|  | uword PayloadStart() const { return PayloadStart(ptr()); } | 
|  | uword MonomorphicEntryPoint() const { return MonomorphicEntryPoint(ptr()); } | 
|  | uword EntryPoint() const { return EntryPoint(ptr()); } | 
|  | static uword PayloadStart(const InstructionsPtr instr) { | 
|  | return reinterpret_cast<uword>(instr->untag()) + HeaderSize(); | 
|  | } | 
|  |  | 
|  | // Note: We keep the checked entrypoint offsets even (emitting NOPs if | 
|  | // necessary) to allow them to be seen as Smis by the GC. | 
|  | #if defined(TARGET_ARCH_IA32) | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetJIT = 6; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetJIT = 36; | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetAOT = 0; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetAOT = 0; | 
|  | #elif defined(TARGET_ARCH_X64) | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetJIT = 8; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetJIT = 42; | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetAOT = 8; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetAOT = 22; | 
|  | #elif defined(TARGET_ARCH_ARM) | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetJIT = 0; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetJIT = 44; | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetAOT = 0; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetAOT = 16; | 
|  | #elif defined(TARGET_ARCH_ARM64) | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetJIT = 8; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetJIT = 52; | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetAOT = 8; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetAOT = 24; | 
|  | #elif defined(TARGET_ARCH_RISCV32) | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetJIT = 6; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetJIT = 44; | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetAOT = 6; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetAOT = 18; | 
|  | #elif defined(TARGET_ARCH_RISCV64) | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetJIT = 6; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetJIT = 44; | 
|  | static constexpr intptr_t kMonomorphicEntryOffsetAOT = 6; | 
|  | static constexpr intptr_t kPolymorphicEntryOffsetAOT = 18; | 
|  | #else | 
|  | #error Missing entry offsets for current architecture | 
|  | #endif | 
|  |  | 
|  | static uword MonomorphicEntryPoint(const InstructionsPtr instr) { | 
|  | uword entry = PayloadStart(instr); | 
|  | if (HasMonomorphicEntry(instr)) { | 
|  | entry += !FLAG_precompiled_mode ? kMonomorphicEntryOffsetJIT | 
|  | : kMonomorphicEntryOffsetAOT; | 
|  | } | 
|  | return entry; | 
|  | } | 
|  |  | 
|  | static uword EntryPoint(const InstructionsPtr instr) { | 
|  | uword entry = PayloadStart(instr); | 
|  | if (HasMonomorphicEntry(instr)) { | 
|  | entry += !FLAG_precompiled_mode ? kPolymorphicEntryOffsetJIT | 
|  | : kPolymorphicEntryOffsetAOT; | 
|  | } | 
|  | return entry; | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t kMaxElements = | 
|  | (kMaxInt32 - (sizeof(UntaggedInstructions) + sizeof(UntaggedObject) + | 
|  | (2 * kObjectStartAlignment))); | 
|  |  | 
|  | // Currently, we align bare instruction payloads on 4 byte boundaries. | 
|  | // | 
|  | // If we later decide to align on larger boundaries to put entries at the | 
|  | // start of cache lines, make sure to account for entry points that are | 
|  | // _not_ at the start of the payload. | 
|  | static constexpr intptr_t kBarePayloadAlignment = 4; | 
|  |  | 
|  | // When instructions reside in the heap we align the payloads on word | 
|  | // boundaries. | 
|  | static constexpr intptr_t kNonBarePayloadAlignment = kWordSize; | 
|  |  | 
|  | // In the precompiled runtime when running in bare instructions mode, | 
|  | // Instructions objects don't exist, just their bare payloads, so we | 
|  | // mark them as unreachable in that case. | 
|  |  | 
|  | static intptr_t HeaderSize() { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | #endif | 
|  | return Utils::RoundUp(sizeof(UntaggedInstructions), | 
|  | kNonBarePayloadAlignment); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT_EQUAL(sizeof(UntaggedInstructions), | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedInstructions, data)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize(intptr_t size) { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | #endif | 
|  | return RoundedAllocationSize(HeaderSize() + size); | 
|  | } | 
|  |  | 
|  | static InstructionsPtr FromPayloadStart(uword payload_start) { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | #endif | 
|  | return static_cast<InstructionsPtr>(payload_start - HeaderSize() + | 
|  | kHeapObjectTag); | 
|  | } | 
|  |  | 
|  | bool Equals(const Instructions& other) const { | 
|  | return Equals(ptr(), other.ptr()); | 
|  | } | 
|  |  | 
|  | static bool Equals(InstructionsPtr a, InstructionsPtr b) { | 
|  | // This method should only be called on non-null Instructions objects. | 
|  | ASSERT_EQUAL(a->GetClassId(), kInstructionsCid); | 
|  | ASSERT_EQUAL(b->GetClassId(), kInstructionsCid); | 
|  | // Don't include the object header tags wholesale in the comparison, | 
|  | // because the GC tags may differ in JIT mode. In fact, we can skip checking | 
|  | // the object header entirely, as we're guaranteed that the cids match, | 
|  | // because there are no subclasses for the Instructions class, and the sizes | 
|  | // should match if the content size encoded in size_and_flags_ matches. | 
|  | if (a->untag()->size_and_flags_ != b->untag()->size_and_flags_) { | 
|  | return false; | 
|  | } | 
|  | NoSafepointScope no_safepoint; | 
|  | return memcmp(a->untag()->data(), b->untag()->data(), Size(a)) == 0; | 
|  | } | 
|  |  | 
|  | uint32_t Hash() const { return Hash(ptr()); } | 
|  |  | 
|  | static uint32_t Hash(const InstructionsPtr instr) { | 
|  | return HashBytes(reinterpret_cast<const uint8_t*>(PayloadStart(instr)), | 
|  | Size(instr)); | 
|  | } | 
|  |  | 
|  | CodeStatistics* stats() const; | 
|  | void set_stats(CodeStatistics* stats) const; | 
|  |  | 
|  | private: | 
|  | friend struct RelocatorTestHelper; | 
|  |  | 
|  | void SetSize(intptr_t value) const { | 
|  | ASSERT(value >= 0); | 
|  | StoreNonPointer(&untag()->size_and_flags_, | 
|  | SizeBits::update(value, untag()->size_and_flags_)); | 
|  | } | 
|  |  | 
|  | #define DEFINE_INSTRUCTIONS_FLAG_HANDLING(Name)                                \ | 
|  | void Set##Name(bool value) const {                                           \ | 
|  | StoreNonPointer(&untag()->size_and_flags_,                                 \ | 
|  | Name##Bit::update(value, untag()->size_and_flags_));       \ | 
|  | } | 
|  |  | 
|  | INSTRUCTIONS_FLAGS_LIST(DEFINE_INSTRUCTIONS_FLAG_HANDLING) | 
|  |  | 
|  | #undef DEFINE_INSTRUCTIONS_FLAG_HANDLING | 
|  |  | 
|  | // New is a private method as RawInstruction and RawCode objects should | 
|  | // only be created using the Code::FinalizeCode method. This method creates | 
|  | // the RawInstruction and RawCode objects, sets up the pointer offsets | 
|  | // and links the two in a GC safe manner. | 
|  | static InstructionsPtr New(intptr_t size, | 
|  | bool has_monomorphic_entry, | 
|  | bool should_be_aligned); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Instructions, Object); | 
|  | friend class Class; | 
|  | friend class Code; | 
|  | friend class AssemblyImageWriter; | 
|  | friend class BlobImageWriter; | 
|  | friend class ImageWriter; | 
|  | }; | 
|  |  | 
|  | // An InstructionsSection contains extra information about serialized AOT | 
|  | // snapshots. | 
|  | // | 
|  | // To avoid changing the embedder to return more information about an AOT | 
|  | // snapshot and possibly disturbing existing clients of that interface, we | 
|  | // serialize a single InstructionsSection object at the start of any text | 
|  | // segments. In bare instructions mode, it also has the benefit of providing | 
|  | // memory accounting for the instructions payloads and avoiding special casing | 
|  | // Images with bare instructions payloads in the GC. Otherwise, it is empty | 
|  | // and the Instructions objects come after it in the Image. | 
|  | class InstructionsSection : public Object { | 
|  | public: | 
|  | // Excludes HeaderSize(). | 
|  | static intptr_t Size(const InstructionsSectionPtr instr) { | 
|  | return instr->untag()->payload_length_; | 
|  | } | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedInstructionsSection) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedInstructionsSection, data)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize(intptr_t size) { | 
|  | return Utils::RoundUp(HeaderSize() + size, kObjectAlignment); | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t kPayloadAlignment = 32; | 
|  | static_assert(kPreferredLoopAlignment <= kPayloadAlignment); | 
|  | static_assert(Instructions::kBarePayloadAlignment <= kPayloadAlignment); | 
|  |  | 
|  | static intptr_t HeaderSize() { | 
|  | return Utils::RoundUp(sizeof(UntaggedInstructionsSection), | 
|  | kPayloadAlignment); | 
|  | } | 
|  |  | 
|  | // There are no public instance methods for the InstructionsSection class, as | 
|  | // all access to the contents is handled by methods on the Image class. | 
|  |  | 
|  | private: | 
|  | // Note there are no New() methods for InstructionsSection. Instead, the | 
|  | // serializer writes the UntaggedInstructionsSection object manually at the | 
|  | // start of instructions Images in precompiled snapshots. | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(InstructionsSection, Object); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Table which maps ranges of machine code to [Code] or | 
|  | // [CompressedStackMaps] objects. | 
|  | // Used in AOT in bare instructions mode. | 
|  | class InstructionsTable : public Object { | 
|  | public: | 
|  | static intptr_t InstanceSize() { return sizeof(UntaggedInstructionsTable); } | 
|  |  | 
|  | static InstructionsTablePtr New(intptr_t length, | 
|  | uword start_pc, | 
|  | uword end_pc, | 
|  | uword rodata); | 
|  |  | 
|  | void SetCodeAt(intptr_t index, CodePtr code) const; | 
|  |  | 
|  | bool ContainsPc(uword pc) const { return ContainsPc(ptr(), pc); } | 
|  | static bool ContainsPc(InstructionsTablePtr table, uword pc); | 
|  |  | 
|  | static CodePtr FindCode(InstructionsTablePtr table, uword pc); | 
|  |  | 
|  | static const UntaggedCompressedStackMaps::Payload* | 
|  | FindStackMap(InstructionsTablePtr table, uword pc, uword* start_pc); | 
|  |  | 
|  | static const UntaggedCompressedStackMaps::Payload* GetCanonicalStackMap( | 
|  | InstructionsTablePtr table); | 
|  |  | 
|  | const UntaggedInstructionsTable::Data* rodata() const { | 
|  | return ptr()->untag()->rodata_; | 
|  | } | 
|  |  | 
|  | // Returns start address of the instructions entry with given index. | 
|  | uword PayloadStartAt(intptr_t index) const { | 
|  | return InstructionsTable::PayloadStartAt(this->ptr(), index); | 
|  | } | 
|  | static uword PayloadStartAt(InstructionsTablePtr table, intptr_t index); | 
|  |  | 
|  | // Returns entry point of the instructions with given index. | 
|  | uword EntryPointAt(intptr_t index) const; | 
|  |  | 
|  | private: | 
|  | uword start_pc() const { return InstructionsTable::start_pc(this->ptr()); } | 
|  | static uword start_pc(InstructionsTablePtr table) { | 
|  | return table->untag()->start_pc_; | 
|  | } | 
|  |  | 
|  | uword end_pc() const { return InstructionsTable::end_pc(this->ptr()); } | 
|  | static uword end_pc(InstructionsTablePtr table) { | 
|  | return table->untag()->end_pc_; | 
|  | } | 
|  |  | 
|  | ArrayPtr code_objects() const { return untag()->code_objects_; } | 
|  |  | 
|  | void set_length(intptr_t value) const; | 
|  | void set_start_pc(uword value) const; | 
|  | void set_end_pc(uword value) const; | 
|  | void set_code_objects(const Array& value) const; | 
|  | void set_rodata(uword rodata) const; | 
|  |  | 
|  | uint32_t ConvertPcToOffset(uword pc) const { | 
|  | return InstructionsTable::ConvertPcToOffset(this->ptr(), pc); | 
|  | } | 
|  | static uint32_t ConvertPcToOffset(InstructionsTablePtr table, uword pc); | 
|  |  | 
|  | static intptr_t FindEntry(InstructionsTablePtr table, | 
|  | uword pc, | 
|  | intptr_t start_index = 0); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(InstructionsTable, Object); | 
|  | friend class Class; | 
|  | friend class Deserializer; | 
|  | }; | 
|  |  | 
|  | class LocalVarDescriptors : public Object { | 
|  | public: | 
|  | intptr_t Length() const; | 
|  |  | 
|  | StringPtr GetName(intptr_t var_index) const; | 
|  |  | 
|  | void SetVar(intptr_t var_index, | 
|  | const String& name, | 
|  | UntaggedLocalVarDescriptors::VarInfo* info) const; | 
|  |  | 
|  | void GetInfo(intptr_t var_index, | 
|  | UntaggedLocalVarDescriptors::VarInfo* info) const; | 
|  |  | 
|  | static constexpr intptr_t kBytesPerElement = | 
|  | sizeof(UntaggedLocalVarDescriptors::VarInfo); | 
|  | static constexpr intptr_t kMaxElements = | 
|  | UntaggedLocalVarDescriptors::VarInfo::kMaxIndex; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedLocalVarDescriptors) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedLocalVarDescriptors, names)); | 
|  | return 0; | 
|  | } | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | ASSERT(0 <= len && len <= kMaxElements); | 
|  | return RoundedAllocationSize( | 
|  | sizeof(UntaggedLocalVarDescriptors) + | 
|  | (len * kWordSize)  // RawStrings for names. | 
|  | + (len * sizeof(UntaggedLocalVarDescriptors::VarInfo))); | 
|  | } | 
|  |  | 
|  | static LocalVarDescriptorsPtr New(intptr_t num_variables); | 
|  |  | 
|  | static const char* KindToCString( | 
|  | UntaggedLocalVarDescriptors::VarInfoKind kind); | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(LocalVarDescriptors, Object); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | class PcDescriptors : public Object { | 
|  | public: | 
|  | static constexpr intptr_t kBytesPerElement = 1; | 
|  | static constexpr intptr_t kMaxElements = kMaxInt32 / kBytesPerElement; | 
|  |  | 
|  | static intptr_t HeaderSize() { return sizeof(UntaggedPcDescriptors); } | 
|  | static intptr_t UnroundedSize(PcDescriptorsPtr desc) { | 
|  | return UnroundedSize(desc->untag()->length_); | 
|  | } | 
|  | static intptr_t UnroundedSize(intptr_t len) { return HeaderSize() + len; } | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT_EQUAL(sizeof(UntaggedPcDescriptors), | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedPcDescriptors, data)); | 
|  | return 0; | 
|  | } | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | ASSERT(0 <= len && len <= kMaxElements); | 
|  | return RoundedAllocationSize(UnroundedSize(len)); | 
|  | } | 
|  |  | 
|  | static PcDescriptorsPtr New(const void* delta_encoded_data, intptr_t size); | 
|  |  | 
|  | // Verify (assert) assumptions about pc descriptors in debug mode. | 
|  | void Verify(const Function& function) const; | 
|  |  | 
|  | void PrintToJSONObject(JSONObject* jsobj, bool ref) const; | 
|  |  | 
|  | // We would have a VisitPointers function here to traverse the | 
|  | // pc descriptors table to visit objects if any in the table. | 
|  | // Note: never return a reference to a UntaggedPcDescriptors::PcDescriptorRec | 
|  | // as the object can move. | 
|  | class Iterator : public ValueObject { | 
|  | public: | 
|  | Iterator(const PcDescriptors& descriptors, intptr_t kind_mask) | 
|  | : descriptors_(descriptors), | 
|  | kind_mask_(kind_mask), | 
|  | byte_index_(0), | 
|  | cur_pc_offset_(0), | 
|  | cur_kind_(0), | 
|  | cur_deopt_id_(0), | 
|  | cur_token_pos_(0), | 
|  | cur_try_index_(0), | 
|  | cur_yield_index_(UntaggedPcDescriptors::kInvalidYieldIndex) {} | 
|  |  | 
|  | bool MoveNext() { | 
|  | NoSafepointScope scope; | 
|  | ReadStream stream(descriptors_.untag()->data(), descriptors_.Length(), | 
|  | byte_index_); | 
|  | // Moves to record that matches kind_mask_. | 
|  | while (byte_index_ < descriptors_.Length()) { | 
|  | const int32_t kind_and_metadata = stream.ReadSLEB128<int32_t>(); | 
|  | cur_kind_ = UntaggedPcDescriptors::KindAndMetadata::DecodeKind( | 
|  | kind_and_metadata); | 
|  | cur_try_index_ = UntaggedPcDescriptors::KindAndMetadata::DecodeTryIndex( | 
|  | kind_and_metadata); | 
|  | cur_yield_index_ = | 
|  | UntaggedPcDescriptors::KindAndMetadata::DecodeYieldIndex( | 
|  | kind_and_metadata); | 
|  |  | 
|  | cur_pc_offset_ += stream.ReadSLEB128(); | 
|  |  | 
|  | if (!FLAG_precompiled_mode) { | 
|  | cur_deopt_id_ += stream.ReadSLEB128(); | 
|  | cur_token_pos_ = Utils::AddWithWrapAround( | 
|  | cur_token_pos_, stream.ReadSLEB128<int32_t>()); | 
|  | } | 
|  | byte_index_ = stream.Position(); | 
|  |  | 
|  | if ((cur_kind_ & kind_mask_) != 0) { | 
|  | return true;  // Current is valid. | 
|  | } | 
|  | } | 
|  | return false; | 
|  | } | 
|  |  | 
|  | uword PcOffset() const { return cur_pc_offset_; } | 
|  | intptr_t DeoptId() const { return cur_deopt_id_; } | 
|  | TokenPosition TokenPos() const { | 
|  | return TokenPosition::Deserialize(cur_token_pos_); | 
|  | } | 
|  | intptr_t TryIndex() const { return cur_try_index_; } | 
|  | intptr_t YieldIndex() const { return cur_yield_index_; } | 
|  | UntaggedPcDescriptors::Kind Kind() const { | 
|  | return static_cast<UntaggedPcDescriptors::Kind>(cur_kind_); | 
|  | } | 
|  |  | 
|  | private: | 
|  | friend class PcDescriptors; | 
|  |  | 
|  | // For nested iterations, starting at element after. | 
|  | explicit Iterator(const Iterator& iter) | 
|  | : ValueObject(), | 
|  | descriptors_(iter.descriptors_), | 
|  | kind_mask_(iter.kind_mask_), | 
|  | byte_index_(iter.byte_index_), | 
|  | cur_pc_offset_(iter.cur_pc_offset_), | 
|  | cur_kind_(iter.cur_kind_), | 
|  | cur_deopt_id_(iter.cur_deopt_id_), | 
|  | cur_token_pos_(iter.cur_token_pos_), | 
|  | cur_try_index_(iter.cur_try_index_), | 
|  | cur_yield_index_(iter.cur_yield_index_) {} | 
|  |  | 
|  | const PcDescriptors& descriptors_; | 
|  | const intptr_t kind_mask_; | 
|  | intptr_t byte_index_; | 
|  |  | 
|  | intptr_t cur_pc_offset_; | 
|  | intptr_t cur_kind_; | 
|  | intptr_t cur_deopt_id_; | 
|  | int32_t cur_token_pos_; | 
|  | intptr_t cur_try_index_; | 
|  | intptr_t cur_yield_index_; | 
|  | }; | 
|  |  | 
|  | intptr_t Length() const; | 
|  | bool Equals(const PcDescriptors& other) const { | 
|  | if (Length() != other.Length()) { | 
|  | return false; | 
|  | } | 
|  | NoSafepointScope no_safepoint; | 
|  | return memcmp(untag()->data(), other.untag()->data(), Length()) == 0; | 
|  | } | 
|  |  | 
|  | // Writes the contents of the PcDescriptors object to the given buffer. | 
|  | // The base argument is added to the PC offset for each entry. | 
|  | void WriteToBuffer(BaseTextBuffer* buffer, uword base) const; | 
|  |  | 
|  | private: | 
|  | static const char* KindAsStr(UntaggedPcDescriptors::Kind kind); | 
|  |  | 
|  | static PcDescriptorsPtr New(intptr_t length); | 
|  |  | 
|  | void SetLength(intptr_t value) const; | 
|  | void CopyData(const void* bytes, intptr_t size); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(PcDescriptors, Object); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | class CodeSourceMap : public Object { | 
|  | public: | 
|  | static constexpr intptr_t kBytesPerElement = 1; | 
|  | static constexpr intptr_t kMaxElements = kMaxInt32 / kBytesPerElement; | 
|  |  | 
|  | static intptr_t HeaderSize() { return sizeof(UntaggedCodeSourceMap); } | 
|  | static intptr_t UnroundedSize(CodeSourceMapPtr map) { | 
|  | return UnroundedSize(map->untag()->length_); | 
|  | } | 
|  | static intptr_t UnroundedSize(intptr_t len) { return HeaderSize() + len; } | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT_EQUAL(sizeof(UntaggedCodeSourceMap), | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedCodeSourceMap, data)); | 
|  | return 0; | 
|  | } | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | ASSERT(0 <= len && len <= kMaxElements); | 
|  | return RoundedAllocationSize(UnroundedSize(len)); | 
|  | } | 
|  |  | 
|  | static CodeSourceMapPtr New(intptr_t length); | 
|  |  | 
|  | intptr_t Length() const { return untag()->length_; } | 
|  | uint8_t* Data() const { return UnsafeMutableNonPointer(&untag()->data()[0]); } | 
|  |  | 
|  | bool Equals(const CodeSourceMap& other) const { | 
|  | if (Length() != other.Length()) { | 
|  | return false; | 
|  | } | 
|  | NoSafepointScope no_safepoint; | 
|  | return memcmp(untag()->data(), other.untag()->data(), Length()) == 0; | 
|  | } | 
|  |  | 
|  | uint32_t Hash() const { | 
|  | NoSafepointScope no_safepoint; | 
|  | return HashBytes(Data(), Length()); | 
|  | } | 
|  |  | 
|  | void PrintToJSONObject(JSONObject* jsobj, bool ref) const; | 
|  |  | 
|  | private: | 
|  | void SetLength(intptr_t value) const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(CodeSourceMap, Object); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | class CompressedStackMaps : public Object { | 
|  | public: | 
|  | uintptr_t payload_size() const { return PayloadSizeOf(ptr()); } | 
|  | static uintptr_t PayloadSizeOf(const CompressedStackMapsPtr raw) { | 
|  | return UntaggedCompressedStackMaps::SizeField::decode( | 
|  | raw->untag()->payload()->flags_and_size()); | 
|  | } | 
|  |  | 
|  | const uint8_t* data() const { return ptr()->untag()->payload()->data(); } | 
|  |  | 
|  | // Methods to allow use with PointerKeyValueTrait to create sets of CSMs. | 
|  | bool Equals(const CompressedStackMaps& other) const { | 
|  | // All of the table flags and payload size must match. | 
|  | if (untag()->payload()->flags_and_size() != | 
|  | other.untag()->payload()->flags_and_size()) { | 
|  | return false; | 
|  | } | 
|  | NoSafepointScope no_safepoint; | 
|  | return memcmp(data(), other.data(), payload_size()) == 0; | 
|  | } | 
|  | uword Hash() const; | 
|  |  | 
|  | static intptr_t HeaderSize() { | 
|  | return sizeof(UntaggedCompressedStackMaps) + | 
|  | sizeof(UntaggedCompressedStackMaps::Payload::FlagsAndSizeHeader); | 
|  | } | 
|  | static intptr_t UnroundedSize(CompressedStackMapsPtr maps) { | 
|  | return UnroundedSize(CompressedStackMaps::PayloadSizeOf(maps)); | 
|  | } | 
|  | static intptr_t UnroundedSize(intptr_t length) { | 
|  | return HeaderSize() + length; | 
|  | } | 
|  | static intptr_t InstanceSize() { return 0; } | 
|  | static intptr_t InstanceSize(intptr_t length) { | 
|  | return RoundedAllocationSize(UnroundedSize(length)); | 
|  | } | 
|  |  | 
|  | bool UsesGlobalTable() const { return UsesGlobalTable(ptr()); } | 
|  | static bool UsesGlobalTable(const CompressedStackMapsPtr raw) { | 
|  | return UntaggedCompressedStackMaps::UsesTableBit::decode( | 
|  | raw->untag()->payload()->flags_and_size()); | 
|  | } | 
|  |  | 
|  | bool IsGlobalTable() const { return IsGlobalTable(ptr()); } | 
|  | static bool IsGlobalTable(const CompressedStackMapsPtr raw) { | 
|  | return UntaggedCompressedStackMaps::GlobalTableBit::decode( | 
|  | raw->untag()->payload()->flags_and_size()); | 
|  | } | 
|  |  | 
|  | static CompressedStackMapsPtr NewInlined(const void* payload, intptr_t size) { | 
|  | return New(payload, size, /*is_global_table=*/false, | 
|  | /*uses_global_table=*/false); | 
|  | } | 
|  | static CompressedStackMapsPtr NewUsingTable(const void* payload, | 
|  | intptr_t size) { | 
|  | return New(payload, size, /*is_global_table=*/false, | 
|  | /*uses_global_table=*/true); | 
|  | } | 
|  |  | 
|  | static CompressedStackMapsPtr NewGlobalTable(const void* payload, | 
|  | intptr_t size) { | 
|  | return New(payload, size, /*is_global_table=*/true, | 
|  | /*uses_global_table=*/false); | 
|  | } | 
|  |  | 
|  | class RawPayloadHandle { | 
|  | public: | 
|  | RawPayloadHandle() {} | 
|  | RawPayloadHandle(const RawPayloadHandle&) = default; | 
|  | RawPayloadHandle& operator=(const RawPayloadHandle&) = default; | 
|  |  | 
|  | const UntaggedCompressedStackMaps::Payload* payload() const { | 
|  | return payload_; | 
|  | } | 
|  | bool IsNull() const { return payload_ == nullptr; } | 
|  |  | 
|  | RawPayloadHandle& operator=( | 
|  | const UntaggedCompressedStackMaps::Payload* payload) { | 
|  | payload_ = payload; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | RawPayloadHandle& operator=(const CompressedStackMaps& maps) { | 
|  | ASSERT(!maps.IsNull()); | 
|  | payload_ = maps.untag()->payload(); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | RawPayloadHandle& operator=(CompressedStackMapsPtr maps) { | 
|  | ASSERT(maps != CompressedStackMaps::null()); | 
|  | payload_ = maps.untag()->payload(); | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | uintptr_t payload_size() const { | 
|  | return UntaggedCompressedStackMaps::SizeField::decode( | 
|  | payload()->flags_and_size()); | 
|  | } | 
|  | const uint8_t* data() const { return payload()->data(); } | 
|  |  | 
|  | bool UsesGlobalTable() const { | 
|  | return UntaggedCompressedStackMaps::UsesTableBit::decode( | 
|  | payload()->flags_and_size()); | 
|  | } | 
|  |  | 
|  | bool IsGlobalTable() const { | 
|  | return UntaggedCompressedStackMaps::GlobalTableBit::decode( | 
|  | payload()->flags_and_size()); | 
|  | } | 
|  |  | 
|  | private: | 
|  | const UntaggedCompressedStackMaps::Payload* payload_ = nullptr; | 
|  | }; | 
|  |  | 
|  | template <typename PayloadHandle> | 
|  | class Iterator { | 
|  | public: | 
|  | Iterator(const PayloadHandle& maps, const PayloadHandle& global_table) | 
|  | : maps_(maps), | 
|  | bits_container_(maps.UsesGlobalTable() ? global_table : maps) { | 
|  | ASSERT(!maps_.IsNull()); | 
|  | ASSERT(!bits_container_.IsNull()); | 
|  | ASSERT(!maps_.IsGlobalTable()); | 
|  | ASSERT(!maps_.UsesGlobalTable() || bits_container_.IsGlobalTable()); | 
|  | } | 
|  |  | 
|  | Iterator(const Iterator& it) | 
|  | : maps_(it.maps_), | 
|  | bits_container_(it.bits_container_), | 
|  | next_offset_(it.next_offset_), | 
|  | current_pc_offset_(it.current_pc_offset_), | 
|  | current_global_table_offset_(it.current_global_table_offset_), | 
|  | current_spill_slot_bit_count_(it.current_spill_slot_bit_count_), | 
|  | current_non_spill_slot_bit_count_(it.current_spill_slot_bit_count_), | 
|  | current_bits_offset_(it.current_bits_offset_) {} | 
|  |  | 
|  | // Loads the next entry from [maps_], if any. If [maps_] is the null value, | 
|  | // this always returns false. | 
|  | bool MoveNext() { | 
|  | if (next_offset_ >= maps_.payload_size()) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | NoSafepointScope scope; | 
|  | ReadStream stream(maps_.data(), maps_.payload_size(), next_offset_); | 
|  |  | 
|  | auto const pc_delta = stream.ReadLEB128(); | 
|  | ASSERT(pc_delta <= (kMaxUint32 - current_pc_offset_)); | 
|  | current_pc_offset_ += pc_delta; | 
|  |  | 
|  | // Table-using CSMs have a table offset after the PC offset delta, whereas | 
|  | // the post-delta part of inlined entries has the same information as | 
|  | // global table entries. | 
|  | // See comments in UntaggedCompressedStackMaps for description of | 
|  | // encoding. | 
|  | if (maps_.UsesGlobalTable()) { | 
|  | current_global_table_offset_ = stream.ReadLEB128(); | 
|  | ASSERT(current_global_table_offset_ < bits_container_.payload_size()); | 
|  |  | 
|  | // Since generally we only use entries in the GC and the GC only needs | 
|  | // the rest of the entry information if the PC offset matches, we lazily | 
|  | // load and cache the information stored in the global object when it is | 
|  | // actually requested. | 
|  | current_spill_slot_bit_count_ = -1; | 
|  | current_non_spill_slot_bit_count_ = -1; | 
|  | current_bits_offset_ = -1; | 
|  |  | 
|  | next_offset_ = stream.Position(); | 
|  | } else { | 
|  | current_spill_slot_bit_count_ = stream.ReadLEB128(); | 
|  | ASSERT(current_spill_slot_bit_count_ >= 0); | 
|  |  | 
|  | current_non_spill_slot_bit_count_ = stream.ReadLEB128(); | 
|  | ASSERT(current_non_spill_slot_bit_count_ >= 0); | 
|  |  | 
|  | const auto stackmap_bits = | 
|  | current_spill_slot_bit_count_ + current_non_spill_slot_bit_count_; | 
|  | const uintptr_t stackmap_size = | 
|  | Utils::RoundUp(stackmap_bits, kBitsPerByte) >> kBitsPerByteLog2; | 
|  | ASSERT(stackmap_size <= (maps_.payload_size() - stream.Position())); | 
|  |  | 
|  | current_bits_offset_ = stream.Position(); | 
|  | next_offset_ = current_bits_offset_ + stackmap_size; | 
|  | } | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | // Finds the entry with the given PC offset starting at the current position | 
|  | // of the iterator. If [maps_] is the null value, this always returns false. | 
|  | bool Find(uint32_t pc_offset) { | 
|  | // We should never have an entry with a PC offset of 0 inside an | 
|  | // non-empty CSM, so fail. | 
|  | if (pc_offset == 0) return false; | 
|  | do { | 
|  | if (current_pc_offset_ >= pc_offset) break; | 
|  | } while (MoveNext()); | 
|  | return current_pc_offset_ == pc_offset; | 
|  | } | 
|  |  | 
|  | // Methods for accessing parts of an entry should not be called until | 
|  | // a successful MoveNext() or Find() call has been made. | 
|  |  | 
|  | // Returns the PC offset of the loaded entry. | 
|  | uint32_t pc_offset() const { | 
|  | ASSERT(HasLoadedEntry()); | 
|  | return current_pc_offset_; | 
|  | } | 
|  |  | 
|  | // Returns the bit length of the loaded entry. | 
|  | intptr_t Length() const { | 
|  | EnsureFullyLoadedEntry(); | 
|  | return current_spill_slot_bit_count_ + current_non_spill_slot_bit_count_; | 
|  | } | 
|  | // Returns the number of spill slot bits of the loaded entry. | 
|  | intptr_t SpillSlotBitCount() const { | 
|  | EnsureFullyLoadedEntry(); | 
|  | return current_spill_slot_bit_count_; | 
|  | } | 
|  | // Returns whether the stack entry represented by the offset contains | 
|  | // a tagged object. | 
|  | bool IsObject(intptr_t bit_index) const { | 
|  | EnsureFullyLoadedEntry(); | 
|  | ASSERT(bit_index >= 0 && bit_index < Length()); | 
|  | const intptr_t byte_index = bit_index >> kBitsPerByteLog2; | 
|  | const intptr_t bit_remainder = bit_index & (kBitsPerByte - 1); | 
|  | uint8_t byte_mask = 1U << bit_remainder; | 
|  | const intptr_t byte_offset = current_bits_offset_ + byte_index; | 
|  | NoSafepointScope scope; | 
|  | return (bits_container_.data()[byte_offset] & byte_mask) != 0; | 
|  | } | 
|  |  | 
|  | private: | 
|  | bool HasLoadedEntry() const { return next_offset_ > 0; } | 
|  |  | 
|  | // Caches the corresponding values from the global table in the mutable | 
|  | // fields. We lazily load these as some clients only need the PC offset. | 
|  | void LazyLoadGlobalTableEntry() const { | 
|  | ASSERT(maps_.UsesGlobalTable()); | 
|  | ASSERT(HasLoadedEntry()); | 
|  | ASSERT(current_global_table_offset_ < bits_container_.payload_size()); | 
|  |  | 
|  | NoSafepointScope scope; | 
|  | ReadStream stream(bits_container_.data(), bits_container_.payload_size(), | 
|  | current_global_table_offset_); | 
|  |  | 
|  | current_spill_slot_bit_count_ = stream.ReadLEB128(); | 
|  | ASSERT(current_spill_slot_bit_count_ >= 0); | 
|  |  | 
|  | current_non_spill_slot_bit_count_ = stream.ReadLEB128(); | 
|  | ASSERT(current_non_spill_slot_bit_count_ >= 0); | 
|  |  | 
|  | const auto stackmap_bits = Length(); | 
|  | const uintptr_t stackmap_size = | 
|  | Utils::RoundUp(stackmap_bits, kBitsPerByte) >> kBitsPerByteLog2; | 
|  | ASSERT(stackmap_size <= | 
|  | (bits_container_.payload_size() - stream.Position())); | 
|  |  | 
|  | current_bits_offset_ = stream.Position(); | 
|  | } | 
|  |  | 
|  | void EnsureFullyLoadedEntry() const { | 
|  | ASSERT(HasLoadedEntry()); | 
|  | if (current_spill_slot_bit_count_ < 0) { | 
|  | LazyLoadGlobalTableEntry(); | 
|  | ASSERT(current_spill_slot_bit_count_ >= 0); | 
|  | } | 
|  | } | 
|  |  | 
|  | const PayloadHandle& maps_; | 
|  | const PayloadHandle& bits_container_; | 
|  |  | 
|  | uintptr_t next_offset_ = 0; | 
|  | uint32_t current_pc_offset_ = 0; | 
|  | // Only used when looking up non-PC information in the global table. | 
|  | uintptr_t current_global_table_offset_ = 0; | 
|  | // Marked as mutable as these fields may be updated with lazily loaded | 
|  | // values from the global table when their associated accessor is called, | 
|  | // but those values will never change for a given entry once loaded.. | 
|  | mutable intptr_t current_spill_slot_bit_count_ = -1; | 
|  | mutable intptr_t current_non_spill_slot_bit_count_ = -1; | 
|  | mutable intptr_t current_bits_offset_ = -1; | 
|  |  | 
|  | friend class StackMapEntry; | 
|  | }; | 
|  |  | 
|  | Iterator<CompressedStackMaps> iterator(Thread* thread) const; | 
|  |  | 
|  | // Writes the contents of the CompressedStackMaps object to the given buffer. | 
|  | // The base argument is added to the PC offset for each entry, and the | 
|  | // separator string is inserted between each entry. | 
|  | void WriteToBuffer(BaseTextBuffer* buffer, | 
|  | uword base, | 
|  | const char* separator) const; | 
|  |  | 
|  | private: | 
|  | static CompressedStackMapsPtr New(const void* payload, | 
|  | intptr_t size, | 
|  | bool is_global_table, | 
|  | bool uses_global_table); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(CompressedStackMaps, Object); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class ExceptionHandlers : public Object { | 
|  | public: | 
|  | static constexpr intptr_t kInvalidPcOffset = 0; | 
|  |  | 
|  | intptr_t num_entries() const; | 
|  |  | 
|  | bool has_async_handler() const; | 
|  | void set_has_async_handler(bool value) const; | 
|  |  | 
|  | void GetHandlerInfo(intptr_t try_index, ExceptionHandlerInfo* info) const; | 
|  |  | 
|  | uword HandlerPCOffset(intptr_t try_index) const; | 
|  | intptr_t OuterTryIndex(intptr_t try_index) const; | 
|  | bool NeedsStackTrace(intptr_t try_index) const; | 
|  | bool IsGenerated(intptr_t try_index) const; | 
|  |  | 
|  | void SetHandlerInfo(intptr_t try_index, | 
|  | intptr_t outer_try_index, | 
|  | uword handler_pc_offset, | 
|  | bool needs_stacktrace, | 
|  | bool has_catch_all, | 
|  | bool is_generated) const; | 
|  |  | 
|  | ArrayPtr GetHandledTypes(intptr_t try_index) const; | 
|  | void SetHandledTypes(intptr_t try_index, const Array& handled_types) const; | 
|  | bool HasCatchAll(intptr_t try_index) const; | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { | 
|  | return sizeof(UntaggedExceptionHandlers); | 
|  | } | 
|  | static constexpr intptr_t kElementSize = sizeof(ExceptionHandlerInfo); | 
|  | }; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedExceptionHandlers) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedExceptionHandlers, data)); | 
|  | return 0; | 
|  | } | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | return RoundedAllocationSize(sizeof(UntaggedExceptionHandlers) + | 
|  | (len * sizeof(ExceptionHandlerInfo))); | 
|  | } | 
|  |  | 
|  | static ExceptionHandlersPtr New(intptr_t num_handlers); | 
|  | static ExceptionHandlersPtr New(const Array& handled_types_data); | 
|  |  | 
|  | // We would have a VisitPointers function here to traverse the | 
|  | // exception handler table to visit objects if any in the table. | 
|  |  | 
|  | // Writes the contents of the ExceptionHandlers object to the given buffer. | 
|  | // The base argument is added to the PC offset for each entry. | 
|  | void WriteToBuffer(BaseTextBuffer* buffer, uword base) const; | 
|  |  | 
|  | private: | 
|  | // Pick somewhat arbitrary maximum number of exception handlers | 
|  | // for a function. This value is used to catch potentially | 
|  | // malicious code. | 
|  | static constexpr intptr_t kMaxHandlers = 1024 * 1024; | 
|  |  | 
|  | void set_handled_types_data(const Array& value) const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(ExceptionHandlers, Object); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | // A WeakSerializationReference (WSR) denotes a type of weak reference to a | 
|  | // target object. In particular, objects that can only be reached from roots via | 
|  | // WSR edges during serialization of AOT snapshots should not be serialized, but | 
|  | // instead references to these objects should be replaced with a reference to | 
|  | // the provided replacement object. | 
|  | // | 
|  | // Of course, the target object may still be serialized if there are paths to | 
|  | // the object from the roots that do not go through one of these objects. In | 
|  | // this case, references through WSRs are serialized as direct references to | 
|  | // the target. | 
|  | // | 
|  | // Unfortunately a WSR is not a proxy for the original object, so WSRs may | 
|  | // only currently be used with ObjectPtr fields. To ease this situation for | 
|  | // fields that are normally a non-ObjectPtr type outside of the precompiler, | 
|  | // use the following macros, which avoid the need to adjust other code to | 
|  | // handle the WSR case: | 
|  | // | 
|  | // * WSR_*POINTER_FIELD() in raw_object.h (i.e., just append WSR_ to the | 
|  | //   original field declaration). | 
|  | // * PRECOMPILER_WSR_FIELD_DECLARATION() in object.h | 
|  | // * PRECOMPILER_WSR_FIELD_DEFINITION() in object.cc | 
|  | class WeakSerializationReference : public Object { | 
|  | public: | 
|  | ObjectPtr target() const { return TargetOf(ptr()); } | 
|  | static ObjectPtr TargetOf(const WeakSerializationReferencePtr obj) { | 
|  | return obj->untag()->target(); | 
|  | } | 
|  |  | 
|  | static ObjectPtr Unwrap(ObjectPtr obj) { | 
|  | #if defined(DART_PRECOMPILER) | 
|  | if (obj->IsHeapObject() && obj->IsWeakSerializationReference()) { | 
|  | return TargetOf(static_cast<WeakSerializationReferencePtr>(obj)); | 
|  | } | 
|  | #endif | 
|  | return obj; | 
|  | } | 
|  | static ObjectPtr Unwrap(const Object& obj) { return Unwrap(obj.ptr()); } | 
|  | static ObjectPtr UnwrapIfTarget(ObjectPtr obj) { return Unwrap(obj); } | 
|  | static ObjectPtr UnwrapIfTarget(const Object& obj) { return Unwrap(obj); } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedWeakSerializationReference)); | 
|  | } | 
|  |  | 
|  | // Returns an ObjectPtr as the target may not need wrapping (e.g., it | 
|  | // is guaranteed to be serialized). | 
|  | static ObjectPtr New(const Object& target, const Object& replacement); | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(WeakSerializationReference, Object); | 
|  |  | 
|  | ObjectPtr replacement() const { return untag()->replacement(); } | 
|  |  | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class WeakArray : public Object { | 
|  | public: | 
|  | intptr_t Length() const { return LengthOf(ptr()); } | 
|  | static inline intptr_t LengthOf(const WeakArrayPtr array); | 
|  |  | 
|  | static intptr_t length_offset() { | 
|  | return OFFSET_OF(UntaggedWeakArray, length_); | 
|  | } | 
|  | static intptr_t data_offset() { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedWeakArray, data); | 
|  | } | 
|  | static intptr_t element_offset(intptr_t index) { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedWeakArray, data) + | 
|  | kBytesPerElement * index; | 
|  | } | 
|  | static intptr_t index_at_offset(intptr_t offset_in_bytes) { | 
|  | intptr_t index = (offset_in_bytes - data_offset()) / kBytesPerElement; | 
|  | ASSERT(index >= 0); | 
|  | return index; | 
|  | } | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { return WeakArray::data_offset(); } | 
|  |  | 
|  | static constexpr intptr_t kElementSize = kCompressedWordSize; | 
|  | }; | 
|  |  | 
|  | ObjectPtr At(intptr_t index) const { return untag()->element(index); } | 
|  | void SetAt(intptr_t index, const Object& value) const { | 
|  | untag()->set_element(index, value.ptr()); | 
|  | } | 
|  |  | 
|  | // Access to the array with acquire release semantics. | 
|  | ObjectPtr AtAcquire(intptr_t index) const { | 
|  | return untag()->element<std::memory_order_acquire>(index); | 
|  | } | 
|  | void SetAtRelease(intptr_t index, const Object& value) const { | 
|  | untag()->set_element<std::memory_order_release>(index, value.ptr()); | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t kBytesPerElement = kCompressedWordSize; | 
|  | static constexpr intptr_t kMaxElements = kSmiMax / kBytesPerElement; | 
|  |  | 
|  | static constexpr bool IsValidLength(intptr_t length) { | 
|  | return 0 <= length && length <= kMaxElements; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedWeakArray) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedWeakArray, data)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t InstanceSize(intptr_t len) { | 
|  | return RoundedAllocationSize(sizeof(UntaggedWeakArray) + | 
|  | (len * kBytesPerElement)); | 
|  | } | 
|  |  | 
|  | static WeakArrayPtr New(intptr_t length, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(WeakArray, Object); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | class Code : public Object { | 
|  | public: | 
|  | // When dual mapping, this returns the executable view. | 
|  | InstructionsPtr active_instructions() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | return nullptr; | 
|  | #else | 
|  | return untag()->active_instructions(); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | // When dual mapping, these return the executable view. | 
|  | InstructionsPtr instructions() const { return untag()->instructions(); } | 
|  | static InstructionsPtr InstructionsOf(const CodePtr code) { | 
|  | return code->untag()->instructions(); | 
|  | } | 
|  |  | 
|  | static intptr_t instructions_offset() { | 
|  | return OFFSET_OF(UntaggedCode, instructions_); | 
|  | } | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | static intptr_t active_instructions_offset() { | 
|  | return OFFSET_OF(UntaggedCode, active_instructions_); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | using EntryKind = CodeEntryKind; | 
|  |  | 
|  | static const char* EntryKindToCString(EntryKind kind); | 
|  | static bool ParseEntryKind(const char* str, EntryKind* out); | 
|  |  | 
|  | static intptr_t entry_point_offset(EntryKind kind = EntryKind::kNormal) { | 
|  | switch (kind) { | 
|  | case EntryKind::kNormal: | 
|  | return OFFSET_OF(UntaggedCode, entry_point_); | 
|  | case EntryKind::kUnchecked: | 
|  | return OFFSET_OF(UntaggedCode, unchecked_entry_point_); | 
|  | case EntryKind::kMonomorphic: | 
|  | return OFFSET_OF(UntaggedCode, monomorphic_entry_point_); | 
|  | case EntryKind::kMonomorphicUnchecked: | 
|  | return OFFSET_OF(UntaggedCode, monomorphic_unchecked_entry_point_); | 
|  | default: | 
|  | UNREACHABLE(); | 
|  | } | 
|  | } | 
|  |  | 
|  | ObjectPoolPtr object_pool() const { return untag()->object_pool(); } | 
|  | static intptr_t object_pool_offset() { | 
|  | return OFFSET_OF(UntaggedCode, object_pool_); | 
|  | } | 
|  |  | 
|  | intptr_t pointer_offsets_length() const { | 
|  | return PtrOffBits::decode(untag()->state_bits_); | 
|  | } | 
|  |  | 
|  | bool is_optimized() const { | 
|  | return OptimizedBit::decode(untag()->state_bits_); | 
|  | } | 
|  | void set_is_optimized(bool value) const; | 
|  | static bool IsOptimized(CodePtr code) { | 
|  | return Code::OptimizedBit::decode(code->untag()->state_bits_); | 
|  | } | 
|  |  | 
|  | bool is_force_optimized() const { | 
|  | return ForceOptimizedBit::decode(untag()->state_bits_); | 
|  | } | 
|  | void set_is_force_optimized(bool value) const; | 
|  |  | 
|  | bool is_alive() const { return AliveBit::decode(untag()->state_bits_); } | 
|  | void set_is_alive(bool value) const; | 
|  |  | 
|  | bool is_discarded() const { return IsDiscarded(ptr()); } | 
|  | static bool IsDiscarded(const CodePtr code) { | 
|  | return DiscardedBit::decode(code->untag()->state_bits_); | 
|  | } | 
|  | void set_is_discarded(bool value) const; | 
|  |  | 
|  | bool HasMonomorphicEntry() const { return HasMonomorphicEntry(ptr()); } | 
|  | static bool HasMonomorphicEntry(const CodePtr code) { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return code->untag()->entry_point_ != | 
|  | code->untag()->monomorphic_entry_point_; | 
|  | #else | 
|  | return Instructions::HasMonomorphicEntry(InstructionsOf(code)); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | // Returns the payload start of [instructions()]. | 
|  | uword PayloadStart() const { return PayloadStartOf(ptr()); } | 
|  | static uword PayloadStartOf(const CodePtr code) { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | if (IsUnknownDartCode(code)) return 0; | 
|  | const uword entry_offset = HasMonomorphicEntry(code) | 
|  | ? Instructions::kPolymorphicEntryOffsetAOT | 
|  | : 0; | 
|  | return EntryPointOf(code) - entry_offset; | 
|  | #else | 
|  | return Instructions::PayloadStart(InstructionsOf(code)); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | // Returns the entry point of [instructions()]. | 
|  | uword EntryPoint() const { return EntryPointOf(ptr()); } | 
|  | static uword EntryPointOf(const CodePtr code) { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return code->untag()->entry_point_; | 
|  | #else | 
|  | return Instructions::EntryPoint(InstructionsOf(code)); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | static uword UncheckedEntryPointOf(const CodePtr code) { | 
|  | return code->untag()->unchecked_entry_point_; | 
|  | } | 
|  |  | 
|  | // Returns the unchecked entry point of [instructions()]. | 
|  | uword UncheckedEntryPoint() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return untag()->unchecked_entry_point_; | 
|  | #else | 
|  | return EntryPoint() + untag()->unchecked_offset_; | 
|  | #endif | 
|  | } | 
|  | // Returns the monomorphic entry point of [instructions()]. | 
|  | uword MonomorphicEntryPoint() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return untag()->monomorphic_entry_point_; | 
|  | #else | 
|  | return Instructions::MonomorphicEntryPoint(instructions()); | 
|  | #endif | 
|  | } | 
|  | // Returns the unchecked monomorphic entry point of [instructions()]. | 
|  | uword MonomorphicUncheckedEntryPoint() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return untag()->monomorphic_unchecked_entry_point_; | 
|  | #else | 
|  | return MonomorphicEntryPoint() + untag()->unchecked_offset_; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | // Returns the size of [instructions()]. | 
|  | uword Size() const { return PayloadSizeOf(ptr()); } | 
|  | static uword PayloadSizeOf(const CodePtr code) { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | if (IsUnknownDartCode(code)) return kUwordMax; | 
|  | return code->untag()->instructions_length_; | 
|  | #else | 
|  | return Instructions::Size(InstructionsOf(code)); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | ObjectPoolPtr GetObjectPool() const; | 
|  | // Returns whether the given PC address is in [instructions()]. | 
|  | bool ContainsInstructionAt(uword addr) const { | 
|  | return ContainsInstructionAt(ptr(), addr); | 
|  | } | 
|  |  | 
|  | // Returns whether the given PC address is in [InstructionsOf(code)]. | 
|  | static bool ContainsInstructionAt(const CodePtr code, uword pc) { | 
|  | return UntaggedCode::ContainsPC(code, pc); | 
|  | } | 
|  |  | 
|  | // Returns true if there is a debugger breakpoint set in this code object. | 
|  | bool HasBreakpoint() const; | 
|  |  | 
|  | PcDescriptorsPtr pc_descriptors() const { return untag()->pc_descriptors(); } | 
|  | void set_pc_descriptors(const PcDescriptors& descriptors) const { | 
|  | ASSERT(descriptors.IsOld()); | 
|  | untag()->set_pc_descriptors(descriptors.ptr()); | 
|  | } | 
|  |  | 
|  | CodeSourceMapPtr code_source_map() const { | 
|  | return untag()->code_source_map(); | 
|  | } | 
|  |  | 
|  | void set_code_source_map(const CodeSourceMap& code_source_map) const { | 
|  | ASSERT(code_source_map.IsOld()); | 
|  | untag()->set_code_source_map(code_source_map.ptr()); | 
|  | } | 
|  |  | 
|  | // Array of DeoptInfo objects. | 
|  | ArrayPtr deopt_info_array() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | return nullptr; | 
|  | #else | 
|  | return untag()->deopt_info_array(); | 
|  | #endif | 
|  | } | 
|  | void set_deopt_info_array(const Array& array) const; | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | intptr_t num_variables() const; | 
|  | void set_num_variables(intptr_t num_variables) const; | 
|  | #endif | 
|  |  | 
|  | TypedDataPtr catch_entry_moves_maps() const; | 
|  | void set_catch_entry_moves_maps(const TypedData& maps) const; | 
|  |  | 
|  | CompressedStackMapsPtr compressed_stackmaps() const { | 
|  | return untag()->compressed_stackmaps(); | 
|  | } | 
|  | void set_compressed_stackmaps(const CompressedStackMaps& maps) const; | 
|  |  | 
|  | enum CallKind { | 
|  | kPcRelativeCall = 1, | 
|  | kPcRelativeTTSCall = 2, | 
|  | kPcRelativeTailCall = 3, | 
|  | kCallViaCode = 4, | 
|  | }; | 
|  | using KindField = | 
|  | BitField<intptr_t, CallKind, 0, Utils::BitLength(kCallViaCode)>; | 
|  |  | 
|  | enum CallEntryPoint { | 
|  | kDefaultEntry, | 
|  | kUncheckedEntry, | 
|  | }; | 
|  | using EntryPointField = BitField<intptr_t, | 
|  | CallEntryPoint, | 
|  | KindField::kNextBit, | 
|  | Utils::BitLength(kUncheckedEntry)>; | 
|  |  | 
|  | using OffsetField = BitField<intptr_t, intptr_t, EntryPointField::kNextBit>; | 
|  |  | 
|  | enum SCallTableEntry { | 
|  | kSCallTableKindAndOffset = 0, | 
|  | kSCallTableCodeOrTypeTarget = 1, | 
|  | kSCallTableFunctionTarget = 2, | 
|  | kSCallTableEntryLength = 3, | 
|  | }; | 
|  |  | 
|  | enum class PoolAttachment { | 
|  | kAttachPool, | 
|  | kNotAttachPool, | 
|  | }; | 
|  |  | 
|  | void set_static_calls_target_table(const Array& value) const; | 
|  | ArrayPtr static_calls_target_table() const { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | return nullptr; | 
|  | #else | 
|  | return untag()->static_calls_target_table(); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | TypedDataPtr GetDeoptInfoAtPc(uword pc, | 
|  | ICData::DeoptReasonId* deopt_reason, | 
|  | uint32_t* deopt_flags) const; | 
|  |  | 
|  | // Returns null if there is no static call at 'pc'. | 
|  | FunctionPtr GetStaticCallTargetFunctionAt(uword pc) const; | 
|  | // Aborts if there is no static call at 'pc'. | 
|  | void SetStaticCallTargetCodeAt(uword pc, const Code& code) const; | 
|  | void SetStubCallTargetCodeAt(uword pc, const Code& code) const; | 
|  |  | 
|  | void Disassemble(DisassemblyFormatter* formatter = nullptr) const; | 
|  |  | 
|  | #if defined(INCLUDE_IL_PRINTER) | 
|  | class Comments : public ZoneAllocated, public CodeComments { | 
|  | public: | 
|  | static Comments& New(intptr_t count); | 
|  |  | 
|  | intptr_t Length() const override; | 
|  |  | 
|  | void SetPCOffsetAt(intptr_t idx, intptr_t pc_offset); | 
|  | void SetCommentAt(intptr_t idx, const String& comment); | 
|  |  | 
|  | intptr_t PCOffsetAt(intptr_t idx) const override; | 
|  | const char* CommentAt(intptr_t idx) const override; | 
|  |  | 
|  | private: | 
|  | explicit Comments(const Array& comments); | 
|  |  | 
|  | // Layout of entries describing comments. | 
|  | enum { | 
|  | kPCOffsetEntry = 0,  // PC offset to a comment as a Smi. | 
|  | kCommentEntry,       // Comment text as a String. | 
|  | kNumberOfEntries | 
|  | }; | 
|  |  | 
|  | const Array& comments_; | 
|  | String& string_; | 
|  |  | 
|  | friend class Code; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(Comments); | 
|  | }; | 
|  |  | 
|  | const CodeComments& comments() const; | 
|  | void set_comments(const CodeComments& comments) const; | 
|  | #endif  // defined(INCLUDE_IL_PRINTER) | 
|  |  | 
|  | ObjectPtr return_address_metadata() const { | 
|  | #if defined(PRODUCT) | 
|  | UNREACHABLE(); | 
|  | return nullptr; | 
|  | #else | 
|  | return untag()->return_address_metadata(); | 
|  | #endif | 
|  | } | 
|  | // Sets |return_address_metadata|. | 
|  | void SetPrologueOffset(intptr_t offset) const; | 
|  | // Returns -1 if no prologue offset is available. | 
|  | intptr_t GetPrologueOffset() const; | 
|  |  | 
|  | ArrayPtr inlined_id_to_function() const; | 
|  | void set_inlined_id_to_function(const Array& value) const; | 
|  |  | 
|  | // Provides the call stack at the given pc offset, with the top-of-stack in | 
|  | // the last element and the root function (this) as the first element, along | 
|  | // with the corresponding source positions. Note the token position for each | 
|  | // function except the top-of-stack is the position of the call to the next | 
|  | // function. The stack will be empty if we lack the metadata to produce it, | 
|  | // which happens for stub code. | 
|  | // The pc offset is interpreted as an instruction address (as needed by the | 
|  | // disassembler or the top frame of a profiler sample). | 
|  | void GetInlinedFunctionsAtInstruction( | 
|  | intptr_t pc_offset, | 
|  | GrowableArray<const Function*>* functions, | 
|  | GrowableArray<TokenPosition>* token_positions) const; | 
|  | // Same as above, except the pc is interpreted as a return address (as needed | 
|  | // for a stack trace or the bottom frames of a profiler sample). | 
|  | void GetInlinedFunctionsAtReturnAddress( | 
|  | intptr_t pc_offset, | 
|  | GrowableArray<const Function*>* functions, | 
|  | GrowableArray<TokenPosition>* token_positions) const { | 
|  | GetInlinedFunctionsAtInstruction(pc_offset - 1, functions, token_positions); | 
|  | } | 
|  |  | 
|  | NOT_IN_PRODUCT(void PrintJSONInlineIntervals(JSONObject* object) const); | 
|  | void DumpInlineIntervals() const; | 
|  | void DumpSourcePositions(bool relative_addresses = false) const; | 
|  |  | 
|  | LocalVarDescriptorsPtr var_descriptors() const { | 
|  | #if defined(PRODUCT) | 
|  | UNREACHABLE(); | 
|  | return nullptr; | 
|  | #else | 
|  | return untag()->var_descriptors<std::memory_order_acquire>(); | 
|  | #endif | 
|  | } | 
|  | void set_var_descriptors(const LocalVarDescriptors& value) const { | 
|  | #if defined(PRODUCT) | 
|  | UNREACHABLE(); | 
|  | #else | 
|  | ASSERT(value.IsOld()); | 
|  | untag()->set_var_descriptors<std::memory_order_release>(value.ptr()); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | // Will compute local var descriptors if necessary. | 
|  | LocalVarDescriptorsPtr GetLocalVarDescriptors() const; | 
|  |  | 
|  | ExceptionHandlersPtr exception_handlers() const { | 
|  | return untag()->exception_handlers(); | 
|  | } | 
|  | void set_exception_handlers(const ExceptionHandlers& handlers) const { | 
|  | ASSERT(handlers.IsOld()); | 
|  | untag()->set_exception_handlers(handlers.ptr()); | 
|  | } | 
|  |  | 
|  | // WARNING: function() returns the owner which is not guaranteed to be | 
|  | // a Function. It is up to the caller to guarantee it isn't a stub, class, | 
|  | // or something else. | 
|  | // TODO(turnidge): Consider dropping this function and making | 
|  | // everybody use owner().  Currently this function is misused - even | 
|  | // while generating the snapshot. | 
|  | FunctionPtr function() const { | 
|  | ASSERT(IsFunctionCode()); | 
|  | return Function::RawCast(owner()); | 
|  | } | 
|  |  | 
|  | ObjectPtr owner() const { | 
|  | return WeakSerializationReference::Unwrap(untag()->owner()); | 
|  | } | 
|  | void set_owner(const Object& owner) const; | 
|  |  | 
|  | classid_t OwnerClassId() const { return OwnerClassIdOf(ptr()); } | 
|  | static classid_t OwnerClassIdOf(CodePtr raw) { | 
|  | ObjectPtr owner = WeakSerializationReference::Unwrap(raw->untag()->owner()); | 
|  | if (!owner->IsHeapObject()) { | 
|  | return RawSmiValue(static_cast<SmiPtr>(owner)); | 
|  | } | 
|  | return owner->GetClassIdOfHeapObject(); | 
|  | } | 
|  |  | 
|  | static intptr_t owner_offset() { return OFFSET_OF(UntaggedCode, owner_); } | 
|  |  | 
|  | // We would have a VisitPointers function here to traverse all the | 
|  | // embedded objects in the instructions using pointer_offsets. | 
|  |  | 
|  | static constexpr intptr_t kBytesPerElement = | 
|  | sizeof(reinterpret_cast<UntaggedCode*>(kOffsetOfPtr)->data()[0]); | 
|  | static constexpr intptr_t kMaxElements = kSmiMax / kBytesPerElement; | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { return sizeof(UntaggedCode); } | 
|  | static constexpr intptr_t kElementSize = kBytesPerElement; | 
|  | }; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedCode) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedCode, data)); | 
|  | return 0; | 
|  | } | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | ASSERT(0 <= len && len <= kMaxElements); | 
|  | return RoundedAllocationSize(sizeof(UntaggedCode) + | 
|  | (len * kBytesPerElement)); | 
|  | } | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | // Finalizes the generated code, by generating various kinds of metadata (e.g. | 
|  | // stack maps, pc descriptors, ...) and attach them to a newly generated | 
|  | // [Code] object. | 
|  | // | 
|  | // If Code::PoolAttachment::kAttachPool is specified for [pool_attachment] | 
|  | // then a new [ObjectPool] will be attached to the code object as well. | 
|  | // Otherwise the caller is responsible for doing this via | 
|  | // `Object::set_object_pool()`. | 
|  | static CodePtr FinalizeCode(FlowGraphCompiler* compiler, | 
|  | compiler::Assembler* assembler, | 
|  | PoolAttachment pool_attachment, | 
|  | bool optimized, | 
|  | CodeStatistics* stats); | 
|  |  | 
|  | // Notifies all active [CodeObserver]s. | 
|  | static void NotifyCodeObservers(const Code& code, bool optimized); | 
|  | static void NotifyCodeObservers(const Function& function, | 
|  | const Code& code, | 
|  | bool optimized); | 
|  | static void NotifyCodeObservers(const char* name, | 
|  | const Code& code, | 
|  | bool optimized); | 
|  |  | 
|  | // Calls [FinalizeCode] and also notifies [CodeObserver]s. | 
|  | static CodePtr FinalizeCodeAndNotify(const Function& function, | 
|  | FlowGraphCompiler* compiler, | 
|  | compiler::Assembler* assembler, | 
|  | PoolAttachment pool_attachment, | 
|  | bool optimized = false, | 
|  | CodeStatistics* stats = nullptr); | 
|  | static CodePtr FinalizeCodeAndNotify(const char* name, | 
|  | FlowGraphCompiler* compiler, | 
|  | compiler::Assembler* assembler, | 
|  | PoolAttachment pool_attachment, | 
|  | bool optimized = false, | 
|  | CodeStatistics* stats = nullptr); | 
|  |  | 
|  | #endif | 
|  | static CodePtr FindCode(uword pc, int64_t timestamp); | 
|  | static CodePtr FindCodeUnsafe(uword pc); | 
|  |  | 
|  | int32_t GetPointerOffsetAt(int index) const { | 
|  | NoSafepointScope no_safepoint; | 
|  | return *PointerOffsetAddrAt(index); | 
|  | } | 
|  | TokenPosition GetTokenIndexOfPC(uword pc) const; | 
|  |  | 
|  | // Find pc, return 0 if not found. | 
|  | uword GetPcForDeoptId(intptr_t deopt_id, | 
|  | UntaggedPcDescriptors::Kind kind) const; | 
|  | intptr_t GetDeoptIdForOsr(uword pc) const; | 
|  |  | 
|  | uint32_t Hash() const; | 
|  | const char* Name() const; | 
|  | const char* QualifiedName(const NameFormattingParams& params) const; | 
|  |  | 
|  | int64_t compile_timestamp() const { | 
|  | #if defined(PRODUCT) | 
|  | return 0; | 
|  | #else | 
|  | return untag()->compile_timestamp_; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | bool IsStubCode() const; | 
|  | bool IsAllocationStubCode() const; | 
|  | bool IsTypeTestStubCode() const; | 
|  | bool IsFunctionCode() const; | 
|  |  | 
|  | // Returns true if this Code object represents | 
|  | // Dart function code without any additional information. | 
|  | bool IsUnknownDartCode() const { return IsUnknownDartCode(ptr()); } | 
|  | static bool IsUnknownDartCode(CodePtr code); | 
|  |  | 
|  | void DisableDartCode() const; | 
|  |  | 
|  | void DisableStubCode(bool is_cls_parameterized) const; | 
|  |  | 
|  | void Enable() const { | 
|  | if (!IsDisabled()) return; | 
|  | ResetActiveInstructions(); | 
|  | } | 
|  |  | 
|  | bool IsDisabled() const { return IsDisabled(ptr()); } | 
|  | static bool IsDisabled(CodePtr code) { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | return false; | 
|  | #else | 
|  | return code->untag()->instructions() != | 
|  | code->untag()->active_instructions(); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void set_object_pool(ObjectPoolPtr object_pool) const { | 
|  | untag()->set_object_pool(object_pool); | 
|  | } | 
|  |  | 
|  | private: | 
|  | void set_state_bits(intptr_t bits) const; | 
|  |  | 
|  | friend class UntaggedObject;  // For UntaggedObject::SizeFromClass(). | 
|  | friend class UntaggedCode; | 
|  | friend struct RelocatorTestHelper; | 
|  |  | 
|  | using OptimizedBit = BitField<decltype(UntaggedCode::state_bits_), bool>; | 
|  |  | 
|  | // Force-optimized is true if the Code was generated for a function with | 
|  | // Function::ForceOptimize(). | 
|  | using ForceOptimizedBit = BitField<decltype(UntaggedCode::state_bits_), | 
|  | bool, | 
|  | OptimizedBit::kNextBit>; | 
|  |  | 
|  | using AliveBit = BitField<decltype(UntaggedCode::state_bits_), | 
|  | bool, | 
|  | ForceOptimizedBit::kNextBit>; | 
|  |  | 
|  | // Set by precompiler if this Code object doesn't contain | 
|  | // useful information besides instructions and compressed stack map. | 
|  | // Such objects are serialized in a shorter form and replaced with | 
|  | // StubCode::UnknownDartCode() during snapshot deserialization. | 
|  | using DiscardedBit = | 
|  | BitField<decltype(UntaggedCode::state_bits_), bool, AliveBit::kNextBit>; | 
|  |  | 
|  | using PtrOffBits = BitField<decltype(UntaggedCode::state_bits_), | 
|  | intptr_t, | 
|  | DiscardedBit::kNextBit>; | 
|  |  | 
|  | static constexpr intptr_t kEntrySize = sizeof(int32_t);  // NOLINT | 
|  |  | 
|  | void set_compile_timestamp(int64_t timestamp) const { | 
|  | #if defined(PRODUCT) | 
|  | UNREACHABLE(); | 
|  | #else | 
|  | StoreNonPointer(&untag()->compile_timestamp_, timestamp); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | // Initializes the cached entrypoint addresses in [code] as calculated | 
|  | // from [instructions] and [unchecked_offset]. | 
|  | static void InitializeCachedEntryPointsFrom(CodePtr code, | 
|  | InstructionsPtr instructions, | 
|  | uint32_t unchecked_offset); | 
|  |  | 
|  | // Sets [active_instructions_] to [instructions] and updates the cached | 
|  | // entry point addresses. | 
|  | void SetActiveInstructions(const Instructions& instructions, | 
|  | uint32_t unchecked_offset) const; | 
|  | void SetActiveInstructionsSafe(const Instructions& instructions, | 
|  | uint32_t unchecked_offset) const; | 
|  |  | 
|  | // Resets [active_instructions_] to its original value of [instructions_] and | 
|  | // updates the cached entry point addresses to match. | 
|  | void ResetActiveInstructions() const; | 
|  |  | 
|  | void set_instructions(const Instructions& instructions) const { | 
|  | ASSERT(Thread::Current()->IsDartMutatorThread() || !is_alive()); | 
|  | untag()->set_instructions(instructions.ptr()); | 
|  | } | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void set_unchecked_offset(uword offset) const { | 
|  | StoreNonPointer(&untag()->unchecked_offset_, offset); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | // Returns the unchecked entry point offset for [instructions_]. | 
|  | uint32_t UncheckedEntryPointOffset() const { | 
|  | return UncheckedEntryPointOffsetOf(ptr()); | 
|  | } | 
|  | static uint32_t UncheckedEntryPointOffsetOf(CodePtr code) { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | UNREACHABLE(); | 
|  | #else | 
|  | return code->untag()->unchecked_offset_; | 
|  | #endif | 
|  | } | 
|  |  | 
|  | void set_pointer_offsets_length(intptr_t value) { | 
|  | // The number of fixups is limited to 1-billion. | 
|  | ASSERT(Utils::IsUint(30, value)); | 
|  | set_state_bits(PtrOffBits::update(value, untag()->state_bits_)); | 
|  | } | 
|  | int32_t* PointerOffsetAddrAt(int index) const { | 
|  | ASSERT(index >= 0); | 
|  | ASSERT(index < pointer_offsets_length()); | 
|  | // TODO(iposva): Unit test is missing for this functionality. | 
|  | return &UnsafeMutableNonPointer(untag()->data())[index]; | 
|  | } | 
|  | void SetPointerOffsetAt(int index, int32_t offset_in_instructions) { | 
|  | NoSafepointScope no_safepoint; | 
|  | *PointerOffsetAddrAt(index) = offset_in_instructions; | 
|  | } | 
|  |  | 
|  | intptr_t BinarySearchInSCallTable(uword pc) const; | 
|  | static CodePtr LookupCodeInIsolateGroup(IsolateGroup* isolate_group, | 
|  | uword pc); | 
|  |  | 
|  | // New is a private method as RawInstruction and RawCode objects should | 
|  | // only be created using the Code::FinalizeCode method. This method creates | 
|  | // the RawInstruction and RawCode objects, sets up the pointer offsets | 
|  | // and links the two in a GC safe manner. | 
|  | static CodePtr New(intptr_t pointer_offsets_length); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Code, Object); | 
|  | friend class Class; | 
|  | friend class CodeTestHelper; | 
|  | friend class StubCode;     // for set_object_pool | 
|  | friend class Precompiler;  // for set_object_pool | 
|  | friend class FunctionSerializationCluster; | 
|  | friend class CodeSerializationCluster; | 
|  | friend class CodeDeserializationCluster; | 
|  | friend class Deserializer;           // for InitializeCachedEntryPointsFrom | 
|  | friend class StubCode;               // for set_object_pool | 
|  | friend class MegamorphicCacheTable;  // for set_object_pool | 
|  | friend class CodePatcher;            // for set_instructions | 
|  | friend class ProgramVisitor;         // for set_instructions | 
|  | // So that the UntaggedFunction pointer visitor can determine whether code the | 
|  | // function points to is optimized. | 
|  | friend class UntaggedFunction; | 
|  | friend class CallSiteResetter; | 
|  | friend class CodeKeyValueTrait;  // for UncheckedEntryPointOffset | 
|  | friend class InstanceCall;       // for StorePointerUnaligned | 
|  | friend class StaticCall;         // for StorePointerUnaligned | 
|  | friend void DumpStackFrame(intptr_t frame_index, uword pc, uword fp); | 
|  | }; | 
|  |  | 
|  | class Bytecode : public Object { | 
|  | public: | 
|  | uword instructions() const { return untag()->instructions_; } | 
|  |  | 
|  | uword PayloadStart() const { return instructions(); } | 
|  | intptr_t Size() const { return untag()->instructions_size_; } | 
|  |  | 
|  | ObjectPoolPtr object_pool() const { return untag()->object_pool(); } | 
|  |  | 
|  | bool ContainsInstructionAt(uword addr) const { | 
|  | return UntaggedBytecode::ContainsPC(ptr(), addr); | 
|  | } | 
|  |  | 
|  | PcDescriptorsPtr pc_descriptors() const { return untag()->pc_descriptors(); } | 
|  | void set_pc_descriptors(const PcDescriptors& descriptors) const { | 
|  | ASSERT(descriptors.IsOld()); | 
|  | untag()->set_pc_descriptors(descriptors.ptr()); | 
|  | } | 
|  |  | 
|  | void Disassemble(DisassemblyFormatter* formatter = NULL) const; | 
|  |  | 
|  | ExceptionHandlersPtr exception_handlers() const { | 
|  | return untag()->exception_handlers(); | 
|  | } | 
|  | void set_exception_handlers(const ExceptionHandlers& handlers) const { | 
|  | ASSERT(handlers.IsOld()); | 
|  | untag()->set_exception_handlers(handlers.ptr()); | 
|  | } | 
|  |  | 
|  | FunctionPtr function() const { return untag()->function(); } | 
|  |  | 
|  | void set_function(const Function& function) const { | 
|  | ASSERT(function.IsOld()); | 
|  | untag()->set_function(function.ptr()); | 
|  | } | 
|  |  | 
|  | TypedDataBasePtr binary() const { return untag()->binary(); } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedBytecode)); | 
|  | } | 
|  | static BytecodePtr New(uword instructions, | 
|  | intptr_t instructions_size, | 
|  | intptr_t instructions_offset, | 
|  | const TypedDataBase& binary, | 
|  | const ObjectPool& object_pool); | 
|  |  | 
|  | TokenPosition GetTokenIndexOfPC(uword return_address) const; | 
|  | intptr_t GetTryIndexAtPc(uword return_address) const; | 
|  |  | 
|  | // Return the pc of the first 'DebugCheck' opcode of the bytecode. | 
|  | // Return 0 if none is found. | 
|  | uword GetFirstDebugCheckOpcodePc() const; | 
|  |  | 
|  | // Return the pc after the first 'debug checked' opcode in the range. | 
|  | // Return 0 if none is found. | 
|  | uword GetDebugCheckedOpcodeReturnAddress(uword from_offset, | 
|  | uword to_offset) const; | 
|  |  | 
|  | intptr_t instructions_binary_offset() const { | 
|  | return untag()->instructions_binary_offset_; | 
|  | } | 
|  | void set_instructions_binary_offset(intptr_t value) const { | 
|  | StoreNonPointer(&untag()->instructions_binary_offset_, value); | 
|  | } | 
|  |  | 
|  | intptr_t code_offset() const { return untag()->code_offset_; } | 
|  | void set_code_offset(intptr_t value) const { | 
|  | StoreNonPointer(&untag()->code_offset_, value); | 
|  | } | 
|  |  | 
|  | intptr_t source_positions_binary_offset() const { | 
|  | return untag()->source_positions_binary_offset_; | 
|  | } | 
|  | void set_source_positions_binary_offset(intptr_t value) const { | 
|  | StoreNonPointer(&untag()->source_positions_binary_offset_, value); | 
|  | } | 
|  | bool HasSourcePositions() const { | 
|  | return (source_positions_binary_offset() != 0); | 
|  | } | 
|  |  | 
|  | const char* Name() const; | 
|  | const char* QualifiedName() const; | 
|  | const char* FullyQualifiedName() const; | 
|  |  | 
|  | private: | 
|  | void set_instructions(uword instructions) const { | 
|  | StoreNonPointer(&untag()->instructions_, instructions); | 
|  | } | 
|  | void set_instructions_size(intptr_t size) const { | 
|  | StoreNonPointer(&untag()->instructions_size_, size); | 
|  | } | 
|  | void set_object_pool(const ObjectPool& object_pool) const { | 
|  | untag()->set_object_pool(object_pool.ptr()); | 
|  | } | 
|  | void set_binary(const TypedDataBase& binary) const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Bytecode, Object); | 
|  | friend class BytecodeDeserializationCluster; | 
|  | friend class Class; | 
|  | friend class SnapshotWriter; | 
|  | }; | 
|  |  | 
|  | class Context : public Object { | 
|  | public: | 
|  | ContextPtr parent() const { return untag()->parent(); } | 
|  | void set_parent(const Context& parent) const { | 
|  | untag()->set_parent(parent.ptr()); | 
|  | } | 
|  | static intptr_t parent_offset() { | 
|  | return OFFSET_OF(UntaggedContext, parent_); | 
|  | } | 
|  |  | 
|  | intptr_t num_variables() const { return untag()->num_variables_; } | 
|  | static intptr_t num_variables_offset() { | 
|  | return OFFSET_OF(UntaggedContext, num_variables_); | 
|  | } | 
|  | static intptr_t NumVariables(const ContextPtr context) { | 
|  | return context->untag()->num_variables_; | 
|  | } | 
|  |  | 
|  | ObjectPtr At(intptr_t context_index) const { | 
|  | return untag()->element(context_index); | 
|  | } | 
|  | inline void SetAt(intptr_t context_index, const Object& value) const; | 
|  |  | 
|  | intptr_t GetLevel() const; | 
|  |  | 
|  | void Dump(int indent = 0) const; | 
|  |  | 
|  | static constexpr intptr_t kBytesPerElement = kCompressedWordSize; | 
|  | static constexpr intptr_t kMaxElements = kSmiMax / kBytesPerElement; | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { return sizeof(UntaggedContext); } | 
|  | static constexpr intptr_t kElementSize = kBytesPerElement; | 
|  | }; | 
|  |  | 
|  | static intptr_t variable_offset(intptr_t context_index) { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedContext, data) + | 
|  | (kBytesPerElement * context_index); | 
|  | } | 
|  |  | 
|  | static bool IsValidLength(intptr_t len) { | 
|  | return 0 <= len && len <= compiler::target::Context::kMaxElements; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedContext) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedContext, data)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | ASSERT(IsValidLength(len)); | 
|  | return RoundedAllocationSize(sizeof(UntaggedContext) + | 
|  | (len * kBytesPerElement)); | 
|  | } | 
|  |  | 
|  | static ContextPtr New(intptr_t num_variables, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | void set_num_variables(intptr_t num_variables) const { | 
|  | StoreNonPointer(&untag()->num_variables_, num_variables); | 
|  | } | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Context, Object); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | // The ContextScope class makes it possible to delay the compilation of a local | 
|  | // function until it is invoked. A ContextScope instance collects the local | 
|  | // variables that are referenced by the local function to be compiled and that | 
|  | // belong to the outer scopes, that is, to the local scopes of (possibly nested) | 
|  | // functions enclosing the local function. Each captured variable is represented | 
|  | // by its token position in the source, its name, its type, its allocation index | 
|  | // in the context, and its context level. The function nesting level and loop | 
|  | // nesting level are not preserved, since they are only used until the context | 
|  | // level is assigned. In addition the ContextScope has a field 'is_implicit' | 
|  | // which is true if the ContextScope was created for an implicit closure. | 
|  | class ContextScope : public Object { | 
|  | public: | 
|  | intptr_t num_variables() const { return untag()->num_variables_; } | 
|  |  | 
|  | TokenPosition TokenIndexAt(intptr_t scope_index) const; | 
|  | void SetTokenIndexAt(intptr_t scope_index, TokenPosition token_pos) const; | 
|  |  | 
|  | TokenPosition DeclarationTokenIndexAt(intptr_t scope_index) const; | 
|  | void SetDeclarationTokenIndexAt(intptr_t scope_index, | 
|  | TokenPosition declaration_token_pos) const; | 
|  |  | 
|  | StringPtr NameAt(intptr_t scope_index) const; | 
|  | void SetNameAt(intptr_t scope_index, const String& name) const; | 
|  |  | 
|  | void ClearFlagsAt(intptr_t scope_index) const; | 
|  |  | 
|  | intptr_t LateInitOffsetAt(intptr_t scope_index) const; | 
|  | void SetLateInitOffsetAt(intptr_t scope_index, | 
|  | intptr_t late_init_offset) const; | 
|  |  | 
|  | #define DECLARE_FLAG_ACCESSORS(Name)                                           \ | 
|  | bool Is##Name##At(intptr_t scope_index) const;                               \ | 
|  | void SetIs##Name##At(intptr_t scope_index, bool value) const; | 
|  |  | 
|  | CONTEXT_SCOPE_VARIABLE_DESC_FLAG_LIST(DECLARE_FLAG_ACCESSORS) | 
|  | #undef DECLARE_FLAG_ACCESSORS | 
|  |  | 
|  | AbstractTypePtr TypeAt(intptr_t scope_index) const; | 
|  | void SetTypeAt(intptr_t scope_index, const AbstractType& type) const; | 
|  |  | 
|  | intptr_t CidAt(intptr_t scope_index) const; | 
|  | void SetCidAt(intptr_t scope_index, intptr_t cid) const; | 
|  |  | 
|  | intptr_t ContextIndexAt(intptr_t scope_index) const; | 
|  | void SetContextIndexAt(intptr_t scope_index, intptr_t context_index) const; | 
|  |  | 
|  | intptr_t ContextLevelAt(intptr_t scope_index) const; | 
|  | void SetContextLevelAt(intptr_t scope_index, intptr_t context_level) const; | 
|  |  | 
|  | intptr_t KernelOffsetAt(intptr_t scope_index) const; | 
|  | void SetKernelOffsetAt(intptr_t scope_index, intptr_t kernel_offset) const; | 
|  |  | 
|  | static constexpr intptr_t kBytesPerElement = | 
|  | sizeof(UntaggedContextScope::VariableDesc); | 
|  | static constexpr intptr_t kMaxElements = kSmiMax / kBytesPerElement; | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { | 
|  | return sizeof(UntaggedContextScope); | 
|  | } | 
|  | static constexpr intptr_t kElementSize = kBytesPerElement; | 
|  | }; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedContextScope) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedContextScope, data)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | ASSERT(0 <= len && len <= kMaxElements); | 
|  | return RoundedAllocationSize(sizeof(UntaggedContextScope) + | 
|  | (len * kBytesPerElement)); | 
|  | } | 
|  |  | 
|  | static ContextScopePtr New(intptr_t num_variables, bool is_implicit); | 
|  |  | 
|  | private: | 
|  | void set_num_variables(intptr_t num_variables) const { | 
|  | StoreNonPointer(&untag()->num_variables_, num_variables); | 
|  | } | 
|  |  | 
|  | void set_is_implicit(bool is_implicit) const { | 
|  | StoreNonPointer(&untag()->is_implicit_, is_implicit); | 
|  | } | 
|  |  | 
|  | const UntaggedContextScope::VariableDesc* VariableDescAddr( | 
|  | intptr_t index) const { | 
|  | ASSERT((index >= 0) && (index < num_variables())); | 
|  | return untag()->VariableDescAddr(index); | 
|  | } | 
|  |  | 
|  | bool GetFlagAt(intptr_t scope_index, intptr_t bit_index) const; | 
|  | void SetFlagAt(intptr_t scope_index, intptr_t bit_index, bool value) const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(ContextScope, Object); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | // Class of special sentinel values: | 
|  | // - Object::sentinel() is a value that cannot be produced by Dart code. | 
|  | // It can be used to mark special values, for example to distinguish | 
|  | // "uninitialized" fields. | 
|  | // - Object::unknown_constant() and Object::non_constant() are optimizing | 
|  | // compiler's constant propagation constants. | 
|  | // - Object::optimized_out() result from deopt environment pruning or failure | 
|  | // to capture variables in a closure's context | 
|  | class Sentinel : public Object { | 
|  | public: | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedSentinel)); | 
|  | } | 
|  |  | 
|  | static SentinelPtr New(); | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Sentinel, Object); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | class MegamorphicCache : public CallSiteData { | 
|  | public: | 
|  | static constexpr intptr_t kInitialCapacity = 16; | 
|  | static constexpr intptr_t kSpreadFactor = 7; | 
|  | static constexpr double kLoadFactor = 0.50; | 
|  |  | 
|  | enum EntryType { | 
|  | kClassIdIndex, | 
|  | kTargetFunctionIndex, | 
|  | kEntryLength, | 
|  | }; | 
|  |  | 
|  | ArrayPtr buckets() const; | 
|  | void set_buckets(const Array& buckets) const; | 
|  |  | 
|  | intptr_t mask() const; | 
|  | void set_mask(intptr_t mask) const; | 
|  |  | 
|  | intptr_t filled_entry_count() const; | 
|  | void set_filled_entry_count(intptr_t num) const; | 
|  |  | 
|  | static intptr_t buckets_offset() { | 
|  | return OFFSET_OF(UntaggedMegamorphicCache, buckets_); | 
|  | } | 
|  | static intptr_t mask_offset() { | 
|  | return OFFSET_OF(UntaggedMegamorphicCache, mask_); | 
|  | } | 
|  | static intptr_t arguments_descriptor_offset() { | 
|  | return OFFSET_OF(UntaggedMegamorphicCache, args_descriptor_); | 
|  | } | 
|  |  | 
|  | static MegamorphicCachePtr New(const String& target_name, | 
|  | const Array& arguments_descriptor); | 
|  |  | 
|  | void EnsureContains(const Smi& class_id, const Object& target) const; | 
|  | ObjectPtr Lookup(const Smi& class_id) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedMegamorphicCache)); | 
|  | } | 
|  |  | 
|  | private: | 
|  | friend class Class; | 
|  | friend class MegamorphicCacheTable; | 
|  | friend class ProgramVisitor; | 
|  |  | 
|  | static MegamorphicCachePtr New(); | 
|  |  | 
|  | // The caller must hold IsolateGroup::type_feedback_mutex(). | 
|  | void InsertLocked(const Smi& class_id, const Object& target) const; | 
|  | void EnsureCapacityLocked() const; | 
|  | ObjectPtr LookupLocked(const Smi& class_id) const; | 
|  |  | 
|  | void InsertEntryLocked(const Smi& class_id, const Object& target) const; | 
|  |  | 
|  | static inline void SetEntry(const Array& array, | 
|  | intptr_t index, | 
|  | const Smi& class_id, | 
|  | const Object& target); | 
|  |  | 
|  | static inline ObjectPtr GetClassId(const Array& array, intptr_t index); | 
|  | static inline ObjectPtr GetTargetFunction(const Array& array, intptr_t index); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(MegamorphicCache, CallSiteData); | 
|  | }; | 
|  |  | 
|  | class SubtypeTestCache : public Object { | 
|  | public: | 
|  | // The contents of the backing array storage is a number of entry tuples. | 
|  | // Any entry that is unoccupied has the null value as its first component. | 
|  | // | 
|  | // If the cache is linear, the entries can be accessed in a linear fashion: | 
|  | // all occupied entries come first, followed by at least one unoccupied | 
|  | // entry to mark the end of the cache. Guaranteeing at least one unoccupied | 
|  | // entry avoids the need for a length check when iterating over the contents | 
|  | // of the linear cache in stubs. | 
|  | // | 
|  | // If the cache is hash-based, the array is instead treated as a hash table | 
|  | // probed by using a hash value derived from the inputs. | 
|  |  | 
|  | // The tuple of values stored in a given entry. | 
|  | // | 
|  | // Note that occupied entry contents are never modified. That means reading a | 
|  | // non-null instance cid or signature means the rest of the entry can be | 
|  | // loaded without worrying about concurrent modification. Thus, we always set | 
|  | // the instance cid or signature last when making an occupied entry. | 
|  | // | 
|  | // Also note that each STC, when created, has a set number of used inputs. | 
|  | // The value of any unused input is unspecified, so for example, if the | 
|  | // STC only uses 3 inputs, then no assumptions can be made about the value | 
|  | // stored in the instantiator type arguments slot. | 
|  | enum Entries { | 
|  | kInstanceCidOrSignature = 0, | 
|  | kInstanceTypeArguments = 1, | 
|  | kInstantiatorTypeArguments = 2, | 
|  | kFunctionTypeArguments = 3, | 
|  | kInstanceParentFunctionTypeArguments = 4, | 
|  | kInstanceDelayedFunctionTypeArguments = 5, | 
|  | kDestinationType = 6, | 
|  | kTestResult = 7, | 
|  | kTestEntryLength = 8, | 
|  | }; | 
|  |  | 
|  | // Assumes only one non-input entry in the array, kTestResult. | 
|  | static_assert(kInstanceCidOrSignature == 0 && | 
|  | kDestinationType + 1 == kTestResult && | 
|  | kTestResult + 1 == kTestEntryLength, | 
|  | "Need to adjust number of max inputs"); | 
|  | static constexpr intptr_t kMaxInputs = kTestResult; | 
|  |  | 
|  | // Returns the number of occupied entries stored in the cache. | 
|  | intptr_t NumberOfChecks() const; | 
|  |  | 
|  | // Retrieves the number of entries (occupied or unoccupied) in the cache. | 
|  | intptr_t NumEntries() const; | 
|  |  | 
|  | // Adds a check, returning the index of the new entry in the cache. | 
|  | intptr_t AddCheck( | 
|  | const Object& instance_class_id_or_signature, | 
|  | const AbstractType& destination_type, | 
|  | const TypeArguments& instance_type_arguments, | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | const TypeArguments& instance_parent_function_type_arguments, | 
|  | const TypeArguments& instance_delayed_type_arguments, | 
|  | const Bool& test_result) const; | 
|  | void GetCheck(intptr_t ix, | 
|  | Object* instance_class_id_or_signature, | 
|  | AbstractType* destination_type, | 
|  | TypeArguments* instance_type_arguments, | 
|  | TypeArguments* instantiator_type_arguments, | 
|  | TypeArguments* function_type_arguments, | 
|  | TypeArguments* instance_parent_function_type_arguments, | 
|  | TypeArguments* instance_delayed_type_arguments, | 
|  | Bool* test_result) const; | 
|  |  | 
|  | // Like GetCheck(), but does not require the subtype test cache mutex and so | 
|  | // may see an outdated view of the cache. | 
|  | void GetCurrentCheck(intptr_t ix, | 
|  | Object* instance_class_id_or_signature, | 
|  | AbstractType* destination_type, | 
|  | TypeArguments* instance_type_arguments, | 
|  | TypeArguments* instantiator_type_arguments, | 
|  | TypeArguments* function_type_arguments, | 
|  | TypeArguments* instance_parent_function_type_arguments, | 
|  | TypeArguments* instance_delayed_type_arguments, | 
|  | Bool* test_result) const; | 
|  |  | 
|  | // Like GetCheck(), but returns the contents of the first occupied entry | 
|  | // at or after the initial contents of [ix]. Returns whether an occupied entry | 
|  | // was found, and if an occupied entry was found, [ix] is updated to the entry | 
|  | // index following the occupied entry. | 
|  | bool GetNextCheck(intptr_t* ix, | 
|  | Object* instance_class_id_or_signature, | 
|  | AbstractType* destination_type, | 
|  | TypeArguments* instance_type_arguments, | 
|  | TypeArguments* instantiator_type_arguments, | 
|  | TypeArguments* function_type_arguments, | 
|  | TypeArguments* instance_parent_function_type_arguments, | 
|  | TypeArguments* instance_delayed_type_arguments, | 
|  | Bool* test_result) const; | 
|  |  | 
|  | // Returns whether all the elements of an existing cache entry, excluding | 
|  | // the result, match the non-pointer arguments. The pointer arguments are | 
|  | // out parameters as follows: | 
|  | // | 
|  | // If [index] is not nullptr, then it is set to the matching entry's index. | 
|  | // If [result] is not nullptr, then it is set to the matching entry's result. | 
|  | // | 
|  | // If called without the STC mutex lock, may return outdated information: | 
|  | // * May return a false negative if the entry was added concurrently. | 
|  | // * The [index] field may be invalid for the STC if the backing array is | 
|  | //   grown concurrently and the new backing array is hash-based. | 
|  | bool HasCheck(const Object& instance_class_id_or_signature, | 
|  | const AbstractType& destination_type, | 
|  | const TypeArguments& instance_type_arguments, | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | const TypeArguments& instance_parent_function_type_arguments, | 
|  | const TypeArguments& instance_delayed_type_arguments, | 
|  | intptr_t* index, | 
|  | Bool* result) const; | 
|  |  | 
|  | // Writes the cache entry at index [index] to the given text buffer. | 
|  | // | 
|  | // The output is comma separated on a single line if [line_prefix] is nullptr, | 
|  | // otherwise line breaks followed by [line_prefix] is used as a separator. | 
|  | void WriteEntryToBuffer(Zone* zone, | 
|  | BaseTextBuffer* buffer, | 
|  | intptr_t index, | 
|  | const char* line_prefix = nullptr) const; | 
|  |  | 
|  | // Writes the contents of this SubtypeTestCache to the given text buffer. | 
|  | void WriteToBuffer(Zone* zone, | 
|  | BaseTextBuffer* buffer, | 
|  | const char* line_prefix = nullptr) const; | 
|  |  | 
|  | void Reset() const; | 
|  |  | 
|  | // Tests that [other] contains the same entries in the same order. | 
|  | bool Equals(const SubtypeTestCache& other) const; | 
|  |  | 
|  | // Returns whether the cache backed by the given storage is hash-based. | 
|  | bool IsHash() const; | 
|  |  | 
|  | // Creates a separate copy of the current STC contents. | 
|  | SubtypeTestCachePtr Copy(Thread* thread) const; | 
|  |  | 
|  | static SubtypeTestCachePtr New(intptr_t num_inputs); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedSubtypeTestCache)); | 
|  | } | 
|  |  | 
|  | static intptr_t cache_offset() { | 
|  | return OFFSET_OF(UntaggedSubtypeTestCache, cache_); | 
|  | } | 
|  | ArrayPtr cache() const; | 
|  |  | 
|  | static intptr_t num_inputs_offset() { | 
|  | return OFFSET_OF(UntaggedSubtypeTestCache, num_inputs_); | 
|  | } | 
|  | intptr_t num_inputs() const { return untag()->num_inputs_; } | 
|  |  | 
|  | intptr_t num_occupied() const { return untag()->num_occupied_; } | 
|  |  | 
|  | // The maximum number of occupied entries for a linear subtype test cache | 
|  | // before swapping to a hash table-based cache. Exposed publicly for tests. | 
|  | #if defined(TARGET_ARCH_IA32) | 
|  | // We don't generate hash cache probing in the stub on IA32, so larger caches | 
|  | // force runtime checks. | 
|  | static constexpr intptr_t kMaxLinearCacheEntries = 100; | 
|  | #else | 
|  | static constexpr intptr_t kMaxLinearCacheEntries = 30; | 
|  | #endif | 
|  |  | 
|  | // Whether the entry at the given index in the cache is occupied. Exposed | 
|  | // publicly for tests. | 
|  | bool IsOccupied(intptr_t index) const; | 
|  |  | 
|  | // Returns the number of inputs needed to cache entries for the given type. | 
|  | static intptr_t UsedInputsForType(const AbstractType& type); | 
|  |  | 
|  | // Given a minimum entry count, calculates an entry count that won't force | 
|  | // additional allocation but minimizes the number of unoccupied entries. | 
|  | // Used to calculate an appropriate value for FLAG_max_subtype_cache_entries. | 
|  | static constexpr intptr_t MaxEntriesForCacheAllocatedFor(intptr_t count) { | 
|  | // If the cache would be linear, just return the count unchanged. | 
|  | if (count <= kMaxLinearCacheEntries) return count; | 
|  | intptr_t allocated_entries = Utils::RoundUpToPowerOfTwo(count); | 
|  | if (LoadFactor(count, allocated_entries) >= kMaxLoadFactor) { | 
|  | allocated_entries *= 2; | 
|  | } | 
|  | const intptr_t max_entries = | 
|  | static_cast<intptr_t>(kMaxLoadFactor * allocated_entries); | 
|  | assert(LoadFactor(max_entries, allocated_entries) < kMaxLoadFactor); | 
|  | assert(max_entries >= count); | 
|  | return max_entries; | 
|  | } | 
|  |  | 
|  | private: | 
|  | static constexpr double LoadFactor(intptr_t occupied, intptr_t capacity) { | 
|  | return occupied / static_cast<double>(capacity); | 
|  | } | 
|  |  | 
|  | // Retrieves the number of entries (occupied or unoccupied) in a cache | 
|  | // backed by the given array. | 
|  | static intptr_t NumEntries(const Array& array); | 
|  |  | 
|  | // Returns whether the cache backed by the given storage is linear. | 
|  | static bool IsLinear(const Array& array) { return !IsHash(array); } | 
|  |  | 
|  | // Returns whether the cache backed by the given storage is hash-based. | 
|  | static bool IsHash(const Array& array); | 
|  |  | 
|  | struct KeyLocation { | 
|  | // The entry index if [present] is true, otherwise where the entry would | 
|  | // be located if added afterwards without any intermediate additions. | 
|  | intptr_t entry; | 
|  | bool present;  // Whether an entry already exists in the cache. | 
|  | }; | 
|  |  | 
|  | // If a cache entry in the given array contains the given inputs, returns a | 
|  | // KeyLocation with the index of the entry and true. Otherwise, returns a | 
|  | // KeyLocation with the index that would be used if the instantiation for the | 
|  | // given type arguments is added and false. | 
|  | // | 
|  | // If called without the STC mutex lock, may return outdated information: | 
|  | // * The [present] field may be a false negative if the entry was added | 
|  | //   concurrently. | 
|  | static KeyLocation FindKeyOrUnused( | 
|  | const Array& array, | 
|  | intptr_t num_inputs, | 
|  | const Object& instance_class_id_or_signature, | 
|  | const AbstractType& destination_type, | 
|  | const TypeArguments& instance_type_arguments, | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | const TypeArguments& instance_parent_function_type_arguments, | 
|  | const TypeArguments& instance_delayed_type_arguments); | 
|  |  | 
|  | // If the given array can contain the requested number of entries, returns | 
|  | // the same array and sets [was_grown] to false. | 
|  | // | 
|  | // If the given array cannot contain the requested number of entries, | 
|  | // returns a new array that can and which contains all the entries of the | 
|  | // given array and sets [was_grown] to true. | 
|  | ArrayPtr EnsureCapacity(Zone* zone, | 
|  | const Array& array, | 
|  | intptr_t new_capacity, | 
|  | bool* was_grown) const; | 
|  |  | 
|  | public:  // Used in the StubCodeCompiler. | 
|  | // The maximum size of the array backing a linear cache. All hash based | 
|  | // caches are guaranteed to have sizes larger than this. | 
|  | static constexpr intptr_t kMaxLinearCacheSize = | 
|  | (kMaxLinearCacheEntries + 1) * kTestEntryLength; | 
|  |  | 
|  | private: | 
|  | // The initial number of entries used when converting from a linear to | 
|  | // a hash-based cache. | 
|  | static constexpr intptr_t kNumInitialHashCacheEntries = | 
|  | Utils::RoundUpToPowerOfTwo(2 * kMaxLinearCacheEntries); | 
|  | static_assert(Utils::IsPowerOfTwo(kNumInitialHashCacheEntries), | 
|  | "number of hash-based cache entries must be a power of two"); | 
|  |  | 
|  | // The max load factor allowed in hash-based caches. | 
|  | static constexpr double kMaxLoadFactor = 0.71; | 
|  |  | 
|  | void set_cache(const Array& value) const; | 
|  | void set_num_occupied(intptr_t value) const; | 
|  |  | 
|  | // Like GetCurrentCheck, but takes the backing storage array. | 
|  | static void GetCheckFromArray( | 
|  | const Array& array, | 
|  | intptr_t num_inputs, | 
|  | intptr_t ix, | 
|  | Object* instance_class_id_or_signature, | 
|  | AbstractType* destination_type, | 
|  | TypeArguments* instance_type_arguments, | 
|  | TypeArguments* instantiator_type_arguments, | 
|  | TypeArguments* function_type_arguments, | 
|  | TypeArguments* instance_parent_function_type_arguments, | 
|  | TypeArguments* instance_delayed_type_arguments, | 
|  | Bool* test_result); | 
|  |  | 
|  | // Like WriteEntryToBuffer(), but does not require the subtype test cache | 
|  | // mutex and so may see an incorrect view of the cache if there are concurrent | 
|  | // modifications. | 
|  | void WriteCurrentEntryToBuffer(Zone* zone, | 
|  | BaseTextBuffer* buffer, | 
|  | intptr_t index, | 
|  | const char* line_prefix = nullptr) const; | 
|  |  | 
|  | // Like WriteToBuffer(), but does not require the subtype test cache mutex and | 
|  | // so may see an incorrect view of the cache if there are concurrent | 
|  | // modifications. | 
|  | void WriteToBufferUnlocked(Zone* zone, | 
|  | BaseTextBuffer* buffer, | 
|  | const char* line_prefix = nullptr) const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(SubtypeTestCache, Object); | 
|  | friend class Class; | 
|  | friend class FieldInvalidator; | 
|  | friend class VMSerializationRoots; | 
|  | friend class VMDeserializationRoots; | 
|  | }; | 
|  |  | 
|  | class LoadingUnit : public Object { | 
|  | public: | 
|  | static constexpr intptr_t kIllegalId = 0; | 
|  | COMPILE_ASSERT(kIllegalId == WeakTable::kNoValue); | 
|  | static constexpr intptr_t kRootId = 1; | 
|  |  | 
|  | static LoadingUnitPtr New(intptr_t id, const LoadingUnit& parent); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedLoadingUnit)); | 
|  | } | 
|  |  | 
|  | static intptr_t LoadingUnitOf(const Function& function); | 
|  | static intptr_t LoadingUnitOf(const Code& code); | 
|  |  | 
|  | LoadingUnitPtr parent() const { return untag()->parent(); } | 
|  |  | 
|  | ArrayPtr base_objects() const { return untag()->base_objects(); } | 
|  | void set_base_objects(const Array& value) const; | 
|  |  | 
|  | intptr_t id() const { | 
|  | return untag()->packed_fields_.Read<UntaggedLoadingUnit::IdBits>(); | 
|  | } | 
|  |  | 
|  | // True once the VM deserializes this unit's snapshot. | 
|  | bool loaded() const { | 
|  | return untag()->packed_fields_.Read<UntaggedLoadingUnit::LoadStateBits>() == | 
|  | UntaggedLoadingUnit::kLoaded; | 
|  | } | 
|  | // value is whether the load succeeded or not. | 
|  | void set_loaded(bool value) const { | 
|  | ASSERT(load_outstanding()); | 
|  | auto const expected = | 
|  | value ? UntaggedLoadingUnit::kLoaded : UntaggedLoadingUnit::kNotLoaded; | 
|  | auto const got = untag() | 
|  | ->packed_fields_ | 
|  | .UpdateConditional<UntaggedLoadingUnit::LoadStateBits>( | 
|  | expected, UntaggedLoadingUnit::kLoadOutstanding); | 
|  | // Check that we're in the expected state afterwards. | 
|  | ASSERT_EQUAL(got, expected); | 
|  | } | 
|  |  | 
|  | // True once the VM invokes the embedder's deferred load callback until the | 
|  | // embedder calls Dart_DeferredLoadComplete[Error]. | 
|  | bool load_outstanding() const { | 
|  | return untag()->packed_fields_.Read<UntaggedLoadingUnit::LoadStateBits>() == | 
|  | UntaggedLoadingUnit::kLoadOutstanding; | 
|  | } | 
|  | void set_load_outstanding() const { | 
|  | auto const previous = UntaggedLoadingUnit::kNotLoaded; | 
|  | ASSERT_EQUAL( | 
|  | untag()->packed_fields_.Read<UntaggedLoadingUnit::LoadStateBits>(), | 
|  | previous); | 
|  | auto const expected = UntaggedLoadingUnit::kLoadOutstanding; | 
|  | auto const got = untag() | 
|  | ->packed_fields_ | 
|  | .UpdateConditional<UntaggedLoadingUnit::LoadStateBits>( | 
|  | expected, previous); | 
|  | // Check that we're in the expected state afterwards. | 
|  | ASSERT_EQUAL(got, expected); | 
|  | } | 
|  |  | 
|  | const uint8_t* instructions_image() const { | 
|  | // The instructions image should only be accessed if the load succeeded. | 
|  | ASSERT(loaded()); | 
|  | return untag()->instructions_image_; | 
|  | } | 
|  | void set_instructions_image(const uint8_t* value) const { | 
|  | ASSERT(load_outstanding()); | 
|  | StoreNonPointer(&untag()->instructions_image_, value); | 
|  | } | 
|  | bool has_instructions_image() const { | 
|  | return loaded() && instructions_image() != nullptr; | 
|  | } | 
|  |  | 
|  | ObjectPtr IssueLoad() const; | 
|  | ObjectPtr CompleteLoad(const String& error_message, | 
|  | bool transient_error) const; | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(LoadingUnit, Object); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class Error : public Object { | 
|  | public: | 
|  | virtual const char* ToErrorCString() const; | 
|  |  | 
|  | private: | 
|  | HEAP_OBJECT_IMPLEMENTATION(Error, Object); | 
|  | }; | 
|  |  | 
|  | class ApiError : public Error { | 
|  | public: | 
|  | StringPtr message() const { return untag()->message(); } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedApiError)); | 
|  | } | 
|  |  | 
|  | static ApiErrorPtr New(const String& message, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | virtual const char* ToErrorCString() const; | 
|  |  | 
|  | private: | 
|  | void set_message(const String& message) const; | 
|  |  | 
|  | static ApiErrorPtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(ApiError, Error); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class LanguageError : public Error { | 
|  | public: | 
|  | Report::Kind kind() const { | 
|  | return static_cast<Report::Kind>(untag()->kind_); | 
|  | } | 
|  |  | 
|  | // Build, cache, and return formatted message. | 
|  | StringPtr FormatMessage() const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedLanguageError)); | 
|  | } | 
|  |  | 
|  | // A null script means no source and a negative token_pos means no position. | 
|  | static LanguageErrorPtr NewFormatted(const Error& prev_error, | 
|  | const Script& script, | 
|  | TokenPosition token_pos, | 
|  | bool report_after_token, | 
|  | Report::Kind kind, | 
|  | Heap::Space space, | 
|  | const char* format, | 
|  | ...) PRINTF_ATTRIBUTE(7, 8); | 
|  |  | 
|  | static LanguageErrorPtr NewFormattedV(const Error& prev_error, | 
|  | const Script& script, | 
|  | TokenPosition token_pos, | 
|  | bool report_after_token, | 
|  | Report::Kind kind, | 
|  | Heap::Space space, | 
|  | const char* format, | 
|  | va_list args); | 
|  |  | 
|  | static LanguageErrorPtr New(const String& formatted_message, | 
|  | Report::Kind kind = Report::kError, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | virtual const char* ToErrorCString() const; | 
|  |  | 
|  | TokenPosition token_pos() const { return untag()->token_pos_; } | 
|  |  | 
|  | private: | 
|  | ErrorPtr previous_error() const { return untag()->previous_error(); } | 
|  | void set_previous_error(const Error& value) const; | 
|  |  | 
|  | ScriptPtr script() const { return untag()->script(); } | 
|  | void set_script(const Script& value) const; | 
|  |  | 
|  | void set_token_pos(TokenPosition value) const; | 
|  |  | 
|  | bool report_after_token() const { return untag()->report_after_token_; } | 
|  | void set_report_after_token(bool value) const; | 
|  |  | 
|  | void set_kind(uint8_t value) const; | 
|  |  | 
|  | StringPtr message() const { return untag()->message(); } | 
|  | void set_message(const String& value) const; | 
|  |  | 
|  | StringPtr formatted_message() const { return untag()->formatted_message(); } | 
|  | void set_formatted_message(const String& value) const; | 
|  |  | 
|  | static LanguageErrorPtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(LanguageError, Error); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class UnhandledException : public Error { | 
|  | public: | 
|  | InstancePtr exception() const { return untag()->exception(); } | 
|  | void set_exception(const Instance& exception) const; | 
|  | static intptr_t exception_offset() { | 
|  | return OFFSET_OF(UntaggedUnhandledException, exception_); | 
|  | } | 
|  |  | 
|  | InstancePtr stacktrace() const { return untag()->stacktrace(); } | 
|  | void set_stacktrace(const Instance& stacktrace) const; | 
|  | static intptr_t stacktrace_offset() { | 
|  | return OFFSET_OF(UntaggedUnhandledException, stacktrace_); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedUnhandledException)); | 
|  | } | 
|  |  | 
|  | static UnhandledExceptionPtr New(const Instance& exception, | 
|  | const Instance& stacktrace, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static UnhandledExceptionPtr New(Heap::Space space = Heap::kNew); | 
|  |  | 
|  | virtual const char* ToErrorCString() const; | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(UnhandledException, Error); | 
|  | friend class Class; | 
|  | friend class ObjectStore; | 
|  | }; | 
|  |  | 
|  | class UnwindError : public Error { | 
|  | public: | 
|  | bool is_user_initiated() const { return untag()->is_user_initiated_; } | 
|  | void set_is_user_initiated(bool value) const; | 
|  |  | 
|  | StringPtr message() const { return untag()->message(); } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedUnwindError)); | 
|  | } | 
|  |  | 
|  | static UnwindErrorPtr New(const String& message, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | virtual const char* ToErrorCString() const; | 
|  |  | 
|  | private: | 
|  | void set_message(const String& message) const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(UnwindError, Error); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Instance is the base class for all instance objects (aka the Object class | 
|  | // in Dart source code. | 
|  | class Instance : public Object { | 
|  | public: | 
|  | // Equality and identity testing. | 
|  | // 1. OperatorEquals: true iff 'this == other' is true in Dart code. | 
|  | // 2. IsIdenticalTo: true iff 'identical(this, other)' is true in Dart code. | 
|  | // 3. CanonicalizeEquals: used to canonicalize compile-time constants, e.g., | 
|  | //    using bitwise equality of fields and list elements. | 
|  | // Subclasses where 1 and 3 coincide may also define a plain Equals, e.g., | 
|  | // String and Integer. | 
|  | virtual bool OperatorEquals(const Instance& other) const; | 
|  | bool IsIdenticalTo(const Instance& other) const; | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  |  | 
|  | intptr_t SizeFromClass() const { | 
|  | #if defined(DEBUG) | 
|  | const Class& cls = Class::Handle(clazz()); | 
|  | ASSERT(cls.is_finalized() || cls.is_prefinalized()); | 
|  | #endif | 
|  | return (clazz()->untag()->host_instance_size_in_words_ * | 
|  | kCompressedWordSize); | 
|  | } | 
|  |  | 
|  | InstancePtr Canonicalize(Thread* thread) const; | 
|  | // Caller must hold IsolateGroup::constant_canonicalization_mutex_. | 
|  | virtual InstancePtr CanonicalizeLocked(Thread* thread) const; | 
|  | virtual void CanonicalizeFieldsLocked(Thread* thread) const; | 
|  |  | 
|  | InstancePtr CopyShallowToOldSpace(Thread* thread) const; | 
|  |  | 
|  | ObjectPtr GetField(const Field& field) const; | 
|  |  | 
|  | void SetField(const Field& field, const Object& value) const; | 
|  | void SetFieldWithoutFieldGuard(const Field& field, const Object& value) const; | 
|  |  | 
|  | AbstractTypePtr GetType( | 
|  | Heap::Space space, | 
|  | TypeVisibility visibility = TypeVisibility::kInternalType) const; | 
|  |  | 
|  | // Access the type arguments vector of this [Instance]. | 
|  | // This vector includes type arguments corresponding to type parameters of | 
|  | // instance's class and all its superclasses. | 
|  | virtual TypeArgumentsPtr GetTypeArguments() const; | 
|  | virtual void SetTypeArguments(const TypeArguments& value) const; | 
|  |  | 
|  | // Check if the type of this instance is a subtype of the given other type. | 
|  | // The type argument vectors are used to instantiate the other type if needed. | 
|  | bool IsInstanceOf(const AbstractType& other, | 
|  | const TypeArguments& other_instantiator_type_arguments, | 
|  | const TypeArguments& other_function_type_arguments) const; | 
|  |  | 
|  | // Return true if the null instance can be assigned to a variable of [other] | 
|  | // type. Return false if null cannot be assigned or we cannot tell (if | 
|  | // [other] is a type parameter in NNBD strong mode). Only used for checks at | 
|  | // compile time. | 
|  | static bool NullIsAssignableTo(const AbstractType& other); | 
|  |  | 
|  | // Return true if the null instance can be assigned to a variable of [other] | 
|  | // type. Return false if null cannot be assigned. Used for checks at runtime, | 
|  | // when the instantiator and function type argument vectors are available. | 
|  | static bool NullIsAssignableTo( | 
|  | const AbstractType& other, | 
|  | const TypeArguments& other_instantiator_type_arguments, | 
|  | const TypeArguments& other_function_type_arguments); | 
|  |  | 
|  | bool IsValidNativeIndex(int index) const { | 
|  | return ((index >= 0) && (index < clazz()->untag()->num_native_fields_)); | 
|  | } | 
|  |  | 
|  | intptr_t* NativeFieldsDataAddr() const; | 
|  | inline intptr_t GetNativeField(int index) const; | 
|  | inline void GetNativeFields(uint16_t num_fields, | 
|  | intptr_t* field_values) const; | 
|  | void SetNativeFields(uint16_t num_fields, const intptr_t* field_values) const; | 
|  |  | 
|  | uint16_t NumNativeFields() const { | 
|  | return clazz()->untag()->num_native_fields_; | 
|  | } | 
|  |  | 
|  | void SetNativeField(int index, intptr_t value) const; | 
|  |  | 
|  | // If the instance is a callable object, i.e. a closure or the instance of a | 
|  | // class implementing a 'call' method, return true and set the function | 
|  | // (if not nullptr) to call. | 
|  | bool IsCallable(Function* function) const; | 
|  |  | 
|  | ObjectPtr Invoke(const String& selector, | 
|  | const Array& arguments, | 
|  | const Array& argument_names, | 
|  | bool check_is_entrypoint = true, | 
|  | bool respect_reflectable = true) const; | 
|  | ObjectPtr InvokeGetter(const String& selector, | 
|  | bool check_is_entrypoint = true, | 
|  | bool respect_reflectable = true) const; | 
|  | ObjectPtr InvokeSetter(const String& selector, | 
|  | const Instance& argument, | 
|  | bool check_is_entrypoint = true, | 
|  | bool respect_reflectable = true) const; | 
|  |  | 
|  | ObjectPtr EvaluateCompiledExpression( | 
|  | const Class& klass, | 
|  | const ExternalTypedData& kernel_buffer, | 
|  | const Array& type_definitions, | 
|  | const Array& arguments, | 
|  | const TypeArguments& type_arguments) const; | 
|  |  | 
|  | // Evaluate the given expression as if it appeared in an instance method of | 
|  | // [receiver] and return the resulting value, or an error object if | 
|  | // evaluating the expression fails. The method has the formal (type) | 
|  | // parameters given in (type_)param_names, and is invoked with the (type) | 
|  | // argument values given in (type_)param_values. | 
|  | // | 
|  | // We allow [receiver] to be null/<optimized out> if | 
|  | //   * the evaluation function doesn't access `this` | 
|  | //   * the evaluation function is static | 
|  | static ObjectPtr EvaluateCompiledExpression( | 
|  | Thread* thread, | 
|  | const Object& receiver, | 
|  | const Library& library, | 
|  | const Class& klass, | 
|  | const ExternalTypedData& kernel_buffer, | 
|  | const Array& type_definitions, | 
|  | const Array& param_values, | 
|  | const TypeArguments& type_param_values); | 
|  |  | 
|  | // Equivalent to invoking hashCode on this instance. | 
|  | virtual ObjectPtr HashCode() const; | 
|  |  | 
|  | // Equivalent to invoking identityHashCode with this instance. | 
|  | IntegerPtr IdentityHashCode(Thread* thread) const; | 
|  |  | 
|  | static intptr_t UnroundedSize() { return sizeof(UntaggedInstance); } | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedInstance)); | 
|  | } | 
|  |  | 
|  | static InstancePtr New(const Class& cls, Heap::Space space = Heap::kNew); | 
|  | static InstancePtr NewAlreadyFinalized(const Class& cls, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Array/list element address computations. | 
|  | static intptr_t DataOffsetFor(intptr_t cid); | 
|  | static intptr_t ElementSizeFor(intptr_t cid); | 
|  |  | 
|  | // Pointers may be subtyped, but their subtypes may not get extra fields. | 
|  | // The subtype runtime representation has exactly the same object layout, | 
|  | // only the class_id is different. So, it is safe to use subtype instances in | 
|  | // Pointer handles. | 
|  | virtual bool IsPointer() const; | 
|  |  | 
|  | static intptr_t NextFieldOffset() { return sizeof(UntaggedInstance); } | 
|  |  | 
|  | static intptr_t NativeFieldsOffset() { return sizeof(UntaggedObject); } | 
|  |  | 
|  | protected: | 
|  | #ifndef PRODUCT | 
|  | virtual void PrintSharedInstanceJSON(JSONObject* jsobj, | 
|  | bool ref, | 
|  | bool include_id = true) const; | 
|  | #endif | 
|  |  | 
|  | private: | 
|  | // Return true if the runtimeType of this instance is a subtype of other type. | 
|  | bool RuntimeTypeIsSubtypeOf( | 
|  | const AbstractType& other, | 
|  | const TypeArguments& other_instantiator_type_arguments, | 
|  | const TypeArguments& other_function_type_arguments) const; | 
|  |  | 
|  | // Returns true if the type of this instance is a subtype of FutureOr<T> | 
|  | // specified by instantiated type 'other'. | 
|  | // Returns false if other type is not a FutureOr. | 
|  | bool RuntimeTypeIsSubtypeOfFutureOr(Zone* zone, | 
|  | const AbstractType& other) const; | 
|  |  | 
|  | CompressedObjectPtr* FieldAddrAtOffset(intptr_t offset) const { | 
|  | ASSERT(IsValidFieldOffset(offset)); | 
|  | return reinterpret_cast<CompressedObjectPtr*>(raw_value() - kHeapObjectTag + | 
|  | offset); | 
|  | } | 
|  | CompressedObjectPtr* FieldAddr(const Field& field) const { | 
|  | return FieldAddrAtOffset(field.HostOffset()); | 
|  | } | 
|  | CompressedObjectPtr* NativeFieldsAddr() const { | 
|  | return FieldAddrAtOffset(sizeof(UntaggedObject)); | 
|  | } | 
|  | void SetFieldAtOffset(intptr_t offset, const Object& value) const { | 
|  | StoreCompressedPointer(FieldAddrAtOffset(offset), value.ptr()); | 
|  | } | 
|  | bool IsValidFieldOffset(intptr_t offset) const; | 
|  |  | 
|  | // The following raw methods are used for morphing. | 
|  | // They are needed due to the extraction of the class in IsValidFieldOffset. | 
|  | CompressedObjectPtr* RawFieldAddrAtOffset(intptr_t offset) const { | 
|  | return reinterpret_cast<CompressedObjectPtr*>(raw_value() - kHeapObjectTag + | 
|  | offset); | 
|  | } | 
|  | ObjectPtr RawGetFieldAtOffset(intptr_t offset) const { | 
|  | return RawFieldAddrAtOffset(offset)->Decompress(untag()->heap_base()); | 
|  | } | 
|  | void RawSetFieldAtOffset(intptr_t offset, const Object& value) const { | 
|  | StoreCompressedPointer(RawFieldAddrAtOffset(offset), value.ptr()); | 
|  | } | 
|  | void RawSetFieldAtOffset(intptr_t offset, ObjectPtr value) const { | 
|  | StoreCompressedPointer(RawFieldAddrAtOffset(offset), value); | 
|  | } | 
|  |  | 
|  | template <typename T> | 
|  | T* RawUnboxedFieldAddrAtOffset(intptr_t offset) const { | 
|  | return reinterpret_cast<T*>(raw_value() - kHeapObjectTag + offset); | 
|  | } | 
|  | template <typename T> | 
|  | T RawGetUnboxedFieldAtOffset(intptr_t offset) const { | 
|  | return *RawUnboxedFieldAddrAtOffset<T>(offset); | 
|  | } | 
|  | template <typename T> | 
|  | void RawSetUnboxedFieldAtOffset(intptr_t offset, const T& value) const { | 
|  | *RawUnboxedFieldAddrAtOffset<T>(offset) = value; | 
|  | } | 
|  |  | 
|  | // TODO(iposva): Determine if this gets in the way of Smi. | 
|  | HEAP_OBJECT_IMPLEMENTATION(Instance, Object); | 
|  | friend class ByteBuffer; | 
|  | friend class Class; | 
|  | friend class Closure; | 
|  | friend class Pointer; | 
|  | friend class DeferredObject; | 
|  | friend class FlowGraphSerializer; | 
|  | friend class FlowGraphDeserializer; | 
|  | friend class RegExp; | 
|  | friend class StubCode; | 
|  | friend class TypedDataView; | 
|  | friend class InstanceSerializationCluster; | 
|  | friend class InstanceDeserializationCluster; | 
|  | friend class Interpreter; | 
|  | friend class ClassDeserializationCluster;  // vtable | 
|  | friend class InstanceMorpher; | 
|  | friend class Obfuscator;  // RawGetFieldAtOffset, RawSetFieldAtOffset | 
|  | }; | 
|  |  | 
|  | class LibraryPrefix : public Instance { | 
|  | public: | 
|  | StringPtr name() const { return untag()->name(); } | 
|  | virtual StringPtr DictionaryName() const { return name(); } | 
|  |  | 
|  | ArrayPtr imports() const { return untag()->imports(); } | 
|  | intptr_t num_imports() const { return untag()->num_imports_; } | 
|  | LibraryPtr importer() const { return untag()->importer(); } | 
|  |  | 
|  | LibraryPtr GetLibrary(int index) const; | 
|  | void AddImport(const Namespace& import) const; | 
|  |  | 
|  | bool is_deferred_load() const { return untag()->is_deferred_load_; } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedLibraryPrefix)); | 
|  | } | 
|  |  | 
|  | static LibraryPrefixPtr New(const String& name, | 
|  | const Namespace& import, | 
|  | bool deferred_load, | 
|  | const Library& importer); | 
|  |  | 
|  | private: | 
|  | static constexpr int kInitialSize = 2; | 
|  | static constexpr int kIncrementSize = 2; | 
|  |  | 
|  | void set_name(const String& value) const; | 
|  | void set_imports(const Array& value) const; | 
|  | void set_num_imports(intptr_t value) const; | 
|  | void set_importer(const Library& value) const; | 
|  |  | 
|  | static LibraryPrefixPtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(LibraryPrefix, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // TypeParameters represents a list of formal type parameters with their bounds | 
|  | // and their default values as calculated by CFE. | 
|  | class TypeParameters : public Object { | 
|  | public: | 
|  | intptr_t Length() const; | 
|  |  | 
|  | static intptr_t names_offset() { | 
|  | return OFFSET_OF(UntaggedTypeParameters, names_); | 
|  | } | 
|  | StringPtr NameAt(intptr_t index) const; | 
|  | void SetNameAt(intptr_t index, const String& value) const; | 
|  |  | 
|  | static intptr_t flags_offset() { | 
|  | return OFFSET_OF(UntaggedTypeParameters, flags_); | 
|  | } | 
|  |  | 
|  | static intptr_t bounds_offset() { | 
|  | return OFFSET_OF(UntaggedTypeParameters, bounds_); | 
|  | } | 
|  | AbstractTypePtr BoundAt(intptr_t index) const; | 
|  | void SetBoundAt(intptr_t index, const AbstractType& value) const; | 
|  | bool AllDynamicBounds() const; | 
|  |  | 
|  | static intptr_t defaults_offset() { | 
|  | return OFFSET_OF(UntaggedTypeParameters, defaults_); | 
|  | } | 
|  | AbstractTypePtr DefaultAt(intptr_t index) const; | 
|  | void SetDefaultAt(intptr_t index, const AbstractType& value) const; | 
|  | bool AllDynamicDefaults() const; | 
|  |  | 
|  | // The isGenericCovariantImpl bits are packed into SMIs in the flags array, | 
|  | // but omitted if they're 0. | 
|  | bool IsGenericCovariantImplAt(intptr_t index) const; | 
|  | void SetIsGenericCovariantImplAt(intptr_t index, bool value) const; | 
|  |  | 
|  | // The number of flags per Smi should be a power of 2 in order to simplify the | 
|  | // generated code accessing the flags array. | 
|  | #if !defined(DART_COMPRESSED_POINTERS) | 
|  | static constexpr intptr_t kFlagsPerSmiShift = kBitsPerWordLog2 - 1; | 
|  | #else | 
|  | static constexpr intptr_t kFlagsPerSmiShift = kBitsPerWordLog2 - 2; | 
|  | #endif | 
|  | static constexpr intptr_t kFlagsPerSmi = 1LL << kFlagsPerSmiShift; | 
|  | COMPILE_ASSERT(kFlagsPerSmi < kSmiBits); | 
|  | static constexpr intptr_t kFlagsPerSmiMask = kFlagsPerSmi - 1; | 
|  |  | 
|  | void Print(Thread* thread, | 
|  | Zone* zone, | 
|  | bool are_class_type_parameters, | 
|  | intptr_t base, | 
|  | NameVisibility name_visibility, | 
|  | BaseTextBuffer* printer) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedTypeParameters)); | 
|  | } | 
|  |  | 
|  | static TypeParametersPtr New(Heap::Space space = Heap::kOld); | 
|  | static TypeParametersPtr New(intptr_t count, Heap::Space space = Heap::kOld); | 
|  |  | 
|  | private: | 
|  | ArrayPtr names() const { return untag()->names(); } | 
|  | void set_names(const Array& value) const; | 
|  | ArrayPtr flags() const { return untag()->flags(); } | 
|  | void set_flags(const Array& value) const; | 
|  | TypeArgumentsPtr bounds() const { return untag()->bounds(); } | 
|  | void set_bounds(const TypeArguments& value) const; | 
|  | TypeArgumentsPtr defaults() const { return untag()->defaults(); } | 
|  | void set_defaults(const TypeArguments& value) const; | 
|  |  | 
|  | // Allocate and initialize the flags array to zero. | 
|  | void AllocateFlags(Heap::Space space) const; | 
|  | // Reset the flags array to null if all flags are zero. | 
|  | void OptimizeFlags() const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(TypeParameters, Object); | 
|  | friend class Class; | 
|  | friend class ClassFinalizer; | 
|  | friend class FlowGraphSerializer; | 
|  | friend class FlowGraphDeserializer; | 
|  | friend class Function; | 
|  | friend class FunctionType; | 
|  | friend class Object; | 
|  | friend class Precompiler; | 
|  | friend class Type;  // To determine whether to print type arguments. | 
|  | }; | 
|  |  | 
|  | // A TypeArguments is an array of AbstractType. | 
|  | class TypeArguments : public Instance { | 
|  | public: | 
|  | // Hash value for a type argument vector consisting solely of dynamic types. | 
|  | static constexpr intptr_t kAllDynamicHash = 1; | 
|  |  | 
|  | // Returns whether this TypeArguments vector can be used in a context that | 
|  | // expects a vector of length [count]. Always true for the null vector. | 
|  | bool HasCount(intptr_t count) const; | 
|  | static intptr_t length_offset() { | 
|  | return OFFSET_OF(UntaggedTypeArguments, length_); | 
|  | } | 
|  | intptr_t Length() const; | 
|  | AbstractTypePtr TypeAt(intptr_t index) const; | 
|  | AbstractTypePtr TypeAtNullSafe(intptr_t index) const; | 
|  | static intptr_t types_offset() { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedTypeArguments, types); | 
|  | } | 
|  | static intptr_t type_at_offset(intptr_t index) { | 
|  | return types_offset() + index * kCompressedWordSize; | 
|  | } | 
|  | void SetTypeAt(intptr_t index, const AbstractType& value) const; | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { | 
|  | return TypeArguments::types_offset(); | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t kElementSize = kCompressedWordSize; | 
|  | }; | 
|  |  | 
|  | // The nullability of a type argument vector represents the nullability of its | 
|  | // type elements (up to a maximum number of them, i.e. kNullabilityMaxTypes). | 
|  | // It is used at runtime in some cases (predetermined by the compiler) to | 
|  | // decide whether the instantiator type arguments (ITA) can be shared instead | 
|  | // of performing a more costly instantiation of the uninstantiated type | 
|  | // arguments (UTA). | 
|  | // The vector nullability is stored as a bit vector (in a Smi field), using | 
|  | // 1 bit per type, which is set if the type is nullable. | 
|  | // The nullability is 0 if the vector is longer than kNullabilityMaxTypes. | 
|  | // The condition evaluated at runtime to decide whether UTA can share ITA is | 
|  | //   (UTA.nullability & ITA.nullability) == UTA.nullability | 
|  | // Note that this allows for ITA to be longer than UTA (the bit vector must be | 
|  | // stored in the same order as the corresponding type vector, i.e. with the | 
|  | // least significant bit representing the nullability of the first type). | 
|  | static constexpr intptr_t kNullabilityBitsPerType = 1; | 
|  | static constexpr intptr_t kNullabilityMaxTypes = | 
|  | kSmiBits / kNullabilityBitsPerType; | 
|  | static constexpr intptr_t kNonNullableBit = 0; | 
|  | static constexpr intptr_t kNullableBit = 1; | 
|  | intptr_t nullability() const; | 
|  | static intptr_t nullability_offset() { | 
|  | return OFFSET_OF(UntaggedTypeArguments, nullability_); | 
|  | } | 
|  |  | 
|  | // The name of this type argument vector, e.g. "<T, dynamic, List<T>, Smi>". | 
|  | StringPtr Name() const; | 
|  |  | 
|  | // The name of this type argument vector, e.g. "<T, dynamic, List<T>, int>". | 
|  | // Names of internal classes are mapped to their public interfaces. | 
|  | StringPtr UserVisibleName() const; | 
|  |  | 
|  | // Print the internal or public name of a subvector of this type argument | 
|  | // vector, e.g. "<T, dynamic, List<T>, int>". | 
|  | void PrintSubvectorName(intptr_t from_index, | 
|  | intptr_t len, | 
|  | NameVisibility name_visibility, | 
|  | BaseTextBuffer* printer) const; | 
|  | void PrintTo(BaseTextBuffer* printer) const; | 
|  |  | 
|  | // Check if the subvector of length 'len' starting at 'from_index' of this | 
|  | // type argument vector consists solely of DynamicType. | 
|  | bool IsRaw(intptr_t from_index, intptr_t len) const { | 
|  | return IsDynamicTypes(false, from_index, len); | 
|  | } | 
|  |  | 
|  | // Check if this type argument vector would consist solely of DynamicType if | 
|  | // it was instantiated from both a raw (null) instantiator type arguments and | 
|  | // a raw (null) function type arguments, i.e. consider each class type | 
|  | // parameter and function type parameters as it would be first instantiated | 
|  | // from a vector of dynamic types. | 
|  | // Consider only a prefix of length 'len'. | 
|  | bool IsRawWhenInstantiatedFromRaw(intptr_t len) const { | 
|  | return IsDynamicTypes(true, 0, len); | 
|  | } | 
|  |  | 
|  | TypeArgumentsPtr Prepend(Zone* zone, | 
|  | const TypeArguments& other, | 
|  | intptr_t other_length, | 
|  | intptr_t total_length) const; | 
|  |  | 
|  | // Concatenate [this] and [other] vectors of type parameters. | 
|  | TypeArgumentsPtr ConcatenateTypeParameters(Zone* zone, | 
|  | const TypeArguments& other) const; | 
|  |  | 
|  | // Returns an InstantiationMode for this type argument vector, which | 
|  | // specifies whether the type argument vector requires instantiation and | 
|  | // shortcuts to instantiate the vector when possible. | 
|  | // | 
|  | // If [function] is provided, any function type arguments used in the type | 
|  | // arguments vector are assumed to be bound by the function or its parent | 
|  | // functions. | 
|  | // | 
|  | // If [class] is provided, any class type arguments used in the type arguments | 
|  | // vector are assumed to be bound by that class. If [function] is provided | 
|  | // but [class] is not, then the owning class of the function is retrieved | 
|  | // and used. | 
|  | InstantiationMode GetInstantiationMode(Zone* zone, | 
|  | const Function* function = nullptr, | 
|  | const Class* cls = nullptr) const; | 
|  |  | 
|  | // Check if the vectors are equal (they may be null). | 
|  | bool Equals(const TypeArguments& other) const { | 
|  | return IsSubvectorEquivalent(other, 0, IsNull() ? 0 : Length(), | 
|  | TypeEquality::kCanonical); | 
|  | } | 
|  |  | 
|  | bool IsEquivalent( | 
|  | const TypeArguments& other, | 
|  | TypeEquality kind, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const { | 
|  | // Make a null vector a vector of dynamic as long as the other vector. | 
|  | return IsSubvectorEquivalent(other, 0, IsNull() ? other.Length() : Length(), | 
|  | kind, function_type_equivalence); | 
|  | } | 
|  | bool IsSubvectorEquivalent( | 
|  | const TypeArguments& other, | 
|  | intptr_t from_index, | 
|  | intptr_t len, | 
|  | TypeEquality kind, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | // Check if the vector is instantiated (it must not be null). | 
|  | bool IsInstantiated(Genericity genericity = kAny, | 
|  | intptr_t num_free_fun_type_params = kAllFree) const { | 
|  | return IsSubvectorInstantiated(0, Length(), genericity, | 
|  | num_free_fun_type_params); | 
|  | } | 
|  | bool IsSubvectorInstantiated( | 
|  | intptr_t from_index, | 
|  | intptr_t len, | 
|  | Genericity genericity = kAny, | 
|  | intptr_t num_free_fun_type_params = kAllFree) const; | 
|  | bool IsUninstantiatedIdentity() const; | 
|  |  | 
|  | // Determine whether this uninstantiated type argument vector can share its | 
|  | // instantiator (resp. function) type argument vector instead of being | 
|  | // instantiated at runtime. | 
|  | // If null is passed in for 'with_runtime_check', the answer is unconditional | 
|  | // (i.e. the answer will be false even if a runtime check may allow sharing), | 
|  | // otherwise, in case the function returns true, 'with_runtime_check' | 
|  | // indicates if a check is still required at runtime before allowing sharing. | 
|  | bool CanShareInstantiatorTypeArguments( | 
|  | const Class& instantiator_class, | 
|  | bool* with_runtime_check = nullptr) const; | 
|  | bool CanShareFunctionTypeArguments(const Function& function, | 
|  | bool* with_runtime_check = nullptr) const; | 
|  | TypeArgumentsPtr TruncatedTo(intptr_t length) const; | 
|  |  | 
|  | // Return true if all types of this vector are finalized. | 
|  | bool IsFinalized() const; | 
|  |  | 
|  | // Caller must hold IsolateGroup::constant_canonicalization_mutex_. | 
|  | virtual InstancePtr CanonicalizeLocked(Thread* thread) const { | 
|  | return Canonicalize(thread); | 
|  | } | 
|  |  | 
|  | // Canonicalize only if instantiated, otherwise returns 'this'. | 
|  | TypeArgumentsPtr Canonicalize(Thread* thread) const; | 
|  |  | 
|  | // Shrinks flattened instance type arguments to ordinary type arguments. | 
|  | TypeArgumentsPtr FromInstanceTypeArguments(Thread* thread, | 
|  | const Class& cls) const; | 
|  |  | 
|  | // Expands type arguments to a vector suitable as instantiator type | 
|  | // arguments. | 
|  | // | 
|  | // Only fills positions corresponding to type parameters of [cls], leave | 
|  | // all positions of superclass type parameters blank. | 
|  | // Use [GetInstanceTypeArguments] on a class or a type if full vector is | 
|  | // needed. | 
|  | TypeArgumentsPtr ToInstantiatorTypeArguments(Thread* thread, | 
|  | const Class& cls) const; | 
|  |  | 
|  | // Add the class name and URI of each type argument of this vector to the uris | 
|  | // list and mark ambiguous triplets to be printed. | 
|  | void EnumerateURIs(URIs* uris) const; | 
|  |  | 
|  | // Return 'this' if this type argument vector is instantiated, i.e. if it does | 
|  | // not refer to type parameters. Otherwise, return a new type argument vector | 
|  | // where each reference to a type parameter is replaced with the corresponding | 
|  | // type from the various type argument vectors (class instantiator, function, | 
|  | // or parent functions via the current context). | 
|  | TypeArgumentsPtr InstantiateFrom( | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping = nullptr, | 
|  | intptr_t num_parent_type_args_adjustment = 0) const; | 
|  |  | 
|  | // Update number of parent function type arguments for | 
|  | // all elements of this vector. | 
|  | TypeArgumentsPtr UpdateFunctionTypes( | 
|  | intptr_t num_parent_type_args_adjustment, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping) const; | 
|  |  | 
|  | // Runtime instantiation with canonicalization. Not to be used during type | 
|  | // finalization at compile time. | 
|  | TypeArgumentsPtr InstantiateAndCanonicalizeFrom( | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments) const; | 
|  |  | 
|  | class Cache : public ValueObject { | 
|  | public: | 
|  | // The contents of the backing array storage is a header followed by | 
|  | // a number of entry tuples. Any entry that is unoccupied has | 
|  | // Sentinel() as its first component. | 
|  | // | 
|  | // If the cache is linear, the entries can be accessed in a linear fashion: | 
|  | // all occupied entries come first, followed by at least one unoccupied | 
|  | // entry to mark the end of the cache. Guaranteeing at least one unoccupied | 
|  | // entry avoids the need for a length check when iterating over the contents | 
|  | // of the linear cache in stubs. | 
|  | // | 
|  | // If the cache is hash-based, the array is instead treated as a hash table | 
|  | // probed by using a hash value derived from the instantiator and function | 
|  | // type arguments. | 
|  |  | 
|  | enum Header { | 
|  | // A single Smi that is a bitfield containing two values: | 
|  | // - The number of occupied entries in the cache for all caches. | 
|  | // - For hash-based caches, the upper bits contain log2(N) where N | 
|  | //   is the number of total entries in the cache, so this information can | 
|  | //   be quickly retrieved by stubs. | 
|  | // | 
|  | // Note: accesses outside of the type arguments canonicalization mutex | 
|  | // must have acquire semantics. In C++ code, use NumOccupied to retrieve | 
|  | // the number of occupied entries. | 
|  | kMetadataIndex = 0, | 
|  | kHeaderSize, | 
|  | }; | 
|  |  | 
|  | using NumOccupiedBits = BitField<intptr_t, | 
|  | intptr_t, | 
|  | 0, | 
|  | compiler::target::kSmiBits - | 
|  | compiler::target::kBitsPerWordLog2>; | 
|  | using EntryCountLog2Bits = BitField<intptr_t, | 
|  | intptr_t, | 
|  | NumOccupiedBits::kNextBit, | 
|  | compiler::target::kBitsPerWordLog2>; | 
|  |  | 
|  | // The tuple of values stored in a given entry. | 
|  | // | 
|  | // Note: accesses of the first component outside of the type arguments | 
|  | // canonicalization mutex must have acquire semantics. | 
|  | enum Entry { | 
|  | kSentinelIndex = 0,  // Used when only checking for sentinel values. | 
|  | kInstantiatorTypeArgsIndex = kSentinelIndex, | 
|  | kFunctionTypeArgsIndex, | 
|  | kInstantiatedTypeArgsIndex, | 
|  | kEntrySize, | 
|  | }; | 
|  |  | 
|  | // Requires that the type arguments canonicalization mutex is held. | 
|  | Cache(Zone* zone, const TypeArguments& source); | 
|  |  | 
|  | // Requires that the type arguments canonicalization mutex is held. | 
|  | Cache(Zone* zone, const Array& array); | 
|  |  | 
|  | // Used to check that the state of the backing array is valid. | 
|  | // | 
|  | // Requires that the type arguments canonicalization mutex is held. | 
|  | DEBUG_ONLY(static bool IsValidStorageLocked(const Array& array);) | 
|  |  | 
|  | // Returns the number of entries stored in the cache. | 
|  | intptr_t NumOccupied() const { return NumOccupied(data_); } | 
|  |  | 
|  | struct KeyLocation { | 
|  | // The entry index if [present] is true, otherwise where the entry would | 
|  | // be located if added afterwards without any intermediate additions. | 
|  | intptr_t entry; | 
|  | bool present;  // Whether an entry already exists in the cache. | 
|  | }; | 
|  |  | 
|  | // If an entry contains the given instantiator and function type arguments, | 
|  | // returns a KeyLocation with the index of the entry and true. Otherwise, | 
|  | // returns the index an entry with those keys would have if added and false. | 
|  | KeyLocation FindKeyOrUnused(const TypeArguments& instantiator_tav, | 
|  | const TypeArguments& function_tav) const { | 
|  | return FindKeyOrUnused(data_, instantiator_tav, function_tav); | 
|  | } | 
|  |  | 
|  | // Returns whether the entry at the given index in the cache is occupied. | 
|  | bool IsOccupied(intptr_t entry) const; | 
|  |  | 
|  | // Given an occupied entry index, returns the instantiated TypeArguments. | 
|  | TypeArgumentsPtr Retrieve(intptr_t entry) const; | 
|  |  | 
|  | // Adds a new instantiation mapping to the cache at index [entry]. Assumes | 
|  | // that the entry at index [entry] is unoccupied. | 
|  | // | 
|  | // May replace the underlying storage array, in which case the returned | 
|  | // index of the entry may differ from the requested one. If this Cache was | 
|  | // constructed using a TypeArguments object, its instantiations field is | 
|  | // also updated to point to the new storage. | 
|  | KeyLocation AddEntry(intptr_t entry, | 
|  | const TypeArguments& instantiator_tav, | 
|  | const TypeArguments& function_tav, | 
|  | const TypeArguments& instantiated_tav) const; | 
|  |  | 
|  | // The sentinel value used to mark unoccupied entries. | 
|  | static SmiPtr Sentinel(); | 
|  |  | 
|  | static const Array& EmptyStorage() { | 
|  | return Object::empty_instantiations_cache_array(); | 
|  | } | 
|  |  | 
|  | // Returns whether the cache is linear. | 
|  | bool IsLinear() const { return IsLinear(data_); } | 
|  |  | 
|  | // Returns whether the cache is hash-based. | 
|  | bool IsHash() const { return IsHash(data_); } | 
|  |  | 
|  | private: | 
|  | static constexpr double LoadFactor(intptr_t occupied, intptr_t capacity) { | 
|  | return occupied / static_cast<double>(capacity); | 
|  | } | 
|  |  | 
|  | // Returns the number of entries stored in the cache backed by the given | 
|  | // array. | 
|  | static intptr_t NumOccupied(const Array& array); | 
|  |  | 
|  | // Returns whether the cache backed by the given storage is linear. | 
|  | static bool IsLinear(const Array& array) { return !IsHash(array); } | 
|  |  | 
|  | // Returns whether the cache backed by the given storage is hash-based. | 
|  | static bool IsHash(const Array& array); | 
|  |  | 
|  | // Ensures that the backing store for the cache can hold at least [occupied] | 
|  | // occupied entries. If it cannot, replaces the backing store with one that | 
|  | // can, copying over entries from the old backing store. | 
|  | // | 
|  | // Returns whether the backing store changed. | 
|  | bool EnsureCapacity(intptr_t occupied) const; | 
|  |  | 
|  | public:  // For testing purposes only. | 
|  | // Retrieves the number of entries (occupied or unoccupied) in the cache. | 
|  | intptr_t NumEntries() const { return NumEntries(data_); } | 
|  |  | 
|  | // The maximum number of occupied entries for a linear cache of | 
|  | // instantiations before swapping to a hash table-based cache. | 
|  | #if defined(TARGET_ARCH_IA32) | 
|  | // We don't generate hash cache probing in the stub on IA32. | 
|  | static constexpr intptr_t kMaxLinearCacheEntries = 500; | 
|  | #else | 
|  | static constexpr intptr_t kMaxLinearCacheEntries = 10; | 
|  | #endif | 
|  |  | 
|  | private: | 
|  | // Retrieves the number of entries (occupied or unoccupied) in a cache | 
|  | // backed by the given array. | 
|  | static intptr_t NumEntries(const Array& array); | 
|  |  | 
|  | // If an entry in the given array contains the given instantiator and | 
|  | // function type arguments, returns a KeyLocation with the index of the | 
|  | // entry and true. Otherwise, returns a KeyLocation with the index that | 
|  | // would be used if the instantiation for the given type arguments is | 
|  | // added and false. | 
|  | static KeyLocation FindKeyOrUnused(const Array& array, | 
|  | const TypeArguments& instantiator_tav, | 
|  | const TypeArguments& function_tav); | 
|  |  | 
|  | // The sentinel value in the Smi returned from Sentinel(). | 
|  | static constexpr intptr_t kSentinelValue = 0; | 
|  |  | 
|  | public:  // Used in the StubCodeCompiler. | 
|  | // The maximum size of the array backing a linear cache. All hash based | 
|  | // caches are guaranteed to have sizes larger than this. | 
|  | static constexpr intptr_t kMaxLinearCacheSize = | 
|  | kHeaderSize + (kMaxLinearCacheEntries + 1) * kEntrySize; | 
|  |  | 
|  | private: | 
|  | // The initial number of entries used when converting from a linear to | 
|  | // a hash-based cache. | 
|  | static constexpr intptr_t kNumInitialHashCacheEntries = | 
|  | Utils::RoundUpToPowerOfTwo(2 * kMaxLinearCacheEntries); | 
|  | static_assert(Utils::IsPowerOfTwo(kNumInitialHashCacheEntries), | 
|  | "number of hash-based cache entries must be a power of two"); | 
|  |  | 
|  | // The max load factor allowed in hash-based caches. | 
|  | static constexpr double kMaxLoadFactor = 0.71; | 
|  |  | 
|  | Zone* const zone_; | 
|  | const TypeArguments* const cache_container_; | 
|  | Array& data_; | 
|  | Smi& smi_handle_; | 
|  |  | 
|  | friend class TypeArguments;  // For asserts against data_. | 
|  | }; | 
|  |  | 
|  | // Return true if this type argument vector has cached instantiations. | 
|  | bool HasInstantiations() const; | 
|  |  | 
|  | static intptr_t instantiations_offset() { | 
|  | return OFFSET_OF(UntaggedTypeArguments, instantiations_); | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t kBytesPerElement = kCompressedWordSize; | 
|  | static constexpr intptr_t kMaxElements = kSmiMax / kBytesPerElement; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedTypeArguments) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedTypeArguments, types)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | // Ensure that the types() is not adding to the object size, which includes | 
|  | // 4 fields: instantiations_, length_, hash_, and nullability_. | 
|  | ASSERT(sizeof(UntaggedTypeArguments) == | 
|  | (sizeof(UntaggedObject) + (kNumFields * kCompressedWordSize))); | 
|  | ASSERT(0 <= len && len <= kMaxElements); | 
|  | return RoundedAllocationSize(sizeof(UntaggedTypeArguments) + | 
|  | (len * kBytesPerElement)); | 
|  | } | 
|  |  | 
|  | virtual uint32_t CanonicalizeHash() const { return Hash(); } | 
|  | uword Hash() const; | 
|  | uword HashForRange(intptr_t from_index, intptr_t len) const; | 
|  | static intptr_t hash_offset() { | 
|  | return OFFSET_OF(UntaggedTypeArguments, hash_); | 
|  | } | 
|  |  | 
|  | static TypeArgumentsPtr New(intptr_t len, Heap::Space space = Heap::kOld); | 
|  |  | 
|  | private: | 
|  | intptr_t ComputeNullability() const; | 
|  | void set_nullability(intptr_t value) const; | 
|  |  | 
|  | uword ComputeHash() const; | 
|  | void SetHash(intptr_t value) const; | 
|  |  | 
|  | // Check if the subvector of length 'len' starting at 'from_index' of this | 
|  | // type argument vector consists solely of DynamicType. | 
|  | // If raw_instantiated is true, consider each class type parameter to be first | 
|  | // instantiated from a vector of dynamic types. | 
|  | bool IsDynamicTypes(bool raw_instantiated, | 
|  | intptr_t from_index, | 
|  | intptr_t len) const; | 
|  |  | 
|  | ArrayPtr instantiations() const; | 
|  | void set_instantiations(const Array& value) const; | 
|  | void SetLength(intptr_t value) const; | 
|  | // Number of fields in the raw object is 4: | 
|  | // instantiations_, length_, hash_ and nullability_. | 
|  | static constexpr int kNumFields = 4; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(TypeArguments, Instance); | 
|  | friend class AbstractType; | 
|  | friend class Class; | 
|  | friend class ClearTypeHashVisitor; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | // AbstractType is an abstract superclass. | 
|  | // Subclasses of AbstractType are Type and TypeParameter. | 
|  | class AbstractType : public Instance { | 
|  | public: | 
|  | static intptr_t flags_offset() { | 
|  | return OFFSET_OF(UntaggedAbstractType, flags_); | 
|  | } | 
|  | static intptr_t hash_offset() { | 
|  | return OFFSET_OF(UntaggedAbstractType, hash_); | 
|  | } | 
|  |  | 
|  | bool IsFinalized() const { | 
|  | const auto state = type_state(); | 
|  | return (state == UntaggedAbstractType::kFinalizedInstantiated) || | 
|  | (state == UntaggedAbstractType::kFinalizedUninstantiated); | 
|  | } | 
|  | void SetIsFinalized() const; | 
|  |  | 
|  | Nullability nullability() const { | 
|  | return static_cast<Nullability>( | 
|  | UntaggedAbstractType::NullabilityBit::decode(untag()->flags())); | 
|  | } | 
|  | // Returns true if type has '?' nullability suffix, or it is a | 
|  | // built-in type which is always nullable (Null, dynamic or void). | 
|  | bool IsNullable() const { return nullability() == Nullability::kNullable; } | 
|  | // Returns true if type does not have any nullability suffix. | 
|  | // This function also returns true for type parameters without | 
|  | // nullability suffix ("T") which can be instantiated with | 
|  | // nullable or legacy types. | 
|  | bool IsNonNullable() const { | 
|  | return nullability() == Nullability::kNonNullable; | 
|  | } | 
|  | // Returns true if it is guaranteed that null cannot be | 
|  | // assigned to this type. | 
|  | bool IsStrictlyNonNullable() const; | 
|  |  | 
|  | virtual AbstractTypePtr SetInstantiatedNullability( | 
|  | const TypeParameter& type_param, | 
|  | Heap::Space space) const; | 
|  | virtual AbstractTypePtr NormalizeFutureOrType(Heap::Space space) const; | 
|  |  | 
|  | virtual bool HasTypeClass() const { return type_class_id() != kIllegalCid; } | 
|  | virtual classid_t type_class_id() const; | 
|  | virtual ClassPtr type_class() const; | 
|  | virtual TypeArgumentsPtr arguments() const; | 
|  | virtual bool IsInstantiated( | 
|  | Genericity genericity = kAny, | 
|  | intptr_t num_free_fun_type_params = kAllFree) const; | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const { | 
|  | return Equals(other); | 
|  | } | 
|  | virtual uint32_t CanonicalizeHash() const { return Hash(); } | 
|  | virtual bool Equals(const Instance& other) const { | 
|  | return IsEquivalent(other, TypeEquality::kCanonical); | 
|  | } | 
|  | virtual bool IsEquivalent( | 
|  | const Instance& other, | 
|  | TypeEquality kind, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | // Instantiate this type using the given type argument vectors. | 
|  | // | 
|  | // Note that some type parameters appearing in this type may not require | 
|  | // instantiation. Consider a class C<T> declaring a non-generic method | 
|  | // foo(bar<B>(T t, B b)). Although foo is not a generic method, it takes a | 
|  | // generic function bar<B> as argument and its function type refers to class | 
|  | // type parameter T and function type parameter B. When instantiating the | 
|  | // function type of foo for a particular value of T, function type parameter B | 
|  | // must remain uninstantiated, because only T is a free variable in this type. | 
|  | // | 
|  | // Return a new type, or return 'this' if it is already instantiated. | 
|  | virtual AbstractTypePtr InstantiateFrom( | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping = nullptr, | 
|  | intptr_t num_parent_type_args_adjustment = 0) const; | 
|  |  | 
|  | // Update number of parent function type arguments for the | 
|  | // nested function types and their type parameters. | 
|  | // | 
|  | // This adjustment is needed when nesting one generic function type | 
|  | // inside another. It is also needed when function type is copied | 
|  | // and owners of type parameters need to be adjusted. | 
|  | // | 
|  | // Number of parent function type arguments is adjusted by | 
|  | // [num_parent_type_args_adjustment]. | 
|  | // Type parameters up to [num_free_fun_type_params] are not adjusted. | 
|  | virtual AbstractTypePtr UpdateFunctionTypes( | 
|  | intptr_t num_parent_type_args_adjustment, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping) const; | 
|  |  | 
|  | // Caller must hold IsolateGroup::constant_canonicalization_mutex_. | 
|  | virtual InstancePtr CanonicalizeLocked(Thread* thread) const { | 
|  | return Canonicalize(thread); | 
|  | } | 
|  |  | 
|  | // Return the canonical version of this type. | 
|  | virtual AbstractTypePtr Canonicalize(Thread* thread) const; | 
|  |  | 
|  | // Add the pair <name, uri> to the list, if not already present. | 
|  | static void AddURI(URIs* uris, const String& name, const String& uri); | 
|  |  | 
|  | // Return a formatted string of the uris. | 
|  | static StringPtr PrintURIs(URIs* uris); | 
|  |  | 
|  | // Returns a C-String (possibly "") representing the nullability of this type. | 
|  | // Legacy and undetermined suffixes are only displayed with kInternalName. | 
|  | virtual const char* NullabilitySuffix(NameVisibility name_visibility) const; | 
|  |  | 
|  | // The name of this type, including the names of its type arguments, if any. | 
|  | StringPtr Name() const; | 
|  | const char* NameCString() const; | 
|  |  | 
|  | // The name of this type, including the names of its type arguments, if any. | 
|  | // Names of internal classes are mapped to their public interfaces. | 
|  | StringPtr UserVisibleName() const; | 
|  | const char* UserVisibleNameCString() const; | 
|  |  | 
|  | // The name of this type, including the names of its type arguments, if any. | 
|  | // Privacy suffixes are dropped. | 
|  | StringPtr ScrubbedName() const; | 
|  | const char* ScrubbedNameCString() const; | 
|  |  | 
|  | // Return the internal or public name of this type, including the names of its | 
|  | // type arguments, if any. | 
|  | virtual void PrintName(NameVisibility visibility, | 
|  | BaseTextBuffer* printer) const; | 
|  |  | 
|  | // Add the class name and URI of each occurring type to the uris | 
|  | // list and mark ambiguous triplets to be printed. | 
|  | virtual void EnumerateURIs(URIs* uris) const; | 
|  |  | 
|  | uword Hash() const; | 
|  | virtual uword ComputeHash() const; | 
|  |  | 
|  | // The name of this type's class, i.e. without the type argument names of this | 
|  | // type. | 
|  | StringPtr ClassName() const; | 
|  |  | 
|  | // Check if this type represents the 'dynamic' type. | 
|  | bool IsDynamicType() const { return type_class_id() == kDynamicCid; } | 
|  |  | 
|  | // Check if this type represents the 'void' type. | 
|  | bool IsVoidType() const { return type_class_id() == kVoidCid; } | 
|  |  | 
|  | // Check if this type represents the 'Null' type. | 
|  | bool IsNullType() const; | 
|  |  | 
|  | // Check if this type represents the 'Never' type. | 
|  | bool IsNeverType() const; | 
|  |  | 
|  | // Check if this type represents the 'Sentinel' type. | 
|  | bool IsSentinelType() const; | 
|  |  | 
|  | // Check if this type represents the 'Object' type. | 
|  | bool IsObjectType() const { return type_class_id() == kInstanceCid; } | 
|  |  | 
|  | // Check if this type represents the 'Object?' type. | 
|  | bool IsNullableObjectType() const { | 
|  | return IsObjectType() && (nullability() == Nullability::kNullable); | 
|  | } | 
|  |  | 
|  | // Check if this type represents a top type for subtyping, | 
|  | // assignability and 'as' type tests. | 
|  | // | 
|  | // Returns true if | 
|  | //  - any type is a subtype of this type; | 
|  | //  - any value can be assigned to a variable of this type; | 
|  | //  - 'as' type test always succeeds for this type. | 
|  | bool IsTopTypeForSubtyping() const; | 
|  |  | 
|  | // Check if this type represents a top type for 'is' type tests. | 
|  | // Returns true if 'is' type test always returns true for this type. | 
|  | bool IsTopTypeForInstanceOf() const; | 
|  |  | 
|  | // Check if this type represents the 'bool' type. | 
|  | bool IsBoolType() const { return type_class_id() == kBoolCid; } | 
|  |  | 
|  | // Check if this type represents the 'int' type. | 
|  | bool IsIntType() const; | 
|  |  | 
|  | // Check if this type represents the '_IntegerImplementation' type. | 
|  | bool IsIntegerImplementationType() const; | 
|  |  | 
|  | // Check if this type represents the 'double' type. | 
|  | bool IsDoubleType() const; | 
|  |  | 
|  | // Check if this type represents the 'Float32x4' type. | 
|  | bool IsFloat32x4Type() const; | 
|  |  | 
|  | // Check if this type represents the 'Float64x2' type. | 
|  | bool IsFloat64x2Type() const; | 
|  |  | 
|  | // Check if this type represents the 'Int32x4' type. | 
|  | bool IsInt32x4Type() const; | 
|  |  | 
|  | // Check if this type represents the 'num' type. | 
|  | bool IsNumberType() const { return type_class_id() == kNumberCid; } | 
|  |  | 
|  | // Check if this type represents the '_Smi' type. | 
|  | bool IsSmiType() const { return type_class_id() == kSmiCid; } | 
|  |  | 
|  | // Check if this type represents the '_Mint' type. | 
|  | bool IsMintType() const { return type_class_id() == kMintCid; } | 
|  |  | 
|  | // Check if this type represents the 'String' type. | 
|  | bool IsStringType() const; | 
|  |  | 
|  | // Check if this type represents the Dart 'Function' type. | 
|  | bool IsDartFunctionType() const; | 
|  |  | 
|  | // Check if this type represents the Dart '_Closure' type. | 
|  | bool IsDartClosureType() const; | 
|  |  | 
|  | // Check if this type represents the Dart 'Record' type. | 
|  | bool IsDartRecordType() const; | 
|  |  | 
|  | // Check if this type represents the 'Pointer' type from "dart:ffi". | 
|  | bool IsFfiPointerType() const; | 
|  |  | 
|  | // Check if this type represents the 'FutureOr' type. | 
|  | bool IsFutureOrType() const { return type_class_id() == kFutureOrCid; } | 
|  |  | 
|  | // Returns the type argument of this (possibly nested) 'FutureOr' type. | 
|  | // Returns unmodified type if this type is not a 'FutureOr' type. | 
|  | AbstractTypePtr UnwrapFutureOr() const; | 
|  |  | 
|  | // Returns true if catching this type will catch all exceptions. | 
|  | // Exception objects are guaranteed to be non-nullable, so | 
|  | // non-nullable Object is also a catch-all type. | 
|  | bool IsCatchAllType() const { return IsDynamicType() || IsObjectType(); } | 
|  |  | 
|  | // Returns true if this type has a type class permitted by SendPort.send for | 
|  | // messages between isolates in different groups. Does not recursively visit | 
|  | // type arguments. | 
|  | bool IsTypeClassAllowedBySpawnUri() const; | 
|  |  | 
|  | // Check the subtype relationship. | 
|  | bool IsSubtypeOf( | 
|  | const AbstractType& other, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | // Returns true iff subtype is a subtype of supertype, false otherwise or if | 
|  | // an error occurred. | 
|  | static bool InstantiateAndTestSubtype( | 
|  | AbstractType* subtype, | 
|  | AbstractType* supertype, | 
|  | const TypeArguments& instantiator_type_args, | 
|  | const TypeArguments& function_type_args); | 
|  |  | 
|  | static intptr_t type_test_stub_entry_point_offset() { | 
|  | return OFFSET_OF(UntaggedAbstractType, type_test_stub_entry_point_); | 
|  | } | 
|  |  | 
|  | uword type_test_stub_entry_point() const { | 
|  | return untag()->type_test_stub_entry_point_; | 
|  | } | 
|  | CodePtr type_test_stub() const { return untag()->type_test_stub(); } | 
|  |  | 
|  | // Sets the TTS to [stub]. | 
|  | // | 
|  | // The update will ensure both fields (code as well as the cached entrypoint) | 
|  | // are updated together. | 
|  | // | 
|  | // Can be used concurrently by multiple threads - the updates will be applied | 
|  | // in undetermined order - but always consistently. | 
|  | void SetTypeTestingStub(const Code& stub) const; | 
|  |  | 
|  | // Sets the TTS to the [stub]. | 
|  | // | 
|  | // The caller has to ensure no other thread can concurrently try to update the | 
|  | // TTS. This should mainly be used when initializing newly allocated Type | 
|  | // objects. | 
|  | void InitializeTypeTestingStubNonAtomic(const Code& stub) const; | 
|  |  | 
|  | void UpdateTypeTestingStubEntryPoint() const { | 
|  | StoreNonPointer(&untag()->type_test_stub_entry_point_, | 
|  | Code::EntryPointOf(untag()->type_test_stub())); | 
|  | } | 
|  |  | 
|  | // No instances of type AbstractType are allocated, but InstanceSize() and | 
|  | // NextFieldOffset() are required to register class _AbstractType. | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedAbstractType)); | 
|  | } | 
|  |  | 
|  | static intptr_t NextFieldOffset() { return -kWordSize; } | 
|  |  | 
|  | private: | 
|  | // Returns true if this type is a subtype of FutureOr<T> specified by 'other'. | 
|  | // Returns false if other type is not a FutureOr. | 
|  | bool IsSubtypeOfFutureOr( | 
|  | Zone* zone, | 
|  | const AbstractType& other, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | protected: | 
|  | bool IsNullabilityEquivalent(Thread* thread, | 
|  | const AbstractType& other_type, | 
|  | TypeEquality kind) const; | 
|  |  | 
|  | void SetHash(intptr_t value) const; | 
|  |  | 
|  | UntaggedAbstractType::TypeState type_state() const { | 
|  | return static_cast<UntaggedAbstractType::TypeState>( | 
|  | UntaggedAbstractType::TypeStateBits::decode(untag()->flags())); | 
|  | } | 
|  | void set_flags(uint32_t value) const; | 
|  | void set_type_state(UntaggedAbstractType::TypeState value) const; | 
|  | void set_nullability(Nullability value) const; | 
|  |  | 
|  | HEAP_OBJECT_IMPLEMENTATION(AbstractType, Instance); | 
|  | friend class Class; | 
|  | friend class ClearTypeHashVisitor; | 
|  | friend class Function; | 
|  | friend class TypeArguments; | 
|  | }; | 
|  |  | 
|  | // A Type consists of a class, possibly parameterized with type | 
|  | // arguments. Example: C<T1, T2>. | 
|  | class Type : public AbstractType { | 
|  | public: | 
|  | static intptr_t arguments_offset() { | 
|  | return OFFSET_OF(UntaggedType, arguments_); | 
|  | } | 
|  | virtual bool HasTypeClass() const { | 
|  | ASSERT(type_class_id() != kIllegalCid); | 
|  | return true; | 
|  | } | 
|  | TypePtr ToNullability(Nullability value, Heap::Space space) const; | 
|  | virtual classid_t type_class_id() const; | 
|  | virtual ClassPtr type_class() const; | 
|  | void set_type_class(const Class& value) const; | 
|  | virtual TypeArgumentsPtr arguments() const { return untag()->arguments(); } | 
|  | void set_arguments(const TypeArguments& value) const; | 
|  |  | 
|  | // Returns flattened instance type arguments vector for | 
|  | // instance of this type. | 
|  | TypeArgumentsPtr GetInstanceTypeArguments(Thread* thread, | 
|  | bool canonicalize = true) const; | 
|  |  | 
|  | virtual bool IsInstantiated( | 
|  | Genericity genericity = kAny, | 
|  | intptr_t num_free_fun_type_params = kAllFree) const; | 
|  | virtual bool IsEquivalent( | 
|  | const Instance& other, | 
|  | TypeEquality kind, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | // Return true if this type can be used as the declaration type of cls after | 
|  | // canonicalization (passed-in cls must match type_class()). | 
|  | bool IsDeclarationTypeOf(const Class& cls) const; | 
|  |  | 
|  | virtual AbstractTypePtr InstantiateFrom( | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping = nullptr, | 
|  | intptr_t num_parent_type_args_adjustment = 0) const; | 
|  |  | 
|  | virtual AbstractTypePtr UpdateFunctionTypes( | 
|  | intptr_t num_parent_type_args_adjustment, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping) const; | 
|  |  | 
|  | virtual AbstractTypePtr Canonicalize(Thread* thread) const; | 
|  | virtual void EnumerateURIs(URIs* uris) const; | 
|  | virtual void PrintName(NameVisibility visibility, | 
|  | BaseTextBuffer* printer) const; | 
|  |  | 
|  | virtual uword ComputeHash() const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedType)); | 
|  | } | 
|  |  | 
|  | // The type of the literal 'null'. | 
|  | static TypePtr NullType(); | 
|  |  | 
|  | // The 'dynamic' type. | 
|  | static TypePtr DynamicType(); | 
|  |  | 
|  | // The 'void' type. | 
|  | static TypePtr VoidType(); | 
|  |  | 
|  | // The 'Never' type. | 
|  | static TypePtr NeverType(); | 
|  |  | 
|  | // The 'Object' type. | 
|  | static TypePtr ObjectType(); | 
|  |  | 
|  | // The 'bool' type. | 
|  | static TypePtr BoolType(); | 
|  |  | 
|  | // The 'int' type. | 
|  | static TypePtr IntType(); | 
|  |  | 
|  | // The 'int?' type. | 
|  | static TypePtr NullableIntType(); | 
|  |  | 
|  | // The 'Smi' type. | 
|  | static TypePtr SmiType(); | 
|  |  | 
|  | // The 'Mint' type. | 
|  | static TypePtr MintType(); | 
|  |  | 
|  | // The 'double' type. | 
|  | static TypePtr Double(); | 
|  |  | 
|  | // The 'double?' type. | 
|  | static TypePtr NullableDouble(); | 
|  |  | 
|  | // The 'Float32x4' type. | 
|  | static TypePtr Float32x4(); | 
|  |  | 
|  | // The 'Float64x2' type. | 
|  | static TypePtr Float64x2(); | 
|  |  | 
|  | // The 'Int32x4' type. | 
|  | static TypePtr Int32x4(); | 
|  |  | 
|  | // The 'num' type. | 
|  | static TypePtr Number(); | 
|  |  | 
|  | // The 'num?' type. | 
|  | static TypePtr NullableNumber(); | 
|  |  | 
|  | // The 'String' type. | 
|  | static TypePtr StringType(); | 
|  |  | 
|  | // The 'Array' type. | 
|  | static TypePtr ArrayType(); | 
|  |  | 
|  | // The 'Function' type. | 
|  | static TypePtr DartFunctionType(); | 
|  |  | 
|  | // The 'Type' type. | 
|  | static TypePtr DartTypeType(); | 
|  |  | 
|  | // The finalized type of the given non-parameterized class. | 
|  | static TypePtr NewNonParameterizedType(const Class& type_class); | 
|  |  | 
|  | static TypePtr New(const Class& clazz, | 
|  | const TypeArguments& arguments, | 
|  | Nullability nullability = Nullability::kNonNullable, | 
|  | Heap::Space space = Heap::kOld); | 
|  |  | 
|  | private: | 
|  | // Takes an intptr_t since the cids of some classes are larger than will fit | 
|  | // in ClassIdTagType. This allows us to guard against that case, instead of | 
|  | // silently truncating the cid. | 
|  | void set_type_class_id(intptr_t id) const; | 
|  |  | 
|  | static TypePtr New(Heap::Space space = Heap::kOld); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Type, AbstractType); | 
|  | friend class Class; | 
|  | friend class TypeArguments; | 
|  | }; | 
|  |  | 
|  | // A FunctionType represents the type of a function. It describes most of the | 
|  | // signature of a function, excluding the names of type parameters and names | 
|  | // of parameters, but includes the names of optional named parameters. | 
|  | class FunctionType : public AbstractType { | 
|  | public: | 
|  | // Reexported so they can be used by the flow graph builders. | 
|  | using PackedNumParentTypeArguments = | 
|  | UntaggedFunctionType::PackedNumParentTypeArguments; | 
|  | using PackedNumTypeParameters = UntaggedFunctionType::PackedNumTypeParameters; | 
|  | using PackedHasNamedOptionalParameters = | 
|  | UntaggedFunctionType::PackedHasNamedOptionalParameters; | 
|  | using PackedNumImplicitParameters = | 
|  | UntaggedFunctionType::PackedNumImplicitParameters; | 
|  | using PackedNumFixedParameters = | 
|  | UntaggedFunctionType::PackedNumFixedParameters; | 
|  | using PackedNumOptionalParameters = | 
|  | UntaggedFunctionType::PackedNumOptionalParameters; | 
|  |  | 
|  | virtual bool HasTypeClass() const { return false; } | 
|  | FunctionTypePtr ToNullability(Nullability value, Heap::Space space) const; | 
|  | virtual classid_t type_class_id() const { return kIllegalCid; } | 
|  | virtual bool IsInstantiated( | 
|  | Genericity genericity = kAny, | 
|  | intptr_t num_free_fun_type_params = kAllFree) const; | 
|  | virtual bool IsEquivalent( | 
|  | const Instance& other, | 
|  | TypeEquality kind, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | virtual AbstractTypePtr InstantiateFrom( | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping = nullptr, | 
|  | intptr_t num_parent_type_args_adjustment = 0) const; | 
|  |  | 
|  | virtual AbstractTypePtr UpdateFunctionTypes( | 
|  | intptr_t num_parent_type_args_adjustment, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping) const; | 
|  |  | 
|  | virtual AbstractTypePtr Canonicalize(Thread* thread) const; | 
|  | virtual void EnumerateURIs(URIs* uris) const; | 
|  | virtual void PrintName(NameVisibility visibility, | 
|  | BaseTextBuffer* printer) const; | 
|  |  | 
|  | virtual uword ComputeHash() const; | 
|  |  | 
|  | bool IsSubtypeOf( | 
|  | const FunctionType& other, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | static intptr_t NumParentTypeArgumentsOf(FunctionTypePtr ptr) { | 
|  | return ptr->untag() | 
|  | ->packed_type_parameter_counts_.Read<PackedNumParentTypeArguments>(); | 
|  | } | 
|  | // Return the number of type arguments in the enclosing signature. | 
|  | intptr_t NumParentTypeArguments() const { | 
|  | return NumParentTypeArgumentsOf(ptr()); | 
|  | } | 
|  | void SetNumParentTypeArguments(intptr_t value) const; | 
|  | static intptr_t NumTypeParametersOf(FunctionTypePtr ptr) { | 
|  | return ptr->untag() | 
|  | ->packed_type_parameter_counts_.Read<PackedNumTypeParameters>(); | 
|  | } | 
|  | intptr_t NumTypeParameters() const { return NumTypeParametersOf(ptr()); } | 
|  |  | 
|  | static intptr_t NumTypeArgumentsOf(FunctionTypePtr ptr) { | 
|  | return NumTypeParametersOf(ptr) + NumParentTypeArgumentsOf(ptr); | 
|  | } | 
|  | intptr_t NumTypeArguments() const { return NumTypeArgumentsOf(ptr()); } | 
|  |  | 
|  | intptr_t num_implicit_parameters() const { | 
|  | return untag() | 
|  | ->packed_parameter_counts_.Read<PackedNumImplicitParameters>(); | 
|  | } | 
|  | void set_num_implicit_parameters(intptr_t value) const; | 
|  |  | 
|  | static intptr_t NumFixedParametersOf(FunctionTypePtr ptr) { | 
|  | return ptr->untag() | 
|  | ->packed_parameter_counts_.Read<PackedNumFixedParameters>(); | 
|  | } | 
|  | intptr_t num_fixed_parameters() const { return NumFixedParametersOf(ptr()); } | 
|  | void set_num_fixed_parameters(intptr_t value) const; | 
|  |  | 
|  | static bool HasOptionalParameters(FunctionTypePtr ptr) { | 
|  | return ptr->untag() | 
|  | ->packed_parameter_counts_.Read<PackedNumOptionalParameters>() > | 
|  | 0; | 
|  | } | 
|  | bool HasOptionalParameters() const { return HasOptionalParameters(ptr()); } | 
|  |  | 
|  | static bool HasOptionalNamedParameters(FunctionTypePtr ptr) { | 
|  | return ptr->untag() | 
|  | ->packed_parameter_counts_.Read<PackedHasNamedOptionalParameters>(); | 
|  | } | 
|  | bool HasOptionalNamedParameters() const { | 
|  | return HasOptionalNamedParameters(ptr()); | 
|  | } | 
|  | bool HasRequiredNamedParameters() const; | 
|  |  | 
|  | static bool HasOptionalPositionalParameters(FunctionTypePtr ptr) { | 
|  | return !HasOptionalNamedParameters(ptr) && HasOptionalParameters(ptr); | 
|  | } | 
|  | bool HasOptionalPositionalParameters() const { | 
|  | return HasOptionalPositionalParameters(ptr()); | 
|  | } | 
|  |  | 
|  | static intptr_t NumOptionalParametersOf(FunctionTypePtr ptr) { | 
|  | return ptr->untag() | 
|  | ->packed_parameter_counts_.Read<PackedNumOptionalParameters>(); | 
|  | } | 
|  | intptr_t NumOptionalParameters() const { | 
|  | return NumOptionalParametersOf(ptr()); | 
|  | } | 
|  | void SetNumOptionalParameters(intptr_t num_optional_parameters, | 
|  | bool are_optional_positional) const; | 
|  |  | 
|  | static intptr_t NumOptionalPositionalParametersOf(FunctionTypePtr ptr) { | 
|  | return HasOptionalNamedParameters(ptr) ? 0 : NumOptionalParametersOf(ptr); | 
|  | } | 
|  | intptr_t NumOptionalPositionalParameters() const { | 
|  | return NumOptionalPositionalParametersOf(ptr()); | 
|  | } | 
|  |  | 
|  | static intptr_t NumOptionalNamedParametersOf(FunctionTypePtr ptr) { | 
|  | return HasOptionalNamedParameters(ptr) ? NumOptionalParametersOf(ptr) : 0; | 
|  | } | 
|  | intptr_t NumOptionalNamedParameters() const { | 
|  | return NumOptionalNamedParametersOf(ptr()); | 
|  | } | 
|  |  | 
|  | static intptr_t NumParametersOf(FunctionTypePtr ptr) { | 
|  | return NumFixedParametersOf(ptr) + NumOptionalParametersOf(ptr); | 
|  | } | 
|  | intptr_t NumParameters() const { return NumParametersOf(ptr()); } | 
|  |  | 
|  | uint32_t packed_parameter_counts() const { | 
|  | return untag()->packed_parameter_counts_; | 
|  | } | 
|  | void set_packed_parameter_counts(uint32_t packed_parameter_counts) const; | 
|  | static intptr_t packed_parameter_counts_offset() { | 
|  | return OFFSET_OF(UntaggedFunctionType, packed_parameter_counts_); | 
|  | } | 
|  | uint16_t packed_type_parameter_counts() const { | 
|  | return untag()->packed_type_parameter_counts_; | 
|  | } | 
|  | void set_packed_type_parameter_counts(uint16_t packed_parameter_counts) const; | 
|  | static intptr_t packed_type_parameter_counts_offset() { | 
|  | return OFFSET_OF(UntaggedFunctionType, packed_type_parameter_counts_); | 
|  | } | 
|  |  | 
|  | // Return the type parameter declared at index. | 
|  | TypeParameterPtr TypeParameterAt( | 
|  | intptr_t index, | 
|  | Nullability nullability = Nullability::kNonNullable) const; | 
|  |  | 
|  | AbstractTypePtr result_type() const { return untag()->result_type(); } | 
|  | void set_result_type(const AbstractType& value) const; | 
|  |  | 
|  | // The parameters, starting with NumImplicitParameters() parameters which are | 
|  | // only visible to the VM, but not to Dart users. | 
|  | // Note that type checks exclude implicit parameters. | 
|  | AbstractTypePtr ParameterTypeAt(intptr_t index) const; | 
|  | void SetParameterTypeAt(intptr_t index, const AbstractType& value) const; | 
|  | ArrayPtr parameter_types() const { return untag()->parameter_types(); } | 
|  | void set_parameter_types(const Array& value) const; | 
|  | static intptr_t parameter_types_offset() { | 
|  | return OFFSET_OF(UntaggedFunctionType, parameter_types_); | 
|  | } | 
|  | // Parameter names are only stored for named parameters. If there are no named | 
|  | // parameters, named_parameter_names() is null. | 
|  | // If there are parameter flags (eg required) they're stored at the end of | 
|  | // this array, so the size of this array isn't necessarily | 
|  | // NumOptionalNamedParameters(), but the first NumOptionalNamedParameters() | 
|  | // elements are the names. | 
|  | ArrayPtr named_parameter_names() const { | 
|  | return untag()->named_parameter_names(); | 
|  | } | 
|  | void set_named_parameter_names(const Array& value) const; | 
|  | static intptr_t named_parameter_names_offset() { | 
|  | return OFFSET_OF(UntaggedFunctionType, named_parameter_names_); | 
|  | } | 
|  | // The index for these operations is the absolute index of the parameter, not | 
|  | // the index relative to the start of the named parameters (if any). | 
|  | StringPtr ParameterNameAt(intptr_t index) const; | 
|  | // Only valid for absolute indexes of named parameters. | 
|  | void SetParameterNameAt(intptr_t index, const String& value) const; | 
|  |  | 
|  | // The required flags are stored at the end of the parameter_names. The flags | 
|  | // are packed into SMIs, but omitted if they're 0. | 
|  | bool IsRequiredAt(intptr_t index) const; | 
|  | void SetIsRequiredAt(intptr_t index) const; | 
|  |  | 
|  | // Sets up the signature's parameter name array, including appropriate space | 
|  | // for any possible parameter flags. This may be an overestimate if some | 
|  | // parameters don't have flags, and so FinalizeNameArray() should | 
|  | // be called after all parameter flags have been appropriately set. | 
|  | // | 
|  | // Assumes that the number of fixed and optional parameters for the signature | 
|  | // has already been set. Uses same default space as FunctionType::New. | 
|  | void CreateNameArrayIncludingFlags(Heap::Space space = Heap::kOld) const; | 
|  |  | 
|  | // Truncate the parameter names array to remove any unused flag slots. Make | 
|  | // sure to only do this after calling SetIsRequiredAt as necessary. | 
|  | void FinalizeNameArray() const; | 
|  |  | 
|  | // Returns the length of the parameter names array that is required to store | 
|  | // all the names plus all their flags. This may be an overestimate if some | 
|  | // parameters don't have flags. | 
|  | static intptr_t NameArrayLengthIncludingFlags(intptr_t num_parameters); | 
|  |  | 
|  | // The formal type parameters, their bounds, and defaults, are specified as an | 
|  | // object of type TypeParameters. | 
|  | TypeParametersPtr type_parameters() const { | 
|  | return untag()->type_parameters(); | 
|  | } | 
|  | void SetTypeParameters(const TypeParameters& value) const; | 
|  | static intptr_t type_parameters_offset() { | 
|  | return OFFSET_OF(UntaggedFunctionType, type_parameters_); | 
|  | } | 
|  |  | 
|  | // Returns true if this function type has the same number of type parameters | 
|  | // with equal bounds as the other function type. Type parameter names and | 
|  | // parameter names (unless optional named) are ignored. | 
|  | bool HasSameTypeParametersAndBounds( | 
|  | const FunctionType& other, | 
|  | TypeEquality kind, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | // Return true if this function type declares type parameters. | 
|  | static bool IsGeneric(FunctionTypePtr ptr) { | 
|  | return ptr->untag()->type_parameters() != TypeParameters::null(); | 
|  | } | 
|  | bool IsGeneric() const { return IsGeneric(ptr()); } | 
|  |  | 
|  | // Return true if any enclosing signature of this signature is generic. | 
|  | bool HasGenericParent() const { return NumParentTypeArguments() > 0; } | 
|  |  | 
|  | // Returns true if the type of the formal parameter at the given position in | 
|  | // this function type is contravariant with the type of the other formal | 
|  | // parameter at the given position in the other function type. | 
|  | bool IsContravariantParameter( | 
|  | intptr_t parameter_position, | 
|  | const FunctionType& other, | 
|  | intptr_t other_parameter_position, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_equivalence) const; | 
|  |  | 
|  | // Returns the index in the parameter names array of the corresponding flag | 
|  | // for the given parameter index. Also returns (via flag_mask) the | 
|  | // corresponding mask within the flag. | 
|  | intptr_t GetRequiredFlagIndex(intptr_t index, intptr_t* flag_mask) const; | 
|  |  | 
|  | void Print(NameVisibility name_visibility, BaseTextBuffer* printer) const; | 
|  | void PrintParameters(Thread* thread, | 
|  | Zone* zone, | 
|  | NameVisibility name_visibility, | 
|  | BaseTextBuffer* printer) const; | 
|  |  | 
|  | StringPtr ToUserVisibleString() const; | 
|  | const char* ToUserVisibleCString() const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedFunctionType)); | 
|  | } | 
|  |  | 
|  | static FunctionTypePtr New( | 
|  | intptr_t num_parent_type_arguments = 0, | 
|  | Nullability nullability = Nullability::kNonNullable, | 
|  | Heap::Space space = Heap::kOld); | 
|  |  | 
|  | static FunctionTypePtr Clone(const FunctionType& orig, Heap::Space space); | 
|  |  | 
|  | bool ContainsHandles() const; | 
|  |  | 
|  | private: | 
|  | static FunctionTypePtr New(Heap::Space space); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(FunctionType, AbstractType); | 
|  | friend class Class; | 
|  | friend class Function; | 
|  | }; | 
|  |  | 
|  | // A TypeParameter represents a type parameter of a parameterized class. | 
|  | // It specifies its index (and its name for debugging purposes), as well as its | 
|  | // upper bound. | 
|  | // For example, the type parameter 'V' is specified as index 1 in the context of | 
|  | // the class HashMap<K, V>. At compile time, the TypeParameter is not | 
|  | // instantiated yet, i.e. it is only a place holder. | 
|  | // Upon finalization, the TypeParameter index is changed to reflect its position | 
|  | // as type argument (rather than type parameter) of the parameterized class. | 
|  | // If the type parameter is declared without an extends clause, its bound is set | 
|  | // to the ObjectType. | 
|  | class TypeParameter : public AbstractType { | 
|  | public: | 
|  | TypeParameterPtr ToNullability(Nullability value, Heap::Space space) const; | 
|  | virtual bool HasTypeClass() const { return false; } | 
|  | virtual classid_t type_class_id() const { return kIllegalCid; } | 
|  |  | 
|  | bool IsFunctionTypeParameter() const { | 
|  | return UntaggedTypeParameter::IsFunctionTypeParameter::decode( | 
|  | untag()->flags()); | 
|  | } | 
|  | bool IsClassTypeParameter() const { return !IsFunctionTypeParameter(); } | 
|  |  | 
|  | intptr_t base() const { return untag()->base_; } | 
|  | void set_base(intptr_t value) const; | 
|  | intptr_t index() const { return untag()->index_; } | 
|  | void set_index(intptr_t value) const; | 
|  | static intptr_t index_offset() { | 
|  | return OFFSET_OF(UntaggedTypeParameter, index_); | 
|  | } | 
|  |  | 
|  | classid_t parameterized_class_id() const; | 
|  | void set_parameterized_class_id(classid_t value) const; | 
|  | ClassPtr parameterized_class() const; | 
|  | FunctionTypePtr parameterized_function_type() const; | 
|  |  | 
|  | AbstractTypePtr bound() const; | 
|  |  | 
|  | virtual bool IsInstantiated( | 
|  | Genericity genericity = kAny, | 
|  | intptr_t num_free_fun_type_params = kAllFree) const; | 
|  | virtual bool IsEquivalent( | 
|  | const Instance& other, | 
|  | TypeEquality kind, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  | virtual AbstractTypePtr InstantiateFrom( | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping = nullptr, | 
|  | intptr_t num_parent_type_args_adjustment = 0) const; | 
|  |  | 
|  | virtual AbstractTypePtr UpdateFunctionTypes( | 
|  | intptr_t num_parent_type_args_adjustment, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping) const; | 
|  |  | 
|  | virtual AbstractTypePtr Canonicalize(Thread* thread) const; | 
|  | virtual void EnumerateURIs(URIs* uris) const { return; } | 
|  | virtual void PrintName(NameVisibility visibility, | 
|  | BaseTextBuffer* printer) const; | 
|  |  | 
|  | // Returns type corresponding to [this] type parameter from the | 
|  | // given [instantiator_type_arguments] and [function_type_arguments]. | 
|  | // Unlike InstantiateFrom, nullability of type parameter is not applied to | 
|  | // the result. | 
|  | AbstractTypePtr GetFromTypeArguments( | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments) const; | 
|  |  | 
|  | // Return a constructed name for this nameless type parameter. | 
|  | const char* CanonicalNameCString() const { | 
|  | return CanonicalNameCString(IsClassTypeParameter(), base(), index()); | 
|  | } | 
|  |  | 
|  | static const char* CanonicalNameCString(bool is_class_type_parameter, | 
|  | intptr_t base, | 
|  | intptr_t index); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedTypeParameter)); | 
|  | } | 
|  |  | 
|  | // 'owner' is a Class or FunctionType. | 
|  | static TypeParameterPtr New(const Object& owner, | 
|  | intptr_t base, | 
|  | intptr_t index, | 
|  | Nullability nullability); | 
|  |  | 
|  | private: | 
|  | virtual uword ComputeHash() const; | 
|  |  | 
|  | void set_owner(const Object& value) const; | 
|  |  | 
|  | static TypeParameterPtr New(); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(TypeParameter, AbstractType); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class Number : public Instance { | 
|  | public: | 
|  | // TODO(iposva): Add more useful Number methods. | 
|  | StringPtr ToString(Heap::Space space) const; | 
|  |  | 
|  | private: | 
|  | OBJECT_IMPLEMENTATION(Number, Instance); | 
|  |  | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class Integer : public Number { | 
|  | public: | 
|  | static IntegerPtr New(const String& str, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Creates a new Integer by given uint64_t value. | 
|  | // Silently casts value to int64_t with wrap-around if it is greater | 
|  | // than kMaxInt64. | 
|  | static IntegerPtr NewFromUint64(uint64_t value, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Returns a canonical Integer object allocated in the old gen space. | 
|  | // Returns null if integer is out of range. | 
|  | static IntegerPtr NewCanonical(const String& str); | 
|  | static IntegerPtr NewCanonical(int64_t value); | 
|  |  | 
|  | static IntegerPtr New(int64_t value, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Returns true iff the given uint64_t value is representable as Dart integer. | 
|  | static bool IsValueInRange(uint64_t value); | 
|  |  | 
|  | virtual bool OperatorEquals(const Instance& other) const { | 
|  | return Equals(other); | 
|  | } | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const { | 
|  | return Equals(other); | 
|  | } | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  | virtual bool Equals(const Instance& other) const; | 
|  |  | 
|  | virtual ObjectPtr HashCode() const { return ptr(); } | 
|  |  | 
|  | int64_t Value() const { return Value(ptr()); } | 
|  | static int64_t Value(IntegerPtr obj); | 
|  |  | 
|  | double ToDouble() const { return static_cast<double>(Value()); } | 
|  |  | 
|  | // Returns 0, -1 or 1. | 
|  | int CompareWith(const Integer& other) const; | 
|  |  | 
|  | // Converts integer to hex string. | 
|  | const char* ToHexCString(Zone* zone) const; | 
|  |  | 
|  | // Returns null to indicate that a bigint operation is required. | 
|  | IntegerPtr ArithmeticOp(Token::Kind operation, | 
|  | const Integer& other, | 
|  | Heap::Space space = Heap::kNew) const; | 
|  | IntegerPtr BitOp(Token::Kind operation, | 
|  | const Integer& other, | 
|  | Heap::Space space = Heap::kNew) const; | 
|  | IntegerPtr ShiftOp(Token::Kind operation, | 
|  | const Integer& other, | 
|  | Heap::Space space = Heap::kNew) const; | 
|  |  | 
|  | private: | 
|  | OBJECT_IMPLEMENTATION(Integer, Number); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class Smi : public Integer { | 
|  | public: | 
|  | static constexpr intptr_t kBits = kSmiBits; | 
|  | static constexpr intptr_t kMaxValue = kSmiMax; | 
|  | static constexpr intptr_t kMinValue = kSmiMin; | 
|  |  | 
|  | intptr_t Value() const { return RawSmiValue(ptr()); } | 
|  |  | 
|  | virtual bool Equals(const Instance& other) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { return 0; } | 
|  |  | 
|  | static SmiPtr New(intptr_t value) { | 
|  | SmiPtr raw_smi = static_cast<SmiPtr>( | 
|  | (static_cast<uintptr_t>(value) << kSmiTagShift) | kSmiTag); | 
|  | ASSERT(RawSmiValue(raw_smi) == value); | 
|  | return raw_smi; | 
|  | } | 
|  |  | 
|  | static ClassPtr Class(); | 
|  |  | 
|  | static intptr_t Value(const SmiPtr raw_smi) { return RawSmiValue(raw_smi); } | 
|  | #if defined(DART_COMPRESSED_POINTERS) | 
|  | static intptr_t Value(const CompressedSmiPtr raw_smi) { | 
|  | return Smi::Value(static_cast<SmiPtr>(raw_smi.DecompressSmi())); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | static intptr_t RawValue(intptr_t value) { | 
|  | return static_cast<intptr_t>(New(value)); | 
|  | } | 
|  |  | 
|  | static bool IsValid(int64_t value) { return compiler::target::IsSmi(value); } | 
|  |  | 
|  | void operator=(SmiPtr value) { | 
|  | ptr_ = value; | 
|  | CHECK_HANDLE(); | 
|  | } | 
|  | void operator^=(ObjectPtr value) { | 
|  | ptr_ = value; | 
|  | CHECK_HANDLE(); | 
|  | } | 
|  |  | 
|  | private: | 
|  | static intptr_t NextFieldOffset() { | 
|  | // Indicates this class cannot be extended by dart code. | 
|  | return -kWordSize; | 
|  | } | 
|  |  | 
|  | Smi() : Integer() {} | 
|  | BASE_OBJECT_IMPLEMENTATION(Smi, Integer); | 
|  | OBJECT_SERVICE_SUPPORT(Smi); | 
|  | friend class Api;  // For ValueFromRaw | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | friend class ReusableSmiHandleScope; | 
|  | friend class Thread; | 
|  | }; | 
|  |  | 
|  | class SmiTraits : AllStatic { | 
|  | public: | 
|  | static const char* Name() { return "SmiTraits"; } | 
|  | static bool ReportStats() { return false; } | 
|  |  | 
|  | static bool IsMatch(const Object& a, const Object& b) { | 
|  | return Smi::Cast(a).Value() == Smi::Cast(b).Value(); | 
|  | } | 
|  |  | 
|  | static uword Hash(const Object& obj) { return Smi::Cast(obj).Value(); } | 
|  | }; | 
|  |  | 
|  | class Mint : public Integer { | 
|  | public: | 
|  | static constexpr intptr_t kBits = 63;  // 64-th bit is sign. | 
|  | static constexpr int64_t kMaxValue = | 
|  | static_cast<int64_t>(DART_2PART_UINT64_C(0x7FFFFFFF, FFFFFFFF)); | 
|  | static constexpr int64_t kMinValue = | 
|  | static_cast<int64_t>(DART_2PART_UINT64_C(0x80000000, 00000000)); | 
|  |  | 
|  | static intptr_t value_offset() { return OFFSET_OF(UntaggedMint, value_); } | 
|  |  | 
|  | int64_t Value() const { return untag()->value_; } | 
|  | static int64_t Value(MintPtr mint) { return mint->untag()->value_; } | 
|  |  | 
|  | virtual bool Equals(const Instance& other) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedMint)); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | // Only Integer::NewXXX is allowed to call Mint::NewXXX directly. | 
|  | friend class Integer; | 
|  | friend class MintMessageDeserializationCluster; | 
|  |  | 
|  | static MintPtr New(int64_t value, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static MintPtr NewCanonical(int64_t value); | 
|  |  | 
|  | private: | 
|  | void set_value(int64_t value) const; | 
|  |  | 
|  | MINT_OBJECT_IMPLEMENTATION(Mint, Integer, Integer); | 
|  | friend class Class; | 
|  | friend class Number; | 
|  | }; | 
|  |  | 
|  | inline int64_t Integer::Value(IntegerPtr obj) { | 
|  | if (obj->IsSmi()) { | 
|  | return Smi::Value(Smi::RawCast(obj)); | 
|  | } else { | 
|  | return Mint::Value(Mint::RawCast(obj)); | 
|  | } | 
|  | } | 
|  |  | 
|  | // Class Double represents class Double in corelib_impl, which implements | 
|  | // abstract class double in corelib. | 
|  | class Double : public Number { | 
|  | public: | 
|  | double value() const { return untag()->value_; } | 
|  | static double Value(DoublePtr dbl) { return dbl->untag()->value_; } | 
|  |  | 
|  | bool BitwiseEqualsToDouble(double value) const; | 
|  | virtual bool OperatorEquals(const Instance& other) const; | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  |  | 
|  | static DoublePtr New(double d, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static DoublePtr New(const String& str, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Returns a canonical double object allocated in the old gen space. | 
|  | static DoublePtr NewCanonical(double d); | 
|  |  | 
|  | // Returns a canonical double object (allocated in the old gen space) or | 
|  | // Double::null() if str points to a string that does not convert to a | 
|  | // double value. | 
|  | static DoublePtr NewCanonical(const String& str); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedDouble)); | 
|  | } | 
|  |  | 
|  | static intptr_t value_offset() { return OFFSET_OF(UntaggedDouble, value_); } | 
|  |  | 
|  | private: | 
|  | void set_value(double value) const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Double, Number); | 
|  | friend class Class; | 
|  | friend class Number; | 
|  | }; | 
|  |  | 
|  | // TODO(http://dartbug.com/46716): Recognize Symbol in the VM. | 
|  | class Symbol : public AllStatic { | 
|  | public: | 
|  | static bool IsSymbolCid(Thread* thread, classid_t class_id); | 
|  |  | 
|  | static uint32_t CanonicalizeHash(Thread* thread, const Instance& instance); | 
|  | }; | 
|  |  | 
|  | // String may not be '\0' terminated. | 
|  | class String : public Instance { | 
|  | public: | 
|  | static constexpr intptr_t kOneByteChar = 1; | 
|  | static constexpr intptr_t kTwoByteChar = 2; | 
|  |  | 
|  | // All strings share the same maximum element count to keep things | 
|  | // simple.  We choose a value that will prevent integer overflow for | 
|  | // 2 byte strings, since it is the worst case. | 
|  | #if defined(HASH_IN_OBJECT_HEADER) | 
|  | static constexpr intptr_t kSizeofRawString = | 
|  | sizeof(UntaggedInstance) + kWordSize; | 
|  | #else | 
|  | static constexpr intptr_t kSizeofRawString = | 
|  | sizeof(UntaggedInstance) + 2 * kWordSize; | 
|  | #endif | 
|  | static constexpr intptr_t kMaxElements = kSmiMax / kTwoByteChar; | 
|  |  | 
|  | static intptr_t HeaderSize() { return String::kSizeofRawString; } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedString)); | 
|  | } | 
|  |  | 
|  | class CodePointIterator : public ValueObject { | 
|  | public: | 
|  | explicit CodePointIterator(const String& str) | 
|  | : str_(str), ch_(0), index_(-1), end_(str.Length()) { | 
|  | ASSERT(!str_.IsNull()); | 
|  | } | 
|  |  | 
|  | CodePointIterator(const String& str, intptr_t start, intptr_t length) | 
|  | : str_(str), ch_(0), index_(start - 1), end_(start + length) { | 
|  | ASSERT(start >= 0); | 
|  | ASSERT(end_ <= str.Length()); | 
|  | } | 
|  |  | 
|  | int32_t Current() const { | 
|  | ASSERT(index_ >= 0); | 
|  | ASSERT(index_ < end_); | 
|  | return ch_; | 
|  | } | 
|  |  | 
|  | bool Next(); | 
|  |  | 
|  | private: | 
|  | const String& str_; | 
|  | int32_t ch_; | 
|  | intptr_t index_; | 
|  | intptr_t end_; | 
|  | DISALLOW_IMPLICIT_CONSTRUCTORS(CodePointIterator); | 
|  | }; | 
|  |  | 
|  | intptr_t Length() const { return LengthOf(ptr()); } | 
|  | static intptr_t LengthOf(StringPtr obj) { | 
|  | return Smi::Value(obj->untag()->length()); | 
|  | } | 
|  | static intptr_t length_offset() { return OFFSET_OF(UntaggedString, length_); } | 
|  |  | 
|  | uword Hash() const { | 
|  | uword result = GetCachedHash(ptr()); | 
|  | if (result != 0) { | 
|  | return result; | 
|  | } | 
|  | result = String::Hash(*this, 0, this->Length()); | 
|  | uword set_hash = SetCachedHashIfNotSet(ptr(), result); | 
|  | ASSERT(set_hash == result); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | static uword Hash(StringPtr raw); | 
|  |  | 
|  | bool HasHash() const { | 
|  | ASSERT(Smi::New(0) == nullptr); | 
|  | return GetCachedHash(ptr()) != 0; | 
|  | } | 
|  |  | 
|  | static intptr_t hash_offset() { | 
|  | #if defined(HASH_IN_OBJECT_HEADER) | 
|  | COMPILE_ASSERT(UntaggedObject::HashTag::shift() % kBitsPerByte == 0); | 
|  | return OFFSET_OF(UntaggedObject, tags_) + | 
|  | UntaggedObject::HashTag::shift() / kBitsPerByte; | 
|  | #else | 
|  | return OFFSET_OF(UntaggedString, hash_); | 
|  | #endif | 
|  | } | 
|  | static uword Hash(const String& str, intptr_t begin_index, intptr_t len); | 
|  | static uword Hash(const char* characters, intptr_t len); | 
|  | static uword Hash(const uint16_t* characters, intptr_t len); | 
|  | static uword Hash(const int32_t* characters, intptr_t len); | 
|  | static uword HashRawSymbol(const StringPtr symbol) { | 
|  | ASSERT(symbol->untag()->IsCanonical()); | 
|  | const uword result = GetCachedHash(symbol); | 
|  | ASSERT(result != 0); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | // Returns the hash of str1 + str2. | 
|  | static uword HashConcat(const String& str1, const String& str2); | 
|  |  | 
|  | virtual ObjectPtr HashCode() const { return Integer::New(Hash()); } | 
|  |  | 
|  | uint16_t CharAt(intptr_t index) const { return CharAt(ptr(), index); } | 
|  | static uint16_t CharAt(StringPtr str, intptr_t index); | 
|  |  | 
|  | intptr_t CharSize() const; | 
|  |  | 
|  | inline bool Equals(const String& str) const; | 
|  |  | 
|  | bool Equals(const String& str, | 
|  | intptr_t begin_index,  // begin index on 'str'. | 
|  | intptr_t len) const;   // len on 'str'. | 
|  |  | 
|  | // Compares to a '\0' terminated array of UTF-8 encoded characters. | 
|  | bool Equals(const char* cstr) const; | 
|  |  | 
|  | // Compares to an array of Latin-1 encoded characters. | 
|  | bool EqualsLatin1(const uint8_t* characters, intptr_t len) const { | 
|  | return Equals(characters, len); | 
|  | } | 
|  |  | 
|  | // Compares to an array of UTF-16 encoded characters. | 
|  | bool Equals(const uint16_t* characters, intptr_t len) const; | 
|  |  | 
|  | // Compares to an array of UTF-32 encoded characters. | 
|  | bool Equals(const int32_t* characters, intptr_t len) const; | 
|  |  | 
|  | // True iff this string equals str1 + str2. | 
|  | bool EqualsConcat(const String& str1, const String& str2) const; | 
|  |  | 
|  | virtual bool OperatorEquals(const Instance& other) const { | 
|  | return Equals(other); | 
|  | } | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const { | 
|  | return Equals(other); | 
|  | } | 
|  | virtual uint32_t CanonicalizeHash() const { return Hash(); } | 
|  | virtual bool Equals(const Instance& other) const; | 
|  |  | 
|  | intptr_t CompareTo(const String& other) const; | 
|  |  | 
|  | bool StartsWith(const String& other) const { | 
|  | NoSafepointScope no_safepoint; | 
|  | return StartsWith(ptr(), other.ptr()); | 
|  | } | 
|  | static bool StartsWith(StringPtr str, StringPtr prefix); | 
|  | bool EndsWith(const String& other) const; | 
|  |  | 
|  | // Strings are canonicalized using the symbol table. | 
|  | // Caller must hold IsolateGroup::constant_canonicalization_mutex_. | 
|  | virtual InstancePtr CanonicalizeLocked(Thread* thread) const; | 
|  |  | 
|  | bool IsSymbol() const { return ptr()->untag()->IsCanonical(); } | 
|  |  | 
|  | bool IsOneByteString() const { | 
|  | return ptr()->GetClassIdOfHeapObject() == kOneByteStringCid; | 
|  | } | 
|  |  | 
|  | bool IsTwoByteString() const { | 
|  | return ptr()->GetClassIdOfHeapObject() == kTwoByteStringCid; | 
|  | } | 
|  |  | 
|  | char* ToMallocCString() const; | 
|  | void ToUTF8(uint8_t* utf8_array, intptr_t array_len) const; | 
|  | static const char* ToCString(Thread* thread, StringPtr ptr); | 
|  |  | 
|  | // Creates a new String object from a C string that is assumed to contain | 
|  | // UTF-8 encoded characters and '\0' is considered a termination character. | 
|  | // TODO(7123) - Rename this to FromCString(....). | 
|  | static StringPtr New(const char* cstr, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Creates a new String object from an array of UTF-8 encoded characters. | 
|  | static StringPtr FromUTF8(const uint8_t* utf8_array, | 
|  | intptr_t array_len, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Creates a new String object from an array of Latin-1 encoded characters. | 
|  | static StringPtr FromLatin1(const uint8_t* latin1_array, | 
|  | intptr_t array_len, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Creates a new String object from an array of UTF-16 encoded characters. | 
|  | static StringPtr FromUTF16(const uint16_t* utf16_array, | 
|  | intptr_t array_len, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Creates a new String object from an array of UTF-32 encoded characters. | 
|  | static StringPtr FromUTF32(const int32_t* utf32_array, | 
|  | intptr_t array_len, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Create a new String object from another Dart String instance. | 
|  | static StringPtr New(const String& str, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Creates a new External String object using the specified array of | 
|  | // UTF-8 encoded characters as the external reference. | 
|  | static StringPtr NewExternal(const uint8_t* utf8_array, | 
|  | intptr_t array_len, | 
|  | void* peer, | 
|  | intptr_t external_allocation_size, | 
|  | Dart_HandleFinalizer callback, | 
|  | Heap::Space = Heap::kNew); | 
|  |  | 
|  | // Creates a new External String object using the specified array of | 
|  | // UTF-16 encoded characters as the external reference. | 
|  | static StringPtr NewExternal(const uint16_t* utf16_array, | 
|  | intptr_t array_len, | 
|  | void* peer, | 
|  | intptr_t external_allocation_size, | 
|  | Dart_HandleFinalizer callback, | 
|  | Heap::Space = Heap::kNew); | 
|  |  | 
|  | static void Copy(const String& dst, | 
|  | intptr_t dst_offset, | 
|  | const uint8_t* characters, | 
|  | intptr_t len); | 
|  | static void Copy(const String& dst, | 
|  | intptr_t dst_offset, | 
|  | const uint16_t* characters, | 
|  | intptr_t len); | 
|  | static void Copy(const String& dst, | 
|  | intptr_t dst_offset, | 
|  | const String& src, | 
|  | intptr_t src_offset, | 
|  | intptr_t len); | 
|  |  | 
|  | static StringPtr EscapeSpecialCharacters(const String& str); | 
|  | // Encodes 'str' for use in an Internationalized Resource Identifier (IRI), | 
|  | // a generalization of URI (percent-encoding). See RFC 3987. | 
|  | static const char* EncodeIRI(const String& str); | 
|  | // Returns null if 'str' is not a valid encoding. | 
|  | static StringPtr DecodeIRI(const String& str); | 
|  | static StringPtr Concat(const String& str1, | 
|  | const String& str2, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static StringPtr ConcatAll(const Array& strings, | 
|  | Heap::Space space = Heap::kNew); | 
|  | // Concat all strings in 'strings' from 'start' to 'end' (excluding). | 
|  | static StringPtr ConcatAllRange(const Array& strings, | 
|  | intptr_t start, | 
|  | intptr_t end, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static StringPtr SubString(const String& str, | 
|  | intptr_t begin_index, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static StringPtr SubString(const String& str, | 
|  | intptr_t begin_index, | 
|  | intptr_t length, | 
|  | Heap::Space space = Heap::kNew) { | 
|  | return SubString(Thread::Current(), str, begin_index, length, space); | 
|  | } | 
|  | static StringPtr SubString(Thread* thread, | 
|  | const String& str, | 
|  | intptr_t begin_index, | 
|  | intptr_t length, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static StringPtr Transform(int32_t (*mapping)(int32_t ch), | 
|  | const String& str, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static StringPtr ToUpperCase(const String& str, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static StringPtr ToLowerCase(const String& str, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static StringPtr RemovePrivateKey(const String& name); | 
|  |  | 
|  | static const char* ScrubName(const String& name, bool is_extension = false); | 
|  | static StringPtr ScrubNameRetainPrivate(const String& name, | 
|  | bool is_extension = false); | 
|  |  | 
|  | static bool EqualsIgnoringPrivateKey(const String& str1, const String& str2); | 
|  |  | 
|  | static StringPtr NewFormatted(const char* format, ...) PRINTF_ATTRIBUTE(1, 2); | 
|  | static StringPtr NewFormatted(Heap::Space space, const char* format, ...) | 
|  | PRINTF_ATTRIBUTE(2, 3); | 
|  | static StringPtr NewFormattedV(const char* format, | 
|  | va_list args, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static bool ParseDouble(const String& str, | 
|  | intptr_t start, | 
|  | intptr_t end, | 
|  | double* result); | 
|  |  | 
|  | #if !defined(HASH_IN_OBJECT_HEADER) | 
|  | static uint32_t GetCachedHash(const StringPtr obj) { | 
|  | return Smi::Value(obj->untag()->hash_); | 
|  | } | 
|  |  | 
|  | static uint32_t SetCachedHashIfNotSet(StringPtr obj, uint32_t hash) { | 
|  | ASSERT(Smi::Value(obj->untag()->hash_) == 0 || | 
|  | Smi::Value(obj->untag()->hash_) == static_cast<intptr_t>(hash)); | 
|  | return SetCachedHash(obj, hash); | 
|  | } | 
|  | static uint32_t SetCachedHash(StringPtr obj, uint32_t hash) { | 
|  | obj->untag()->hash_ = Smi::New(hash); | 
|  | return hash; | 
|  | } | 
|  | #else | 
|  | static uint32_t SetCachedHash(StringPtr obj, uint32_t hash) { | 
|  | return Object::SetCachedHashIfNotSet(obj, hash); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | protected: | 
|  | // These two operate on an array of Latin-1 encoded characters. | 
|  | // They are protected to avoid mistaking Latin-1 for UTF-8, but used | 
|  | // by friendly templated code (e.g., Symbols). | 
|  | bool Equals(const uint8_t* characters, intptr_t len) const; | 
|  | static uword Hash(const uint8_t* characters, intptr_t len); | 
|  |  | 
|  | void SetLength(intptr_t value) const { | 
|  | // This is only safe because we create a new Smi, which does not cause | 
|  | // heap allocation. | 
|  | untag()->set_length(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | void SetHash(intptr_t value) const { | 
|  | const intptr_t hash_set = SetCachedHashIfNotSet(ptr(), value); | 
|  | ASSERT(hash_set == value); | 
|  | } | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(String, Instance); | 
|  |  | 
|  | friend class Class; | 
|  | friend class Symbols; | 
|  | friend class StringSlice;  // SetHash | 
|  | template <typename CharType> | 
|  | friend class CharArray;     // SetHash | 
|  | friend class ConcatString;  // SetHash | 
|  | friend class OneByteString; | 
|  | friend class TwoByteString; | 
|  | friend class UntaggedOneByteString; | 
|  | friend class RODataSerializationCluster;  // SetHash | 
|  | friend class Pass2Visitor;                // Stack "handle" | 
|  | }; | 
|  |  | 
|  | // Synchronize with implementation in compiler (intrinsifier). | 
|  | class StringHasher : public ValueObject { | 
|  | public: | 
|  | StringHasher() : hash_(0) {} | 
|  | void Add(uint16_t code_unit) { hash_ = CombineHashes(hash_, code_unit); } | 
|  | void Add(const uint8_t* code_units, intptr_t len) { | 
|  | while (len > 0) { | 
|  | Add(*code_units); | 
|  | code_units++; | 
|  | len--; | 
|  | } | 
|  | } | 
|  | void Add(const uint16_t* code_units, intptr_t len) { | 
|  | while (len > 0) { | 
|  | Add(LoadUnaligned(code_units)); | 
|  | code_units++; | 
|  | len--; | 
|  | } | 
|  | } | 
|  | void Add(const String& str, intptr_t begin_index, intptr_t len); | 
|  | intptr_t Finalize() { return FinalizeHash(hash_, String::kHashBits); } | 
|  |  | 
|  | private: | 
|  | uint32_t hash_; | 
|  | }; | 
|  |  | 
|  | class OneByteString : public AllStatic { | 
|  | public: | 
|  | static uint16_t CharAt(const String& str, intptr_t index) { | 
|  | ASSERT(str.IsOneByteString()); | 
|  | return OneByteString::CharAt(static_cast<OneByteStringPtr>(str.ptr()), | 
|  | index); | 
|  | } | 
|  |  | 
|  | static uint16_t CharAt(OneByteStringPtr str, intptr_t index) { | 
|  | ASSERT(index >= 0 && index < String::LengthOf(str)); | 
|  | return str->untag()->data()[index]; | 
|  | } | 
|  |  | 
|  | static void SetCharAt(const String& str, intptr_t index, uint8_t code_unit) { | 
|  | NoSafepointScope no_safepoint; | 
|  | *CharAddr(str, index) = code_unit; | 
|  | } | 
|  | static OneByteStringPtr EscapeSpecialCharacters(const String& str); | 
|  | // We use the same maximum elements for all strings. | 
|  | static constexpr intptr_t kBytesPerElement = 1; | 
|  | static constexpr intptr_t kMaxElements = String::kMaxElements; | 
|  | static constexpr intptr_t kMaxNewSpaceElements = | 
|  | (kNewAllocatableSize - sizeof(UntaggedOneByteString)) / kBytesPerElement; | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { | 
|  | return sizeof(UntaggedOneByteString); | 
|  | } | 
|  | static constexpr intptr_t kElementSize = kBytesPerElement; | 
|  | }; | 
|  |  | 
|  | static intptr_t data_offset() { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedOneByteString, data); | 
|  | } | 
|  |  | 
|  | static intptr_t UnroundedSize(OneByteStringPtr str) { | 
|  | return UnroundedSize(Smi::Value(str->untag()->length())); | 
|  | } | 
|  | static intptr_t UnroundedSize(intptr_t len) { | 
|  | return sizeof(UntaggedOneByteString) + (len * kBytesPerElement); | 
|  | } | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedOneByteString) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedOneByteString, data)); | 
|  | return 0; | 
|  | } | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | ASSERT(sizeof(UntaggedOneByteString) == String::kSizeofRawString); | 
|  | ASSERT(0 <= len && len <= kMaxElements); | 
|  | return String::RoundedAllocationSize(UnroundedSize(len)); | 
|  | } | 
|  |  | 
|  | static OneByteStringPtr New(intptr_t len, Heap::Space space); | 
|  | static OneByteStringPtr New(const char* c_string, | 
|  | Heap::Space space = Heap::kNew) { | 
|  | return New(reinterpret_cast<const uint8_t*>(c_string), strlen(c_string), | 
|  | space); | 
|  | } | 
|  | static OneByteStringPtr New(const uint8_t* characters, | 
|  | intptr_t len, | 
|  | Heap::Space space); | 
|  | static OneByteStringPtr New(const uint16_t* characters, | 
|  | intptr_t len, | 
|  | Heap::Space space); | 
|  | static OneByteStringPtr New(const int32_t* characters, | 
|  | intptr_t len, | 
|  | Heap::Space space); | 
|  | static OneByteStringPtr New(const String& str, Heap::Space space); | 
|  | // 'other' must be OneByteString. | 
|  | static OneByteStringPtr New(const String& other_one_byte_string, | 
|  | intptr_t other_start_index, | 
|  | intptr_t other_len, | 
|  | Heap::Space space); | 
|  |  | 
|  | static OneByteStringPtr New(const TypedDataBase& other_typed_data, | 
|  | intptr_t other_start_index, | 
|  | intptr_t other_len, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static OneByteStringPtr Concat(const String& str1, | 
|  | const String& str2, | 
|  | Heap::Space space); | 
|  | static OneByteStringPtr ConcatAll(const Array& strings, | 
|  | intptr_t start, | 
|  | intptr_t end, | 
|  | intptr_t len, | 
|  | Heap::Space space); | 
|  |  | 
|  | static OneByteStringPtr Transform(int32_t (*mapping)(int32_t ch), | 
|  | const String& str, | 
|  | Heap::Space space); | 
|  |  | 
|  | // High performance version of substring for one-byte strings. | 
|  | // "str" must be OneByteString. | 
|  | static OneByteStringPtr SubStringUnchecked(const String& str, | 
|  | intptr_t begin_index, | 
|  | intptr_t length, | 
|  | Heap::Space space); | 
|  |  | 
|  | static const ClassId kClassId = kOneByteStringCid; | 
|  |  | 
|  | static OneByteStringPtr null() { | 
|  | return static_cast<OneByteStringPtr>(Object::null()); | 
|  | } | 
|  |  | 
|  | private: | 
|  | static OneByteStringPtr raw(const String& str) { | 
|  | return static_cast<OneByteStringPtr>(str.ptr()); | 
|  | } | 
|  |  | 
|  | static const UntaggedOneByteString* untag(const String& str) { | 
|  | return reinterpret_cast<const UntaggedOneByteString*>(str.untag()); | 
|  | } | 
|  |  | 
|  | static uint8_t* CharAddr(const String& str, intptr_t index) { | 
|  | ASSERT((index >= 0) && (index < str.Length())); | 
|  | ASSERT(str.IsOneByteString()); | 
|  | return &str.UnsafeMutableNonPointer(untag(str)->data())[index]; | 
|  | } | 
|  |  | 
|  | static uint8_t* DataStart(const String& str) { | 
|  | ASSERT(str.IsOneByteString()); | 
|  | return &str.UnsafeMutableNonPointer(untag(str)->data())[0]; | 
|  | } | 
|  |  | 
|  | ALLSTATIC_CONTAINS_COMPRESSED_IMPLEMENTATION(OneByteString, String); | 
|  |  | 
|  | friend class Class; | 
|  | friend class FlowGraphSerializer; | 
|  | friend class ImageWriter; | 
|  | friend class String; | 
|  | friend class StringHasher; | 
|  | friend class Symbols; | 
|  | friend class Utf8; | 
|  | friend class OneByteStringMessageSerializationCluster; | 
|  | friend class Deserializer; | 
|  | friend class JSONWriter; | 
|  | }; | 
|  |  | 
|  | class TwoByteString : public AllStatic { | 
|  | public: | 
|  | static uint16_t CharAt(const String& str, intptr_t index) { | 
|  | ASSERT(str.IsTwoByteString()); | 
|  | return TwoByteString::CharAt(static_cast<TwoByteStringPtr>(str.ptr()), | 
|  | index); | 
|  | } | 
|  |  | 
|  | static uint16_t CharAt(TwoByteStringPtr str, intptr_t index) { | 
|  | ASSERT(index >= 0 && index < String::LengthOf(str)); | 
|  | return str->untag()->data()[index]; | 
|  | } | 
|  |  | 
|  | static void SetCharAt(const String& str, intptr_t index, uint16_t ch) { | 
|  | NoSafepointScope no_safepoint; | 
|  | *CharAddr(str, index) = ch; | 
|  | } | 
|  |  | 
|  | static TwoByteStringPtr EscapeSpecialCharacters(const String& str); | 
|  |  | 
|  | // We use the same maximum elements for all strings. | 
|  | static constexpr intptr_t kBytesPerElement = 2; | 
|  | static constexpr intptr_t kMaxElements = String::kMaxElements; | 
|  | static constexpr intptr_t kMaxNewSpaceElements = | 
|  | (kNewAllocatableSize - sizeof(UntaggedTwoByteString)) / kBytesPerElement; | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { | 
|  | return sizeof(UntaggedTwoByteString); | 
|  | } | 
|  | static constexpr intptr_t kElementSize = kBytesPerElement; | 
|  | }; | 
|  |  | 
|  | static intptr_t data_offset() { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedTwoByteString, data); | 
|  | } | 
|  | static intptr_t UnroundedSize(TwoByteStringPtr str) { | 
|  | return UnroundedSize(Smi::Value(str->untag()->length())); | 
|  | } | 
|  | static intptr_t UnroundedSize(intptr_t len) { | 
|  | return sizeof(UntaggedTwoByteString) + (len * kBytesPerElement); | 
|  | } | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedTwoByteString) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedTwoByteString, data)); | 
|  | return 0; | 
|  | } | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | ASSERT(sizeof(UntaggedTwoByteString) == String::kSizeofRawString); | 
|  | ASSERT(0 <= len && len <= kMaxElements); | 
|  | return String::RoundedAllocationSize(UnroundedSize(len)); | 
|  | } | 
|  |  | 
|  | static TwoByteStringPtr New(intptr_t len, Heap::Space space); | 
|  | static TwoByteStringPtr New(const uint16_t* characters, | 
|  | intptr_t len, | 
|  | Heap::Space space); | 
|  | static TwoByteStringPtr New(intptr_t utf16_len, | 
|  | const int32_t* characters, | 
|  | intptr_t len, | 
|  | Heap::Space space); | 
|  | static TwoByteStringPtr New(const String& str, Heap::Space space); | 
|  |  | 
|  | static TwoByteStringPtr New(const TypedDataBase& other_typed_data, | 
|  | intptr_t other_start_index, | 
|  | intptr_t other_len, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static TwoByteStringPtr Concat(const String& str1, | 
|  | const String& str2, | 
|  | Heap::Space space); | 
|  | static TwoByteStringPtr ConcatAll(const Array& strings, | 
|  | intptr_t start, | 
|  | intptr_t end, | 
|  | intptr_t len, | 
|  | Heap::Space space); | 
|  |  | 
|  | static TwoByteStringPtr Transform(int32_t (*mapping)(int32_t ch), | 
|  | const String& str, | 
|  | Heap::Space space); | 
|  |  | 
|  | static TwoByteStringPtr null() { | 
|  | return static_cast<TwoByteStringPtr>(Object::null()); | 
|  | } | 
|  |  | 
|  | static const ClassId kClassId = kTwoByteStringCid; | 
|  |  | 
|  | private: | 
|  | static TwoByteStringPtr raw(const String& str) { | 
|  | return static_cast<TwoByteStringPtr>(str.ptr()); | 
|  | } | 
|  |  | 
|  | static const UntaggedTwoByteString* untag(const String& str) { | 
|  | return reinterpret_cast<const UntaggedTwoByteString*>(str.untag()); | 
|  | } | 
|  |  | 
|  | static uint16_t* CharAddr(const String& str, intptr_t index) { | 
|  | ASSERT((index >= 0) && (index < str.Length())); | 
|  | ASSERT(str.IsTwoByteString()); | 
|  | return &str.UnsafeMutableNonPointer(untag(str)->data())[index]; | 
|  | } | 
|  |  | 
|  | // Use this instead of CharAddr(0).  It will not assert that the index is < | 
|  | // length. | 
|  | static uint16_t* DataStart(const String& str) { | 
|  | ASSERT(str.IsTwoByteString()); | 
|  | return &str.UnsafeMutableNonPointer(untag(str)->data())[0]; | 
|  | } | 
|  |  | 
|  | ALLSTATIC_CONTAINS_COMPRESSED_IMPLEMENTATION(TwoByteString, String); | 
|  |  | 
|  | friend class Class; | 
|  | friend class FlowGraphSerializer; | 
|  | friend class ImageWriter; | 
|  | friend class String; | 
|  | friend class StringHasher; | 
|  | friend class Symbols; | 
|  | friend class TwoByteStringMessageSerializationCluster; | 
|  | friend class JSONWriter; | 
|  | }; | 
|  |  | 
|  | // Matches null_patch.dart / bool_patch.dart. | 
|  | static constexpr intptr_t kNullIdentityHash = 2011; | 
|  | static constexpr intptr_t kTrueIdentityHash = 1231; | 
|  | static constexpr intptr_t kFalseIdentityHash = 1237; | 
|  |  | 
|  | // Class Bool implements Dart core class bool. | 
|  | class Bool : public Instance { | 
|  | public: | 
|  | bool value() const { return untag()->value_; } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedBool)); | 
|  | } | 
|  |  | 
|  | static const Bool& True() { return Object::bool_true(); } | 
|  |  | 
|  | static const Bool& False() { return Object::bool_false(); } | 
|  |  | 
|  | static const Bool& Get(bool value) { | 
|  | return value ? Bool::True() : Bool::False(); | 
|  | } | 
|  |  | 
|  | virtual uint32_t CanonicalizeHash() const { | 
|  | return ptr() == True().ptr() ? kTrueIdentityHash : kFalseIdentityHash; | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Bool, Instance); | 
|  | friend class Class; | 
|  | friend class Object;  // To initialize the true and false values. | 
|  | }; | 
|  |  | 
|  | class Array : public Instance { | 
|  | public: | 
|  | // Returns `true` if we use card marking for arrays of length [array_length]. | 
|  | static constexpr bool UseCardMarkingForAllocation( | 
|  | const intptr_t array_length) { | 
|  | return Array::InstanceSize(array_length) > kNewAllocatableSize; | 
|  | } | 
|  |  | 
|  | // WB invariant restoration code only applies to arrives which have at most | 
|  | // this many elements. Consequently WB elimination code should not eliminate | 
|  | // WB on arrays of larger lengths across instructions that can cause GC. | 
|  | // Note: we also can't restore WB invariant for arrays which use card marking. | 
|  | static constexpr intptr_t kMaxLengthForWriteBarrierElimination = 8; | 
|  |  | 
|  | intptr_t Length() const { return LengthOf(ptr()); } | 
|  | static intptr_t LengthOf(const ArrayPtr array) { | 
|  | return Smi::Value(array->untag()->length()); | 
|  | } | 
|  |  | 
|  | static intptr_t length_offset() { return OFFSET_OF(UntaggedArray, length_); } | 
|  | static intptr_t data_offset() { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedArray, data); | 
|  | } | 
|  | static intptr_t element_offset(intptr_t index) { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedArray, data) + | 
|  | kBytesPerElement * index; | 
|  | } | 
|  | static intptr_t index_at_offset(intptr_t offset_in_bytes) { | 
|  | intptr_t index = (offset_in_bytes - data_offset()) / kBytesPerElement; | 
|  | ASSERT(index >= 0); | 
|  | return index; | 
|  | } | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { return Array::data_offset(); } | 
|  |  | 
|  | static constexpr intptr_t kElementSize = kCompressedWordSize; | 
|  | }; | 
|  |  | 
|  | static bool Equals(ArrayPtr a, ArrayPtr b) { | 
|  | if (a == b) return true; | 
|  | if (a->IsRawNull() || b->IsRawNull()) return false; | 
|  | if (a->untag()->length() != b->untag()->length()) return false; | 
|  | if (a->untag()->type_arguments() != b->untag()->type_arguments()) { | 
|  | return false; | 
|  | } | 
|  | const intptr_t length = LengthOf(a); | 
|  | return memcmp(a->untag()->data(), b->untag()->data(), | 
|  | kBytesPerElement * length) == 0; | 
|  | } | 
|  | bool Equals(const Array& other) const { | 
|  | NoSafepointScope scope; | 
|  | return Equals(ptr(), other.ptr()); | 
|  | } | 
|  |  | 
|  | static CompressedObjectPtr* DataOf(ArrayPtr array) { | 
|  | return array->untag()->data(); | 
|  | } | 
|  |  | 
|  | template <std::memory_order order = std::memory_order_relaxed> | 
|  | ObjectPtr At(intptr_t index) const { | 
|  | ASSERT((0 <= index) && (index < Length())); | 
|  | return untag()->element<order>(index); | 
|  | } | 
|  | template <std::memory_order order = std::memory_order_relaxed> | 
|  | void SetAt(intptr_t index, const Object& value) const { | 
|  | ASSERT((0 <= index) && (index < Length())); | 
|  | untag()->set_element<order>(index, value.ptr()); | 
|  | } | 
|  | template <std::memory_order order = std::memory_order_relaxed> | 
|  | void SetAt(intptr_t index, const Object& value, Thread* thread) const { | 
|  | ASSERT((0 <= index) && (index < Length())); | 
|  | untag()->set_element<order>(index, value.ptr(), thread); | 
|  | } | 
|  |  | 
|  | // Access to the array with acquire release semantics. | 
|  | ObjectPtr AtAcquire(intptr_t index) const { | 
|  | ASSERT((0 <= index) && (index < Length())); | 
|  | return untag()->element<std::memory_order_acquire>(index); | 
|  | } | 
|  | void SetAtRelease(intptr_t index, const Object& value) const { | 
|  | ASSERT((0 <= index) && (index < Length())); | 
|  | untag()->set_element<std::memory_order_release>(index, value.ptr()); | 
|  | } | 
|  |  | 
|  | bool IsImmutable() const { | 
|  | return ptr()->GetClassIdOfHeapObject() == kImmutableArrayCid; | 
|  | } | 
|  |  | 
|  | // Position of element type in type arguments. | 
|  | static constexpr intptr_t kElementTypeTypeArgPos = 0; | 
|  |  | 
|  | virtual TypeArgumentsPtr GetTypeArguments() const { | 
|  | return untag()->type_arguments(); | 
|  | } | 
|  | virtual void SetTypeArguments(const TypeArguments& value) const { | 
|  | // An Array is raw or takes one type argument. However, its type argument | 
|  | // vector may be longer than 1 due to a type optimization reusing the type | 
|  | // argument vector of the instantiator. | 
|  | ASSERT(value.IsNull() || | 
|  | ((value.Length() >= 1) && | 
|  | value.IsInstantiated() /*&& value.IsCanonical()*/)); | 
|  | // TODO(asiva): Values read from a message snapshot are not properly marked | 
|  | // as canonical. See for example tests/isolate/mandel_isolate_test.dart. | 
|  | StoreArrayPointer(&untag()->type_arguments_, value.ptr()); | 
|  | } | 
|  |  | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  |  | 
|  | static constexpr intptr_t kBytesPerElement = ArrayTraits::kElementSize; | 
|  | static constexpr intptr_t kMaxElements = kSmiMax / kBytesPerElement; | 
|  | static constexpr intptr_t kMaxNewSpaceElements = | 
|  | (kNewAllocatableSize - sizeof(UntaggedArray)) / kBytesPerElement; | 
|  |  | 
|  | static intptr_t type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedArray, type_arguments_); | 
|  | } | 
|  |  | 
|  | static constexpr bool IsValidLength(intptr_t len) { | 
|  | return 0 <= len && len <= kMaxElements; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedArray) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedArray, data)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t UnroundedSize(intptr_t len) { | 
|  | // Ensure that variable length data is not adding to the object length. | 
|  | ASSERT(sizeof(UntaggedArray) == | 
|  | (sizeof(UntaggedInstance) + (2 * kBytesPerElement))); | 
|  | ASSERT(IsValidLength(len)); | 
|  | return sizeof(UntaggedArray) + (len * kBytesPerElement); | 
|  | } | 
|  | static constexpr intptr_t InstanceSize(intptr_t len) { | 
|  | return RoundedAllocationSize(UnroundedSize(len)); | 
|  | } | 
|  |  | 
|  | virtual void CanonicalizeFieldsLocked(Thread* thread) const; | 
|  |  | 
|  | // Make the array immutable to Dart code by switching the class pointer | 
|  | // to ImmutableArray. | 
|  | void MakeImmutable() const; | 
|  |  | 
|  | static ArrayPtr New(intptr_t len, Heap::Space space = Heap::kNew) { | 
|  | return New(kArrayCid, len, space); | 
|  | } | 
|  | // The result's type arguments and elements are GC-safe but not initialized to | 
|  | // null. | 
|  | static ArrayPtr NewUninitialized(intptr_t len, | 
|  | Heap::Space space = Heap::kNew) { | 
|  | return NewUninitialized(kArrayCid, len, space); | 
|  | } | 
|  | static ArrayPtr New(intptr_t len, | 
|  | const AbstractType& element_type, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Creates and returns a new array with 'new_length'. Copies all elements from | 
|  | // 'source' to the new array. 'new_length' must be greater than or equal to | 
|  | // 'source.Length()'. 'source' can be null. | 
|  | static ArrayPtr Grow(const Array& source, | 
|  | intptr_t new_length, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Truncates the array to a given length. 'new_length' must be less than | 
|  | // or equal to 'source.Length()'. The remaining unused part of the array is | 
|  | // marked as an Array object or a regular Object so that it can be traversed | 
|  | // during garbage collection. | 
|  | void Truncate(intptr_t new_length) const; | 
|  |  | 
|  | // Return an Array object that contains all the elements currently present | 
|  | // in the specified Growable Object Array. This is done by first truncating | 
|  | // the Growable Object Array's backing array to the currently used size and | 
|  | // returning the truncated backing array. | 
|  | // The backing array of the original Growable Object Array is | 
|  | // set to an empty array. | 
|  | // If the unique parameter is false, the function is allowed to return | 
|  | // a shared Array instance. | 
|  | static ArrayPtr MakeFixedLength(const GrowableObjectArray& growable_array, | 
|  | bool unique = false); | 
|  |  | 
|  | ArrayPtr Slice(intptr_t start, intptr_t count, bool with_type_argument) const; | 
|  | ArrayPtr Copy() const { | 
|  | return Slice(0, Length(), /*with_type_argument=*/true); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | static ArrayPtr New(intptr_t class_id, | 
|  | intptr_t len, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static ArrayPtr NewUninitialized(intptr_t class_id, | 
|  | intptr_t len, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | CompressedObjectPtr const* ObjectAddr(intptr_t index) const { | 
|  | // TODO(iposva): Determine if we should throw an exception here. | 
|  | ASSERT((index >= 0) && (index < Length())); | 
|  | return &untag()->data()[index]; | 
|  | } | 
|  |  | 
|  | void SetLength(intptr_t value) const { untag()->set_length(Smi::New(value)); } | 
|  | void SetLengthRelease(intptr_t value) const { | 
|  | untag()->set_length<std::memory_order_release>(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | template <typename type, | 
|  | std::memory_order order = std::memory_order_relaxed, | 
|  | typename value_type> | 
|  | void StoreArrayPointer(type const* addr, value_type value) const { | 
|  | ptr()->untag()->StoreArrayPointer<type, order, value_type>(addr, value); | 
|  | } | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Array, Instance); | 
|  | friend class Class; | 
|  | friend class ImmutableArray; | 
|  | friend class Interpreter; | 
|  | friend class Object; | 
|  | friend class String; | 
|  | friend class MessageDeserializer; | 
|  | }; | 
|  |  | 
|  | class ImmutableArray : public AllStatic { | 
|  | public: | 
|  | static constexpr bool ContainsCompressedPointers() { | 
|  | return Array::ContainsCompressedPointers(); | 
|  | } | 
|  |  | 
|  | static ImmutableArrayPtr New(intptr_t len, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static const ClassId kClassId = kImmutableArrayCid; | 
|  |  | 
|  | static intptr_t InstanceSize() { return Array::InstanceSize(); } | 
|  |  | 
|  | static intptr_t InstanceSize(intptr_t len) { | 
|  | return Array::InstanceSize(len); | 
|  | } | 
|  |  | 
|  | private: | 
|  | static intptr_t NextFieldOffset() { | 
|  | // Indicates this class cannot be extended by dart code. | 
|  | return -kWordSize; | 
|  | } | 
|  |  | 
|  | static ImmutableArrayPtr raw(const Array& array) { | 
|  | return static_cast<ImmutableArrayPtr>(array.ptr()); | 
|  | } | 
|  |  | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class GrowableObjectArray : public Instance { | 
|  | public: | 
|  | intptr_t Capacity() const { | 
|  | NoSafepointScope no_safepoint; | 
|  | ASSERT(!IsNull()); | 
|  | return Smi::Value(DataArray()->length()); | 
|  | } | 
|  | intptr_t Length() const { | 
|  | ASSERT(!IsNull()); | 
|  | return Smi::Value(untag()->length()); | 
|  | } | 
|  | void SetLength(intptr_t value) const { | 
|  | // This is only safe because we create a new Smi, which does not cause | 
|  | // heap allocation. | 
|  | untag()->set_length(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | ArrayPtr data() const { return untag()->data(); } | 
|  | void SetData(const Array& value) const { untag()->set_data(value.ptr()); } | 
|  |  | 
|  | ObjectPtr At(intptr_t index) const { | 
|  | NoSafepointScope no_safepoint; | 
|  | ASSERT(!IsNull()); | 
|  | ASSERT(index < Length()); | 
|  | return data()->untag()->element(index); | 
|  | } | 
|  | void SetAt(intptr_t index, const Object& value) const { | 
|  | ASSERT(!IsNull()); | 
|  | ASSERT(index < Length()); | 
|  |  | 
|  | // TODO(iposva): Add storing NoSafepointScope. | 
|  | data()->untag()->set_element(index, value.ptr()); | 
|  | } | 
|  |  | 
|  | void Add(const Object& value, Heap::Space space = Heap::kNew) const; | 
|  |  | 
|  | void Grow(intptr_t new_capacity, Heap::Space space = Heap::kNew) const; | 
|  | ObjectPtr RemoveLast() const; | 
|  |  | 
|  | virtual TypeArgumentsPtr GetTypeArguments() const { | 
|  | return untag()->type_arguments(); | 
|  | } | 
|  | virtual void SetTypeArguments(const TypeArguments& value) const { | 
|  | // A GrowableObjectArray is raw or takes one type argument. However, its | 
|  | // type argument vector may be longer than 1 due to a type optimization | 
|  | // reusing the type argument vector of the instantiator. | 
|  | ASSERT(value.IsNull() || ((value.Length() >= 1) && value.IsInstantiated() && | 
|  | value.IsCanonical())); | 
|  |  | 
|  | untag()->set_type_arguments(value.ptr()); | 
|  | } | 
|  |  | 
|  | // We don't expect a growable object array to be canonicalized. | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const { | 
|  | UNREACHABLE(); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // We don't expect a growable object array to be canonicalized. | 
|  | virtual InstancePtr CanonicalizeLocked(Thread* thread) const { | 
|  | UNREACHABLE(); | 
|  | return Instance::null(); | 
|  | } | 
|  |  | 
|  | static intptr_t type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedGrowableObjectArray, type_arguments_); | 
|  | } | 
|  |  | 
|  | static intptr_t length_offset() { | 
|  | return OFFSET_OF(UntaggedGrowableObjectArray, length_); | 
|  | } | 
|  | static intptr_t data_offset() { | 
|  | return OFFSET_OF(UntaggedGrowableObjectArray, data_); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedGrowableObjectArray)); | 
|  | } | 
|  |  | 
|  | static GrowableObjectArrayPtr New(Heap::Space space = Heap::kNew) { | 
|  | return New(kDefaultInitialCapacity, space); | 
|  | } | 
|  | static GrowableObjectArrayPtr New(intptr_t capacity, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static GrowableObjectArrayPtr New(const Array& array, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static SmiPtr NoSafepointLength(const GrowableObjectArrayPtr array) { | 
|  | return array->untag()->length(); | 
|  | } | 
|  |  | 
|  | static ArrayPtr NoSafepointData(const GrowableObjectArrayPtr array) { | 
|  | return array->untag()->data(); | 
|  | } | 
|  |  | 
|  | private: | 
|  | UntaggedArray* DataArray() const { return data()->untag(); } | 
|  |  | 
|  | static constexpr int kDefaultInitialCapacity = 0; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(GrowableObjectArray, Instance); | 
|  | friend class Array; | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class Float32x4 : public Instance { | 
|  | public: | 
|  | static Float32x4Ptr New(float value0, | 
|  | float value1, | 
|  | float value2, | 
|  | float value3, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static Float32x4Ptr New(simd128_value_t value, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | float x() const; | 
|  | float y() const; | 
|  | float z() const; | 
|  | float w() const; | 
|  |  | 
|  | void set_x(float x) const; | 
|  | void set_y(float y) const; | 
|  | void set_z(float z) const; | 
|  | void set_w(float w) const; | 
|  |  | 
|  | simd128_value_t value() const; | 
|  | void set_value(simd128_value_t value) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedFloat32x4)); | 
|  | } | 
|  |  | 
|  | static intptr_t value_offset() { | 
|  | return OFFSET_OF(UntaggedFloat32x4, value_); | 
|  | } | 
|  |  | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Float32x4, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class Int32x4 : public Instance { | 
|  | public: | 
|  | static Int32x4Ptr New(int32_t value0, | 
|  | int32_t value1, | 
|  | int32_t value2, | 
|  | int32_t value3, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static Int32x4Ptr New(simd128_value_t value, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | int32_t x() const; | 
|  | int32_t y() const; | 
|  | int32_t z() const; | 
|  | int32_t w() const; | 
|  |  | 
|  | void set_x(int32_t x) const; | 
|  | void set_y(int32_t y) const; | 
|  | void set_z(int32_t z) const; | 
|  | void set_w(int32_t w) const; | 
|  |  | 
|  | simd128_value_t value() const; | 
|  | void set_value(simd128_value_t value) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedInt32x4)); | 
|  | } | 
|  |  | 
|  | static intptr_t value_offset() { return OFFSET_OF(UntaggedInt32x4, value_); } | 
|  |  | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Int32x4, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class Float64x2 : public Instance { | 
|  | public: | 
|  | static Float64x2Ptr New(double value0, | 
|  | double value1, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static Float64x2Ptr New(simd128_value_t value, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | double x() const; | 
|  | double y() const; | 
|  |  | 
|  | void set_x(double x) const; | 
|  | void set_y(double y) const; | 
|  |  | 
|  | simd128_value_t value() const; | 
|  | void set_value(simd128_value_t value) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedFloat64x2)); | 
|  | } | 
|  |  | 
|  | static intptr_t value_offset() { | 
|  | return OFFSET_OF(UntaggedFloat64x2, value_); | 
|  | } | 
|  |  | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Float64x2, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Packed representation of record shape (number of fields and field names). | 
|  | class RecordShape { | 
|  | using NumFieldsBitField = BitField<intptr_t, intptr_t, 0, 16>; | 
|  | using FieldNamesIndexBitField = | 
|  | BitField<intptr_t, | 
|  | intptr_t, | 
|  | NumFieldsBitField::kNextBit, | 
|  | kSmiBits - NumFieldsBitField::kNextBit>; | 
|  |  | 
|  | public: | 
|  | static constexpr intptr_t kNumFieldsMask = NumFieldsBitField::mask(); | 
|  | static constexpr intptr_t kMaxNumFields = NumFieldsBitField::max(); | 
|  | static constexpr intptr_t kFieldNamesIndexMask = | 
|  | FieldNamesIndexBitField::mask(); | 
|  | static constexpr intptr_t kFieldNamesIndexShift = | 
|  | FieldNamesIndexBitField::shift(); | 
|  | static constexpr intptr_t kMaxFieldNamesIndex = | 
|  | FieldNamesIndexBitField::max(); | 
|  |  | 
|  | explicit RecordShape(intptr_t value) : value_(value) { ASSERT(value_ >= 0); } | 
|  | explicit RecordShape(SmiPtr smi_value) : value_(Smi::Value(smi_value)) { | 
|  | ASSERT(value_ >= 0); | 
|  | } | 
|  | RecordShape(intptr_t num_fields, intptr_t field_names_index) | 
|  | : value_(NumFieldsBitField::encode(num_fields) | | 
|  | FieldNamesIndexBitField::encode(field_names_index)) { | 
|  | ASSERT(value_ >= 0); | 
|  | } | 
|  | static RecordShape ForUnnamed(intptr_t num_fields) { | 
|  | return RecordShape(num_fields, 0); | 
|  | } | 
|  |  | 
|  | bool HasNamedFields() const { return field_names_index() != 0; } | 
|  |  | 
|  | intptr_t num_fields() const { return NumFieldsBitField::decode(value_); } | 
|  |  | 
|  | intptr_t field_names_index() const { | 
|  | return FieldNamesIndexBitField::decode(value_); | 
|  | } | 
|  |  | 
|  | SmiPtr AsSmi() const { return Smi::New(value_); } | 
|  |  | 
|  | intptr_t AsInt() const { return value_; } | 
|  |  | 
|  | bool operator==(const RecordShape& other) const { | 
|  | return value_ == other.value_; | 
|  | } | 
|  | bool operator!=(const RecordShape& other) const { | 
|  | return value_ != other.value_; | 
|  | } | 
|  |  | 
|  | // Registers record shape with [num_fields] and [field_names] in the current | 
|  | // isolate group. | 
|  | static RecordShape Register(Thread* thread, | 
|  | intptr_t num_fields, | 
|  | const Array& field_names); | 
|  |  | 
|  | // Retrieves an array of field names. | 
|  | ArrayPtr GetFieldNames(Thread* thread) const; | 
|  |  | 
|  | private: | 
|  | intptr_t value_; | 
|  |  | 
|  | DISALLOW_ALLOCATION(); | 
|  | }; | 
|  |  | 
|  | // A RecordType represents the type of a record. It describes | 
|  | // number of named and positional fields, field types and | 
|  | // names of the named fields. | 
|  | class RecordType : public AbstractType { | 
|  | public: | 
|  | virtual bool HasTypeClass() const { return false; } | 
|  | RecordTypePtr ToNullability(Nullability value, Heap::Space space) const; | 
|  | virtual classid_t type_class_id() const { return kIllegalCid; } | 
|  | virtual bool IsInstantiated( | 
|  | Genericity genericity = kAny, | 
|  | intptr_t num_free_fun_type_params = kAllFree) const; | 
|  | virtual bool IsEquivalent( | 
|  | const Instance& other, | 
|  | TypeEquality kind, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | virtual AbstractTypePtr InstantiateFrom( | 
|  | const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping = nullptr, | 
|  | intptr_t num_parent_type_args_adjustment = 0) const; | 
|  |  | 
|  | virtual AbstractTypePtr UpdateFunctionTypes( | 
|  | intptr_t num_parent_type_args_adjustment, | 
|  | intptr_t num_free_fun_type_params, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_mapping) const; | 
|  |  | 
|  | virtual AbstractTypePtr Canonicalize(Thread* thread) const; | 
|  | virtual void EnumerateURIs(URIs* uris) const; | 
|  | virtual void PrintName(NameVisibility visibility, | 
|  | BaseTextBuffer* printer) const; | 
|  |  | 
|  | virtual uword ComputeHash() const; | 
|  |  | 
|  | bool IsSubtypeOf( | 
|  | const RecordType& other, | 
|  | Heap::Space space, | 
|  | FunctionTypeMapping* function_type_equivalence = nullptr) const; | 
|  |  | 
|  | RecordShape shape() const { return RecordShape(untag()->shape()); } | 
|  |  | 
|  | ArrayPtr field_types() const { return untag()->field_types(); } | 
|  |  | 
|  | AbstractTypePtr FieldTypeAt(intptr_t index) const; | 
|  | void SetFieldTypeAt(intptr_t index, const AbstractType& value) const; | 
|  |  | 
|  | // Names of the named fields, sorted. | 
|  | ArrayPtr GetFieldNames(Thread* thread) const; | 
|  |  | 
|  | intptr_t NumFields() const; | 
|  |  | 
|  | void Print(NameVisibility name_visibility, BaseTextBuffer* printer) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedRecordType)); | 
|  | } | 
|  |  | 
|  | static RecordTypePtr New(RecordShape shape, | 
|  | const Array& field_types, | 
|  | Nullability nullability = Nullability::kNonNullable, | 
|  | Heap::Space space = Heap::kOld); | 
|  |  | 
|  | private: | 
|  | void set_shape(RecordShape shape) const; | 
|  | void set_field_types(const Array& value) const; | 
|  |  | 
|  | static RecordTypePtr New(Heap::Space space); | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(RecordType, AbstractType); | 
|  | friend class Class; | 
|  | friend class ClassFinalizer; | 
|  | friend class Record; | 
|  | }; | 
|  |  | 
|  | class Record : public Instance { | 
|  | public: | 
|  | intptr_t num_fields() const { return NumFields(ptr()); } | 
|  | static intptr_t NumFields(RecordPtr ptr) { | 
|  | return RecordShape(ptr->untag()->shape()).num_fields(); | 
|  | } | 
|  |  | 
|  | RecordShape shape() const { return RecordShape(untag()->shape()); } | 
|  | static intptr_t shape_offset() { return OFFSET_OF(UntaggedRecord, shape_); } | 
|  |  | 
|  | ObjectPtr FieldAt(intptr_t field_index) const { | 
|  | return untag()->field(field_index); | 
|  | } | 
|  | void SetFieldAt(intptr_t field_index, const Object& value) const { | 
|  | untag()->set_field(field_index, value.ptr()); | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t kBytesPerElement = kCompressedWordSize; | 
|  | static constexpr intptr_t kMaxElements = RecordShape::kMaxNumFields; | 
|  |  | 
|  | struct ArrayTraits { | 
|  | static intptr_t elements_start_offset() { return sizeof(UntaggedRecord); } | 
|  | static constexpr intptr_t kElementSize = kBytesPerElement; | 
|  | }; | 
|  |  | 
|  | static intptr_t field_offset(intptr_t index) { | 
|  | return OFFSET_OF_RETURNED_VALUE(UntaggedRecord, data) + | 
|  | kBytesPerElement * index; | 
|  | } | 
|  | static intptr_t field_index_at_offset(intptr_t offset_in_bytes) { | 
|  | const intptr_t index = | 
|  | (offset_in_bytes - OFFSET_OF_RETURNED_VALUE(UntaggedRecord, data)) / | 
|  | kBytesPerElement; | 
|  | ASSERT(index >= 0); | 
|  | return index; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedRecord) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedRecord, data)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize(intptr_t num_fields) { | 
|  | return RoundedAllocationSize(sizeof(UntaggedRecord) + | 
|  | (num_fields * kBytesPerElement)); | 
|  | } | 
|  |  | 
|  | static RecordPtr New(RecordShape shape, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  | virtual void CanonicalizeFieldsLocked(Thread* thread) const; | 
|  |  | 
|  | // Returns RecordType representing runtime type of this record instance. | 
|  | // It is not created eagerly when record instance is allocated because | 
|  | // it depends on runtime types of values if its fields, which can be | 
|  | // quite expensive to query. | 
|  | RecordTypePtr GetRecordType(TypeVisibility visibility) const; | 
|  |  | 
|  | // Parses positional field name and return its index, | 
|  | // or -1 if [field_name] is not a valid positional field name. | 
|  | static intptr_t GetPositionalFieldIndexFromFieldName( | 
|  | const String& field_name); | 
|  |  | 
|  | // Returns index of the field with given name, or -1 | 
|  | // if such field doesn't exist. | 
|  | // Supports positional field names ("$1", "$2", etc). | 
|  | intptr_t GetFieldIndexByName(Thread* thread, const String& field_name) const; | 
|  |  | 
|  | ArrayPtr GetFieldNames(Thread* thread) const { | 
|  | return shape().GetFieldNames(thread); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Record, Instance); | 
|  | friend class Class; | 
|  | friend class Object; | 
|  | }; | 
|  |  | 
|  | class PointerBase : public Instance { | 
|  | public: | 
|  | static intptr_t data_offset() { | 
|  | return OFFSET_OF(UntaggedPointerBase, data_); | 
|  | } | 
|  | }; | 
|  |  | 
|  | class TypedDataBase : public PointerBase { | 
|  | public: | 
|  | static intptr_t length_offset() { | 
|  | return OFFSET_OF(UntaggedTypedDataBase, length_); | 
|  | } | 
|  |  | 
|  | SmiPtr length() const { return untag()->length(); } | 
|  |  | 
|  | intptr_t Length() const { | 
|  | ASSERT(!IsNull()); | 
|  | return Smi::Value(untag()->length()); | 
|  | } | 
|  |  | 
|  | intptr_t LengthInBytes() const { | 
|  | return ElementSizeInBytes(ptr()->GetClassIdOfHeapObject()) * Length(); | 
|  | } | 
|  |  | 
|  | TypedDataElementType ElementType() const { | 
|  | return ElementType(ptr()->GetClassIdOfHeapObject()); | 
|  | } | 
|  |  | 
|  | intptr_t ElementSizeInBytes() const { | 
|  | return element_size(ElementType(ptr()->GetClassIdOfHeapObject())); | 
|  | } | 
|  |  | 
|  | static intptr_t ElementSizeInBytes(classid_t cid) { | 
|  | return element_size(ElementType(cid)); | 
|  | } | 
|  |  | 
|  | static TypedDataElementType ElementType(classid_t cid) { | 
|  | if (cid == kByteDataViewCid || cid == kUnmodifiableByteDataViewCid) { | 
|  | return kUint8ArrayElement; | 
|  | } else if (IsTypedDataClassId(cid)) { | 
|  | const intptr_t index = | 
|  | (cid - kFirstTypedDataCid - kTypedDataCidRemainderInternal) / | 
|  | kNumTypedDataCidRemainders; | 
|  | return static_cast<TypedDataElementType>(index); | 
|  | } else if (IsTypedDataViewClassId(cid)) { | 
|  | const intptr_t index = | 
|  | (cid - kFirstTypedDataCid - kTypedDataCidRemainderView) / | 
|  | kNumTypedDataCidRemainders; | 
|  | return static_cast<TypedDataElementType>(index); | 
|  | } else if (IsExternalTypedDataClassId(cid)) { | 
|  | const intptr_t index = | 
|  | (cid - kFirstTypedDataCid - kTypedDataCidRemainderExternal) / | 
|  | kNumTypedDataCidRemainders; | 
|  | return static_cast<TypedDataElementType>(index); | 
|  | } else { | 
|  | ASSERT(IsUnmodifiableTypedDataViewClassId(cid)); | 
|  | const intptr_t index = | 
|  | (cid - kFirstTypedDataCid - kTypedDataCidRemainderUnmodifiable) / | 
|  | kNumTypedDataCidRemainders; | 
|  | return static_cast<TypedDataElementType>(index); | 
|  | } | 
|  | } | 
|  |  | 
|  | bool IsExternalOrExternalView() const; | 
|  | TypedDataViewPtr ViewFromTo(intptr_t start, | 
|  | intptr_t end, | 
|  | Heap::Space space = Heap::kNew) const; | 
|  |  | 
|  | void* DataAddr(intptr_t byte_offset) const { | 
|  | ASSERT((byte_offset == 0) || | 
|  | ((byte_offset > 0) && (byte_offset < LengthInBytes()))); | 
|  | return reinterpret_cast<void*>(Validate(untag()->data_) + byte_offset); | 
|  | } | 
|  |  | 
|  | #define TYPED_GETTER_SETTER(name, type)                                        \ | 
|  | type Get##name(intptr_t byte_offset) const {                                 \ | 
|  | ASSERT(static_cast<uintptr_t>(byte_offset) <=                              \ | 
|  | static_cast<uintptr_t>(LengthInBytes()) - sizeof(type));            \ | 
|  | return LoadUnaligned(                                                      \ | 
|  | reinterpret_cast<type*>(untag()->data_ + byte_offset));                \ | 
|  | }                                                                            \ | 
|  | void Set##name(intptr_t byte_offset, type value) const {                     \ | 
|  | ASSERT(static_cast<uintptr_t>(byte_offset) <=                              \ | 
|  | static_cast<uintptr_t>(LengthInBytes()) - sizeof(type));            \ | 
|  | StoreUnaligned(reinterpret_cast<type*>(untag()->data_ + byte_offset),      \ | 
|  | value);                                                     \ | 
|  | } | 
|  |  | 
|  | TYPED_GETTER_SETTER(Int8, int8_t) | 
|  | TYPED_GETTER_SETTER(Uint8, uint8_t) | 
|  | TYPED_GETTER_SETTER(Int16, int16_t) | 
|  | TYPED_GETTER_SETTER(Uint16, uint16_t) | 
|  | TYPED_GETTER_SETTER(Int32, int32_t) | 
|  | TYPED_GETTER_SETTER(Uint32, uint32_t) | 
|  | TYPED_GETTER_SETTER(Int64, int64_t) | 
|  | TYPED_GETTER_SETTER(Uint64, uint64_t) | 
|  | TYPED_GETTER_SETTER(Float32, float) | 
|  | TYPED_GETTER_SETTER(Float64, double) | 
|  | TYPED_GETTER_SETTER(Float32x4, simd128_value_t) | 
|  | TYPED_GETTER_SETTER(Int32x4, simd128_value_t) | 
|  | TYPED_GETTER_SETTER(Float64x2, simd128_value_t) | 
|  |  | 
|  | #undef TYPED_GETTER_SETTER | 
|  |  | 
|  | protected: | 
|  | void SetLength(intptr_t value) const { | 
|  | ASSERT(value <= Smi::kMaxValue); | 
|  | untag()->set_length(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | virtual uint8_t* Validate(uint8_t* data) const { | 
|  | return UnsafeMutableNonPointer(data); | 
|  | } | 
|  |  | 
|  | private: | 
|  | friend class Class; | 
|  |  | 
|  | static intptr_t element_size(intptr_t index) { | 
|  | ASSERT(0 <= index && index < kNumElementSizes); | 
|  | intptr_t size = element_size_table[index]; | 
|  | ASSERT(size != 0); | 
|  | return size; | 
|  | } | 
|  | static constexpr intptr_t kNumElementSizes = | 
|  | ((kLastTypedDataCid + 1) - kFirstTypedDataCid) / | 
|  | kNumTypedDataCidRemainders; | 
|  | static const intptr_t element_size_table[kNumElementSizes]; | 
|  |  | 
|  | HEAP_OBJECT_IMPLEMENTATION(TypedDataBase, PointerBase); | 
|  | }; | 
|  |  | 
|  | class TypedData : public TypedDataBase { | 
|  | public: | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  |  | 
|  | #define TYPED_GETTER_SETTER(name, type)                                        \ | 
|  | type Get##name(intptr_t byte_offset) const {                                 \ | 
|  | ASSERT(static_cast<uintptr_t>(byte_offset) <=                              \ | 
|  | static_cast<uintptr_t>(LengthInBytes()) - sizeof(type));            \ | 
|  | return LoadUnaligned(                                                      \ | 
|  | reinterpret_cast<const type*>(untag()->data() + byte_offset));         \ | 
|  | }                                                                            \ | 
|  | void Set##name(intptr_t byte_offset, type value) const {                     \ | 
|  | ASSERT(static_cast<uintptr_t>(byte_offset) <=                              \ | 
|  | static_cast<uintptr_t>(LengthInBytes()) - sizeof(type));            \ | 
|  | return StoreUnaligned(                                                     \ | 
|  | reinterpret_cast<type*>(untag()->data() + byte_offset), value);        \ | 
|  | } | 
|  |  | 
|  | TYPED_GETTER_SETTER(Int8, int8_t) | 
|  | TYPED_GETTER_SETTER(Uint8, uint8_t) | 
|  | TYPED_GETTER_SETTER(Int16, int16_t) | 
|  | TYPED_GETTER_SETTER(Uint16, uint16_t) | 
|  | TYPED_GETTER_SETTER(Int32, int32_t) | 
|  | TYPED_GETTER_SETTER(Uint32, uint32_t) | 
|  | TYPED_GETTER_SETTER(Int64, int64_t) | 
|  | TYPED_GETTER_SETTER(Uint64, uint64_t) | 
|  | TYPED_GETTER_SETTER(Float32, float) | 
|  | TYPED_GETTER_SETTER(Float64, double) | 
|  | TYPED_GETTER_SETTER(Float32x4, simd128_value_t) | 
|  | TYPED_GETTER_SETTER(Int32x4, simd128_value_t) | 
|  | TYPED_GETTER_SETTER(Float64x2, simd128_value_t) | 
|  |  | 
|  | #undef TYPED_GETTER_SETTER | 
|  |  | 
|  | static intptr_t payload_offset() { | 
|  | return UntaggedTypedData::payload_offset(); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT(sizeof(UntaggedTypedData) == | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedTypedData, internal_data)); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize(intptr_t lengthInBytes) { | 
|  | ASSERT(0 <= lengthInBytes && lengthInBytes <= kSmiMax); | 
|  | return RoundedAllocationSize(sizeof(UntaggedTypedData) + lengthInBytes); | 
|  | } | 
|  |  | 
|  | static intptr_t MaxElements(intptr_t class_id) { | 
|  | ASSERT(IsTypedDataClassId(class_id)); | 
|  | return (kSmiMax / ElementSizeInBytes(class_id)); | 
|  | } | 
|  |  | 
|  | static intptr_t MaxNewSpaceElements(intptr_t class_id) { | 
|  | ASSERT(IsTypedDataClassId(class_id)); | 
|  | return (kNewAllocatableSize - sizeof(UntaggedTypedData)) / | 
|  | ElementSizeInBytes(class_id); | 
|  | } | 
|  |  | 
|  | static TypedDataPtr New(intptr_t class_id, | 
|  | intptr_t len, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static TypedDataPtr Grow(const TypedData& current, | 
|  | intptr_t len, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static bool IsTypedData(const Instance& obj) { | 
|  | ASSERT(!obj.IsNull()); | 
|  | intptr_t cid = obj.ptr()->GetClassId(); | 
|  | return IsTypedDataClassId(cid); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | void RecomputeDataField() { ptr()->untag()->RecomputeDataField(); } | 
|  |  | 
|  | private: | 
|  | // Provides const access to non-pointer, non-aligned data within the object. | 
|  | // Such access does not need a write barrier, but it is *not* GC-safe, since | 
|  | // the object might move. | 
|  | // | 
|  | // Therefore this method is private and the call-sites in this class need to | 
|  | // ensure the returned pointer does not escape. | 
|  | template <typename FieldType> | 
|  | const FieldType* ReadOnlyDataAddr(intptr_t byte_offset) const { | 
|  | return reinterpret_cast<const FieldType*>((untag()->data()) + byte_offset); | 
|  | } | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(TypedData, TypedDataBase); | 
|  | friend class Class; | 
|  | friend class ExternalTypedData; | 
|  | friend class TypedDataView; | 
|  | }; | 
|  |  | 
|  | class ExternalTypedData : public TypedDataBase { | 
|  | public: | 
|  | // Alignment of data when serializing ExternalTypedData in a clustered | 
|  | // snapshot. Should be independent of word size. | 
|  | static constexpr int kDataSerializationAlignment = 8; | 
|  |  | 
|  | FinalizablePersistentHandle* AddFinalizer(void* peer, | 
|  | Dart_HandleFinalizer callback, | 
|  | intptr_t external_size) const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedExternalTypedData)); | 
|  | } | 
|  |  | 
|  | static intptr_t MaxElements(intptr_t class_id) { | 
|  | ASSERT(IsExternalTypedDataClassId(class_id)); | 
|  | return (kSmiMax / ElementSizeInBytes(class_id)); | 
|  | } | 
|  |  | 
|  | static ExternalTypedDataPtr New( | 
|  | intptr_t class_id, | 
|  | uint8_t* data, | 
|  | intptr_t len, | 
|  | Heap::Space space = Heap::kNew, | 
|  | bool perform_eager_msan_initialization_check = true); | 
|  |  | 
|  | static ExternalTypedDataPtr NewFinalizeWithFree(uint8_t* data, intptr_t len); | 
|  |  | 
|  | static bool IsExternalTypedData(const Instance& obj) { | 
|  | ASSERT(!obj.IsNull()); | 
|  | intptr_t cid = obj.ptr()->GetClassId(); | 
|  | return IsExternalTypedDataClassId(cid); | 
|  | } | 
|  |  | 
|  | protected: | 
|  | virtual uint8_t* Validate(uint8_t* data) const { return data; } | 
|  |  | 
|  | void SetLength(intptr_t value) const { | 
|  | ASSERT(value <= Smi::kMaxValue); | 
|  | untag()->set_length(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | void SetData(uint8_t* data) const { | 
|  | ASSERT(!IsolateGroup::Current()->heap()->Contains( | 
|  | reinterpret_cast<uword>(data))); | 
|  | StoreNonPointer(&untag()->data_, data); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(ExternalTypedData, TypedDataBase); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class TypedDataView : public TypedDataBase { | 
|  | public: | 
|  | static TypedDataViewPtr New(intptr_t class_id, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static TypedDataViewPtr New(intptr_t class_id, | 
|  | const TypedDataBase& typed_data, | 
|  | intptr_t offset_in_bytes, | 
|  | intptr_t length, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedTypedDataView)); | 
|  | } | 
|  |  | 
|  | static InstancePtr Data(const TypedDataView& view) { | 
|  | return view.typed_data(); | 
|  | } | 
|  |  | 
|  | static SmiPtr OffsetInBytes(const TypedDataView& view) { | 
|  | return view.offset_in_bytes(); | 
|  | } | 
|  |  | 
|  | static bool IsExternalTypedDataView(const TypedDataView& view_obj) { | 
|  | const auto& data = Instance::Handle(Data(view_obj)); | 
|  | intptr_t cid = data.ptr()->GetClassIdOfHeapObject(); | 
|  | ASSERT(IsTypedDataClassId(cid) || IsExternalTypedDataClassId(cid)); | 
|  | return IsExternalTypedDataClassId(cid); | 
|  | } | 
|  |  | 
|  | static intptr_t typed_data_offset() { | 
|  | return OFFSET_OF(UntaggedTypedDataView, typed_data_); | 
|  | } | 
|  |  | 
|  | static intptr_t offset_in_bytes_offset() { | 
|  | return OFFSET_OF(UntaggedTypedDataView, offset_in_bytes_); | 
|  | } | 
|  |  | 
|  | TypedDataBasePtr typed_data() const { return untag()->typed_data(); } | 
|  |  | 
|  | void InitializeWith(const TypedDataBase& typed_data, | 
|  | intptr_t offset_in_bytes, | 
|  | intptr_t length) { | 
|  | const classid_t cid = typed_data.GetClassId(); | 
|  | ASSERT(IsTypedDataClassId(cid) || IsExternalTypedDataClassId(cid)); | 
|  | untag()->set_typed_data(typed_data.ptr()); | 
|  | untag()->set_length(Smi::New(length)); | 
|  | untag()->set_offset_in_bytes(Smi::New(offset_in_bytes)); | 
|  |  | 
|  | // Update the inner pointer. | 
|  | RecomputeDataField(); | 
|  | } | 
|  |  | 
|  | SmiPtr offset_in_bytes() const { return untag()->offset_in_bytes(); } | 
|  |  | 
|  | protected: | 
|  | virtual uint8_t* Validate(uint8_t* data) const { return data; } | 
|  |  | 
|  | private: | 
|  | void RecomputeDataField() const { ptr()->untag()->RecomputeDataField(); } | 
|  |  | 
|  | void Clear() { | 
|  | untag()->set_length(Smi::New(0)); | 
|  | untag()->set_offset_in_bytes(Smi::New(0)); | 
|  | StoreNonPointer(&untag()->data_, nullptr); | 
|  | untag()->set_typed_data(TypedDataBase::RawCast(Object::null())); | 
|  | } | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(TypedDataView, TypedDataBase); | 
|  | friend class Class; | 
|  | friend class DeferredObject; | 
|  | friend class Object; | 
|  | friend class TypedDataViewDeserializationCluster; | 
|  | }; | 
|  |  | 
|  | class ByteBuffer : public AllStatic { | 
|  | public: | 
|  | static constexpr bool ContainsCompressedPointers() { | 
|  | return Instance::ContainsCompressedPointers(); | 
|  | } | 
|  |  | 
|  | static InstancePtr Data(const Instance& view_obj) { | 
|  | ASSERT(!view_obj.IsNull()); | 
|  | return reinterpret_cast<CompressedInstancePtr*>( | 
|  | reinterpret_cast<uword>(view_obj.untag()) + data_offset()) | 
|  | ->Decompress(view_obj.untag()->heap_base()); | 
|  | } | 
|  |  | 
|  | static intptr_t NumberOfFields() { return kNumFields; } | 
|  |  | 
|  | static intptr_t data_offset() { | 
|  | return sizeof(UntaggedObject) + (kCompressedWordSize * kDataIndex); | 
|  | } | 
|  |  | 
|  | private: | 
|  | enum { | 
|  | kDataIndex = 0, | 
|  | kNumFields = 1, | 
|  | }; | 
|  | }; | 
|  |  | 
|  | class Pointer : public Instance { | 
|  | public: | 
|  | static PointerPtr New(uword native_address, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedPointer)); | 
|  | } | 
|  |  | 
|  | static bool IsPointer(const Instance& obj); | 
|  |  | 
|  | size_t NativeAddress() const { | 
|  | return reinterpret_cast<size_t>(untag()->data_); | 
|  | } | 
|  |  | 
|  | void SetNativeAddress(size_t address) const { | 
|  | uint8_t* value = reinterpret_cast<uint8_t*>(address); | 
|  | StoreNonPointer(&untag()->data_, value); | 
|  | } | 
|  |  | 
|  | static intptr_t type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedPointer, type_arguments_); | 
|  | } | 
|  |  | 
|  | static constexpr intptr_t kNativeTypeArgPos = 0; | 
|  |  | 
|  | // Fetches the NativeType type argument. | 
|  | AbstractTypePtr type_argument() const { | 
|  | TypeArguments& type_args = TypeArguments::Handle(GetTypeArguments()); | 
|  | return type_args.TypeAtNullSafe(Pointer::kNativeTypeArgPos); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Pointer, Instance); | 
|  |  | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class DynamicLibrary : public Instance { | 
|  | public: | 
|  | static DynamicLibraryPtr New(void* handle, | 
|  | bool canBeClosed, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedDynamicLibrary)); | 
|  | } | 
|  |  | 
|  | static bool IsDynamicLibrary(const Instance& obj) { | 
|  | ASSERT(!obj.IsNull()); | 
|  | intptr_t cid = obj.ptr()->GetClassId(); | 
|  | return IsFfiDynamicLibraryClassId(cid); | 
|  | } | 
|  |  | 
|  | void* GetHandle() const { | 
|  | ASSERT(!IsNull()); | 
|  | return untag()->handle_; | 
|  | } | 
|  |  | 
|  | void SetHandle(void* value) const { | 
|  | StoreNonPointer(&untag()->handle_, value); | 
|  | } | 
|  |  | 
|  | bool CanBeClosed() const { | 
|  | ASSERT(!IsNull()); | 
|  | return untag()->canBeClosed_; | 
|  | } | 
|  |  | 
|  | void SetCanBeClosed(bool value) const { | 
|  | ASSERT(!IsNull()); | 
|  | StoreNonPointer(&untag()->canBeClosed_, value); | 
|  | } | 
|  |  | 
|  | bool IsClosed() const { | 
|  | ASSERT(!IsNull()); | 
|  | return untag()->isClosed_; | 
|  | } | 
|  |  | 
|  | void SetClosed(bool value) const { | 
|  | StoreNonPointer(&untag()->isClosed_, value); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(DynamicLibrary, Instance); | 
|  |  | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class LinkedHashBase : public Instance { | 
|  | public: | 
|  | // Keep consistent with _indexSizeToHashMask in compact_hash.dart. | 
|  | static intptr_t IndexSizeToHashMask(intptr_t index_size) { | 
|  | ASSERT(index_size >= kInitialIndexSize); | 
|  | intptr_t index_bits = Utils::BitLength(index_size) - 2; | 
|  | #if defined(HAS_SMI_63_BITS) | 
|  | return (1 << (32 - index_bits)) - 1; | 
|  | #else | 
|  | return (1 << (Object::kHashBits - index_bits)) - 1; | 
|  | #endif | 
|  | } | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedLinkedHashBase)); | 
|  | } | 
|  |  | 
|  | static intptr_t type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedLinkedHashBase, type_arguments_); | 
|  | } | 
|  |  | 
|  | static intptr_t index_offset() { | 
|  | return OFFSET_OF(UntaggedLinkedHashBase, index_); | 
|  | } | 
|  |  | 
|  | static intptr_t data_offset() { | 
|  | return OFFSET_OF(UntaggedLinkedHashBase, data_); | 
|  | } | 
|  |  | 
|  | static intptr_t hash_mask_offset() { | 
|  | return OFFSET_OF(UntaggedLinkedHashBase, hash_mask_); | 
|  | } | 
|  |  | 
|  | static intptr_t used_data_offset() { | 
|  | return OFFSET_OF(UntaggedLinkedHashBase, used_data_); | 
|  | } | 
|  |  | 
|  | static intptr_t deleted_keys_offset() { | 
|  | return OFFSET_OF(UntaggedLinkedHashBase, deleted_keys_); | 
|  | } | 
|  |  | 
|  | static const LinkedHashBase& Cast(const Object& obj) { | 
|  | ASSERT(obj.IsMap() || obj.IsSet()); | 
|  | return static_cast<const LinkedHashBase&>(obj); | 
|  | } | 
|  |  | 
|  | bool IsImmutable() const { | 
|  | return GetClassId() == kConstMapCid || GetClassId() == kConstSetCid; | 
|  | } | 
|  |  | 
|  | virtual TypeArgumentsPtr GetTypeArguments() const { | 
|  | return untag()->type_arguments(); | 
|  | } | 
|  | virtual void SetTypeArguments(const TypeArguments& value) const { | 
|  | const intptr_t num_type_args = IsMap() ? 2 : 1; | 
|  | ASSERT(value.IsNull() || | 
|  | ((value.Length() >= num_type_args) && | 
|  | value.IsInstantiated() /*&& value.IsCanonical()*/)); | 
|  | // TODO(asiva): Values read from a message snapshot are not properly marked | 
|  | // as canonical. See for example tests/isolate/message3_test.dart. | 
|  | untag()->set_type_arguments(value.ptr()); | 
|  | } | 
|  |  | 
|  | TypedDataPtr index() const { return untag()->index(); } | 
|  | void set_index(const TypedData& value) const { | 
|  | ASSERT(!value.IsNull()); | 
|  | untag()->set_index(value.ptr()); | 
|  | } | 
|  |  | 
|  | ArrayPtr data() const { return untag()->data(); } | 
|  | void set_data(const Array& value) const { untag()->set_data(value.ptr()); } | 
|  |  | 
|  | SmiPtr hash_mask() const { return untag()->hash_mask(); } | 
|  | void set_hash_mask(intptr_t value) const { | 
|  | untag()->set_hash_mask(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | SmiPtr used_data() const { return untag()->used_data(); } | 
|  | void set_used_data(intptr_t value) const { | 
|  | untag()->set_used_data(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | SmiPtr deleted_keys() const { return untag()->deleted_keys(); } | 
|  | void set_deleted_keys(intptr_t value) const { | 
|  | untag()->set_deleted_keys(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | intptr_t Length() const { | 
|  | // The map or set may be uninitialized. | 
|  | if (untag()->used_data() == Object::null()) return 0; | 
|  | if (untag()->deleted_keys() == Object::null()) return 0; | 
|  |  | 
|  | intptr_t used = Smi::Value(untag()->used_data()); | 
|  | if (IsMap()) { | 
|  | used >>= 1; | 
|  | } | 
|  | const intptr_t deleted = Smi::Value(untag()->deleted_keys()); | 
|  | return used - deleted; | 
|  | } | 
|  |  | 
|  | // We do not compute the indices in the VM, but we do precompute the hash | 
|  | // mask to avoid a load acquire barrier on reading the combination of index | 
|  | // and hash mask. | 
|  | void ComputeAndSetHashMask() const; | 
|  |  | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  | virtual void CanonicalizeFieldsLocked(Thread* thread) const; | 
|  |  | 
|  | // Keep this in sync with Dart implementation (lib/compact_hash.dart). | 
|  | static constexpr intptr_t kInitialIndexBits = 2; | 
|  | static constexpr intptr_t kInitialIndexSize = 1 << (kInitialIndexBits + 1); | 
|  | static constexpr intptr_t kUninitializedIndexSize = 1; | 
|  |  | 
|  | private: | 
|  | LinkedHashBasePtr ptr() const { return static_cast<LinkedHashBasePtr>(ptr_); } | 
|  | UntaggedLinkedHashBase* untag() const { | 
|  | ASSERT(ptr() != null()); | 
|  | return const_cast<UntaggedLinkedHashBase*>(ptr()->untag()); | 
|  | } | 
|  |  | 
|  | friend class Class; | 
|  | friend class ImmutableLinkedHashBase; | 
|  | friend class LinkedHashBaseDeserializationCluster; | 
|  | }; | 
|  |  | 
|  | class ImmutableLinkedHashBase : public AllStatic { | 
|  | public: | 
|  | static constexpr bool ContainsCompressedPointers() { | 
|  | return LinkedHashBase::ContainsCompressedPointers(); | 
|  | } | 
|  |  | 
|  | static intptr_t data_offset() { return LinkedHashBase::data_offset(); } | 
|  | }; | 
|  |  | 
|  | // Corresponds to | 
|  | // - _Map in dart:_compact_hash | 
|  | // - "new Map()", | 
|  | // - non-const map literals, and | 
|  | // - the default constructor of LinkedHashMap in dart:collection. | 
|  | class Map : public LinkedHashBase { | 
|  | public: | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedMap)); | 
|  | } | 
|  |  | 
|  | // Allocates a map with some default capacity, just like "new Map()". | 
|  | static MapPtr NewDefault(intptr_t class_id = kMapCid, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static MapPtr New(intptr_t class_id, | 
|  | const Array& data, | 
|  | const TypedData& index, | 
|  | intptr_t hash_mask, | 
|  | intptr_t used_data, | 
|  | intptr_t deleted_keys, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // This iterator differs somewhat from its Dart counterpart (_CompactIterator | 
|  | // in runtime/lib/compact_hash.dart): | 
|  | //  - There are no checks for concurrent modifications. | 
|  | //  - Accessing a key or value before the first call to MoveNext and after | 
|  | //    MoveNext returns false will result in crashes. | 
|  | class Iterator : public ValueObject { | 
|  | public: | 
|  | explicit Iterator(const Map& map) | 
|  | : data_(Array::Handle(map.data())), | 
|  | scratch_(Object::Handle()), | 
|  | offset_(-2), | 
|  | length_(Smi::Value(map.used_data())) {} | 
|  |  | 
|  | bool MoveNext() { | 
|  | while (true) { | 
|  | offset_ += 2; | 
|  | if (offset_ >= length_) { | 
|  | return false; | 
|  | } | 
|  | scratch_ = data_.At(offset_); | 
|  | if (scratch_.ptr() != data_.ptr()) { | 
|  | // Slot is not deleted (self-reference indicates deletion). | 
|  | return true; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | ObjectPtr CurrentKey() const { return data_.At(offset_); } | 
|  |  | 
|  | ObjectPtr CurrentValue() const { return data_.At(offset_ + 1); } | 
|  |  | 
|  | private: | 
|  | const Array& data_; | 
|  | Object& scratch_; | 
|  | intptr_t offset_; | 
|  | const intptr_t length_; | 
|  | }; | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Map, LinkedHashBase); | 
|  |  | 
|  | // Allocate a map, but leave all fields set to null. | 
|  | // Used during deserialization (since map might contain itself as key/value). | 
|  | static MapPtr NewUninitialized(intptr_t class_id, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | friend class Class; | 
|  | friend class ConstMap; | 
|  | friend class MapDeserializationCluster; | 
|  | }; | 
|  |  | 
|  | // Corresponds to | 
|  | // - _ConstMap in dart:_compact_hash | 
|  | // - const map literals | 
|  | class ConstMap : public AllStatic { | 
|  | public: | 
|  | static constexpr bool ContainsCompressedPointers() { | 
|  | return Map::ContainsCompressedPointers(); | 
|  | } | 
|  |  | 
|  | static ConstMapPtr NewDefault(Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static ConstMapPtr NewUninitialized(Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static const ClassId kClassId = kConstMapCid; | 
|  |  | 
|  | static intptr_t InstanceSize() { return Map::InstanceSize(); } | 
|  |  | 
|  | private: | 
|  | static intptr_t NextFieldOffset() { | 
|  | // Indicates this class cannot be extended by dart code. | 
|  | return -kWordSize; | 
|  | } | 
|  |  | 
|  | static ConstMapPtr raw(const Map& map) { | 
|  | return static_cast<ConstMapPtr>(map.ptr()); | 
|  | } | 
|  |  | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Corresponds to | 
|  | // - _Set in dart:_compact_hash, | 
|  | // - "new Set()", | 
|  | // - non-const set literals, and | 
|  | // - the default constructor of LinkedHashSet in dart:collection. | 
|  | class Set : public LinkedHashBase { | 
|  | public: | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedSet)); | 
|  | } | 
|  |  | 
|  | // Allocates a set with some default capacity, just like "new Set()". | 
|  | static SetPtr NewDefault(intptr_t class_id = kSetCid, | 
|  | Heap::Space space = Heap::kNew); | 
|  | static SetPtr New(intptr_t class_id, | 
|  | const Array& data, | 
|  | const TypedData& index, | 
|  | intptr_t hash_mask, | 
|  | intptr_t used_data, | 
|  | intptr_t deleted_keys, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // This iterator differs somewhat from its Dart counterpart (_CompactIterator | 
|  | // in runtime/lib/compact_hash.dart): | 
|  | //  - There are no checks for concurrent modifications. | 
|  | //  - Accessing a key or value before the first call to MoveNext and after | 
|  | //    MoveNext returns false will result in crashes. | 
|  | class Iterator : public ValueObject { | 
|  | public: | 
|  | explicit Iterator(const Set& set) | 
|  | : data_(Array::Handle(set.data())), | 
|  | scratch_(Object::Handle()), | 
|  | offset_(-1), | 
|  | length_(Smi::Value(set.used_data())) {} | 
|  |  | 
|  | bool MoveNext() { | 
|  | while (true) { | 
|  | offset_++; | 
|  | if (offset_ >= length_) { | 
|  | return false; | 
|  | } | 
|  | scratch_ = data_.At(offset_); | 
|  | if (scratch_.ptr() != data_.ptr()) { | 
|  | // Slot is not deleted (self-reference indicates deletion). | 
|  | return true; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | ObjectPtr CurrentKey() const { return data_.At(offset_); } | 
|  |  | 
|  | private: | 
|  | const Array& data_; | 
|  | Object& scratch_; | 
|  | intptr_t offset_; | 
|  | const intptr_t length_; | 
|  | }; | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Set, LinkedHashBase); | 
|  |  | 
|  | // Allocate a set, but leave all fields set to null. | 
|  | // Used during deserialization (since set might contain itself as key/value). | 
|  | static SetPtr NewUninitialized(intptr_t class_id, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | friend class Class; | 
|  | friend class ConstSet; | 
|  | friend class SetDeserializationCluster; | 
|  | }; | 
|  |  | 
|  | // Corresponds to | 
|  | // - _ConstSet in dart:_compact_hash | 
|  | // - const set literals | 
|  | class ConstSet : public AllStatic { | 
|  | public: | 
|  | static constexpr bool ContainsCompressedPointers() { | 
|  | return Set::ContainsCompressedPointers(); | 
|  | } | 
|  |  | 
|  | static ConstSetPtr NewDefault(Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static ConstSetPtr NewUninitialized(Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static const ClassId kClassId = kConstSetCid; | 
|  |  | 
|  | static intptr_t InstanceSize() { return Set::InstanceSize(); } | 
|  |  | 
|  | private: | 
|  | static intptr_t NextFieldOffset() { | 
|  | // Indicates this class cannot be extended by dart code. | 
|  | return -kWordSize; | 
|  | } | 
|  |  | 
|  | static ConstSetPtr raw(const Set& map) { | 
|  | return static_cast<ConstSetPtr>(map.ptr()); | 
|  | } | 
|  |  | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class Closure : public Instance { | 
|  | public: | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | uword entry_point() const { return untag()->entry_point_; } | 
|  | void set_entry_point(uword entry_point) const { | 
|  | StoreNonPointer(&untag()->entry_point_, entry_point); | 
|  | } | 
|  | static intptr_t entry_point_offset() { | 
|  | return OFFSET_OF(UntaggedClosure, entry_point_); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | TypeArgumentsPtr instantiator_type_arguments() const { | 
|  | return untag()->instantiator_type_arguments(); | 
|  | } | 
|  | void set_instantiator_type_arguments(const TypeArguments& args) const { | 
|  | untag()->set_instantiator_type_arguments(args.ptr()); | 
|  | } | 
|  | static intptr_t instantiator_type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedClosure, instantiator_type_arguments_); | 
|  | } | 
|  |  | 
|  | TypeArgumentsPtr function_type_arguments() const { | 
|  | return untag()->function_type_arguments(); | 
|  | } | 
|  | void set_function_type_arguments(const TypeArguments& args) const { | 
|  | untag()->set_function_type_arguments(args.ptr()); | 
|  | } | 
|  | static intptr_t function_type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedClosure, function_type_arguments_); | 
|  | } | 
|  |  | 
|  | TypeArgumentsPtr delayed_type_arguments() const { | 
|  | return untag()->delayed_type_arguments(); | 
|  | } | 
|  | void set_delayed_type_arguments(const TypeArguments& args) const { | 
|  | untag()->set_delayed_type_arguments(args.ptr()); | 
|  | } | 
|  | static intptr_t delayed_type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedClosure, delayed_type_arguments_); | 
|  | } | 
|  |  | 
|  | FunctionPtr function() const { return untag()->function(); } | 
|  | static intptr_t function_offset() { | 
|  | return OFFSET_OF(UntaggedClosure, function_); | 
|  | } | 
|  | static FunctionPtr FunctionOf(ClosurePtr closure) { | 
|  | return closure.untag()->function(); | 
|  | } | 
|  |  | 
|  | ObjectPtr RawContext() const { return untag()->context(); } | 
|  |  | 
|  | ContextPtr GetContext() const { | 
|  | ASSERT(!Function::IsImplicitClosureFunction(function())); | 
|  | return Context::RawCast(RawContext()); | 
|  | } | 
|  |  | 
|  | InstancePtr GetImplicitClosureReceiver() const { | 
|  | ASSERT(Function::IsImplicitInstanceClosureFunction(function())); | 
|  | return Instance::RawCast(RawContext()); | 
|  | } | 
|  |  | 
|  | static intptr_t context_offset() { | 
|  | return OFFSET_OF(UntaggedClosure, context_); | 
|  | } | 
|  |  | 
|  | // Returns whether the closure is generic, that is, it has a generic closure | 
|  | // function and no delayed type arguments. | 
|  | bool IsGeneric() const { | 
|  | return delayed_type_arguments() == Object::empty_type_arguments().ptr(); | 
|  | } | 
|  |  | 
|  | SmiPtr hash() const { return untag()->hash(); } | 
|  | static intptr_t hash_offset() { return OFFSET_OF(UntaggedClosure, hash_); } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedClosure)); | 
|  | } | 
|  |  | 
|  | virtual void CanonicalizeFieldsLocked(Thread* thread) const; | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const { | 
|  | return Function::Handle(function()).Hash(); | 
|  | } | 
|  | uword ComputeHash() const; | 
|  |  | 
|  | static ClosurePtr New(const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | const Function& function, | 
|  | const Object& context, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static ClosurePtr New(const TypeArguments& instantiator_type_arguments, | 
|  | const TypeArguments& function_type_arguments, | 
|  | const TypeArguments& delayed_type_arguments, | 
|  | const Function& function, | 
|  | const Object& context, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | FunctionTypePtr GetInstantiatedSignature(Zone* zone) const; | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Closure, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Corresponds to _Capability in dart:isolate. | 
|  | class Capability : public Instance { | 
|  | public: | 
|  | uint64_t Id() const { return untag()->id_; } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedCapability)); | 
|  | } | 
|  | static CapabilityPtr New(uint64_t id, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Capability, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Corresponds to _RawReceivePort in dart:isolate. | 
|  | class ReceivePort : public Instance { | 
|  | public: | 
|  | SendPortPtr send_port() const { return untag()->send_port(); } | 
|  | static intptr_t send_port_offset() { | 
|  | return OFFSET_OF(UntaggedReceivePort, send_port_); | 
|  | } | 
|  | Dart_Port Id() const { return send_port()->untag()->id_; } | 
|  |  | 
|  | InstancePtr handler() const { return untag()->handler(); } | 
|  | void set_handler(const Instance& value) const { | 
|  | untag()->set_handler(value.ptr()); | 
|  | } | 
|  | static intptr_t handler_offset() { | 
|  | return OFFSET_OF(UntaggedReceivePort, handler_); | 
|  | } | 
|  |  | 
|  | bool is_open() const { | 
|  | return IsOpen::decode(Smi::Value(untag()->bitfield())); | 
|  | } | 
|  | void set_is_open(bool value) const { | 
|  | const auto updated = IsOpen::update(value, Smi::Value(untag()->bitfield())); | 
|  | untag()->set_bitfield(Smi::New(updated)); | 
|  | } | 
|  |  | 
|  | bool keep_isolate_alive() const { | 
|  | return IsKeepIsolateAlive::decode(Smi::Value(untag()->bitfield())); | 
|  | } | 
|  | void set_keep_isolate_alive(bool value) const { | 
|  | const auto updated = | 
|  | IsKeepIsolateAlive::update(value, Smi::Value(untag()->bitfield())); | 
|  | untag()->set_bitfield(Smi::New(updated)); | 
|  | } | 
|  |  | 
|  | #if !defined(PRODUCT) | 
|  | StackTracePtr allocation_location() const { | 
|  | return untag()->allocation_location(); | 
|  | } | 
|  |  | 
|  | StringPtr debug_name() const { return untag()->debug_name(); } | 
|  | #endif | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedReceivePort)); | 
|  | } | 
|  | static ReceivePortPtr New(Dart_Port id, | 
|  | const String& debug_name, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | using IsOpen = BitField<intptr_t, bool>; | 
|  | using IsKeepIsolateAlive = BitField<intptr_t, bool, IsOpen::kNextBit>; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(ReceivePort, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Corresponds to _SendPort in dart:isolate. | 
|  | class SendPort : public Instance { | 
|  | public: | 
|  | Dart_Port Id() const { return untag()->id_; } | 
|  |  | 
|  | Dart_Port origin_id() const { return untag()->origin_id_; } | 
|  | void set_origin_id(Dart_Port id) const { | 
|  | ASSERT(origin_id() == 0); | 
|  | StoreNonPointer(&(untag()->origin_id_), id); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedSendPort)); | 
|  | } | 
|  | static SendPortPtr New(Dart_Port id, Heap::Space space = Heap::kNew); | 
|  | static SendPortPtr New(Dart_Port id, | 
|  | Dart_Port origin_id, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(SendPort, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // This is allocated when new instance of TransferableTypedData is created in | 
|  | // [TransferableTypedData::New]. | 
|  | class TransferableTypedDataPeer { | 
|  | public: | 
|  | // [data] backing store should be malloc'ed, not new'ed. | 
|  | TransferableTypedDataPeer(uint8_t* data, intptr_t length) | 
|  | : data_(data), length_(length), handle_(nullptr) {} | 
|  |  | 
|  | ~TransferableTypedDataPeer() { free(data_); } | 
|  |  | 
|  | uint8_t* data() const { return data_; } | 
|  | intptr_t length() const { return length_; } | 
|  | FinalizablePersistentHandle* handle() const { return handle_; } | 
|  | void set_handle(FinalizablePersistentHandle* handle) { handle_ = handle; } | 
|  |  | 
|  | void ClearData() { | 
|  | data_ = nullptr; | 
|  | length_ = 0; | 
|  | handle_ = nullptr; | 
|  | } | 
|  |  | 
|  | private: | 
|  | uint8_t* data_; | 
|  | intptr_t length_; | 
|  | FinalizablePersistentHandle* handle_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(TransferableTypedDataPeer); | 
|  | }; | 
|  |  | 
|  | class TransferableTypedData : public Instance { | 
|  | public: | 
|  | static TransferableTypedDataPtr New(uint8_t* data, intptr_t len); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedTransferableTypedData)); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(TransferableTypedData, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class DebuggerStackTrace; | 
|  |  | 
|  | // Internal stacktrace object used in exceptions for printing stack traces. | 
|  | class StackTrace : public Instance { | 
|  | public: | 
|  | static constexpr int kFixedOOMStackdepth = 90; | 
|  |  | 
|  | intptr_t Length() const; | 
|  |  | 
|  | StackTracePtr async_link() const { return untag()->async_link(); } | 
|  | void set_async_link(const StackTrace& async_link) const; | 
|  | void set_expand_inlined(bool value) const; | 
|  |  | 
|  | ArrayPtr code_array() const { return untag()->code_array(); } | 
|  | ObjectPtr CodeAtFrame(intptr_t frame_index) const; | 
|  | void SetCodeAtFrame(intptr_t frame_index, const Object& code) const; | 
|  |  | 
|  | TypedDataPtr pc_offset_array() const { return untag()->pc_offset_array(); } | 
|  | uword PcOffsetAtFrame(intptr_t frame_index) const; | 
|  | void SetPcOffsetAtFrame(intptr_t frame_index, uword pc_offset) const; | 
|  |  | 
|  | bool skip_sync_start_in_parent_stack() const; | 
|  | void set_skip_sync_start_in_parent_stack(bool value) const; | 
|  |  | 
|  | // The number of frames that should be cut off the top of an async stack trace | 
|  | // if it's appended to a synchronous stack trace along a sync-async call. | 
|  | // | 
|  | // Without cropping, the border would look like: | 
|  | // | 
|  | // <async function> | 
|  | // --------------------------- | 
|  | // <asynchronous gap marker> | 
|  | // <async function> | 
|  | // | 
|  | // Since it's not actually an async call, we crop off the last two | 
|  | // frames when concatenating the sync and async stacktraces. | 
|  | static constexpr intptr_t kSyncAsyncCroppedFrames = 2; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedStackTrace)); | 
|  | } | 
|  | static StackTracePtr New(const Array& code_array, | 
|  | const TypedData& pc_offset_array, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static StackTracePtr New(const Array& code_array, | 
|  | const TypedData& pc_offset_array, | 
|  | const StackTrace& async_link, | 
|  | bool skip_sync_start_in_parent_stack, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | void set_code_array(const Array& code_array) const; | 
|  | void set_pc_offset_array(const TypedData& pc_offset_array) const; | 
|  | bool expand_inlined() const; | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(StackTrace, Instance); | 
|  | friend class Class; | 
|  | friend class DebuggerStackTrace; | 
|  | }; | 
|  |  | 
|  | class SuspendState : public Instance { | 
|  | public: | 
|  | // :suspend_state local variable index | 
|  | static constexpr intptr_t kSuspendStateVarIndex = 0; | 
|  |  | 
|  | static intptr_t HeaderSize() { return sizeof(UntaggedSuspendState); } | 
|  | static intptr_t UnroundedSize(SuspendStatePtr ptr) { | 
|  | return UnroundedSize(ptr->untag()->frame_capacity()); | 
|  | } | 
|  | static intptr_t UnroundedSize(intptr_t frame_capacity) { | 
|  | return HeaderSize() + frame_capacity; | 
|  | } | 
|  | static intptr_t InstanceSize() { | 
|  | ASSERT_EQUAL(sizeof(UntaggedSuspendState), | 
|  | OFFSET_OF_RETURNED_VALUE(UntaggedSuspendState, payload)); | 
|  | return 0; | 
|  | } | 
|  | static intptr_t InstanceSize(intptr_t frame_capacity) { | 
|  | return RoundedAllocationSize(UnroundedSize(frame_capacity)); | 
|  | } | 
|  |  | 
|  | // Number of extra words reserved for growth of frame size | 
|  | // during SuspendState allocation. Frames do not grow in AOT. | 
|  | static intptr_t FrameSizeGrowthGap() { | 
|  | return ONLY_IN_PRECOMPILED(0) NOT_IN_PRECOMPILED(2); | 
|  | } | 
|  |  | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | static intptr_t frame_capacity_offset() { | 
|  | return OFFSET_OF(UntaggedSuspendState, frame_capacity_); | 
|  | } | 
|  | #endif | 
|  | static intptr_t frame_size_offset() { | 
|  | return OFFSET_OF(UntaggedSuspendState, frame_size_); | 
|  | } | 
|  | static intptr_t pc_offset() { return OFFSET_OF(UntaggedSuspendState, pc_); } | 
|  | static intptr_t function_data_offset() { | 
|  | return OFFSET_OF(UntaggedSuspendState, function_data_); | 
|  | } | 
|  | static intptr_t then_callback_offset() { | 
|  | return OFFSET_OF(UntaggedSuspendState, then_callback_); | 
|  | } | 
|  | static intptr_t error_callback_offset() { | 
|  | return OFFSET_OF(UntaggedSuspendState, error_callback_); | 
|  | } | 
|  | static intptr_t payload_offset() { | 
|  | return UntaggedSuspendState::payload_offset(); | 
|  | } | 
|  |  | 
|  | static SuspendStatePtr New(intptr_t frame_size, | 
|  | const Instance& function_data, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | // Makes a copy of [src] object. | 
|  | // The object should be holding a suspended frame. | 
|  | static SuspendStatePtr Clone(Thread* thread, | 
|  | const SuspendState& src, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | uword pc() const { return untag()->pc_; } | 
|  |  | 
|  | intptr_t frame_size() const { return untag()->frame_size_; } | 
|  |  | 
|  | InstancePtr function_data() const { return untag()->function_data(); } | 
|  |  | 
|  | ClosurePtr then_callback() const { return untag()->then_callback(); } | 
|  |  | 
|  | ClosurePtr error_callback() const { return untag()->error_callback(); } | 
|  |  | 
|  | // Returns Code object corresponding to the suspended function. | 
|  | CodePtr GetCodeObject() const; | 
|  |  | 
|  | private: | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | void set_frame_capacity(intptr_t frame_capcity) const; | 
|  | #endif | 
|  | void set_frame_size(intptr_t frame_size) const; | 
|  | void set_pc(uword pc) const; | 
|  | void set_function_data(const Instance& function_data) const; | 
|  | void set_then_callback(const Closure& then_callback) const; | 
|  | void set_error_callback(const Closure& error_callback) const; | 
|  |  | 
|  | uint8_t* payload() const { return untag()->payload(); } | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(SuspendState, Instance); | 
|  | friend class Class; | 
|  | friend class Interpreter; | 
|  | }; | 
|  |  | 
|  | class RegExpFlags { | 
|  | public: | 
|  | // Flags are passed to a regex object as follows: | 
|  | // 'i': ignore case, 'g': do global matches, 'm': pattern is multi line, | 
|  | // 'u': pattern is full Unicode, not just BMP, 's': '.' in pattern matches | 
|  | // all characters including line terminators. | 
|  | enum Flags { | 
|  | kNone = 0, | 
|  | kGlobal = 1, | 
|  | kIgnoreCase = 2, | 
|  | kMultiLine = 4, | 
|  | kUnicode = 8, | 
|  | kDotAll = 16, | 
|  | }; | 
|  |  | 
|  | static constexpr int kDefaultFlags = 0; | 
|  |  | 
|  | RegExpFlags() : value_(kDefaultFlags) {} | 
|  | explicit RegExpFlags(int value) : value_(value) {} | 
|  |  | 
|  | inline bool IsGlobal() const { return (value_ & kGlobal) != 0; } | 
|  | inline bool IgnoreCase() const { return (value_ & kIgnoreCase) != 0; } | 
|  | inline bool IsMultiLine() const { return (value_ & kMultiLine) != 0; } | 
|  | inline bool IsUnicode() const { return (value_ & kUnicode) != 0; } | 
|  | inline bool IsDotAll() const { return (value_ & kDotAll) != 0; } | 
|  |  | 
|  | inline bool NeedsUnicodeCaseEquivalents() { | 
|  | // Both unicode and ignore_case flags are set. We need to use ICU to find | 
|  | // the closure over case equivalents. | 
|  | return IsUnicode() && IgnoreCase(); | 
|  | } | 
|  |  | 
|  | void SetGlobal() { value_ |= kGlobal; } | 
|  | void SetIgnoreCase() { value_ |= kIgnoreCase; } | 
|  | void SetMultiLine() { value_ |= kMultiLine; } | 
|  | void SetUnicode() { value_ |= kUnicode; } | 
|  | void SetDotAll() { value_ |= kDotAll; } | 
|  |  | 
|  | const char* ToCString() const; | 
|  |  | 
|  | int value() const { return value_; } | 
|  |  | 
|  | bool operator==(const RegExpFlags& other) const { | 
|  | return value_ == other.value_; | 
|  | } | 
|  | bool operator!=(const RegExpFlags& other) const { | 
|  | return value_ != other.value_; | 
|  | } | 
|  |  | 
|  | private: | 
|  | int value_; | 
|  | }; | 
|  |  | 
|  | // Internal JavaScript regular expression object. | 
|  | class RegExp : public Instance { | 
|  | public: | 
|  | // Meaning of RegExType: | 
|  | // kUninitialized: the type of th regexp has not been initialized yet. | 
|  | // kSimple: A simple pattern to match against, using string indexOf operation. | 
|  | // kComplex: A complex pattern to match. | 
|  | enum RegExType { | 
|  | kUninitialized = 0, | 
|  | kSimple = 1, | 
|  | kComplex = 2, | 
|  | }; | 
|  |  | 
|  | using TypeBits = BitField<int8_t, RegExType, 0, 2>; | 
|  |  | 
|  | // Must be kept in sync with RegExFlags::Flags. | 
|  | using GlobalBit = BitField<int8_t, bool, TypeBits::kNextBit>; | 
|  | using IgnoreCaseBit = BitField<int8_t, bool, GlobalBit::kNextBit>; | 
|  | using MultiLineBit = BitField<int8_t, bool, IgnoreCaseBit::kNextBit>; | 
|  | using UnicodeBit = BitField<int8_t, bool, MultiLineBit::kNextBit>; | 
|  | using DotAllBit = BitField<int8_t, bool, UnicodeBit::kNextBit>; | 
|  |  | 
|  | // The portion of the bitfield container that contains all the above | 
|  | // bool bits, which is passed to the constructor for RegExFlags. | 
|  | using FlagsBits = BitField<int8_t, | 
|  | int8_t, | 
|  | TypeBits::kNextBit, | 
|  | DotAllBit::kNextBit - TypeBits::kNextBit>; | 
|  |  | 
|  | bool is_initialized() const { return (type() != kUninitialized); } | 
|  | bool is_simple() const { return (type() == kSimple); } | 
|  | bool is_complex() const { return (type() == kComplex); } | 
|  |  | 
|  | intptr_t num_registers(bool is_one_byte) const { | 
|  | return LoadNonPointer<intptr_t, std::memory_order_relaxed>( | 
|  | is_one_byte ? &untag()->num_one_byte_registers_ | 
|  | : &untag()->num_two_byte_registers_); | 
|  | } | 
|  |  | 
|  | StringPtr pattern() const { return untag()->pattern(); } | 
|  | intptr_t num_bracket_expressions() const { | 
|  | return untag()->num_bracket_expressions_; | 
|  | } | 
|  | ArrayPtr capture_name_map() const { return untag()->capture_name_map(); } | 
|  |  | 
|  | TypedDataPtr bytecode(bool is_one_byte, bool sticky) const { | 
|  | if (sticky) { | 
|  | return TypedData::RawCast( | 
|  | is_one_byte ? untag()->one_byte_sticky<std::memory_order_acquire>() | 
|  | : untag()->two_byte_sticky<std::memory_order_acquire>()); | 
|  | } else { | 
|  | return TypedData::RawCast( | 
|  | is_one_byte ? untag()->one_byte<std::memory_order_acquire>() | 
|  | : untag()->two_byte<std::memory_order_acquire>()); | 
|  | } | 
|  | } | 
|  |  | 
|  | static intptr_t function_offset(intptr_t cid, bool sticky) { | 
|  | if (sticky) { | 
|  | switch (cid) { | 
|  | case kOneByteStringCid: | 
|  | return OFFSET_OF(UntaggedRegExp, one_byte_sticky_); | 
|  | case kTwoByteStringCid: | 
|  | return OFFSET_OF(UntaggedRegExp, two_byte_sticky_); | 
|  | } | 
|  | } else { | 
|  | switch (cid) { | 
|  | case kOneByteStringCid: | 
|  | return OFFSET_OF(UntaggedRegExp, one_byte_); | 
|  | case kTwoByteStringCid: | 
|  | return OFFSET_OF(UntaggedRegExp, two_byte_); | 
|  | } | 
|  | } | 
|  |  | 
|  | UNREACHABLE(); | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | FunctionPtr function(intptr_t cid, bool sticky) const { | 
|  | if (sticky) { | 
|  | switch (cid) { | 
|  | case kOneByteStringCid: | 
|  | return static_cast<FunctionPtr>(untag()->one_byte_sticky()); | 
|  | case kTwoByteStringCid: | 
|  | return static_cast<FunctionPtr>(untag()->two_byte_sticky()); | 
|  | } | 
|  | } else { | 
|  | switch (cid) { | 
|  | case kOneByteStringCid: | 
|  | return static_cast<FunctionPtr>(untag()->one_byte()); | 
|  | case kTwoByteStringCid: | 
|  | return static_cast<FunctionPtr>(untag()->two_byte()); | 
|  | } | 
|  | } | 
|  |  | 
|  | UNREACHABLE(); | 
|  | return Function::null(); | 
|  | } | 
|  |  | 
|  | void set_pattern(const String& pattern) const; | 
|  | void set_function(intptr_t cid, bool sticky, const Function& value) const; | 
|  | void set_bytecode(bool is_one_byte, | 
|  | bool sticky, | 
|  | const TypedData& bytecode) const; | 
|  |  | 
|  | void set_num_bracket_expressions(SmiPtr value) const; | 
|  | void set_num_bracket_expressions(const Smi& value) const; | 
|  | void set_num_bracket_expressions(intptr_t value) const; | 
|  | void set_capture_name_map(const Array& array) const; | 
|  | void set_is_global() const { | 
|  | untag()->type_flags_.UpdateBool<GlobalBit>(true); | 
|  | } | 
|  | void set_is_ignore_case() const { | 
|  | untag()->type_flags_.UpdateBool<IgnoreCaseBit>(true); | 
|  | } | 
|  | void set_is_multi_line() const { | 
|  | untag()->type_flags_.UpdateBool<MultiLineBit>(true); | 
|  | } | 
|  | void set_is_unicode() const { | 
|  | untag()->type_flags_.UpdateBool<UnicodeBit>(true); | 
|  | } | 
|  | void set_is_dot_all() const { | 
|  | untag()->type_flags_.UpdateBool<DotAllBit>(true); | 
|  | } | 
|  | void set_is_simple() const { set_type(kSimple); } | 
|  | void set_is_complex() const { set_type(kComplex); } | 
|  | void set_num_registers(bool is_one_byte, intptr_t value) const { | 
|  | StoreNonPointer<intptr_t, intptr_t, std::memory_order_relaxed>( | 
|  | is_one_byte ? &untag()->num_one_byte_registers_ | 
|  | : &untag()->num_two_byte_registers_, | 
|  | value); | 
|  | } | 
|  |  | 
|  | RegExpFlags flags() const { | 
|  | return RegExpFlags(untag()->type_flags_.Read<FlagsBits>()); | 
|  | } | 
|  | void set_flags(RegExpFlags flags) const { | 
|  | untag()->type_flags_.Update<FlagsBits>(flags.value()); | 
|  | } | 
|  |  | 
|  | virtual bool CanonicalizeEquals(const Instance& other) const; | 
|  | virtual uint32_t CanonicalizeHash() const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedRegExp)); | 
|  | } | 
|  |  | 
|  | static RegExpPtr New(Zone* zone, Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | void set_type(RegExType type) const { | 
|  | untag()->type_flags_.Update<TypeBits>(type); | 
|  | } | 
|  | RegExType type() const { return untag()->type_flags_.Read<TypeBits>(); } | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(RegExp, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Corresponds to _WeakProperty in dart:core. | 
|  | class WeakProperty : public Instance { | 
|  | public: | 
|  | ObjectPtr key() const { return untag()->key(); } | 
|  | void set_key(const Object& key) const { untag()->set_key(key.ptr()); } | 
|  | static intptr_t key_offset() { return OFFSET_OF(UntaggedWeakProperty, key_); } | 
|  |  | 
|  | ObjectPtr value() const { return untag()->value(); } | 
|  | void set_value(const Object& value) const { untag()->set_value(value.ptr()); } | 
|  | static intptr_t value_offset() { | 
|  | return OFFSET_OF(UntaggedWeakProperty, value_); | 
|  | } | 
|  |  | 
|  | static WeakPropertyPtr New(Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedWeakProperty)); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(WeakProperty, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Corresponds to _WeakReference in dart:core. | 
|  | class WeakReference : public Instance { | 
|  | public: | 
|  | ObjectPtr target() const { return untag()->target(); } | 
|  | void set_target(const Object& target) const { | 
|  | untag()->set_target(target.ptr()); | 
|  | } | 
|  | static intptr_t target_offset() { | 
|  | return OFFSET_OF(UntaggedWeakReference, target_); | 
|  | } | 
|  |  | 
|  | static intptr_t type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedWeakReference, type_arguments_); | 
|  | } | 
|  |  | 
|  | static WeakReferencePtr New(Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedWeakReference)); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(WeakReference, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class FinalizerBase; | 
|  | class FinalizerEntry : public Instance { | 
|  | public: | 
|  | ObjectPtr value() const { return untag()->value(); } | 
|  | void set_value(const Object& value) const { untag()->set_value(value.ptr()); } | 
|  | static intptr_t value_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizerEntry, value_); | 
|  | } | 
|  |  | 
|  | ObjectPtr detach() const { return untag()->detach(); } | 
|  | void set_detach(const Object& value) const { | 
|  | untag()->set_detach(value.ptr()); | 
|  | } | 
|  | static intptr_t detach_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizerEntry, detach_); | 
|  | } | 
|  |  | 
|  | ObjectPtr token() const { return untag()->token(); } | 
|  | void set_token(const Object& value) const { untag()->set_token(value.ptr()); } | 
|  | static intptr_t token_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizerEntry, token_); | 
|  | } | 
|  |  | 
|  | FinalizerBasePtr finalizer() const { return untag()->finalizer(); } | 
|  | void set_finalizer(const FinalizerBase& value) const; | 
|  | static intptr_t finalizer_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizerEntry, finalizer_); | 
|  | } | 
|  |  | 
|  | FinalizerEntryPtr next() const { return untag()->next(); } | 
|  | void set_next(const FinalizerEntry& value) const { | 
|  | untag()->set_next(value.ptr()); | 
|  | } | 
|  | static intptr_t next_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizerEntry, next_); | 
|  | } | 
|  |  | 
|  | intptr_t external_size() const { return untag()->external_size(); } | 
|  | void set_external_size(intptr_t value) const { | 
|  | untag()->set_external_size(value); | 
|  | } | 
|  | static intptr_t external_size_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizerEntry, external_size_); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedFinalizerEntry)); | 
|  | } | 
|  |  | 
|  | // Allocates a new FinalizerEntry, initializing the external size (to 0) and | 
|  | // finalizer. | 
|  | // | 
|  | // Should only be used for object tests. | 
|  | // | 
|  | // Does not initialize `value`, `token`, and `detach` to allow for flexible | 
|  | // testing code setting those manually. | 
|  | // | 
|  | // Does _not_ add the entry to the finalizer. We could add the entry to | 
|  | // finalizer.all_entries.data, but we have no way of initializing the hashset | 
|  | // index. | 
|  | static FinalizerEntryPtr New(const FinalizerBase& finalizer, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(FinalizerEntry, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class FinalizerBase : public Instance { | 
|  | public: | 
|  | static intptr_t isolate_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizerBase, isolate_); | 
|  | } | 
|  | Isolate* isolate() const { return untag()->isolate_; } | 
|  | void set_isolate(Isolate* value) const { untag()->isolate_ = value; } | 
|  |  | 
|  | static intptr_t detachments_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizerBase, detachments_); | 
|  | } | 
|  |  | 
|  | SetPtr all_entries() const { return untag()->all_entries(); } | 
|  | void set_all_entries(const Set& value) const { | 
|  | untag()->set_all_entries(value.ptr()); | 
|  | } | 
|  | static intptr_t all_entries_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizerBase, all_entries_); | 
|  | } | 
|  |  | 
|  | FinalizerEntryPtr entries_collected() const { | 
|  | return untag()->entries_collected(); | 
|  | } | 
|  | void set_entries_collected(const FinalizerEntry& value) const { | 
|  | untag()->set_entries_collected(value.ptr()); | 
|  | } | 
|  | static intptr_t entries_collected_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizer, entries_collected_); | 
|  | } | 
|  |  | 
|  | private: | 
|  | HEAP_OBJECT_IMPLEMENTATION(FinalizerBase, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class Finalizer : public FinalizerBase { | 
|  | public: | 
|  | static intptr_t type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizer, type_arguments_); | 
|  | } | 
|  |  | 
|  | ObjectPtr callback() const { return untag()->callback(); } | 
|  | static intptr_t callback_offset() { | 
|  | return OFFSET_OF(UntaggedFinalizer, callback_); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedFinalizer)); | 
|  | } | 
|  |  | 
|  | static FinalizerPtr New(Heap::Space space = Heap::kNew); | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(Finalizer, FinalizerBase); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class NativeFinalizer : public FinalizerBase { | 
|  | public: | 
|  | typedef void (*Callback)(void*); | 
|  |  | 
|  | PointerPtr callback() const { return untag()->callback(); } | 
|  | void set_callback(const Pointer& value) const { | 
|  | untag()->set_callback(value.ptr()); | 
|  | } | 
|  | static intptr_t callback_offset() { | 
|  | return OFFSET_OF(UntaggedNativeFinalizer, callback_); | 
|  | } | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedNativeFinalizer)); | 
|  | } | 
|  |  | 
|  | static NativeFinalizerPtr New(Heap::Space space = Heap::kNew); | 
|  |  | 
|  | void RunCallback(const FinalizerEntry& entry, | 
|  | const char* trace_context) const; | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(NativeFinalizer, FinalizerBase); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class MirrorReference : public Instance { | 
|  | public: | 
|  | ObjectPtr referent() const { return untag()->referent(); } | 
|  |  | 
|  | void set_referent(const Object& referent) const { | 
|  | untag()->set_referent(referent.ptr()); | 
|  | } | 
|  |  | 
|  | AbstractTypePtr GetAbstractTypeReferent() const; | 
|  |  | 
|  | ClassPtr GetClassReferent() const; | 
|  |  | 
|  | FieldPtr GetFieldReferent() const; | 
|  |  | 
|  | FunctionPtr GetFunctionReferent() const; | 
|  |  | 
|  | FunctionTypePtr GetFunctionTypeReferent() const; | 
|  |  | 
|  | LibraryPtr GetLibraryReferent() const; | 
|  |  | 
|  | TypeParameterPtr GetTypeParameterReferent() const; | 
|  |  | 
|  | static MirrorReferencePtr New(const Object& referent, | 
|  | Heap::Space space = Heap::kNew); | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedMirrorReference)); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(MirrorReference, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | class UserTag : public Instance { | 
|  | public: | 
|  | uword tag() const { return untag()->tag(); } | 
|  | void set_tag(uword t) const { | 
|  | ASSERT(t >= UserTags::kUserTagIdOffset); | 
|  | ASSERT(t < UserTags::kUserTagIdOffset + UserTags::kMaxUserTags); | 
|  | StoreNonPointer(&untag()->tag_, t); | 
|  | } | 
|  |  | 
|  | bool streamable() const { return untag()->streamable(); } | 
|  | void set_streamable(bool streamable) { | 
|  | StoreNonPointer(&untag()->streamable_, streamable); | 
|  | } | 
|  |  | 
|  | static intptr_t tag_offset() { return OFFSET_OF(UntaggedUserTag, tag_); } | 
|  |  | 
|  | StringPtr label() const { return untag()->label(); } | 
|  |  | 
|  | UserTagPtr MakeActive() const; | 
|  |  | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedUserTag)); | 
|  | } | 
|  |  | 
|  | static UserTagPtr New(const String& label, Heap::Space space = Heap::kOld); | 
|  | static UserTagPtr DefaultTag(); | 
|  |  | 
|  | static bool TagTableIsFull(Thread* thread); | 
|  | static UserTagPtr FindTagById(const Isolate* isolate, uword tag_id); | 
|  | static UserTagPtr FindTagInIsolate(Isolate* isolate, | 
|  | Thread* thread, | 
|  | const String& label); | 
|  |  | 
|  | private: | 
|  | static UserTagPtr FindTagInIsolate(Thread* thread, const String& label); | 
|  | static void AddTagToIsolate(Thread* thread, const UserTag& tag); | 
|  |  | 
|  | void set_label(const String& tag_label) const { | 
|  | untag()->set_label(tag_label.ptr()); | 
|  | } | 
|  |  | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(UserTag, Instance); | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Represents abstract FutureOr class in dart:async. | 
|  | class FutureOr : public Instance { | 
|  | public: | 
|  | static intptr_t InstanceSize() { | 
|  | return RoundedAllocationSize(sizeof(UntaggedFutureOr)); | 
|  | } | 
|  |  | 
|  | virtual TypeArgumentsPtr GetTypeArguments() const { | 
|  | return untag()->type_arguments(); | 
|  | } | 
|  | static intptr_t type_arguments_offset() { | 
|  | return OFFSET_OF(UntaggedFutureOr, type_arguments_); | 
|  | } | 
|  |  | 
|  | private: | 
|  | FINAL_HEAP_OBJECT_IMPLEMENTATION(FutureOr, Instance); | 
|  |  | 
|  | friend class Class; | 
|  | }; | 
|  |  | 
|  | // Breaking cycles and loops. | 
|  | ClassPtr Object::clazz() const { | 
|  | uword raw_value = static_cast<uword>(ptr_); | 
|  | if ((raw_value & kSmiTagMask) == kSmiTag) { | 
|  | return Smi::Class(); | 
|  | } | 
|  | return IsolateGroup::Current()->class_table()->At( | 
|  | ptr()->GetClassIdOfHeapObject()); | 
|  | } | 
|  |  | 
|  | DART_FORCE_INLINE | 
|  | void Object::setPtr(ObjectPtr value, intptr_t default_cid) { | 
|  | ptr_ = value; | 
|  | intptr_t cid = value->GetClassId(); | 
|  | // Free-list elements cannot be wrapped in a handle. | 
|  | ASSERT(cid != kFreeListElement); | 
|  | ASSERT(cid != kForwardingCorpse); | 
|  | if (cid == kNullCid) { | 
|  | cid = default_cid; | 
|  | } else if (cid >= kNumPredefinedCids) { | 
|  | cid = kInstanceCid; | 
|  | } | 
|  | set_vtable(builtin_vtables_[cid]); | 
|  | } | 
|  |  | 
|  | #if defined(DART_DYNAMIC_MODULES) | 
|  | BytecodePtr Function::GetBytecode() const { | 
|  | return GetBytecode(ptr()); | 
|  | } | 
|  |  | 
|  | BytecodePtr Function::GetBytecode(FunctionPtr function) { | 
|  | return Bytecode::RawCast(function.untag()->ic_data_array_or_bytecode()); | 
|  | } | 
|  |  | 
|  | bool Function::HasBytecode() const { | 
|  | return HasBytecode(ptr()); | 
|  | } | 
|  |  | 
|  | bool Function::HasBytecode(FunctionPtr function) { | 
|  | return function.untag()->ic_data_array_or_bytecode()->IsBytecode(); | 
|  | } | 
|  | #endif  // defined(DART_DYNAMIC_MODULES) | 
|  |  | 
|  | intptr_t Field::HostOffset() const { | 
|  | ASSERT(is_instance());  // Valid only for dart instance fields. | 
|  | return (Smi::Value(untag()->host_offset_or_field_id()) * kCompressedWordSize); | 
|  | } | 
|  |  | 
|  | intptr_t Field::TargetOffset() const { | 
|  | ASSERT(is_instance());  // Valid only for dart instance fields. | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | return (untag()->target_offset_ * compiler::target::kCompressedWordSize); | 
|  | #else | 
|  | return HostOffset(); | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | inline intptr_t Field::TargetOffsetOf(const FieldPtr field) { | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | return field->untag()->target_offset_; | 
|  | #else | 
|  | return Smi::Value(field->untag()->host_offset_or_field_id()); | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | void Field::SetOffset(intptr_t host_offset_in_bytes, | 
|  | intptr_t target_offset_in_bytes) const { | 
|  | ASSERT(is_instance());  // Valid only for dart instance fields. | 
|  | ASSERT(kCompressedWordSize != 0); | 
|  | untag()->set_host_offset_or_field_id( | 
|  | Smi::New(host_offset_in_bytes / kCompressedWordSize)); | 
|  | #if !defined(DART_PRECOMPILED_RUNTIME) | 
|  | ASSERT(compiler::target::kCompressedWordSize != 0); | 
|  | StoreNonPointer( | 
|  | &untag()->target_offset_, | 
|  | target_offset_in_bytes / compiler::target::kCompressedWordSize); | 
|  | #else | 
|  | ASSERT(host_offset_in_bytes == target_offset_in_bytes); | 
|  | #endif  //  !defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | ObjectPtr Field::StaticValue() const { | 
|  | ASSERT(is_static());  // Valid only for static dart fields. | 
|  | return Isolate::Current()->field_table()->At(field_id()); | 
|  | } | 
|  |  | 
|  | inline intptr_t Field::field_id() const { | 
|  | return Smi::Value(untag()->host_offset_or_field_id()); | 
|  | } | 
|  |  | 
|  | void Field::set_field_id(intptr_t field_id) const { | 
|  | DEBUG_ASSERT( | 
|  | IsolateGroup::Current()->program_lock()->IsCurrentThreadWriter()); | 
|  | set_field_id_unsafe(field_id); | 
|  | } | 
|  |  | 
|  | void Field::set_field_id_unsafe(intptr_t field_id) const { | 
|  | ASSERT(is_static()); | 
|  | untag()->set_host_offset_or_field_id(Smi::New(field_id)); | 
|  | } | 
|  |  | 
|  | intptr_t WeakArray::LengthOf(const WeakArrayPtr array) { | 
|  | return Smi::Value(array->untag()->length()); | 
|  | } | 
|  |  | 
|  | void Context::SetAt(intptr_t index, const Object& value) const { | 
|  | untag()->set_element(index, value.ptr()); | 
|  | } | 
|  |  | 
|  | intptr_t Instance::GetNativeField(int index) const { | 
|  | ASSERT(IsValidNativeIndex(index)); | 
|  | NoSafepointScope no_safepoint; | 
|  | TypedDataPtr native_fields = static_cast<TypedDataPtr>( | 
|  | NativeFieldsAddr()->Decompress(untag()->heap_base())); | 
|  | if (native_fields == TypedData::null()) { | 
|  | return 0; | 
|  | } | 
|  | return reinterpret_cast<intptr_t*>(native_fields->untag()->data())[index]; | 
|  | } | 
|  |  | 
|  | void Instance::GetNativeFields(uint16_t num_fields, | 
|  | intptr_t* field_values) const { | 
|  | NoSafepointScope no_safepoint; | 
|  | ASSERT(num_fields == NumNativeFields()); | 
|  | ASSERT(field_values != nullptr); | 
|  | TypedDataPtr native_fields = static_cast<TypedDataPtr>( | 
|  | NativeFieldsAddr()->Decompress(untag()->heap_base())); | 
|  | if (native_fields == TypedData::null()) { | 
|  | for (intptr_t i = 0; i < num_fields; i++) { | 
|  | field_values[i] = 0; | 
|  | } | 
|  | } | 
|  | intptr_t* fields = | 
|  | reinterpret_cast<intptr_t*>(native_fields->untag()->data()); | 
|  | for (intptr_t i = 0; i < num_fields; i++) { | 
|  | field_values[i] = fields[i]; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool String::Equals(const String& str) const { | 
|  | if (ptr() == str.ptr()) { | 
|  | return true;  // Both handles point to the same raw instance. | 
|  | } | 
|  | if (str.IsNull()) { | 
|  | return false; | 
|  | } | 
|  | if (IsCanonical() && str.IsCanonical()) { | 
|  | return false;  // Two symbols that aren't identical aren't equal. | 
|  | } | 
|  | if (HasHash() && str.HasHash() && (Hash() != str.Hash())) { | 
|  | return false;  // Both sides have hash codes and they do not match. | 
|  | } | 
|  | return Equals(str, 0, str.Length()); | 
|  | } | 
|  |  | 
|  | intptr_t Library::UrlHash() const { | 
|  | intptr_t result = String::GetCachedHash(url()); | 
|  | ASSERT(result != 0); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | void MegamorphicCache::SetEntry(const Array& array, | 
|  | intptr_t index, | 
|  | const Smi& class_id, | 
|  | const Object& target) { | 
|  | ASSERT(target.IsNull() || target.IsFunction() || target.IsSmi()); | 
|  | array.SetAt((index * kEntryLength) + kClassIdIndex, class_id); | 
|  | array.SetAt((index * kEntryLength) + kTargetFunctionIndex, target); | 
|  | } | 
|  |  | 
|  | ObjectPtr MegamorphicCache::GetClassId(const Array& array, intptr_t index) { | 
|  | return array.At((index * kEntryLength) + kClassIdIndex); | 
|  | } | 
|  |  | 
|  | ObjectPtr MegamorphicCache::GetTargetFunction(const Array& array, | 
|  | intptr_t index) { | 
|  | return array.At((index * kEntryLength) + kTargetFunctionIndex); | 
|  | } | 
|  |  | 
|  | inline uword AbstractType::Hash() const { | 
|  | ASSERT(IsFinalized()); | 
|  | intptr_t result = Smi::Value(untag()->hash()); | 
|  | if (result != 0) { | 
|  | return result; | 
|  | } | 
|  | return ComputeHash(); | 
|  | } | 
|  |  | 
|  | inline void AbstractType::SetHash(intptr_t value) const { | 
|  | // This is only safe because we create a new Smi, which does not cause | 
|  | // heap allocation. | 
|  | untag()->set_hash(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | inline intptr_t RecordType::NumFields() const { | 
|  | return Array::LengthOf(field_types()); | 
|  | } | 
|  |  | 
|  | inline uword TypeArguments::Hash() const { | 
|  | if (IsNull()) return kAllDynamicHash; | 
|  | intptr_t result = Smi::Value(untag()->hash()); | 
|  | if (result != 0) { | 
|  | return result; | 
|  | } | 
|  | return ComputeHash(); | 
|  | } | 
|  |  | 
|  | inline void TypeArguments::SetHash(intptr_t value) const { | 
|  | // This is only safe because we create a new Smi, which does not cause | 
|  | // heap allocation. | 
|  | untag()->set_hash(Smi::New(value)); | 
|  | } | 
|  |  | 
|  | inline uint16_t String::CharAt(StringPtr str, intptr_t index) { | 
|  | switch (str->GetClassIdOfHeapObject()) { | 
|  | case kOneByteStringCid: | 
|  | return OneByteString::CharAt(static_cast<OneByteStringPtr>(str), index); | 
|  | case kTwoByteStringCid: | 
|  | return TwoByteString::CharAt(static_cast<TwoByteStringPtr>(str), index); | 
|  | } | 
|  | UNREACHABLE(); | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | // A view on an [Array] as a list of tuples, optionally starting at an offset. | 
|  | // | 
|  | // Example: We store a list of (kind, function, code) tuples into the | 
|  | // [Code::static_calls_target_table] array of type [Array]. | 
|  | // | 
|  | // This helper class can then be used via | 
|  | // | 
|  | //     using CallTableView = ArrayOfTuplesView< | 
|  | //         Code::Kind, std::tuple<Smi, Function, Code>>; | 
|  | // | 
|  | //     auto& array = Array::Handle(code.static_calls_targets_table()); | 
|  | //     CallTableView static_calls(array); | 
|  | // | 
|  | //     // Using convenient for loop. | 
|  | //     auto& function = Function::Handle(); | 
|  | //     for (auto& call : static_calls) { | 
|  | //       function = call.Get<Code::kSCallTableFunctionTarget>(); | 
|  | //       call.Set<Code::kSCallTableFunctionTarget>(function); | 
|  | //     } | 
|  | // | 
|  | //     // Using manual loop. | 
|  | //     auto& function = Function::Handle(); | 
|  | //     for (intptr_t i = 0; i < static_calls.Length(); ++i) { | 
|  | //       auto call = static_calls[i]; | 
|  | //       function = call.Get<Code::kSCallTableFunctionTarget>(); | 
|  | //       call.Set<Code::kSCallTableFunctionTarget>(function); | 
|  | //     } | 
|  | // | 
|  | // | 
|  | // Template parameters: | 
|  | // | 
|  | //   * [EnumType] must be a normal enum which enumerates the entries of the | 
|  | //     tuple | 
|  | // | 
|  | //   * [kStartOffset] is the offset at which the first tuple in the array | 
|  | //     starts (can be 0). | 
|  | // | 
|  | //   * [TupleT] must be a std::tuple<...> where "..." are the heap object handle | 
|  | //     classes (e.g. 'Code', 'Smi', 'Object') | 
|  | template <typename EnumType, typename TupleT, int kStartOffset = 0> | 
|  | class ArrayOfTuplesView { | 
|  | public: | 
|  | static constexpr intptr_t EntrySize = std::tuple_size<TupleT>::value; | 
|  |  | 
|  | class Iterator; | 
|  |  | 
|  | class TupleView { | 
|  | public: | 
|  | TupleView(const Array& array, intptr_t index) | 
|  | : array_(array), index_(index) {} | 
|  |  | 
|  | template <EnumType kElement, | 
|  | std::memory_order order = std::memory_order_relaxed> | 
|  | typename std::tuple_element<kElement, TupleT>::type::ObjectPtrType Get() | 
|  | const { | 
|  | using object_type = typename std::tuple_element<kElement, TupleT>::type; | 
|  | return object_type::RawCast(array_.At<order>(index_ + kElement)); | 
|  | } | 
|  |  | 
|  | template <EnumType kElement, | 
|  | std::memory_order order = std::memory_order_relaxed> | 
|  | void Set(const typename std::tuple_element<kElement, TupleT>::type& value) | 
|  | const { | 
|  | array_.SetAt<order>(index_ + kElement, value); | 
|  | } | 
|  |  | 
|  | intptr_t index() const { return (index_ - kStartOffset) / EntrySize; } | 
|  |  | 
|  | private: | 
|  | const Array& array_; | 
|  | intptr_t index_; | 
|  |  | 
|  | friend class Iterator; | 
|  | }; | 
|  |  | 
|  | class Iterator { | 
|  | public: | 
|  | Iterator(const Array& array, intptr_t index) : entry_(array, index) {} | 
|  |  | 
|  | bool operator==(const Iterator& other) { | 
|  | return entry_.index_ == other.entry_.index_; | 
|  | } | 
|  | bool operator!=(const Iterator& other) { | 
|  | return entry_.index_ != other.entry_.index_; | 
|  | } | 
|  |  | 
|  | const TupleView& operator*() const { return entry_; } | 
|  |  | 
|  | Iterator& operator++() { | 
|  | entry_.index_ += EntrySize; | 
|  | return *this; | 
|  | } | 
|  |  | 
|  | private: | 
|  | TupleView entry_; | 
|  | }; | 
|  |  | 
|  | explicit ArrayOfTuplesView(const Array& array) : array_(array) { | 
|  | ASSERT(!array.IsNull()); | 
|  | ASSERT(array.Length() >= kStartOffset); | 
|  | ASSERT(array.Length() % EntrySize == kStartOffset); | 
|  | } | 
|  |  | 
|  | intptr_t Length() const { | 
|  | return (array_.Length() - kStartOffset) / EntrySize; | 
|  | } | 
|  |  | 
|  | TupleView At(intptr_t i) const { | 
|  | return TupleView(array_, kStartOffset + i * EntrySize); | 
|  | } | 
|  |  | 
|  | TupleView operator[](intptr_t i) const { return At(i); } | 
|  |  | 
|  | Iterator begin() const { return Iterator(array_, kStartOffset); } | 
|  |  | 
|  | Iterator end() const { | 
|  | return Iterator(array_, kStartOffset + Length() * EntrySize); | 
|  | } | 
|  |  | 
|  | private: | 
|  | const Array& array_; | 
|  | }; | 
|  |  | 
|  | using StaticCallsTable = | 
|  | ArrayOfTuplesView<Code::SCallTableEntry, std::tuple<Smi, Object, Function>>; | 
|  |  | 
|  | using StaticCallsTableEntry = StaticCallsTable::TupleView; | 
|  |  | 
|  | using SubtypeTestCacheTable = ArrayOfTuplesView<SubtypeTestCache::Entries, | 
|  | std::tuple<Object, | 
|  | TypeArguments, | 
|  | TypeArguments, | 
|  | TypeArguments, | 
|  | TypeArguments, | 
|  | TypeArguments, | 
|  | AbstractType, | 
|  | Bool>>; | 
|  |  | 
|  | using MegamorphicCacheEntries = | 
|  | ArrayOfTuplesView<MegamorphicCache::EntryType, std::tuple<Smi, Object>>; | 
|  |  | 
|  | using InstantiationsCacheTable = | 
|  | ArrayOfTuplesView<TypeArguments::Cache::Entry, | 
|  | std::tuple<Object, TypeArguments, TypeArguments>, | 
|  | TypeArguments::Cache::kHeaderSize>; | 
|  |  | 
|  | void DumpTypeTable(Isolate* isolate); | 
|  | void DumpTypeParameterTable(Isolate* isolate); | 
|  | void DumpTypeArgumentsTable(Isolate* isolate); | 
|  |  | 
|  | bool FindPragmaInMetadata(Thread* T, | 
|  | const Object& metadata_obj, | 
|  | const String& pragma_name, | 
|  | bool multiple = false, | 
|  | Object* options = nullptr); | 
|  |  | 
|  | EntryPointPragma FindEntryPointPragma(IsolateGroup* isolate_group, | 
|  | const Array& metadata, | 
|  | Field* reusable_field_handle, | 
|  | Object* reusable_object_handle); | 
|  |  | 
|  | #undef PRECOMPILER_WSR_FIELD_DECLARATION | 
|  |  | 
|  | }  // namespace dart | 
|  |  | 
|  | #endif  // RUNTIME_VM_OBJECT_H_ |