OLD | NEW |
1 // Copyright 2012 the V8 project authors. All rights reserved. | 1 // Copyright 2012 the V8 project authors. All rights reserved. |
2 // Redistribution and use in source and binary forms, with or without | 2 // Redistribution and use in source and binary forms, with or without |
3 // modification, are permitted provided that the following conditions are | 3 // modification, are permitted provided that the following conditions are |
4 // met: | 4 // met: |
5 // | 5 // |
6 // * Redistributions of source code must retain the above copyright | 6 // * Redistributions of source code must retain the above copyright |
7 // notice, this list of conditions and the following disclaimer. | 7 // notice, this list of conditions and the following disclaimer. |
8 // * Redistributions in binary form must reproduce the above | 8 // * Redistributions in binary form must reproduce the above |
9 // copyright notice, this list of conditions and the following | 9 // copyright notice, this list of conditions and the following |
10 // disclaimer in the documentation and/or other materials provided | 10 // disclaimer in the documentation and/or other materials provided |
(...skipping 939 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
950 index_ = index; | 950 index_ = index; |
951 to_ = to; | 951 to_ = to; |
952 } | 952 } |
953 | 953 |
954 | 954 |
955 void HeapGraphEdge::Init(int child_index, int index, HeapEntry* to) { | 955 void HeapGraphEdge::Init(int child_index, int index, HeapEntry* to) { |
956 Init(child_index, kElement, index, to); | 956 Init(child_index, kElement, index, to); |
957 } | 957 } |
958 | 958 |
959 | 959 |
| 960 HeapEntry* HeapGraphEdge::From() { |
| 961 return reinterpret_cast<HeapEntry*>(this - child_index_) - 1; |
| 962 } |
| 963 |
| 964 |
960 void HeapEntry::Init(HeapSnapshot* snapshot, | 965 void HeapEntry::Init(HeapSnapshot* snapshot, |
961 Type type, | 966 Type type, |
962 const char* name, | 967 const char* name, |
963 SnapshotObjectId id, | 968 SnapshotObjectId id, |
964 int self_size, | 969 int self_size, |
965 int children_count, | 970 int children_count, |
966 int retainers_count) { | 971 int retainers_count) { |
967 snapshot_ = snapshot; | 972 snapshot_ = snapshot; |
968 type_ = type; | 973 type_ = type; |
969 painted_ = false; | 974 painted_ = false; |
970 reachable_from_window_ = false; | |
971 name_ = name; | 975 name_ = name; |
972 self_size_ = self_size; | 976 self_size_ = self_size; |
973 retained_size_ = 0; | 977 retained_size_ = 0; |
974 entry_index_ = -1; | 978 entry_index_ = -1; |
975 children_count_ = children_count; | 979 children_count_ = children_count; |
976 retainers_count_ = retainers_count; | 980 retainers_count_ = retainers_count; |
977 dominator_ = NULL; | 981 dominator_ = NULL; |
978 id_ = id; | 982 id_ = id; |
979 } | 983 } |
980 | 984 |
(...skipping 1186 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2167 break; | 2171 break; |
2168 } | 2172 } |
2169 } | 2173 } |
2170 } else { | 2174 } else { |
2171 StringDictionary* dictionary = js_obj->property_dictionary(); | 2175 StringDictionary* dictionary = js_obj->property_dictionary(); |
2172 int length = dictionary->Capacity(); | 2176 int length = dictionary->Capacity(); |
2173 for (int i = 0; i < length; ++i) { | 2177 for (int i = 0; i < length; ++i) { |
2174 Object* k = dictionary->KeyAt(i); | 2178 Object* k = dictionary->KeyAt(i); |
2175 if (dictionary->IsKey(k)) { | 2179 if (dictionary->IsKey(k)) { |
2176 Object* target = dictionary->ValueAt(i); | 2180 Object* target = dictionary->ValueAt(i); |
| 2181 SetPropertyReference( |
| 2182 js_obj, entry, String::cast(k), target); |
2177 // We assume that global objects can only have slow properties. | 2183 // We assume that global objects can only have slow properties. |
2178 Object* value = target->IsJSGlobalPropertyCell() | 2184 if (target->IsJSGlobalPropertyCell()) { |
2179 ? JSGlobalPropertyCell::cast(target)->value() | 2185 SetPropertyShortcutReference(js_obj, |
2180 : target; | 2186 entry, |
2181 if (String::cast(k)->length() > 0) { | 2187 String::cast(k), |
2182 SetPropertyReference(js_obj, entry, String::cast(k), value); | 2188 JSGlobalPropertyCell::cast( |
2183 } else { | 2189 target)->value()); |
2184 TagObject(value, "(hidden properties)"); | |
2185 SetInternalReference(js_obj, entry, "hidden_properties", value); | |
2186 } | 2190 } |
2187 } | 2191 } |
2188 } | 2192 } |
2189 } | 2193 } |
2190 } | 2194 } |
2191 | 2195 |
2192 | 2196 |
2193 void V8HeapExplorer::ExtractElementReferences(JSObject* js_obj, | 2197 void V8HeapExplorer::ExtractElementReferences(JSObject* js_obj, |
2194 HeapEntry* entry) { | 2198 HeapEntry* entry) { |
2195 if (js_obj->HasFastElements()) { | 2199 if (js_obj->HasFastElements()) { |
(...skipping 430 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
2626 isolate->factory()->NewStringFromAscii(CStrVector("document")); | 2630 isolate->factory()->NewStringFromAscii(CStrVector("document")); |
2627 Handle<String> url_string = | 2631 Handle<String> url_string = |
2628 isolate->factory()->NewStringFromAscii(CStrVector("URL")); | 2632 isolate->factory()->NewStringFromAscii(CStrVector("URL")); |
2629 const char** urls = NewArray<const char*>(enumerator.count()); | 2633 const char** urls = NewArray<const char*>(enumerator.count()); |
2630 for (int i = 0, l = enumerator.count(); i < l; ++i) { | 2634 for (int i = 0, l = enumerator.count(); i < l; ++i) { |
2631 urls[i] = NULL; | 2635 urls[i] = NULL; |
2632 HandleScope scope; | 2636 HandleScope scope; |
2633 Handle<JSGlobalObject> global_obj = enumerator.at(i); | 2637 Handle<JSGlobalObject> global_obj = enumerator.at(i); |
2634 Object* obj_document; | 2638 Object* obj_document; |
2635 if (global_obj->GetProperty(*document_string)->ToObject(&obj_document) && | 2639 if (global_obj->GetProperty(*document_string)->ToObject(&obj_document) && |
2636 obj_document->IsJSObject()) { | 2640 obj_document->IsJSObject()) { |
2637 JSObject* document = JSObject::cast(obj_document); | 2641 JSObject* document = JSObject::cast(obj_document); |
2638 Object* obj_url; | 2642 Object* obj_url; |
2639 if (document->GetProperty(*url_string)->ToObject(&obj_url) && | 2643 if (document->GetProperty(*url_string)->ToObject(&obj_url) && |
2640 obj_url->IsString()) { | 2644 obj_url->IsString()) { |
2641 urls[i] = collection_->names()->GetName(String::cast(obj_url)); | 2645 urls[i] = collection_->names()->GetName(String::cast(obj_url)); |
2642 } | 2646 } |
2643 } | 2647 } |
2644 } | 2648 } |
2645 | 2649 |
2646 AssertNoAllocation no_allocation; | 2650 AssertNoAllocation no_allocation; |
(...skipping 533 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3180 // in turn may relocate objects in property maps thus changing the heap | 3184 // in turn may relocate objects in property maps thus changing the heap |
3181 // layout and affecting retainer counts. This is not acceptable because | 3185 // layout and affecting retainer counts. This is not acceptable because |
3182 // number of retainers must not change between count and fill passes. | 3186 // number of retainers must not change between count and fill passes. |
3183 // To avoid this there's a separate postpass that set object names. | 3187 // To avoid this there's a separate postpass that set object names. |
3184 return v8_heap_explorer_.IterateAndExtractReferences(&filler) | 3188 return v8_heap_explorer_.IterateAndExtractReferences(&filler) |
3185 && dom_explorer_.IterateAndExtractReferences(&filler) | 3189 && dom_explorer_.IterateAndExtractReferences(&filler) |
3186 && v8_heap_explorer_.IterateAndSetObjectNames(&filler); | 3190 && v8_heap_explorer_.IterateAndSetObjectNames(&filler); |
3187 } | 3191 } |
3188 | 3192 |
3189 | 3193 |
3190 void HeapSnapshotGenerator::MarkWindowReachableObjects() { | 3194 void HeapSnapshotGenerator::FillReversePostorderIndexes( |
3191 List<HeapEntry*> worklist; | |
3192 | |
3193 Vector<HeapGraphEdge> children = snapshot_->root()->children(); | |
3194 for (int i = 0; i < children.length(); ++i) { | |
3195 if (children[i].type() == HeapGraphEdge::kShortcut) { | |
3196 worklist.Add(children[i].to()); | |
3197 } | |
3198 } | |
3199 | |
3200 while (!worklist.is_empty()) { | |
3201 HeapEntry* entry = worklist.RemoveLast(); | |
3202 if (entry->reachable_from_window()) continue; | |
3203 entry->set_reachable_from_window(); | |
3204 Vector<HeapGraphEdge> children = entry->children(); | |
3205 for (int i = 0; i < children.length(); ++i) { | |
3206 HeapEntry* child = children[i].to(); | |
3207 if (!child->reachable_from_window()) { | |
3208 worklist.Add(child); | |
3209 } | |
3210 } | |
3211 } | |
3212 } | |
3213 | |
3214 | |
3215 static bool IsRetainingEdge(HeapGraphEdge* edge) { | |
3216 if (edge->type() == HeapGraphEdge::kShortcut) return false; | |
3217 // The edge is not retaining if it goes from system domain | |
3218 // (i.e. an object not reachable from window) to the user domain | |
3219 // (i.e. a reachable object). | |
3220 return edge->from()->reachable_from_window() | |
3221 || !edge->to()->reachable_from_window(); | |
3222 } | |
3223 | |
3224 | |
3225 void HeapSnapshotGenerator::FillPostorderIndexes( | |
3226 Vector<HeapEntry*>* entries) { | 3195 Vector<HeapEntry*>* entries) { |
3227 snapshot_->ClearPaint(); | 3196 snapshot_->ClearPaint(); |
3228 int current_entry = 0; | 3197 int current_entry = 0; |
3229 List<HeapEntry*> nodes_to_visit; | 3198 List<HeapEntry*> nodes_to_visit; |
3230 HeapEntry* root = snapshot_->root(); | 3199 nodes_to_visit.Add(snapshot_->root()); |
3231 nodes_to_visit.Add(root); | |
3232 snapshot_->root()->paint(); | 3200 snapshot_->root()->paint(); |
3233 while (!nodes_to_visit.is_empty()) { | 3201 while (!nodes_to_visit.is_empty()) { |
3234 HeapEntry* entry = nodes_to_visit.last(); | 3202 HeapEntry* entry = nodes_to_visit.last(); |
3235 Vector<HeapGraphEdge> children = entry->children(); | 3203 Vector<HeapGraphEdge> children = entry->children(); |
3236 bool has_new_edges = false; | 3204 bool has_new_edges = false; |
3237 for (int i = 0; i < children.length(); ++i) { | 3205 for (int i = 0; i < children.length(); ++i) { |
3238 if (entry != root && !IsRetainingEdge(&children[i])) continue; | 3206 if (children[i].type() == HeapGraphEdge::kShortcut) continue; |
3239 HeapEntry* child = children[i].to(); | 3207 HeapEntry* child = children[i].to(); |
3240 if (!child->painted()) { | 3208 if (!child->painted()) { |
3241 nodes_to_visit.Add(child); | 3209 nodes_to_visit.Add(child); |
3242 child->paint(); | 3210 child->paint(); |
3243 has_new_edges = true; | 3211 has_new_edges = true; |
3244 } | 3212 } |
3245 } | 3213 } |
3246 if (!has_new_edges) { | 3214 if (!has_new_edges) { |
3247 entry->set_ordered_index(current_entry); | 3215 entry->set_ordered_index(current_entry); |
3248 (*entries)[current_entry++] = entry; | 3216 (*entries)[current_entry++] = entry; |
(...skipping 14 matching lines...) Expand all Loading... |
3263 } | 3231 } |
3264 | 3232 |
3265 | 3233 |
3266 // The algorithm is based on the article: | 3234 // The algorithm is based on the article: |
3267 // K. Cooper, T. Harvey and K. Kennedy "A Simple, Fast Dominance Algorithm" | 3235 // K. Cooper, T. Harvey and K. Kennedy "A Simple, Fast Dominance Algorithm" |
3268 // Softw. Pract. Exper. 4 (2001), pp. 1-10. | 3236 // Softw. Pract. Exper. 4 (2001), pp. 1-10. |
3269 bool HeapSnapshotGenerator::BuildDominatorTree( | 3237 bool HeapSnapshotGenerator::BuildDominatorTree( |
3270 const Vector<HeapEntry*>& entries, | 3238 const Vector<HeapEntry*>& entries, |
3271 Vector<int>* dominators) { | 3239 Vector<int>* dominators) { |
3272 if (entries.length() == 0) return true; | 3240 if (entries.length() == 0) return true; |
3273 HeapEntry* root = snapshot_->root(); | |
3274 const int entries_length = entries.length(), root_index = entries_length - 1; | 3241 const int entries_length = entries.length(), root_index = entries_length - 1; |
3275 static const int kNoDominator = -1; | 3242 static const int kNoDominator = -1; |
3276 for (int i = 0; i < root_index; ++i) (*dominators)[i] = kNoDominator; | 3243 for (int i = 0; i < root_index; ++i) (*dominators)[i] = kNoDominator; |
3277 (*dominators)[root_index] = root_index; | 3244 (*dominators)[root_index] = root_index; |
3278 | 3245 |
3279 // The affected array is used to mark entries which dominators | 3246 // The affected array is used to mark entries which dominators |
3280 // have to be racalculated because of changes in their retainers. | 3247 // have to be racalculated because of changes in their retainers. |
3281 ScopedVector<bool> affected(entries_length); | 3248 ScopedVector<bool> affected(entries_length); |
3282 for (int i = 0; i < affected.length(); ++i) affected[i] = false; | 3249 for (int i = 0; i < affected.length(); ++i) affected[i] = false; |
3283 // Mark the root direct children as affected. | 3250 // Mark the root direct children as affected. |
3284 Vector<HeapGraphEdge> children = entries[root_index]->children(); | 3251 Vector<HeapGraphEdge> children = entries[root_index]->children(); |
3285 for (int i = 0; i < children.length(); ++i) { | 3252 for (int i = 0; i < children.length(); ++i) { |
3286 affected[children[i].to()->ordered_index()] = true; | 3253 affected[children[i].to()->ordered_index()] = true; |
3287 } | 3254 } |
3288 | 3255 |
3289 bool changed = true; | 3256 bool changed = true; |
3290 while (changed) { | 3257 while (changed) { |
3291 changed = false; | 3258 changed = false; |
3292 if (!ProgressReport(true)) return false; | 3259 if (!ProgressReport(true)) return false; |
3293 for (int i = root_index - 1; i >= 0; --i) { | 3260 for (int i = root_index - 1; i >= 0; --i) { |
3294 if (!affected[i]) continue; | 3261 if (!affected[i]) continue; |
3295 affected[i] = false; | 3262 affected[i] = false; |
3296 // If dominator of the entry has already been set to root, | 3263 // If dominator of the entry has already been set to root, |
3297 // then it can't propagate any further. | 3264 // then it can't propagate any further. |
3298 if ((*dominators)[i] == root_index) continue; | 3265 if ((*dominators)[i] == root_index) continue; |
3299 int new_idom_index = kNoDominator; | 3266 int new_idom_index = kNoDominator; |
3300 Vector<HeapGraphEdge*> rets = entries[i]->retainers(); | 3267 Vector<HeapGraphEdge*> rets = entries[i]->retainers(); |
3301 for (int j = 0; j < rets.length(); ++j) { | 3268 for (int j = 0; j < rets.length(); ++j) { |
3302 if (rets[j]->from() != root && !IsRetainingEdge(rets[j])) continue; | 3269 if (rets[j]->type() == HeapGraphEdge::kShortcut) continue; |
3303 int ret_index = rets[j]->from()->ordered_index(); | 3270 int ret_index = rets[j]->From()->ordered_index(); |
3304 if (dominators->at(ret_index) != kNoDominator) { | 3271 if (dominators->at(ret_index) != kNoDominator) { |
3305 new_idom_index = new_idom_index == kNoDominator | 3272 new_idom_index = new_idom_index == kNoDominator |
3306 ? ret_index | 3273 ? ret_index |
3307 : Intersect(ret_index, new_idom_index, *dominators); | 3274 : Intersect(ret_index, new_idom_index, *dominators); |
3308 // If idom has already reached the root, it doesn't make sense | 3275 // If idom has already reached the root, it doesn't make sense |
3309 // to check other retainers. | 3276 // to check other retainers. |
3310 if (new_idom_index == root_index) break; | 3277 if (new_idom_index == root_index) break; |
3311 } | 3278 } |
3312 } | 3279 } |
3313 if (new_idom_index != kNoDominator | 3280 if (new_idom_index != kNoDominator |
3314 && dominators->at(i) != new_idom_index) { | 3281 && dominators->at(i) != new_idom_index) { |
3315 (*dominators)[i] = new_idom_index; | 3282 (*dominators)[i] = new_idom_index; |
3316 changed = true; | 3283 changed = true; |
3317 Vector<HeapGraphEdge> children = entries[i]->children(); | 3284 Vector<HeapGraphEdge> children = entries[i]->children(); |
3318 for (int j = 0; j < children.length(); ++j) { | 3285 for (int j = 0; j < children.length(); ++j) { |
3319 affected[children[j].to()->ordered_index()] = true; | 3286 affected[children[j].to()->ordered_index()] = true; |
3320 } | 3287 } |
3321 } | 3288 } |
3322 } | 3289 } |
3323 } | 3290 } |
3324 return true; | 3291 return true; |
3325 } | 3292 } |
3326 | 3293 |
3327 | 3294 |
3328 bool HeapSnapshotGenerator::SetEntriesDominators() { | 3295 bool HeapSnapshotGenerator::SetEntriesDominators() { |
3329 MarkWindowReachableObjects(); | 3296 // This array is used for maintaining reverse postorder of nodes. |
3330 // This array is used for maintaining postorder of nodes. | |
3331 ScopedVector<HeapEntry*> ordered_entries(snapshot_->entries()->length()); | 3297 ScopedVector<HeapEntry*> ordered_entries(snapshot_->entries()->length()); |
3332 FillPostorderIndexes(&ordered_entries); | 3298 FillReversePostorderIndexes(&ordered_entries); |
3333 ScopedVector<int> dominators(ordered_entries.length()); | 3299 ScopedVector<int> dominators(ordered_entries.length()); |
3334 if (!BuildDominatorTree(ordered_entries, &dominators)) return false; | 3300 if (!BuildDominatorTree(ordered_entries, &dominators)) return false; |
3335 for (int i = 0; i < ordered_entries.length(); ++i) { | 3301 for (int i = 0; i < ordered_entries.length(); ++i) { |
3336 ASSERT(dominators[i] >= 0); | 3302 ASSERT(dominators[i] >= 0); |
3337 ordered_entries[i]->set_dominator(ordered_entries[dominators[i]]); | 3303 ordered_entries[i]->set_dominator(ordered_entries[dominators[i]]); |
3338 } | 3304 } |
3339 return true; | 3305 return true; |
3340 } | 3306 } |
3341 | 3307 |
3342 | 3308 |
(...skipping 447 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
3790 | 3756 |
3791 | 3757 |
3792 void HeapSnapshotJSONSerializer::SortHashMap( | 3758 void HeapSnapshotJSONSerializer::SortHashMap( |
3793 HashMap* map, List<HashMap::Entry*>* sorted_entries) { | 3759 HashMap* map, List<HashMap::Entry*>* sorted_entries) { |
3794 for (HashMap::Entry* p = map->Start(); p != NULL; p = map->Next(p)) | 3760 for (HashMap::Entry* p = map->Start(); p != NULL; p = map->Next(p)) |
3795 sorted_entries->Add(p); | 3761 sorted_entries->Add(p); |
3796 sorted_entries->Sort(SortUsingEntryValue); | 3762 sorted_entries->Sort(SortUsingEntryValue); |
3797 } | 3763 } |
3798 | 3764 |
3799 } } // namespace v8::internal | 3765 } } // namespace v8::internal |
OLD | NEW |