blob: 09fd5cdacb78b68a4aef6fec75ab98f271bf580c [file] [log] [blame]
// Copyright (c) 2013, 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.
#include "vm/heap_histogram.h"
#include "platform/assert.h"
#include "vm/flags.h"
#include "vm/object.h"
namespace dart {
DEFINE_FLAG(bool, print_object_histogram, false,
"Print average object histogram at isolate shutdown");
class ObjectHistogramVisitor : public ObjectVisitor {
public:
explicit ObjectHistogramVisitor(Isolate* isolate) : ObjectVisitor(isolate) { }
virtual void VisitObject(RawObject* obj) {
isolate()->object_histogram()->Add(obj);
}
private:
DISALLOW_COPY_AND_ASSIGN(ObjectHistogramVisitor);
};
void ObjectHistogram::Collect() {
major_gc_count_++;
ObjectHistogramVisitor object_visitor(isolate_);
isolate_->heap()->IterateObjects(&object_visitor);
}
ObjectHistogram::ObjectHistogram(Isolate* isolate) {
isolate_ = isolate;
major_gc_count_ = 0;
table_length_ = 512;
table_ = reinterpret_cast<Element*>(
calloc(table_length_, sizeof(Element))); // NOLINT
for (int index = 0; index < table_length_; index++) {
table_[index].class_id_ = index;
}
}
ObjectHistogram::~ObjectHistogram() {
free(table_);
}
void ObjectHistogram::RegisterClass(const Class& cls) {
int class_id = cls.id();
if (class_id < table_length_) return;
// Resize the table.
int new_table_length = table_length_ * 2;
Element* new_table = reinterpret_cast<Element*>(
realloc(table_, new_table_length * sizeof(Element))); // NOLINT
for (int i = table_length_; i < new_table_length; i++) {
new_table[i].class_id_ = i;
new_table[i].count_ = 0;
new_table[i].size_ = 0;
}
table_ = new_table;
table_length_ = new_table_length;
ASSERT(class_id < table_length_);
}
void ObjectHistogram::Add(RawObject* obj) {
intptr_t class_id = obj->GetClassId();
if (class_id == kFreeListElement) return;
ASSERT(class_id < table_length_);
table_[class_id].Add(obj->Size());
}
int ObjectHistogram::compare(const Element** a, const Element** b) {
return (*b)->size_ - (*a)->size_;
}
void ObjectHistogram::Print() {
OS::Print("Printing Object Histogram\n");
OS::Print("____bytes___count_description____________\n");
// First count the number of non empty entries.
int length = 0;
for (int index = 0; index < table_length_; index++) {
if (table_[index].count_ > 0) length++;
}
// Then add them to a new array and sort.
Element** array = reinterpret_cast<Element**>(
calloc(length, sizeof(Element*))); // NOLINT
int pos = 0;
for (int index = 0; index < table_length_; index++) {
if (table_[index].count_ > 0) array[pos++] = &table_[index];
}
typedef int (*CmpFunc)(const void*, const void*);
qsort(array, length, sizeof(Element*), // NOLINT
reinterpret_cast<CmpFunc>(compare));
// Finally print the sorted array.
Class& cls = Class::Handle();
String& str = String::Handle();
Library& lib = Library::Handle();
for (pos = 0; pos < length; pos++) {
Element* e = array[pos];
if (e->count_ > 0) {
cls = isolate_->class_table()->At(e->class_id_);
str = cls.Name();
lib = cls.library();
OS::Print("%9"Pd" %7"Pd" ",
e->size_ / major_gc_count_,
e->count_ / major_gc_count_);
if (e->class_id_ < kInstanceCid) {
OS::Print("`%s`", str.ToCString()); // VM names.
} else {
OS::Print("%s", str.ToCString());
}
if (lib.IsNull()) {
OS::Print("\n");
} else {
str = lib.url();
OS::Print(", library \'%s\'\n", str.ToCString());
}
}
}
// Deallocate the array for sorting.
free(array);
}
} // namespace dart