6 #include "src/heap/object-stats.h" 8 #include <unordered_set> 10 #include "src/assembler-inl.h" 11 #include "src/base/bits.h" 12 #include "src/compilation-cache.h" 13 #include "src/counters.h" 14 #include "src/globals.h" 15 #include "src/heap/heap-inl.h" 16 #include "src/heap/mark-compact.h" 17 #include "src/isolate.h" 18 #include "src/objects/compilation-cache-inl.h" 19 #include "src/objects/heap-object.h" 20 #include "src/objects/js-collection-inl.h" 21 #include "src/objects/literal-objects-inl.h" 22 #include "src/objects/slots.h" 23 #include "src/objects/templates.h" 24 #include "src/utils.h" 29 static base::LazyMutex object_stats_mutex = LAZY_MUTEX_INITIALIZER;
34 size_t* embedder_fields_count,
35 size_t* unboxed_double_fields_count,
36 size_t* raw_fields_count)
37 : tagged_fields_count_(tagged_fields_count),
38 embedder_fields_count_(embedder_fields_count),
39 unboxed_double_fields_count_(unboxed_double_fields_count),
40 raw_fields_count_(raw_fields_count) {}
43 size_t old_pointer_fields_count = *tagged_fields_count_;
45 size_t tagged_fields_count_in_object =
46 *tagged_fields_count_ - old_pointer_fields_count;
48 int object_size_in_words = host->Size() / kPointerSize;
49 DCHECK_LE(tagged_fields_count_in_object, object_size_in_words);
50 size_t raw_fields_count_in_object =
51 object_size_in_words - tagged_fields_count_in_object;
53 if (host->IsJSObject()) {
54 JSObjectFieldStats field_stats = GetInobjectFieldStats(host->map());
56 DCHECK_LE(field_stats.embedded_fields_count_,
57 tagged_fields_count_in_object);
58 tagged_fields_count_in_object -= field_stats.embedded_fields_count_;
59 *tagged_fields_count_ -= field_stats.embedded_fields_count_;
60 *embedder_fields_count_ += field_stats.embedded_fields_count_;
63 DCHECK_LE(field_stats.unboxed_double_fields_count_,
64 raw_fields_count_in_object);
65 raw_fields_count_in_object -= field_stats.unboxed_double_fields_count_;
66 *unboxed_double_fields_count_ += field_stats.unboxed_double_fields_count_;
68 *raw_fields_count_ += raw_fields_count_in_object;
73 *tagged_fields_count_ += (end - start);
77 *tagged_fields_count_ += (end - start);
81 struct JSObjectFieldStats {
83 : embedded_fields_count_(0), unboxed_double_fields_count_(0) {}
85 unsigned embedded_fields_count_ : kDescriptorIndexBitCount;
86 unsigned unboxed_double_fields_count_ : kDescriptorIndexBitCount;
88 std::unordered_map<Map, JSObjectFieldStats, ObjectPtr::Hasher>
91 JSObjectFieldStats GetInobjectFieldStats(
Map map);
93 size_t*
const tagged_fields_count_;
94 size_t*
const embedder_fields_count_;
95 size_t*
const unboxed_double_fields_count_;
96 size_t*
const raw_fields_count_;
99 FieldStatsCollector::JSObjectFieldStats
100 FieldStatsCollector::GetInobjectFieldStats(
Map map) {
101 auto iter = field_stats_cache_.find(map);
102 if (iter != field_stats_cache_.end()) {
106 JSObjectFieldStats stats;
107 stats.embedded_fields_count_ = JSObject::GetEmbedderFieldCount(map);
108 if (!map->is_dictionary_map()) {
109 int nof = map->NumberOfOwnDescriptors();
110 DescriptorArray* descriptors = map->instance_descriptors();
111 for (
int descriptor = 0; descriptor < nof; descriptor++) {
112 PropertyDetails details = descriptors->GetDetails(descriptor);
113 if (details.location() == kField) {
114 FieldIndex index = FieldIndex::ForDescriptor(map, descriptor);
116 if (!index.is_inobject())
break;
117 if (details.representation().IsDouble() &&
118 map->IsUnboxedDoubleField(index)) {
119 ++stats.unboxed_double_fields_count_;
124 field_stats_cache_.insert(std::make_pair(map, stats));
128 void ObjectStats::ClearObjectStats(
bool clear_last_time_stats) {
129 memset(object_counts_, 0,
sizeof(object_counts_));
130 memset(object_sizes_, 0,
sizeof(object_sizes_));
131 memset(over_allocated_, 0,
sizeof(over_allocated_));
132 memset(size_histogram_, 0,
sizeof(size_histogram_));
133 memset(over_allocated_histogram_, 0,
sizeof(over_allocated_histogram_));
134 if (clear_last_time_stats) {
135 memset(object_counts_last_time_, 0,
sizeof(object_counts_last_time_));
136 memset(object_sizes_last_time_, 0,
sizeof(object_sizes_last_time_));
138 tagged_fields_count_ = 0;
139 embedder_fields_count_ = 0;
140 unboxed_double_fields_count_ = 0;
141 raw_fields_count_ = 0;
147 V8_NOINLINE
static void PrintJSONArray(
size_t* array,
const int len) {
149 for (
int i = 0;
i < len;
i++) {
150 PrintF(
"%zu", array[
i]);
151 if (
i != (len - 1)) PrintF(
", ");
156 V8_NOINLINE
static void DumpJSONArray(std::stringstream& stream,
size_t* array,
158 stream << PrintCollection(Vector<size_t>(array, len));
161 void ObjectStats::PrintKeyAndId(
const char* key,
int gc_count) {
162 PrintF(
"\"isolate\": \"%p\", \"id\": %d, \"key\": \"%s\", ",
163 reinterpret_cast<void*>(isolate()), gc_count, key);
166 void ObjectStats::PrintInstanceTypeJSON(
const char* key,
int gc_count,
167 const char* name,
int index) {
169 PrintKeyAndId(key, gc_count);
170 PrintF(
"\"type\": \"instance_type_data\", ");
171 PrintF(
"\"instance_type\": %d, ", index);
172 PrintF(
"\"instance_type_name\": \"%s\", ", name);
173 PrintF(
"\"overall\": %zu, ", object_sizes_[index]);
174 PrintF(
"\"count\": %zu, ", object_counts_[index]);
175 PrintF(
"\"over_allocated\": %zu, ", over_allocated_[index]);
176 PrintF(
"\"histogram\": ");
177 PrintJSONArray(size_histogram_[index], kNumberOfBuckets);
179 PrintF(
"\"over_allocated_histogram\": ");
180 PrintJSONArray(over_allocated_histogram_[index], kNumberOfBuckets);
184 void ObjectStats::PrintJSON(
const char* key) {
185 double time = isolate()->time_millis_since_init();
186 int gc_count = heap()->gc_count();
190 PrintKeyAndId(key, gc_count);
191 PrintF(
"\"type\": \"gc_descriptor\", \"time\": %f }\n", time);
194 PrintKeyAndId(key, gc_count);
195 PrintF(
"\"type\": \"field_data\"");
196 PrintF(
", \"tagged_fields\": %zu", tagged_fields_count_ * kPointerSize);
197 PrintF(
", \"embedder_fields\": %zu", embedder_fields_count_ * kPointerSize);
198 PrintF(
", \"unboxed_double_fields\": %zu",
199 unboxed_double_fields_count_ * kDoubleSize);
200 PrintF(
", \"other_raw_fields\": %zu", raw_fields_count_ * kPointerSize);
204 PrintKeyAndId(key, gc_count);
205 PrintF(
"\"type\": \"bucket_sizes\", \"sizes\": [ ");
206 for (
int i = 0;
i < kNumberOfBuckets;
i++) {
207 PrintF(
"%d", 1 << (kFirstBucketShift +
i));
208 if (
i != (kNumberOfBuckets - 1)) PrintF(
", ");
212 #define INSTANCE_TYPE_WRAPPER(name) \ 213 PrintInstanceTypeJSON(key, gc_count, #name, name); 215 #define VIRTUAL_INSTANCE_TYPE_WRAPPER(name) \ 216 PrintInstanceTypeJSON(key, gc_count, #name, FIRST_VIRTUAL_TYPE + name); 218 INSTANCE_TYPE_LIST(INSTANCE_TYPE_WRAPPER)
219 VIRTUAL_INSTANCE_TYPE_LIST(VIRTUAL_INSTANCE_TYPE_WRAPPER)
221 #undef INSTANCE_TYPE_WRAPPER 222 #undef VIRTUAL_INSTANCE_TYPE_WRAPPER 225 void ObjectStats::DumpInstanceTypeData(std::stringstream& stream,
226 const char* name,
int index) {
227 stream <<
"\"" << name <<
"\":{";
228 stream <<
"\"type\":" <<
static_cast<int>(index) <<
",";
229 stream <<
"\"overall\":" << object_sizes_[index] <<
",";
230 stream <<
"\"count\":" << object_counts_[index] <<
",";
231 stream <<
"\"over_allocated\":" << over_allocated_[index] <<
",";
232 stream <<
"\"histogram\":";
233 DumpJSONArray(stream, size_histogram_[index], kNumberOfBuckets);
234 stream <<
",\"over_allocated_histogram\":";
235 DumpJSONArray(stream, over_allocated_histogram_[index], kNumberOfBuckets);
239 void ObjectStats::Dump(std::stringstream& stream) {
240 double time = isolate()->time_millis_since_init();
241 int gc_count = heap()->gc_count();
244 stream <<
"\"isolate\":\"" <<
reinterpret_cast<void*
>(isolate()) <<
"\",";
245 stream <<
"\"id\":" << gc_count <<
",";
246 stream <<
"\"time\":" << time <<
",";
249 stream <<
"\"field_data\":{";
250 stream <<
"\"tagged_fields\":" << (tagged_fields_count_ * kPointerSize);
251 stream <<
",\"embedder_fields\":" << (embedder_fields_count_ * kPointerSize);
252 stream <<
",\"unboxed_double_fields\": " 253 << (unboxed_double_fields_count_ * kDoubleSize);
254 stream <<
",\"other_raw_fields\":" << (raw_fields_count_ * kPointerSize);
257 stream <<
"\"bucket_sizes\":[";
258 for (
int i = 0;
i < kNumberOfBuckets;
i++) {
259 stream << (1 << (kFirstBucketShift +
i));
260 if (
i != (kNumberOfBuckets - 1)) stream <<
",";
263 stream <<
"\"type_data\":{";
265 #define INSTANCE_TYPE_WRAPPER(name) DumpInstanceTypeData(stream, #name, name); 267 #define VIRTUAL_INSTANCE_TYPE_WRAPPER(name) \ 268 DumpInstanceTypeData(stream, #name, FIRST_VIRTUAL_TYPE + name); 270 INSTANCE_TYPE_LIST(INSTANCE_TYPE_WRAPPER);
271 VIRTUAL_INSTANCE_TYPE_LIST(VIRTUAL_INSTANCE_TYPE_WRAPPER)
272 stream <<
"\"END\":{}}}";
274 #undef INSTANCE_TYPE_WRAPPER 275 #undef VIRTUAL_INSTANCE_TYPE_WRAPPER 278 void ObjectStats::CheckpointObjectStats() {
279 base::MutexGuard lock_guard(object_stats_mutex.Pointer());
280 MemCopy(object_counts_last_time_, object_counts_,
sizeof(object_counts_));
281 MemCopy(object_sizes_last_time_, object_sizes_,
sizeof(object_sizes_));
287 int Log2ForSize(
size_t size) {
289 return kSizetSize * 8 - 1 - base::bits::CountLeadingZeros(size);
294 int ObjectStats::HistogramIndexFromSize(
size_t size) {
295 if (size == 0)
return 0;
296 return Min(Max(Log2ForSize(size) + 1 - kFirstBucketShift, 0),
297 kLastValueBucketIndex);
300 void ObjectStats::RecordObjectStats(InstanceType type,
size_t size) {
301 DCHECK_LE(type, LAST_TYPE);
302 object_counts_[type]++;
303 object_sizes_[type] += size;
304 size_histogram_[type][HistogramIndexFromSize(size)]++;
307 void ObjectStats::RecordVirtualObjectStats(VirtualInstanceType type,
308 size_t size,
size_t over_allocated) {
309 DCHECK_LE(type, LAST_VIRTUAL_TYPE);
310 object_counts_[FIRST_VIRTUAL_TYPE + type]++;
311 object_sizes_[FIRST_VIRTUAL_TYPE + type] += size;
312 size_histogram_[FIRST_VIRTUAL_TYPE + type][HistogramIndexFromSize(size)]++;
313 over_allocated_[FIRST_VIRTUAL_TYPE + type] += over_allocated;
314 over_allocated_histogram_[FIRST_VIRTUAL_TYPE + type]
315 [HistogramIndexFromSize(size)]++;
318 Isolate* ObjectStats::isolate() {
return heap()->isolate(); }
326 static const int kNumberOfPhases = kPhase2 + 1;
330 void CollectGlobalStatistics();
332 enum class CollectFieldStats { kNo, kYes };
333 void CollectStatistics(
HeapObject* obj, Phase phase,
334 CollectFieldStats collect_field_stats);
342 Isolate* isolate() {
return heap_->isolate(); }
345 ObjectStats::VirtualInstanceType
type,
346 size_t size,
size_t over_allocated,
347 CowMode check_cow_array = kCheckCow);
348 void RecordExternalResourceStats(
Address resource,
349 ObjectStats::VirtualInstanceType
type,
353 ObjectStats::VirtualInstanceType
type);
355 void RecordHashTableVirtualObjectStats(
HeapObject* parent,
357 ObjectStats::VirtualInstanceType
type);
366 bool ShouldRecordObject(
HeapObject*
object, CowMode check_cow_array);
368 void RecordObjectStats(
HeapObject* obj, InstanceType
type,
size_t size);
372 void RecordVirtualObjectsForConstantPoolOrEmbeddedObjects(
374 ObjectStats::VirtualInstanceType
type);
378 void RecordVirtualBytecodeArrayDetails(
BytecodeArray bytecode);
379 void RecordVirtualCodeDetails(
Code code);
380 void RecordVirtualContext(
Context context);
382 void RecordVirtualFixedArrayDetails(
FixedArray array);
385 void RecordVirtualJSCollectionDetails(
JSObject*
object);
386 void RecordVirtualJSObjectDetails(
JSObject*
object);
387 void RecordVirtualMapDetails(
Map map);
388 void RecordVirtualScriptDetails(
Script* script);
391 void RecordVirtualJSFunctionDetails(
JSFunction*
function);
393 void RecordVirtualArrayBoilerplateDescription(
398 std::unordered_set<HeapObject*> virtual_objects_;
399 std::unordered_set<Address> external_resources_;
403 ObjectStatsCollectorImpl::ObjectStatsCollectorImpl(
Heap* heap,
408 heap->mark_compact_collector()->non_atomic_marking_state()),
409 field_stats_collector_(
410 &stats->tagged_fields_count_, &stats->embedder_fields_count_,
411 &stats->unboxed_double_fields_count_, &stats->raw_fields_count_) {}
413 bool ObjectStatsCollectorImpl::ShouldRecordObject(HeapObject* obj,
414 CowMode check_cow_array) {
415 if (obj->IsFixedArrayExact()) {
416 FixedArray fixed_array = FixedArray::cast(obj);
417 bool cow_check = check_cow_array == kIgnoreCow || !IsCowArray(fixed_array);
418 return CanRecordFixedArray(fixed_array) && cow_check;
420 if (obj == ReadOnlyRoots(heap_).empty_property_array())
return false;
424 void ObjectStatsCollectorImpl::RecordHashTableVirtualObjectStats(
425 HeapObject* parent, FixedArray hash_table,
426 ObjectStats::VirtualInstanceType type) {
427 CHECK(hash_table->IsHashTable());
429 RecordVirtualObjectStats(parent, hash_table, type, hash_table->Size(),
430 ObjectStats::kNoOverAllocation);
433 bool ObjectStatsCollectorImpl::RecordSimpleVirtualObjectStats(
434 HeapObject* parent, HeapObject* obj,
435 ObjectStats::VirtualInstanceType type) {
436 return RecordVirtualObjectStats(parent, obj, type, obj->Size(),
437 ObjectStats::kNoOverAllocation, kCheckCow);
440 bool ObjectStatsCollectorImpl::RecordVirtualObjectStats(
441 HeapObject* parent, HeapObject* obj, ObjectStats::VirtualInstanceType type,
442 size_t size,
size_t over_allocated, CowMode check_cow_array) {
443 if (!SameLiveness(parent, obj) || !ShouldRecordObject(obj, check_cow_array)) {
447 if (virtual_objects_.find(obj) == virtual_objects_.end()) {
448 virtual_objects_.insert(obj);
449 stats_->RecordVirtualObjectStats(type, size, over_allocated);
455 void ObjectStatsCollectorImpl::RecordExternalResourceStats(
456 Address resource, ObjectStats::VirtualInstanceType type,
size_t size) {
457 if (external_resources_.find(resource) == external_resources_.end()) {
458 external_resources_.insert(resource);
459 stats_->RecordVirtualObjectStats(type, size, 0);
463 void ObjectStatsCollectorImpl::RecordVirtualAllocationSiteDetails(
464 AllocationSite* site) {
465 if (!site->PointsToLiteral())
return;
466 JSObject* boilerplate = site->boilerplate();
467 if (boilerplate->IsJSArray()) {
468 RecordSimpleVirtualObjectStats(site, boilerplate,
469 ObjectStats::JS_ARRAY_BOILERPLATE_TYPE);
472 RecordVirtualObjectStats(
473 site, boilerplate, ObjectStats::JS_OBJECT_BOILERPLATE_TYPE,
474 boilerplate->Size(), ObjectStats::kNoOverAllocation);
475 if (boilerplate->HasFastProperties()) {
478 PropertyArray properties = boilerplate->property_array();
479 RecordSimpleVirtualObjectStats(
480 site, properties, ObjectStats::BOILERPLATE_PROPERTY_ARRAY_TYPE);
482 NameDictionary properties = boilerplate->property_dictionary();
483 RecordSimpleVirtualObjectStats(
484 site, properties, ObjectStats::BOILERPLATE_PROPERTY_DICTIONARY_TYPE);
487 FixedArrayBase elements = boilerplate->elements();
488 RecordSimpleVirtualObjectStats(site, elements,
489 ObjectStats::BOILERPLATE_ELEMENTS_TYPE);
492 void ObjectStatsCollectorImpl::RecordVirtualFunctionTemplateInfoDetails(
493 FunctionTemplateInfo* fti) {
496 if (!fti->call_code()->IsUndefined(isolate())) {
497 RecordSimpleVirtualObjectStats(
498 fti, CallHandlerInfo::cast(fti->call_code()),
499 ObjectStats::FUNCTION_TEMPLATE_INFO_ENTRIES_TYPE);
501 if (!fti->GetInstanceCallHandler()->IsUndefined(isolate())) {
502 RecordSimpleVirtualObjectStats(
503 fti, CallHandlerInfo::cast(fti->GetInstanceCallHandler()),
504 ObjectStats::FUNCTION_TEMPLATE_INFO_ENTRIES_TYPE);
508 void ObjectStatsCollectorImpl::RecordVirtualJSGlobalObjectDetails(
509 JSGlobalObject*
object) {
511 GlobalDictionary properties =
object->global_dictionary();
512 RecordHashTableVirtualObjectStats(
object, properties,
513 ObjectStats::GLOBAL_PROPERTIES_TYPE);
515 FixedArrayBase elements =
object->elements();
516 RecordSimpleVirtualObjectStats(
object, elements,
517 ObjectStats::GLOBAL_ELEMENTS_TYPE);
520 void ObjectStatsCollectorImpl::RecordVirtualJSCollectionDetails(
522 if (object->IsJSMap()) {
523 RecordSimpleVirtualObjectStats(
524 object, FixedArray::cast(JSMap::cast(
object)->table()),
525 ObjectStats::JS_COLLECTION_TABLE_TYPE);
527 if (object->IsJSSet()) {
528 RecordSimpleVirtualObjectStats(
529 object, FixedArray::cast(JSSet::cast(
object)->table()),
530 ObjectStats::JS_COLLECTION_TABLE_TYPE);
534 void ObjectStatsCollectorImpl::RecordVirtualJSObjectDetails(JSObject*
object) {
536 if (object->IsJSGlobalObject())
return;
539 if (object->HasFastProperties()) {
540 PropertyArray properties =
object->property_array();
541 CHECK_EQ(PROPERTY_ARRAY_TYPE, properties->map()->instance_type());
543 NameDictionary properties =
object->property_dictionary();
544 RecordHashTableVirtualObjectStats(
545 object, properties, ObjectStats::OBJECT_PROPERTY_DICTIONARY_TYPE);
548 FixedArrayBase elements =
object->elements();
549 RecordSimpleVirtualObjectStats(
object, elements, ObjectStats::ELEMENTS_TYPE);
552 static ObjectStats::VirtualInstanceType GetFeedbackSlotType(
553 MaybeObject maybe_obj, FeedbackSlotKind kind, Isolate* isolate) {
554 if (maybe_obj->IsCleared())
555 return ObjectStats::FEEDBACK_VECTOR_SLOT_OTHER_TYPE;
556 Object* obj = maybe_obj->GetHeapObjectOrSmi();
558 case FeedbackSlotKind::kCall:
559 if (obj == *isolate->factory()->uninitialized_symbol() ||
560 obj == *isolate->factory()->premonomorphic_symbol()) {
561 return ObjectStats::FEEDBACK_VECTOR_SLOT_CALL_UNUSED_TYPE;
563 return ObjectStats::FEEDBACK_VECTOR_SLOT_CALL_TYPE;
565 case FeedbackSlotKind::kLoadProperty:
566 case FeedbackSlotKind::kLoadGlobalInsideTypeof:
567 case FeedbackSlotKind::kLoadGlobalNotInsideTypeof:
568 case FeedbackSlotKind::kLoadKeyed:
569 if (obj == *isolate->factory()->uninitialized_symbol() ||
570 obj == *isolate->factory()->premonomorphic_symbol()) {
571 return ObjectStats::FEEDBACK_VECTOR_SLOT_LOAD_UNUSED_TYPE;
573 return ObjectStats::FEEDBACK_VECTOR_SLOT_LOAD_TYPE;
575 case FeedbackSlotKind::kStoreNamedSloppy:
576 case FeedbackSlotKind::kStoreNamedStrict:
577 case FeedbackSlotKind::kStoreOwnNamed:
578 case FeedbackSlotKind::kStoreGlobalSloppy:
579 case FeedbackSlotKind::kStoreGlobalStrict:
580 case FeedbackSlotKind::kStoreKeyedSloppy:
581 case FeedbackSlotKind::kStoreKeyedStrict:
582 if (obj == *isolate->factory()->uninitialized_symbol() ||
583 obj == *isolate->factory()->premonomorphic_symbol()) {
584 return ObjectStats::FEEDBACK_VECTOR_SLOT_STORE_UNUSED_TYPE;
586 return ObjectStats::FEEDBACK_VECTOR_SLOT_STORE_TYPE;
588 case FeedbackSlotKind::kBinaryOp:
589 case FeedbackSlotKind::kCompareOp:
590 return ObjectStats::FEEDBACK_VECTOR_SLOT_ENUM_TYPE;
593 return ObjectStats::FEEDBACK_VECTOR_SLOT_OTHER_TYPE;
597 void ObjectStatsCollectorImpl::RecordVirtualFeedbackVectorDetails(
598 FeedbackVector* vector) {
599 if (virtual_objects_.find(vector) == virtual_objects_.end()) {
602 virtual_objects_.insert(vector);
604 size_t calculated_size = 0;
607 size_t header_size = vector->slots_start().address() - vector->address();
608 stats_->RecordVirtualObjectStats(ObjectStats::FEEDBACK_VECTOR_HEADER_TYPE,
610 ObjectStats::kNoOverAllocation);
611 calculated_size += header_size;
614 if (!vector->shared_function_info()->HasFeedbackMetadata())
return;
616 FeedbackMetadataIterator it(vector->metadata());
617 while (it.HasNext()) {
618 FeedbackSlot slot = it.Next();
620 size_t slot_size = it.entry_size() * kPointerSize;
621 stats_->RecordVirtualObjectStats(
622 GetFeedbackSlotType(vector->Get(slot), it.kind(), heap_->isolate()),
623 slot_size, ObjectStats::kNoOverAllocation);
624 calculated_size += slot_size;
627 for (
int i = 0;
i < it.entry_size();
i++) {
628 MaybeObject raw_object = vector->get(slot.ToInt() +
i);
630 if (raw_object->GetHeapObject(&
object)) {
631 if (object->IsCell() ||
object->IsWeakFixedArray()) {
632 RecordSimpleVirtualObjectStats(
633 vector,
object, ObjectStats::FEEDBACK_VECTOR_ENTRY_TYPE);
639 CHECK_EQ(calculated_size, vector->Size());
643 void ObjectStatsCollectorImpl::RecordVirtualFixedArrayDetails(
645 if (IsCowArray(array)) {
646 RecordVirtualObjectStats(
nullptr, array, ObjectStats::COW_ARRAY_TYPE,
647 array->Size(), ObjectStats::kNoOverAllocation,
652 void ObjectStatsCollectorImpl::CollectStatistics(
653 HeapObject* obj, Phase phase, CollectFieldStats collect_field_stats) {
654 Map map = obj->map();
657 if (obj->IsFeedbackVector()) {
658 RecordVirtualFeedbackVectorDetails(FeedbackVector::cast(obj));
659 }
else if (obj->IsMap()) {
660 RecordVirtualMapDetails(Map::cast(obj));
661 }
else if (obj->IsBytecodeArray()) {
662 RecordVirtualBytecodeArrayDetails(BytecodeArray::cast(obj));
663 }
else if (obj->IsCode()) {
664 RecordVirtualCodeDetails(Code::cast(obj));
665 }
else if (obj->IsFunctionTemplateInfo()) {
666 RecordVirtualFunctionTemplateInfoDetails(
667 FunctionTemplateInfo::cast(obj));
668 }
else if (obj->IsJSFunction()) {
669 RecordVirtualJSFunctionDetails(JSFunction::cast(obj));
670 }
else if (obj->IsJSGlobalObject()) {
671 RecordVirtualJSGlobalObjectDetails(JSGlobalObject::cast(obj));
672 }
else if (obj->IsJSObject()) {
675 RecordVirtualJSObjectDetails(JSObject::cast(obj));
676 }
else if (obj->IsJSCollection()) {
677 RecordVirtualJSCollectionDetails(JSObject::cast(obj));
678 }
else if (obj->IsSharedFunctionInfo()) {
679 RecordVirtualSharedFunctionInfoDetails(SharedFunctionInfo::cast(obj));
680 }
else if (obj->IsContext()) {
681 RecordVirtualContext(Context::cast(obj));
682 }
else if (obj->IsScript()) {
683 RecordVirtualScriptDetails(Script::cast(obj));
684 }
else if (obj->IsArrayBoilerplateDescription()) {
685 RecordVirtualArrayBoilerplateDescription(
686 ArrayBoilerplateDescription::cast(obj));
687 }
else if (obj->IsFixedArrayExact()) {
689 RecordVirtualFixedArrayDetails(FixedArray::cast(obj));
693 if (obj->IsExternalString()) {
696 RecordVirtualExternalStringDetails(ExternalString::cast(obj));
698 RecordObjectStats(obj, map->instance_type(), obj->Size());
699 if (collect_field_stats == CollectFieldStats::kYes) {
700 field_stats_collector_.RecordStats(obj);
706 void ObjectStatsCollectorImpl::CollectGlobalStatistics() {
708 Object* list = heap_->allocation_sites_list();
709 while (list->IsAllocationSite()) {
710 AllocationSite* site = AllocationSite::cast(list);
711 RecordVirtualAllocationSiteDetails(site);
712 list = site->weak_next();
716 RecordSimpleVirtualObjectStats(
nullptr, heap_->serialized_objects(),
717 ObjectStats::SERIALIZED_OBJECTS_TYPE);
718 RecordSimpleVirtualObjectStats(
nullptr, heap_->number_string_cache(),
719 ObjectStats::NUMBER_STRING_CACHE_TYPE);
720 RecordSimpleVirtualObjectStats(
721 nullptr, heap_->single_character_string_cache(),
722 ObjectStats::SINGLE_CHARACTER_STRING_CACHE_TYPE);
723 RecordSimpleVirtualObjectStats(
nullptr, heap_->string_split_cache(),
724 ObjectStats::STRING_SPLIT_CACHE_TYPE);
725 RecordSimpleVirtualObjectStats(
nullptr, heap_->regexp_multiple_cache(),
726 ObjectStats::REGEXP_MULTIPLE_CACHE_TYPE);
727 RecordSimpleVirtualObjectStats(
nullptr, heap_->retained_maps(),
728 ObjectStats::RETAINED_MAPS_TYPE);
731 RecordSimpleVirtualObjectStats(
732 nullptr, WeakArrayList::cast(heap_->noscript_shared_function_infos()),
733 ObjectStats::NOSCRIPT_SHARED_FUNCTION_INFOS_TYPE);
734 RecordSimpleVirtualObjectStats(
nullptr,
735 WeakArrayList::cast(heap_->script_list()),
736 ObjectStats::SCRIPT_LIST_TYPE);
739 RecordHashTableVirtualObjectStats(
nullptr, heap_->code_stubs(),
740 ObjectStats::CODE_STUBS_TABLE_TYPE);
743 void ObjectStatsCollectorImpl::RecordObjectStats(HeapObject* obj,
746 if (virtual_objects_.find(obj) == virtual_objects_.end()) {
747 stats_->RecordObjectStats(type, size);
751 bool ObjectStatsCollectorImpl::CanRecordFixedArray(FixedArrayBase array) {
752 ReadOnlyRoots roots(heap_);
753 return array != roots.empty_fixed_array() &&
754 array != roots.empty_sloppy_arguments_elements() &&
755 array != roots.empty_slow_element_dictionary() &&
756 array != roots.empty_property_dictionary();
759 bool ObjectStatsCollectorImpl::IsCowArray(FixedArrayBase array) {
760 return array->map() == ReadOnlyRoots(heap_).fixed_cow_array_map();
763 bool ObjectStatsCollectorImpl::SameLiveness(HeapObject* obj1,
765 return obj1 ==
nullptr || obj2 ==
nullptr ||
766 marking_state_->Color(obj1) == marking_state_->Color(obj2);
769 void ObjectStatsCollectorImpl::RecordVirtualMapDetails(Map map) {
772 DescriptorArray* array = map->instance_descriptors();
773 if (map->owns_descriptors() &&
774 array != ReadOnlyRoots(heap_).empty_descriptor_array()) {
776 EnumCache* enum_cache = array->enum_cache();
777 RecordSimpleVirtualObjectStats(array, enum_cache->keys(),
778 ObjectStats::ENUM_CACHE_TYPE);
779 RecordSimpleVirtualObjectStats(array, enum_cache->indices(),
780 ObjectStats::ENUM_INDICES_CACHE_TYPE);
783 if (map->is_prototype_map()) {
784 if (map->prototype_info()->IsPrototypeInfo()) {
785 PrototypeInfo* info = PrototypeInfo::cast(map->prototype_info());
786 Object* users = info->prototype_users();
787 if (users->IsWeakFixedArray()) {
788 RecordSimpleVirtualObjectStats(map, WeakArrayList::cast(users),
789 ObjectStats::PROTOTYPE_USERS_TYPE);
795 void ObjectStatsCollectorImpl::RecordVirtualScriptDetails(Script* script) {
796 RecordSimpleVirtualObjectStats(
797 script, script->shared_function_infos(),
798 ObjectStats::SCRIPT_SHARED_FUNCTION_INFOS_TYPE);
801 Object* raw_source = script->source();
802 if (raw_source->IsExternalString()) {
806 ExternalString
string = ExternalString::cast(raw_source);
807 Address resource =
string->resource_as_address();
808 size_t off_heap_size =
string->ExternalPayloadSize();
809 RecordExternalResourceStats(
811 string->IsOneByteRepresentation()
812 ? ObjectStats::SCRIPT_SOURCE_EXTERNAL_ONE_BYTE_TYPE
813 : ObjectStats::SCRIPT_SOURCE_EXTERNAL_TWO_BYTE_TYPE,
815 }
else if (raw_source->IsString()) {
816 String source = String::cast(raw_source);
817 RecordSimpleVirtualObjectStats(
819 source->IsOneByteRepresentation()
820 ? ObjectStats::SCRIPT_SOURCE_NON_EXTERNAL_ONE_BYTE_TYPE
821 : ObjectStats::SCRIPT_SOURCE_NON_EXTERNAL_TWO_BYTE_TYPE);
825 void ObjectStatsCollectorImpl::RecordVirtualExternalStringDetails(
826 ExternalString
string) {
829 Address resource =
string->resource_as_address();
830 size_t off_heap_size =
string->ExternalPayloadSize();
831 RecordExternalResourceStats(
833 string->IsOneByteRepresentation()
834 ? ObjectStats::STRING_EXTERNAL_RESOURCE_ONE_BYTE_TYPE
835 : ObjectStats::STRING_EXTERNAL_RESOURCE_TWO_BYTE_TYPE,
839 void ObjectStatsCollectorImpl::RecordVirtualSharedFunctionInfoDetails(
840 SharedFunctionInfo* info) {
842 if (!info->is_compiled()) {
843 RecordSimpleVirtualObjectStats(
844 nullptr, info, ObjectStats::UNCOMPILED_SHARED_FUNCTION_INFO_TYPE);
848 void ObjectStatsCollectorImpl::RecordVirtualJSFunctionDetails(
849 JSFunction*
function) {
851 if (!function->is_compiled()) {
852 RecordSimpleVirtualObjectStats(
nullptr,
function,
853 ObjectStats::UNCOMPILED_JS_FUNCTION_TYPE);
856 void ObjectStatsCollectorImpl::RecordVirtualArrayBoilerplateDescription(
857 ArrayBoilerplateDescription* description) {
858 RecordVirtualObjectsForConstantPoolOrEmbeddedObjects(
859 description, description->constant_elements(),
860 ObjectStats::ARRAY_BOILERPLATE_DESCRIPTION_ELEMENTS_TYPE);
863 void ObjectStatsCollectorImpl::
864 RecordVirtualObjectsForConstantPoolOrEmbeddedObjects(
865 HeapObject* parent, HeapObject*
object,
866 ObjectStats::VirtualInstanceType type) {
867 if (!RecordSimpleVirtualObjectStats(parent,
object, type))
return;
868 if (object->IsFixedArrayExact()) {
869 FixedArray array = FixedArray::cast(
object);
870 for (
int i = 0;
i < array->length();
i++) {
871 Object* entry = array->get(
i);
872 if (!entry->IsHeapObject())
continue;
873 RecordVirtualObjectsForConstantPoolOrEmbeddedObjects(
874 array, HeapObject::cast(entry), type);
879 void ObjectStatsCollectorImpl::RecordVirtualBytecodeArrayDetails(
880 BytecodeArray bytecode) {
881 RecordSimpleVirtualObjectStats(
882 bytecode, bytecode->constant_pool(),
883 ObjectStats::BYTECODE_ARRAY_CONSTANT_POOL_TYPE);
886 FixedArray constant_pool = FixedArray::cast(bytecode->constant_pool());
887 for (
int i = 0;
i < constant_pool->length();
i++) {
888 Object* entry = constant_pool->get(
i);
889 if (entry->IsFixedArrayExact()) {
890 RecordVirtualObjectsForConstantPoolOrEmbeddedObjects(
891 constant_pool, HeapObject::cast(entry),
892 ObjectStats::EMBEDDED_OBJECT_TYPE);
895 RecordSimpleVirtualObjectStats(
896 bytecode, bytecode->handler_table(),
897 ObjectStats::BYTECODE_ARRAY_HANDLER_TABLE_TYPE);
898 RecordSimpleVirtualObjectStats(bytecode, bytecode->SourcePositionTable(),
899 ObjectStats::SOURCE_POSITION_TABLE_TYPE);
904 ObjectStats::VirtualInstanceType CodeKindToVirtualInstanceType(
907 #define CODE_KIND_CASE(type) \ 909 return ObjectStats::type; 910 CODE_KIND_LIST(CODE_KIND_CASE)
911 #undef CODE_KIND_CASE 920 void ObjectStatsCollectorImpl::RecordVirtualCodeDetails(Code code) {
921 RecordSimpleVirtualObjectStats(
nullptr, code,
922 CodeKindToVirtualInstanceType(code->kind()));
923 RecordSimpleVirtualObjectStats(code, code->deoptimization_data(),
924 ObjectStats::DEOPTIMIZATION_DATA_TYPE);
925 RecordSimpleVirtualObjectStats(code, code->relocation_info(),
926 ObjectStats::RELOC_INFO_TYPE);
927 Object* source_position_table = code->source_position_table();
928 if (source_position_table->IsSourcePositionTableWithFrameCache()) {
929 RecordSimpleVirtualObjectStats(
931 SourcePositionTableWithFrameCache::cast(source_position_table)
932 ->source_position_table(),
933 ObjectStats::SOURCE_POSITION_TABLE_TYPE);
934 }
else if (source_position_table->IsHeapObject()) {
935 RecordSimpleVirtualObjectStats(code,
936 HeapObject::cast(source_position_table),
937 ObjectStats::SOURCE_POSITION_TABLE_TYPE);
939 if (code->kind() == Code::Kind::OPTIMIZED_FUNCTION) {
940 DeoptimizationData input_data =
941 DeoptimizationData::cast(code->deoptimization_data());
942 if (input_data->length() > 0) {
943 RecordSimpleVirtualObjectStats(code->deoptimization_data(),
944 input_data->LiteralArray(),
945 ObjectStats::OPTIMIZED_CODE_LITERALS_TYPE);
948 int const mode_mask = RelocInfo::ModeMask(RelocInfo::EMBEDDED_OBJECT);
949 for (RelocIterator it(code, mode_mask); !it.done(); it.next()) {
950 RelocInfo::Mode mode = it.rinfo()->rmode();
951 if (mode == RelocInfo::EMBEDDED_OBJECT) {
952 Object* target = it.rinfo()->target_object();
953 if (target->IsFixedArrayExact()) {
954 RecordVirtualObjectsForConstantPoolOrEmbeddedObjects(
955 code, HeapObject::cast(target), ObjectStats::EMBEDDED_OBJECT_TYPE);
961 void ObjectStatsCollectorImpl::RecordVirtualContext(Context context) {
962 if (context->IsNativeContext()) {
963 RecordObjectStats(context, NATIVE_CONTEXT_TYPE, context->Size());
964 }
else if (context->IsFunctionContext()) {
965 RecordObjectStats(context, FUNCTION_CONTEXT_TYPE, context->Size());
967 RecordSimpleVirtualObjectStats(
nullptr, context,
968 ObjectStats::OTHER_CONTEXT_TYPE);
976 ObjectStatsCollectorImpl::Phase phase)
977 : live_collector_(live_collector),
978 dead_collector_(dead_collector),
980 heap->mark_compact_collector()->non_atomic_marking_state()),
984 if (marking_state_->IsBlack(obj)) {
985 live_collector_->CollectStatistics(
986 obj, phase_, ObjectStatsCollectorImpl::CollectFieldStats::kYes);
988 DCHECK(!marking_state_->IsGrey(obj));
989 dead_collector_->CollectStatistics(
990 obj, phase_, ObjectStatsCollectorImpl::CollectFieldStats::kNo);
999 ObjectStatsCollectorImpl::Phase phase_;
1007 while (space_it.has_next()) {
1008 std::unique_ptr<ObjectIterator> it(space_it.next()->GetObjectIterator());
1010 while ((obj = obj_it->Next()) !=
nullptr) {
1011 visitor->Visit(obj, obj->Size());
1018 void ObjectStatsCollector::Collect() {
1019 ObjectStatsCollectorImpl live_collector(heap_, live_);
1020 ObjectStatsCollectorImpl dead_collector(heap_, dead_);
1021 live_collector.CollectGlobalStatistics();
1022 for (
int i = 0;
i < ObjectStatsCollectorImpl::kNumberOfPhases;
i++) {
1023 ObjectStatsVisitor visitor(heap_, &live_collector, &dead_collector,
1024 static_cast<ObjectStatsCollectorImpl::Phase>(
i));
1025 IterateHeap(heap_, &visitor);