Fix HeapSnapshotsDiff test, diff implementation, and a bug introduced

during snapshot size optimization.

Sorry, now I figured out that the diff implementation itself was also
incorrect.  Reachable nodes must be filtered from the beginning,
otherwise, an object that is already disconnected, but not discarded
yet, will not appear as a deleted (thankfully, this bug for some
reason had appeared on the x64 port.)

BUG=868
TEST=HeapSnapshotRootPreservedAfterSorting

Review URL: http://codereview.chromium.org/3531005

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@5570 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent 7228d867
...@@ -952,7 +952,7 @@ void HeapEntry::PaintAllReachable() { ...@@ -952,7 +952,7 @@ void HeapEntry::PaintAllReachable() {
void HeapEntry::Print(int max_depth, int indent) { void HeapEntry::Print(int max_depth, int indent) {
OS::Print("%6d %6d %6d [%ld] ", OS::Print("%6d %6d %6d [%llu] ",
self_size(), ReachableSize(), RetainedSize(), id_); self_size(), ReachableSize(), RetainedSize(), id_);
if (type() != kString) { if (type() != kString) {
OS::Print("%s %.40s\n", TypeAsString(), name_); OS::Print("%s %.40s\n", TypeAsString(), name_);
...@@ -1237,7 +1237,7 @@ HeapSnapshot::HeapSnapshot(HeapSnapshotsCollection* collection, ...@@ -1237,7 +1237,7 @@ HeapSnapshot::HeapSnapshot(HeapSnapshotsCollection* collection,
type_(type), type_(type),
title_(title), title_(title),
uid_(uid), uid_(uid),
root_entry_index_(-1), root_entry_(NULL),
raw_entries_(NULL), raw_entries_(NULL),
entries_sorted_(false) { entries_sorted_(false) {
STATIC_ASSERT( STATIC_ASSERT(
...@@ -1276,11 +1276,11 @@ HeapEntry* HeapSnapshot::AddEntry(HeapObject* object, ...@@ -1276,11 +1276,11 @@ HeapEntry* HeapSnapshot::AddEntry(HeapObject* object,
int children_count, int children_count,
int retainers_count) { int retainers_count) {
if (object == kInternalRootObject) { if (object == kInternalRootObject) {
ASSERT(root_entry_index_ == -1); ASSERT(root_entry_ == NULL);
root_entry_index_ = entries_.length();
ASSERT(retainers_count == 0); ASSERT(retainers_count == 0);
return AddEntry( root_entry_ = AddEntry(
HeapEntry::kInternal, "", 0, 0, children_count, retainers_count); HeapEntry::kInternal, "", 0, 0, children_count, retainers_count);
return root_entry_;
} else if (object->IsJSFunction()) { } else if (object->IsJSFunction()) {
JSFunction* func = JSFunction::cast(object); JSFunction* func = JSFunction::cast(object);
SharedFunctionInfo* shared = func->shared(); SharedFunctionInfo* shared = func->shared();
...@@ -2095,6 +2095,11 @@ HeapSnapshotsComparator::~HeapSnapshotsComparator() { ...@@ -2095,6 +2095,11 @@ HeapSnapshotsComparator::~HeapSnapshotsComparator() {
HeapSnapshotsDiff* HeapSnapshotsComparator::Compare(HeapSnapshot* snapshot1, HeapSnapshotsDiff* HeapSnapshotsComparator::Compare(HeapSnapshot* snapshot1,
HeapSnapshot* snapshot2) { HeapSnapshot* snapshot2) {
snapshot1->ClearPaint();
snapshot1->root()->PaintAllReachable();
snapshot2->ClearPaint();
snapshot2->root()->PaintAllReachable();
List<HeapEntry*>* entries1 = snapshot1->GetSortedEntriesList(); List<HeapEntry*>* entries1 = snapshot1->GetSortedEntriesList();
List<HeapEntry*>* entries2 = snapshot2->GetSortedEntriesList(); List<HeapEntry*>* entries2 = snapshot2->GetSortedEntriesList();
int i = 0, j = 0; int i = 0, j = 0;
...@@ -2103,8 +2108,14 @@ HeapSnapshotsDiff* HeapSnapshotsComparator::Compare(HeapSnapshot* snapshot1, ...@@ -2103,8 +2108,14 @@ HeapSnapshotsDiff* HeapSnapshotsComparator::Compare(HeapSnapshot* snapshot1,
uint64_t id1 = entries1->at(i)->id(); uint64_t id1 = entries1->at(i)->id();
uint64_t id2 = entries2->at(j)->id(); uint64_t id2 = entries2->at(j)->id();
if (id1 == id2) { if (id1 == id2) {
i++; HeapEntry* entry1 = entries1->at(i++);
j++; HeapEntry* entry2 = entries2->at(j++);
if (entry1->painted_reachable() != entry2->painted_reachable()) {
if (entry1->painted_reachable())
deleted_entries.Add(entry1);
else
added_entries.Add(entry2);
}
} else if (id1 < id2) { } else if (id1 < id2) {
HeapEntry* entry = entries1->at(i++); HeapEntry* entry = entries1->at(i++);
deleted_entries.Add(entry); deleted_entries.Add(entry);
...@@ -2122,35 +2133,17 @@ HeapSnapshotsDiff* HeapSnapshotsComparator::Compare(HeapSnapshot* snapshot1, ...@@ -2122,35 +2133,17 @@ HeapSnapshotsDiff* HeapSnapshotsComparator::Compare(HeapSnapshot* snapshot1,
added_entries.Add(entry); added_entries.Add(entry);
} }
snapshot1->ClearPaint();
snapshot1->root()->PaintAllReachable();
snapshot2->ClearPaint();
snapshot2->root()->PaintAllReachable();
int reachable_deleted_entries = 0, reachable_added_entries = 0;
for (int i = 0; i < deleted_entries.length(); ++i) {
HeapEntry* entry = deleted_entries[i];
if (entry->painted_reachable()) ++reachable_deleted_entries;
}
for (int i = 0; i < added_entries.length(); ++i) {
HeapEntry* entry = added_entries[i];
if (entry->painted_reachable()) ++reachable_added_entries;
}
HeapSnapshotsDiff* diff = new HeapSnapshotsDiff(snapshot1, snapshot2); HeapSnapshotsDiff* diff = new HeapSnapshotsDiff(snapshot1, snapshot2);
diffs_.Add(diff); diffs_.Add(diff);
diff->CreateRoots(reachable_added_entries, reachable_deleted_entries); diff->CreateRoots(added_entries.length(), deleted_entries.length());
int del_child_index = 0, deleted_entry_index = 1;
for (int i = 0; i < deleted_entries.length(); ++i) { for (int i = 0; i < deleted_entries.length(); ++i) {
HeapEntry* entry = deleted_entries[i]; HeapEntry* entry = deleted_entries[i];
if (entry->painted_reachable()) diff->AddDeletedEntry(i, i + 1, entry);
diff->AddDeletedEntry(del_child_index++, deleted_entry_index++, entry);
} }
int add_child_index = 0, added_entry_index = 1;
for (int i = 0; i < added_entries.length(); ++i) { for (int i = 0; i < added_entries.length(); ++i) {
HeapEntry* entry = added_entries[i]; HeapEntry* entry = added_entries[i];
if (entry->painted_reachable()) diff->AddAddedEntry(i, i + 1, entry);
diff->AddAddedEntry(add_child_index++, added_entry_index++, entry);
} }
return diff; return diff;
} }
......
...@@ -662,7 +662,7 @@ class HeapSnapshot { ...@@ -662,7 +662,7 @@ class HeapSnapshot {
Type type() { return type_; } Type type() { return type_; }
const char* title() { return title_; } const char* title() { return title_; }
unsigned uid() { return uid_; } unsigned uid() { return uid_; }
HeapEntry* root() { return entries_[root_entry_index_]; } HeapEntry* root() { return root_entry_; }
void AllocateEntries( void AllocateEntries(
int entries_count, int children_count, int retainers_count); int entries_count, int children_count, int retainers_count);
...@@ -704,7 +704,7 @@ class HeapSnapshot { ...@@ -704,7 +704,7 @@ class HeapSnapshot {
Type type_; Type type_;
const char* title_; const char* title_;
unsigned uid_; unsigned uid_;
int root_entry_index_; HeapEntry* root_entry_;
char* raw_entries_; char* raw_entries_;
List<HeapEntry*> entries_; List<HeapEntry*> entries_;
bool entries_sorted_; bool entries_sorted_;
......
...@@ -35,11 +35,6 @@ test-debug/DebuggerAgent: PASS, (PASS || FAIL) if $system == linux ...@@ -35,11 +35,6 @@ test-debug/DebuggerAgent: PASS, (PASS || FAIL) if $system == linux
# BUG(382): Weird test. Can't guarantee that it never times out. # BUG(382): Weird test. Can't guarantee that it never times out.
test-api/ApplyInterruption: PASS || TIMEOUT test-api/ApplyInterruption: PASS || TIMEOUT
# Bug (484): This test which we thought was originally corrected in r5236
# is reappering. Disabled until bug in test is fixed. This only fails
# when snapshot is on, so I am marking it PASS || FAIL
test-heap-profiler/HeapSnapshotsDiff: PASS || FAIL
# These tests always fail. They are here to test test.py. If # These tests always fail. They are here to test test.py. If
# they don't fail then test.py has failed. # they don't fail then test.py has failed.
test-serialize/TestThatAlwaysFails: FAIL test-serialize/TestThatAlwaysFails: FAIL
......
...@@ -787,6 +787,7 @@ TEST(HeapSnapshotsDiff) { ...@@ -787,6 +787,7 @@ TEST(HeapSnapshotsDiff) {
CompileAndRunScript( CompileAndRunScript(
"function A() {}\n" "function A() {}\n"
"function B(x) { this.x = x; }\n" "function B(x) { this.x = x; }\n"
"function A2(a) { for (var i = 0; i < a; ++i) this[i] = i; }\n"
"var a = new A();\n" "var a = new A();\n"
"var b = new B(a);"); "var b = new B(a);");
const v8::HeapSnapshot* snapshot1 = const v8::HeapSnapshot* snapshot1 =
...@@ -795,7 +796,7 @@ TEST(HeapSnapshotsDiff) { ...@@ -795,7 +796,7 @@ TEST(HeapSnapshotsDiff) {
CompileAndRunScript( CompileAndRunScript(
"delete a;\n" "delete a;\n"
"b.x = null;\n" "b.x = null;\n"
"var a = new A();\n" "var a = new A2(20);\n"
"var b2 = new B(a);"); "var b2 = new B(a);");
const v8::HeapSnapshot* snapshot2 = const v8::HeapSnapshot* snapshot2 =
v8::HeapProfiler::TakeSnapshot(v8::String::New("s2")); v8::HeapProfiler::TakeSnapshot(v8::String::New("s2"));
...@@ -811,7 +812,7 @@ TEST(HeapSnapshotsDiff) { ...@@ -811,7 +812,7 @@ TEST(HeapSnapshotsDiff) {
const v8::HeapGraphNode* node = prop->GetToNode(); const v8::HeapGraphNode* node = prop->GetToNode();
if (node->GetType() == v8::HeapGraphNode::kObject) { if (node->GetType() == v8::HeapGraphNode::kObject) {
v8::String::AsciiValue node_name(node->GetName()); v8::String::AsciiValue node_name(node->GetName());
if (strcmp(*node_name, "A") == 0) { if (strcmp(*node_name, "A2") == 0) {
CHECK(IsNodeRetainedAs(node, v8::HeapGraphEdge::kProperty, "a")); CHECK(IsNodeRetainedAs(node, v8::HeapGraphEdge::kProperty, "a"));
CHECK(!found_A); CHECK(!found_A);
found_A = true; found_A = true;
...@@ -849,6 +850,19 @@ TEST(HeapSnapshotsDiff) { ...@@ -849,6 +850,19 @@ TEST(HeapSnapshotsDiff) {
} }
TEST(HeapSnapshotRootPreservedAfterSorting) {
v8::HandleScope scope;
LocalContext env;
const v8::HeapSnapshot* snapshot =
v8::HeapProfiler::TakeSnapshot(v8::String::New("s"));
const v8::HeapGraphNode* root1 = snapshot->GetRoot();
const_cast<i::HeapSnapshot*>(reinterpret_cast<const i::HeapSnapshot*>(
snapshot))->GetSortedEntriesList();
const v8::HeapGraphNode* root2 = snapshot->GetRoot();
CHECK_EQ(root1, root2);
}
namespace v8 { namespace v8 {
namespace internal { namespace internal {
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment