blob: 6b159a98a338c7b2704ed924a59bc401f8425663 [file] [log] [blame]
ager@chromium.org5f0c45f2010-12-17 08:51:21 +00001// Copyright 2009-2010 the V8 project authors. All rights reserved.
christian.plesner.hansen@gmail.com2bc58ef2009-09-22 10:00:30 +00002// Redistribution and use in source and binary forms, with or without
3// modification, are permitted provided that the following conditions are
4// met:
5//
6// * Redistributions of source code must retain the above copyright
7// notice, this list of conditions and the following disclaimer.
8// * Redistributions in binary form must reproduce the above
9// copyright notice, this list of conditions and the following
10// disclaimer in the documentation and/or other materials provided
11// with the distribution.
12// * Neither the name of Google Inc. nor the names of its
13// contributors may be used to endorse or promote products derived
14// from this software without specific prior written permission.
15//
16// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
17// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
18// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
19// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
20// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27
28#include "v8.h"
29
mstarzinger@chromium.orga2e1a402013-10-15 08:25:05 +000030#include "deoptimizer.h"
christian.plesner.hansen@gmail.com2bc58ef2009-09-22 10:00:30 +000031#include "heap-profiler.h"
ulan@chromium.org2e04b582013-02-21 14:06:02 +000032#include "heap-snapshot-generator-inl.h"
christian.plesner.hansen@gmail.com2bc58ef2009-09-22 10:00:30 +000033
34namespace v8 {
35namespace internal {
36
hpayer@chromium.org7c3372b2013-02-13 17:26:04 +000037HeapProfiler::HeapProfiler(Heap* heap)
38 : snapshots_(new HeapSnapshotsCollection(heap)),
mstarzinger@chromium.orga2e1a402013-10-15 08:25:05 +000039 next_snapshot_uid_(1),
40 is_tracking_allocations_(false) {
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000041}
42
43
44HeapProfiler::~HeapProfiler() {
45 delete snapshots_;
46}
47
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +000048
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +000049void HeapProfiler::DeleteAllSnapshots() {
hpayer@chromium.org7c3372b2013-02-13 17:26:04 +000050 Heap* the_heap = heap();
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +000051 delete snapshots_;
hpayer@chromium.org7c3372b2013-02-13 17:26:04 +000052 snapshots_ = new HeapSnapshotsCollection(the_heap);
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +000053}
54
55
whesse@chromium.orgb08986c2011-03-14 16:13:42 +000056void HeapProfiler::DefineWrapperClass(
57 uint16_t class_id, v8::HeapProfiler::WrapperInfoCallback callback) {
whesse@chromium.orgb08986c2011-03-14 16:13:42 +000058 ASSERT(class_id != v8::HeapProfiler::kPersistentHandleNoClassId);
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +000059 if (wrapper_callbacks_.length() <= class_id) {
60 wrapper_callbacks_.AddBlock(
61 NULL, class_id - wrapper_callbacks_.length() + 1);
whesse@chromium.orgb08986c2011-03-14 16:13:42 +000062 }
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +000063 wrapper_callbacks_[class_id] = callback;
whesse@chromium.orgb08986c2011-03-14 16:13:42 +000064}
65
66
67v8::RetainedObjectInfo* HeapProfiler::ExecuteWrapperClassCallback(
68 uint16_t class_id, Object** wrapper) {
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +000069 if (wrapper_callbacks_.length() <= class_id) return NULL;
70 return wrapper_callbacks_[class_id](
whesse@chromium.orgb08986c2011-03-14 16:13:42 +000071 class_id, Utils::ToLocal(Handle<Object>(wrapper)));
72}
73
74
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +000075HeapSnapshot* HeapProfiler::TakeSnapshot(
mstarzinger@chromium.org32280cf2012-12-06 17:32:37 +000076 const char* name,
mstarzinger@chromium.org32280cf2012-12-06 17:32:37 +000077 v8::ActivityControl* control,
78 v8::HeapProfiler::ObjectNameResolver* resolver) {
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +000079 HeapSnapshot* result = snapshots_->NewSnapshot(name, next_snapshot_uid_++);
80 {
81 HeapSnapshotGenerator generator(result, control, resolver, heap());
82 if (!generator.GenerateSnapshot()) {
83 delete result;
84 result = NULL;
erik.corry@gmail.com145eff52010-08-23 11:36:18 +000085 }
ager@chromium.org5f0c45f2010-12-17 08:51:21 +000086 }
87 snapshots_->SnapshotGenerationFinished(result);
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000088 return result;
89}
90
91
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +000092HeapSnapshot* HeapProfiler::TakeSnapshot(
mstarzinger@chromium.org32280cf2012-12-06 17:32:37 +000093 String* name,
mstarzinger@chromium.org32280cf2012-12-06 17:32:37 +000094 v8::ActivityControl* control,
95 v8::HeapProfiler::ObjectNameResolver* resolver) {
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +000096 return TakeSnapshot(snapshots_->names()->GetName(name), control, resolver);
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000097}
98
mstarzinger@chromium.orge0e1b0d2013-07-08 08:38:06 +000099
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +0000100void HeapProfiler::StartHeapObjectsTracking() {
jkummerow@chromium.org28faa982012-04-13 09:58:30 +0000101 snapshots_->StartHeapObjectsTracking();
102}
103
104
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +0000105SnapshotObjectId HeapProfiler::PushHeapObjectsStats(OutputStream* stream) {
rossberg@chromium.org400388e2012-06-06 09:29:22 +0000106 return snapshots_->PushHeapObjectsStats(stream);
jkummerow@chromium.org28faa982012-04-13 09:58:30 +0000107}
108
109
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +0000110void HeapProfiler::StopHeapObjectsTracking() {
jkummerow@chromium.org28faa982012-04-13 09:58:30 +0000111 snapshots_->StopHeapObjectsTracking();
112}
113
whesse@chromium.org2c186ca2010-06-16 11:32:39 +0000114
mmassi@chromium.org7028c052012-06-13 11:51:58 +0000115size_t HeapProfiler::GetMemorySizeUsedByProfiler() {
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +0000116 return snapshots_->GetUsedMemorySize();
mmassi@chromium.org7028c052012-06-13 11:51:58 +0000117}
118
119
whesse@chromium.org2c186ca2010-06-16 11:32:39 +0000120int HeapProfiler::GetSnapshotsCount() {
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +0000121 return snapshots_->snapshots()->length();
whesse@chromium.org2c186ca2010-06-16 11:32:39 +0000122}
123
124
125HeapSnapshot* HeapProfiler::GetSnapshot(int index) {
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +0000126 return snapshots_->snapshots()->at(index);
whesse@chromium.org2c186ca2010-06-16 11:32:39 +0000127}
128
129
erik.corry@gmail.comed49e962012-04-17 11:57:53 +0000130SnapshotObjectId HeapProfiler::GetSnapshotObjectId(Handle<Object> obj) {
131 if (!obj->IsHeapObject())
132 return v8::HeapProfiler::kUnknownObjectId;
mstarzinger@chromium.orgf705b502013-04-04 11:38:09 +0000133 return snapshots_->FindObjectId(HeapObject::cast(*obj)->address());
whesse@chromium.org2c186ca2010-06-16 11:32:39 +0000134}
135
136
mstarzinger@chromium.orga2e1a402013-10-15 08:25:05 +0000137void HeapProfiler::ObjectMoveEvent(Address from, Address to, int size) {
138 snapshots_->ObjectMoveEvent(from, to, size);
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000139}
140
mstarzinger@chromium.orga2e1a402013-10-15 08:25:05 +0000141
142void HeapProfiler::NewObjectEvent(Address addr, int size) {
143 snapshots_->NewObjectEvent(addr, size);
144}
145
146
147void HeapProfiler::UpdateObjectSizeEvent(Address addr, int size) {
148 snapshots_->UpdateObjectSizeEvent(addr, size);
149}
150
151
danno@chromium.orgca29dd82013-04-26 11:59:48 +0000152void HeapProfiler::SetRetainedObjectInfo(UniqueId id,
153 RetainedObjectInfo* info) {
154 // TODO(yurus, marja): Don't route this information through GlobalHandles.
155 heap()->isolate()->global_handles()->SetRetainedObjectInfo(id, info);
156}
christian.plesner.hansen@gmail.com2bc58ef2009-09-22 10:00:30 +0000157
mstarzinger@chromium.orga2e1a402013-10-15 08:25:05 +0000158
159void HeapProfiler::StartHeapAllocationsRecording() {
160 StartHeapObjectsTracking();
161 is_tracking_allocations_ = true;
162 DropCompiledCode();
163 snapshots_->UpdateHeapObjectsMap();
164}
165
166
167void HeapProfiler::StopHeapAllocationsRecording() {
168 StopHeapObjectsTracking();
169 is_tracking_allocations_ = false;
170 DropCompiledCode();
171}
172
173
174void HeapProfiler::RecordObjectAllocationFromMasm(Isolate* isolate,
175 Address obj,
176 int size) {
177 isolate->heap_profiler()->NewObjectEvent(obj, size);
178}
179
180
181void HeapProfiler::DropCompiledCode() {
182 Isolate* isolate = heap()->isolate();
183 HandleScope scope(isolate);
184
185 if (FLAG_concurrent_recompilation) {
186 isolate->optimizing_compiler_thread()->Flush();
187 }
188
189 Deoptimizer::DeoptimizeAll(isolate);
190
191 Handle<Code> lazy_compile =
192 Handle<Code>(isolate->builtins()->builtin(Builtins::kLazyCompile));
193
194 heap()->CollectAllGarbage(Heap::kMakeHeapIterableMask,
195 "switch allocations tracking");
196
197 DisallowHeapAllocation no_allocation;
198
199 HeapIterator iterator(heap());
200 HeapObject* obj = NULL;
201 while (((obj = iterator.next()) != NULL)) {
202 if (obj->IsJSFunction()) {
203 JSFunction* function = JSFunction::cast(obj);
204 SharedFunctionInfo* shared = function->shared();
205
206 if (!shared->allows_lazy_compilation()) continue;
207 if (!shared->script()->IsScript()) continue;
208
209 Code::Kind kind = function->code()->kind();
210 if (kind == Code::FUNCTION || kind == Code::BUILTIN) {
211 function->set_code(*lazy_compile);
212 shared->set_code(*lazy_compile);
213 }
214 }
215 }
216}
217
218
christian.plesner.hansen@gmail.com2bc58ef2009-09-22 10:00:30 +0000219} } // namespace v8::internal