Index: runtime/vm/heap_trace.cc |
=================================================================== |
--- runtime/vm/heap_trace.cc (revision 21785) |
+++ runtime/vm/heap_trace.cc (working copy) |
@@ -1,488 +0,0 @@ |
-// 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. |
- |
-#include "vm/heap_trace.h" |
- |
-#include "include/dart_api.h" |
-#include "vm/dart_api_state.h" |
-#include "vm/debugger.h" |
-#include "vm/isolate.h" |
-#include "vm/object.h" |
-#include "vm/object_set.h" |
-#include "vm/object_store.h" |
-#include "vm/os.h" |
-#include "vm/stack_frame.h" |
-#include "vm/unicode.h" |
- |
-namespace dart { |
- |
-DEFINE_FLAG(bool, heap_trace, false, "Enable heap tracing."); |
- |
-Dart_FileOpenCallback HeapTrace::open_callback_ = NULL; |
-Dart_FileWriteCallback HeapTrace::write_callback_ = NULL; |
-Dart_FileCloseCallback HeapTrace::close_callback_ = NULL; |
-bool HeapTrace::is_enabled_ = false; |
- |
-class HeapTraceVisitor : public ObjectPointerVisitor { |
- public: |
- HeapTraceVisitor(Isolate* isolate, |
- HeapTrace* heap_trace, |
- ObjectSet* object_set) |
- : ObjectPointerVisitor(isolate), |
- heap_trace_(heap_trace), |
- vm_isolate_(Dart::vm_isolate()), |
- object_set_(object_set) { |
- } |
- |
- void VisitPointers(RawObject** first, RawObject** last) { |
- for (RawObject** current = first; current <= last; current++) { |
- RawObject* raw_obj = *current; |
- |
- // We only care about objects in the heap |
- // Also, since this visitor will frequently be encountering redudant |
- // roots, we use an object_set to skip the duplicates. |
- if (raw_obj->IsHeapObject() && |
- raw_obj != reinterpret_cast<RawObject*>(0x1) && |
- raw_obj != reinterpret_cast<RawObject*>(0xabababab) && |
- !object_set_->Contains(raw_obj) && |
- !vm_isolate_->heap()->Contains(RawObject::ToAddr(raw_obj))) { |
- object_set_->Add(raw_obj); |
- uword addr = RawObject::ToAddr(raw_obj); |
- heap_trace_->TraceSingleRoot(addr); |
- } |
- } |
- } |
- |
- private: |
- HeapTrace* heap_trace_; |
- Isolate* vm_isolate_; |
- // TODO(cshapiro): replace with a sparse data structure. |
- ObjectSet* object_set_; |
- DISALLOW_COPY_AND_ASSIGN(HeapTraceVisitor); |
-}; |
- |
- |
-class HeapTraceScopedHandleVisitor : public ObjectPointerVisitor { |
- public: |
- HeapTraceScopedHandleVisitor(Isolate* isolate, HeapTrace* heap_trace) |
- : ObjectPointerVisitor(isolate), heap_trace_(heap_trace) { |
- } |
- |
- void VisitPointers(RawObject** first, RawObject** last) { |
- for (RawObject** current = first; current <= last; current++) { |
- RawObject* raw_obj = *current; |
- Heap* heap = isolate()->heap(); |
- |
- // We only care about objects in the heap |
- if (raw_obj->IsHeapObject() && |
- raw_obj != reinterpret_cast<RawObject*>(0x1) && |
- raw_obj != reinterpret_cast<RawObject*>(0xabababab) && |
- heap->Contains(RawObject::ToAddr(raw_obj))) { |
- uword addr = RawObject::ToAddr(raw_obj); |
- heap_trace_->TraceScopedHandle(addr); |
- } |
- } |
- } |
- |
- private: |
- HeapTrace* heap_trace_; |
- DISALLOW_COPY_AND_ASSIGN(HeapTraceScopedHandleVisitor); |
-}; |
- |
- |
-class HeapTraceObjectStoreVisitor : public ObjectPointerVisitor { |
- public: |
- HeapTraceObjectStoreVisitor(Isolate* isolate, HeapTrace* heap_trace) |
- : ObjectPointerVisitor(isolate), heap_trace_(heap_trace) { |
- } |
- |
- void VisitPointers(RawObject** first, RawObject** last) { |
- for (RawObject** current = first; current <= last; current++) { |
- RawObject* raw_obj = *current; |
- |
- // We only care about obects in the heap. |
- if (raw_obj->IsHeapObject() && |
- raw_obj != reinterpret_cast<RawObject*>(0x1) && |
- raw_obj != reinterpret_cast<RawObject*>(0xabababab)) { |
- uword addr = RawObject::ToAddr(raw_obj); |
- heap_trace_->TraceObjectStorePointer(addr); |
- } |
- } |
- } |
- |
- private: |
- HeapTrace* heap_trace_; |
- DISALLOW_COPY_AND_ASSIGN(HeapTraceObjectStoreVisitor); |
-}; |
- |
- |
-class HeapTraceInitialHeapVisitor : public ObjectVisitor { |
- public: |
- HeapTraceInitialHeapVisitor(Isolate* isolate, HeapTrace* heap_trace) |
- : ObjectVisitor(isolate), heap_trace_(heap_trace) {} |
- |
- void VisitObject(RawObject* raw_obj) { |
- heap_trace_->TraceSnapshotAlloc(raw_obj, raw_obj->Size()); |
- } |
- |
- private: |
- HeapTrace* heap_trace_; |
- DISALLOW_COPY_AND_ASSIGN(HeapTraceInitialHeapVisitor); |
-}; |
- |
- |
-HeapTrace::HeapTrace() : isolate_initialized_(false), output_stream_(NULL) { |
-} |
- |
- |
-HeapTrace::~HeapTrace() { |
- if (isolate_initialized_) { |
- (*close_callback_)(output_stream_); |
- } |
-} |
- |
- |
-void HeapTrace::InitOnce(Dart_FileOpenCallback open_callback, |
- Dart_FileWriteCallback write_callback, |
- Dart_FileCloseCallback close_callback) { |
- ASSERT(open_callback != NULL); |
- ASSERT(write_callback != NULL); |
- ASSERT(close_callback != NULL); |
- HeapTrace::open_callback_ = open_callback; |
- HeapTrace::write_callback_ = write_callback; |
- HeapTrace::close_callback_ = close_callback; |
- HeapTrace::is_enabled_ = true; |
-} |
- |
- |
-ObjectSet* HeapTrace::CreateEmptyObjectSet() const { |
- Isolate* isolate = Isolate::Current(); |
- uword start, end; |
- isolate->heap()->StartEndAddress(&start, &end); |
- |
- Isolate* vm_isolate = Dart::vm_isolate(); |
- uword vm_start, vm_end; |
- vm_isolate->heap()->StartEndAddress(&vm_start, &vm_end); |
- |
- ObjectSet* allocated_set = new ObjectSet(Utils::Minimum(start, vm_start), |
- Utils::Maximum(end, vm_end)); |
- |
- return allocated_set; |
-} |
- |
- |
-void HeapTrace::ResizeObjectSet() { |
- Isolate* isolate = Isolate::Current(); |
- uword start, end; |
- isolate->heap()->StartEndAddress(&start, &end); |
- Isolate* vm_isolate = Dart::vm_isolate(); |
- uword vm_start, vm_end; |
- vm_isolate->heap()->StartEndAddress(&vm_start, &vm_end); |
- object_set_.Resize(Utils::Minimum(start, vm_start), |
- Utils::Maximum(end, vm_end)); |
-} |
- |
- |
-void HeapTrace::Init(Isolate* isolate) { |
- // Do not trace the VM isolate |
- if (isolate == Dart::vm_isolate()) { |
- return; |
- } |
- ASSERT(isolate_initialized_ == false); |
- const char* format = "%s.htrace"; |
- intptr_t len = OS::SNPrint(NULL, 0, format, isolate->name()); |
- char* filename = new char[len + 1]; |
- OS::SNPrint(filename, len + 1, format, isolate->name()); |
- output_stream_ = (*open_callback_)(filename); |
- ASSERT(output_stream_ != NULL); |
- delete[] filename; |
- isolate_initialized_ = true; |
- |
- HeapTraceObjectStoreVisitor object_store_visitor(isolate, this); |
- isolate->object_store()->VisitObjectPointers(&object_store_visitor); |
- |
- // Visit any objects that may have been allocated during startup, |
- // before we started tracing. |
- HeapTraceInitialHeapVisitor heap_visitor(isolate, this); |
- isolate->heap()->IterateObjects(&heap_visitor); |
- TraceRoots(isolate); |
-} |
- |
- |
-// Allocation Record - 'A' (0x41) |
-// |
-// Format: |
-// 'A' |
-// uword - address of allocated object |
-// uword - size of allocated object |
-void HeapTrace::TraceAllocation(uword addr, intptr_t size) { |
- if (isolate_initialized_) { |
- { |
- AllocationRecord rec(this); |
- rec.Write(addr); |
- rec.Write(size); |
- } |
- TraceRoots(Isolate::Current()); |
- } |
-} |
- |
- |
-// Snapshot Allocation Record - 'B' (0x41) |
-// |
-// Format: |
-// 'B' |
-// uword - address of allocated object |
-// uword - size of allocated object |
-void HeapTrace::TraceSnapshotAlloc(RawObject* obj, intptr_t size) { |
- if (isolate_initialized_) { |
- SnapshotAllocationRecord rec(this); |
- rec.Write(RawObject::ToAddr(obj)); |
- rec.Write(static_cast<uword>(size)); |
- } |
-} |
- |
- |
-// Allocate Zone Handle Record - 'Z' (0x5a) |
-// |
-// Format: |
-// 'Z' |
-// uword - handle address (where the handle is pointing) |
-// uword - zone address (address of the zone the handle is in) |
-void HeapTrace::TraceAllocateZoneHandle(uword handle, uword zone_addr) { |
- if (isolate_initialized_) { |
- AllocZoneHandleRecord rec(this); |
- rec.Write(handle); |
- rec.Write(zone_addr); |
- } |
-} |
- |
- |
-// Delete Zone Record - 'z' (0x7a) |
-// |
-// Format: |
-// 'z' |
-// uword - zone address (all the handles in that zone are now gone) |
-void HeapTrace::TraceDeleteZone(Zone* zone) { |
- if (isolate_initialized_) { |
- DeleteZoneRecord rec(this); |
- rec.Write(reinterpret_cast<uword>(zone)); |
- } |
-} |
- |
- |
-// Delete Scoped Hanldes Record - 's' (0x73) |
-// |
-// Format: |
-// 's' |
-void HeapTrace::TraceDeleteScopedHandles() { |
- if (isolate_initialized_) { |
- DeleteScopedHandlesRecord rec(this); |
- } |
-} |
- |
- |
-// Copy Record - 'C' (0x43) |
-// |
-// Format: |
-// 'C' |
-// uword - old address |
-// uword - new address |
-void HeapTrace::TraceCopy(uword from_addr, uword to_addr) { |
- if (isolate_initialized_) { |
- CopyRecord rec(this); |
- rec.Write(from_addr); |
- rec.Write(to_addr); |
- } |
-} |
- |
- |
-// Object Store Recorda - 'O'(0x4f) |
-// |
-// Format: |
-// 'O' |
-// uword - address |
-void HeapTrace::TraceObjectStorePointer(uword addr) { |
- if (isolate_initialized_) { |
- ObjectStoreRecord rec(this); |
- rec.Write(addr); |
- } |
-} |
- |
- |
-// Promotion Records - 'P' (0x50) |
-// |
-// Format: |
-// 'P' |
-// uword - old address |
-// uword - new address |
-void HeapTrace::TracePromotion(uword old_addr, uword promoted_addr) { |
- if (isolate_initialized_) { |
- PromotionRecord rec(this); |
- rec.Write(old_addr); |
- rec.Write(promoted_addr); |
- } |
-} |
- |
- |
-// Death Range Record - 'L' (0x4c) |
-// |
-// Format: |
-// 'L' |
-// uword - inclusive start address of the space being left |
-// uword - exclusive end address of the space being left |
-void HeapTrace::TraceDeathRange(uword inclusive_start, uword exclusive_end) { |
- if (isolate_initialized_) { |
- DeathRangeRecord rec(this); |
- rec.Write(inclusive_start); |
- rec.Write(exclusive_end); |
- } |
-} |
- |
- |
-// Register Class Record - 'K' (0x4b) |
-// |
-// Format: |
-// 'K' |
-// uword - address ( the address of the class) |
-void HeapTrace::TraceRegisterClass(const Class& cls) { |
- if (isolate_initialized_) { |
- RegisterClassRecord rec(this); |
- rec.Write(RawObject::ToAddr(cls.raw())); |
- } |
-} |
- |
- |
-// Scoped Handle Record - 'H' (0x48) |
-// |
-// Format: |
-// 'H' |
-// uword - adress of the scoped handle (where it is pointing) |
-void HeapTrace::TraceScopedHandle(uword handle) { |
- if (isolate_initialized_) { |
- AllocScopedHandleRecord rec(this); |
- rec.Write(handle); |
- } |
-} |
- |
- |
-// Root Record - 'R' (0x52) |
-// |
-// Format: |
-// 'R' |
-// uword - address |
-void HeapTrace::TraceSingleRoot(uword root_addr) { |
- if (isolate_initialized_) { |
- RootRecord rec(this); |
- rec.Write(root_addr); |
- } |
-} |
- |
- |
-// Sweep Record - 'S' |
-// |
-// Format: |
-// 'S' |
-// uword - address |
-void HeapTrace::TraceSweep(uword sweept_addr) { |
- if (isolate_initialized_) { |
- SweepRecord rec(this); |
- rec.Write(sweept_addr); |
- } |
-} |
- |
- |
-// Does not output any records directly, |
-// but does call TraceSingleRoot |
-void HeapTrace::TraceRoots(Isolate* isolate) { |
- if (isolate_initialized_) { |
- ResizeObjectSet(); |
- HeapTraceVisitor visitor(isolate, this, &object_set_); |
- HeapTraceScopedHandleVisitor handle_visitor(isolate, this); |
- |
- bool visit_prologue_weak_handles = true; |
- bool validate_frames = false; |
- |
- // Visit objects in per isolate stubs. |
- StubCode::VisitObjectPointers(&visitor); |
- |
- // stack |
- StackFrameIterator frames_iterator(validate_frames); |
- StackFrame* frame = frames_iterator.NextFrame(); |
- while (frame != NULL) { |
- frame->VisitObjectPointers(&visitor); |
- frame = frames_iterator.NextFrame(); |
- } |
- |
- if (isolate->api_state() != NULL) { |
- isolate->api_state()->VisitObjectPointers(&visitor, |
- visit_prologue_weak_handles); |
- } |
- |
- // Visit the top context which is stored in the isolate. |
- RawContext* top_context = isolate->top_context(); |
- visitor.VisitPointer(reinterpret_cast<RawObject**>(&top_context)); |
- |
- // Visit the currently active IC data array. |
- RawArray* ic_data_array = isolate->ic_data_array(); |
- visitor.VisitPointer(reinterpret_cast<RawObject**>(&ic_data_array)); |
- |
- // Visit objects in the debugger. |
- isolate->debugger()->VisitObjectPointers(&visitor); |
- |
- isolate->current_zone()->handles()-> |
- VisitUnvisitedScopedHandles(&handle_visitor); |
- |
- object_set_.FastClear(); |
- } |
-} |
- |
- |
-// Store Record - 'U' (0x55) |
-// |
-// Format: |
-// 'U' |
-// uword - originating object address (where a pointer is being stored) |
-// uword - byte offset into origin where the pointer is being stored |
-// uword - value of the pointer being stored |
-void HeapTrace::TraceStoreIntoObject(uword object, |
- uword field_addr, |
- uword value) { |
- if (isolate_initialized_) { |
- // We don't care about pointers into the VM_Islate heap, so skip them. |
- // There should not be any pointers /out/ of the VM isolate; so we |
- // do not check object. |
- if (Isolate::Current()->heap()->Contains(value)) { |
- StoreRecord rec(this); |
- uword slot_offset = field_addr - object; |
- |
- rec.Write(object); |
- rec.Write(slot_offset); |
- rec.Write(value); |
- } |
- } |
-} |
- |
- |
-// Mark Sweep Start Record - '{' (0x7b) |
-// |
-// Format: |
-// '{' |
-void HeapTrace::TraceMarkSweepStart() { |
- if (isolate_initialized_) { |
- MarkSweepStartRecord rec(this); |
- } |
-} |
- |
- |
-// Mark Sweep Finish Record - '}' (0x7d) |
-// |
-// Format: |
-// '}' |
-void HeapTrace::TraceMarkSweepFinish() { |
- if (isolate_initialized_) { |
- MarkSweepFinishRecord rec(this); |
- } |
-} |
- |
-} // namespace dart |