blob: 7e48a87a7ee10e7d015543da3a6a4e80cdc1f69d [file] [log] [blame]
fschneider@chromium.orgfb144a02011-05-04 12:43:48 +00001// Copyright 2011 the V8 project authors. All rights reserved.
christian.plesner.hansen@gmail.com2bc58ef2009-09-22 10:00:30 +00002//
3// Tests for heap profiler
4
christian.plesner.hansen@gmail.com2bc58ef2009-09-22 10:00:30 +00005#include "v8.h"
fschneider@chromium.orgfb144a02011-05-04 12:43:48 +00006
7#include "cctest.h"
christian.plesner.hansen@gmail.com2bc58ef2009-09-22 10:00:30 +00008#include "heap-profiler.h"
whesse@chromium.org2c186ca2010-06-16 11:32:39 +00009#include "snapshot.h"
fschneider@chromium.orgfb144a02011-05-04 12:43:48 +000010#include "utils-inl.h"
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000011#include "../include/v8-profiler.h"
christian.plesner.hansen@gmail.com2bc58ef2009-09-22 10:00:30 +000012
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000013namespace {
14
15class NamedEntriesDetector {
16 public:
17 NamedEntriesDetector()
ager@chromium.org01fe7df2010-11-10 11:59:11 +000018 : has_A2(false), has_B2(false), has_C2(false) {
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000019 }
20
ulan@chromium.org967e2702012-02-28 09:49:15 +000021 void CheckEntry(i::HeapEntry* entry) {
22 if (strcmp(entry->name(), "A2") == 0) has_A2 = true;
23 if (strcmp(entry->name(), "B2") == 0) has_B2 = true;
24 if (strcmp(entry->name(), "C2") == 0) has_C2 = true;
vegorov@chromium.org26c16f82010-08-11 13:41:03 +000025 }
26
ulan@chromium.org967e2702012-02-28 09:49:15 +000027 void CheckAllReachables(i::HeapEntry* root) {
28 i::List<i::HeapEntry*> list(10);
29 list.Add(root);
30 root->paint();
31 CheckEntry(root);
32 while (!list.is_empty()) {
33 i::HeapEntry* entry = list.RemoveLast();
34 i::Vector<i::HeapGraphEdge> children = entry->children();
35 for (int i = 0; i < children.length(); ++i) {
36 if (children[i].type() == i::HeapGraphEdge::kShortcut) continue;
37 i::HeapEntry* child = children[i].to();
38 if (!child->painted()) {
39 list.Add(child);
40 child->paint();
41 CheckEntry(child);
42 }
43 }
44 }
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000045 }
46
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000047 bool has_A2;
48 bool has_B2;
49 bool has_C2;
50};
51
52} // namespace
53
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +000054
55static const v8::HeapGraphNode* GetGlobalObject(
56 const v8::HeapSnapshot* snapshot) {
vegorov@chromium.org21b5e952010-11-23 10:24:40 +000057 CHECK_EQ(2, snapshot->GetRoot()->GetChildrenCount());
58 const v8::HeapGraphNode* global_obj =
59 snapshot->GetRoot()->GetChild(0)->GetToNode();
ricow@chromium.orgd2be9012011-06-01 06:00:58 +000060 CHECK_EQ(0, strncmp("Object", const_cast<i::HeapEntry*>(
61 reinterpret_cast<const i::HeapEntry*>(global_obj))->name(), 6));
vegorov@chromium.org21b5e952010-11-23 10:24:40 +000062 return global_obj;
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +000063}
64
65
66static const v8::HeapGraphNode* GetProperty(const v8::HeapGraphNode* node,
67 v8::HeapGraphEdge::Type type,
68 const char* name) {
69 for (int i = 0, count = node->GetChildrenCount(); i < count; ++i) {
70 const v8::HeapGraphEdge* prop = node->GetChild(i);
71 v8::String::AsciiValue prop_name(prop->GetName());
72 if (prop->GetType() == type && strcmp(name, *prop_name) == 0)
73 return prop->GetToNode();
74 }
75 return NULL;
76}
77
78
79static bool HasString(const v8::HeapGraphNode* node, const char* contents) {
80 for (int i = 0, count = node->GetChildrenCount(); i < count; ++i) {
81 const v8::HeapGraphEdge* prop = node->GetChild(i);
82 const v8::HeapGraphNode* node = prop->GetToNode();
vegorov@chromium.org26c16f82010-08-11 13:41:03 +000083 if (node->GetType() == v8::HeapGraphNode::kString) {
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +000084 v8::String::AsciiValue node_name(node->GetName());
85 if (strcmp(contents, *node_name) == 0) return true;
86 }
87 }
88 return false;
89}
90
91
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000092TEST(HeapSnapshot) {
93 v8::HandleScope scope;
ricow@chromium.org4980dff2010-07-19 08:33:45 +000094 LocalContext env2;
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000095
vegorov@chromium.org42841962010-10-18 11:18:59 +000096 CompileRun(
whesse@chromium.org2c186ca2010-06-16 11:32:39 +000097 "function A2() {}\n"
98 "function B2(x) { return function() { return typeof x; }; }\n"
99 "function C2(x) { this.x1 = x; this.x2 = x; this[1] = x; }\n"
100 "var a2 = new A2();\n"
101 "var b2_1 = new B2(a2), b2_2 = new B2(a2);\n"
102 "var c2 = new C2(a2);");
103 const v8::HeapSnapshot* snapshot_env2 =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000104 v8::HeapProfiler::TakeSnapshot(v8_str("env2"));
whesse@chromium.org2c186ca2010-06-16 11:32:39 +0000105 i::HeapSnapshot* i_snapshot_env2 =
106 const_cast<i::HeapSnapshot*>(
107 reinterpret_cast<const i::HeapSnapshot*>(snapshot_env2));
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000108 const v8::HeapGraphNode* global_env2 = GetGlobalObject(snapshot_env2);
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000109
ager@chromium.org01fe7df2010-11-10 11:59:11 +0000110 // Verify, that JS global object of env2 has '..2' properties.
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000111 const v8::HeapGraphNode* a2_node =
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000112 GetProperty(global_env2, v8::HeapGraphEdge::kShortcut, "a2");
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000113 CHECK_NE(NULL, a2_node);
114 CHECK_NE(
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000115 NULL, GetProperty(global_env2, v8::HeapGraphEdge::kShortcut, "b2_1"));
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000116 CHECK_NE(
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000117 NULL, GetProperty(global_env2, v8::HeapGraphEdge::kShortcut, "b2_2"));
118 CHECK_NE(NULL, GetProperty(global_env2, v8::HeapGraphEdge::kShortcut, "c2"));
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000119
ulan@chromium.org967e2702012-02-28 09:49:15 +0000120 // Paint all nodes reachable from global object.
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000121 NamedEntriesDetector det;
ulan@chromium.org967e2702012-02-28 09:49:15 +0000122 i_snapshot_env2->ClearPaint();
123 det.CheckAllReachables(const_cast<i::HeapEntry*>(
124 reinterpret_cast<const i::HeapEntry*>(global_env2)));
whesse@chromium.org2c186ca2010-06-16 11:32:39 +0000125 CHECK(det.has_A2);
126 CHECK(det.has_B2);
127 CHECK(det.has_C2);
whesse@chromium.org2c186ca2010-06-16 11:32:39 +0000128}
129
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000130
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000131TEST(HeapSnapshotObjectSizes) {
132 v8::HandleScope scope;
133 LocalContext env;
134
135 // -a-> X1 --a
136 // x -b-> X2 <-|
vegorov@chromium.org42841962010-10-18 11:18:59 +0000137 CompileRun(
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000138 "function X(a, b) { this.a = a; this.b = b; }\n"
139 "x = new X(new X(), new X());\n"
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000140 "(function() { x.a.a = x.b; })();");
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000141 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000142 v8::HeapProfiler::TakeSnapshot(v8_str("sizes"));
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000143 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
144 const v8::HeapGraphNode* x =
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000145 GetProperty(global, v8::HeapGraphEdge::kShortcut, "x");
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000146 CHECK_NE(NULL, x);
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000147 const v8::HeapGraphNode* x1 =
148 GetProperty(x, v8::HeapGraphEdge::kProperty, "a");
149 CHECK_NE(NULL, x1);
150 const v8::HeapGraphNode* x2 =
151 GetProperty(x, v8::HeapGraphEdge::kProperty, "b");
152 CHECK_NE(NULL, x2);
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000153
ulan@chromium.org967e2702012-02-28 09:49:15 +0000154 // Test sizes.
155 CHECK_EQ(x->GetSelfSize() * 3, x->GetRetainedSize());
156 CHECK_EQ(x1->GetSelfSize(), x1->GetRetainedSize());
157 CHECK_EQ(x2->GetSelfSize(), x2->GetRetainedSize());
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000158}
159
160
ulan@chromium.org65a89c22012-02-14 11:46:07 +0000161TEST(BoundFunctionInSnapshot) {
162 v8::HandleScope scope;
163 LocalContext env;
164 CompileRun(
165 "function myFunction(a, b) { this.a = a; this.b = b; }\n"
166 "function AAAAA() {}\n"
167 "boundFunction = myFunction.bind(new AAAAA(), 20, new Number(12)); \n");
168 const v8::HeapSnapshot* snapshot =
169 v8::HeapProfiler::TakeSnapshot(v8_str("sizes"));
170 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
171 const v8::HeapGraphNode* f =
172 GetProperty(global, v8::HeapGraphEdge::kShortcut, "boundFunction");
173 CHECK(f);
174 CHECK_EQ(v8::String::New("native_bind"), f->GetName());
175 const v8::HeapGraphNode* bindings =
176 GetProperty(f, v8::HeapGraphEdge::kInternal, "bindings");
177 CHECK_NE(NULL, bindings);
178 CHECK_EQ(v8::HeapGraphNode::kArray, bindings->GetType());
179 CHECK_EQ(4, bindings->GetChildrenCount());
180
181 const v8::HeapGraphNode* bound_this = GetProperty(
182 f, v8::HeapGraphEdge::kShortcut, "bound_this");
183 CHECK(bound_this);
184 CHECK_EQ(v8::HeapGraphNode::kObject, bound_this->GetType());
185
186 const v8::HeapGraphNode* bound_function = GetProperty(
187 f, v8::HeapGraphEdge::kShortcut, "bound_function");
188 CHECK(bound_function);
189 CHECK_EQ(v8::HeapGraphNode::kClosure, bound_function->GetType());
190
191 const v8::HeapGraphNode* bound_argument = GetProperty(
192 f, v8::HeapGraphEdge::kShortcut, "bound_argument_1");
193 CHECK(bound_argument);
194 CHECK_EQ(v8::HeapGraphNode::kObject, bound_argument->GetType());
195}
196
197
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000198TEST(HeapSnapshotEntryChildren) {
199 v8::HandleScope scope;
200 LocalContext env;
201
vegorov@chromium.org42841962010-10-18 11:18:59 +0000202 CompileRun(
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000203 "function A() { }\n"
204 "a = new A;");
205 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000206 v8::HeapProfiler::TakeSnapshot(v8_str("children"));
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000207 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
208 for (int i = 0, count = global->GetChildrenCount(); i < count; ++i) {
209 const v8::HeapGraphEdge* prop = global->GetChild(i);
210 CHECK_EQ(global, prop->GetFromNode());
211 }
212 const v8::HeapGraphNode* a =
213 GetProperty(global, v8::HeapGraphEdge::kProperty, "a");
214 CHECK_NE(NULL, a);
215 for (int i = 0, count = a->GetChildrenCount(); i < count; ++i) {
216 const v8::HeapGraphEdge* prop = a->GetChild(i);
217 CHECK_EQ(a, prop->GetFromNode());
218 }
219}
220
221
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000222TEST(HeapSnapshotCodeObjects) {
223 v8::HandleScope scope;
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000224 LocalContext env;
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000225
vegorov@chromium.org42841962010-10-18 11:18:59 +0000226 CompileRun(
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000227 "function lazy(x) { return x - 1; }\n"
228 "function compiled(x) { return x + 1; }\n"
erik.corry@gmail.com145eff52010-08-23 11:36:18 +0000229 "var anonymous = (function() { return function() { return 0; } })();\n"
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000230 "compiled(1)");
231 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000232 v8::HeapProfiler::TakeSnapshot(v8_str("code"));
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000233
234 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
235 const v8::HeapGraphNode* compiled =
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000236 GetProperty(global, v8::HeapGraphEdge::kShortcut, "compiled");
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000237 CHECK_NE(NULL, compiled);
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000238 CHECK_EQ(v8::HeapGraphNode::kClosure, compiled->GetType());
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000239 const v8::HeapGraphNode* lazy =
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000240 GetProperty(global, v8::HeapGraphEdge::kShortcut, "lazy");
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000241 CHECK_NE(NULL, lazy);
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000242 CHECK_EQ(v8::HeapGraphNode::kClosure, lazy->GetType());
erik.corry@gmail.com145eff52010-08-23 11:36:18 +0000243 const v8::HeapGraphNode* anonymous =
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000244 GetProperty(global, v8::HeapGraphEdge::kShortcut, "anonymous");
erik.corry@gmail.com145eff52010-08-23 11:36:18 +0000245 CHECK_NE(NULL, anonymous);
246 CHECK_EQ(v8::HeapGraphNode::kClosure, anonymous->GetType());
247 v8::String::AsciiValue anonymous_name(anonymous->GetName());
vegorov@chromium.org42841962010-10-18 11:18:59 +0000248 CHECK_EQ("", *anonymous_name);
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000249
250 // Find references to code.
251 const v8::HeapGraphNode* compiled_code =
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +0000252 GetProperty(compiled, v8::HeapGraphEdge::kInternal, "shared");
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000253 CHECK_NE(NULL, compiled_code);
254 const v8::HeapGraphNode* lazy_code =
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +0000255 GetProperty(lazy, v8::HeapGraphEdge::kInternal, "shared");
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000256 CHECK_NE(NULL, lazy_code);
257
258 // Verify that non-compiled code doesn't contain references to "x"
ager@chromium.org6a2b0aa2010-07-13 20:58:03 +0000259 // literal, while compiled code does. The scope info is stored in FixedArray
260 // objects attached to the SharedFunctionInfo.
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000261 bool compiled_references_x = false, lazy_references_x = false;
262 for (int i = 0, count = compiled_code->GetChildrenCount(); i < count; ++i) {
263 const v8::HeapGraphEdge* prop = compiled_code->GetChild(i);
264 const v8::HeapGraphNode* node = prop->GetToNode();
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000265 if (node->GetType() == v8::HeapGraphNode::kArray) {
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000266 if (HasString(node, "x")) {
267 compiled_references_x = true;
268 break;
269 }
270 }
271 }
272 for (int i = 0, count = lazy_code->GetChildrenCount(); i < count; ++i) {
273 const v8::HeapGraphEdge* prop = lazy_code->GetChild(i);
274 const v8::HeapGraphNode* node = prop->GetToNode();
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000275 if (node->GetType() == v8::HeapGraphNode::kArray) {
ricow@chromium.org5ad5ace2010-06-23 09:06:43 +0000276 if (HasString(node, "x")) {
277 lazy_references_x = true;
278 break;
279 }
280 }
281 }
282 CHECK(compiled_references_x);
283 CHECK(!lazy_references_x);
284}
285
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000286
vegorov@chromium.org42841962010-10-18 11:18:59 +0000287TEST(HeapSnapshotHeapNumbers) {
288 v8::HandleScope scope;
289 LocalContext env;
290 CompileRun(
291 "a = 1; // a is Smi\n"
292 "b = 2.5; // b is HeapNumber");
293 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000294 v8::HeapProfiler::TakeSnapshot(v8_str("numbers"));
vegorov@chromium.org42841962010-10-18 11:18:59 +0000295 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000296 CHECK_EQ(NULL, GetProperty(global, v8::HeapGraphEdge::kShortcut, "a"));
vegorov@chromium.org42841962010-10-18 11:18:59 +0000297 const v8::HeapGraphNode* b =
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000298 GetProperty(global, v8::HeapGraphEdge::kShortcut, "b");
vegorov@chromium.org42841962010-10-18 11:18:59 +0000299 CHECK_NE(NULL, b);
300 CHECK_EQ(v8::HeapGraphNode::kHeapNumber, b->GetType());
301}
302
erik.corry@gmail.com394dbcf2011-10-27 07:38:48 +0000303TEST(HeapSnapshotSlicedString) {
304 v8::HandleScope scope;
305 LocalContext env;
306 CompileRun(
307 "parent_string = \"123456789.123456789.123456789.123456789.123456789."
308 "123456789.123456789.123456789.123456789.123456789."
309 "123456789.123456789.123456789.123456789.123456789."
310 "123456789.123456789.123456789.123456789.123456789.\";"
311 "child_string = parent_string.slice(100);");
312 const v8::HeapSnapshot* snapshot =
313 v8::HeapProfiler::TakeSnapshot(v8_str("strings"));
314 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
315 const v8::HeapGraphNode* parent_string =
316 GetProperty(global, v8::HeapGraphEdge::kShortcut, "parent_string");
317 CHECK_NE(NULL, parent_string);
318 const v8::HeapGraphNode* child_string =
319 GetProperty(global, v8::HeapGraphEdge::kShortcut, "child_string");
320 CHECK_NE(NULL, child_string);
321 const v8::HeapGraphNode* parent =
322 GetProperty(child_string, v8::HeapGraphEdge::kInternal, "parent");
323 CHECK_EQ(parent_string, parent);
324}
vegorov@chromium.org42841962010-10-18 11:18:59 +0000325
326TEST(HeapSnapshotInternalReferences) {
327 v8::HandleScope scope;
328 v8::Local<v8::ObjectTemplate> global_template = v8::ObjectTemplate::New();
329 global_template->SetInternalFieldCount(2);
330 LocalContext env(NULL, global_template);
331 v8::Handle<v8::Object> global_proxy = env->Global();
332 v8::Handle<v8::Object> global = global_proxy->GetPrototype().As<v8::Object>();
333 CHECK_EQ(2, global->InternalFieldCount());
334 v8::Local<v8::Object> obj = v8::Object::New();
335 global->SetInternalField(0, v8_num(17));
336 global->SetInternalField(1, obj);
337 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000338 v8::HeapProfiler::TakeSnapshot(v8_str("internals"));
vegorov@chromium.org42841962010-10-18 11:18:59 +0000339 const v8::HeapGraphNode* global_node = GetGlobalObject(snapshot);
340 // The first reference will not present, because it's a Smi.
341 CHECK_EQ(NULL, GetProperty(global_node, v8::HeapGraphEdge::kInternal, "0"));
342 // The second reference is to an object.
343 CHECK_NE(NULL, GetProperty(global_node, v8::HeapGraphEdge::kInternal, "1"));
344}
345
346
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000347// Trying to introduce a check helper for uint32_t causes many
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000348// overloading ambiguities, so it seems easier just to cast
349// them to a signed type.
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000350#define CHECK_EQ_SNAPSHOT_OBJECT_ID(a, b) \
351 CHECK_EQ(static_cast<int32_t>(a), static_cast<int32_t>(b))
352#define CHECK_NE_SNAPSHOT_OBJECT_ID(a, b) \
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000353 CHECK((a) != (b)) // NOLINT
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000354
ulan@chromium.org6ff65142012-03-21 09:52:17 +0000355TEST(HeapEntryIdsAndArrayShift) {
356 v8::HandleScope scope;
357 LocalContext env;
358
359 CompileRun(
360 "function AnObject() {\n"
361 " this.first = 'first';\n"
362 " this.second = 'second';\n"
363 "}\n"
364 "var a = new Array();\n"
365 "for (var i = 0; i < 10; ++i)\n"
366 " a.push(new AnObject());\n");
367 const v8::HeapSnapshot* snapshot1 =
368 v8::HeapProfiler::TakeSnapshot(v8_str("s1"));
369
370 CompileRun(
371 "for (var i = 0; i < 1; ++i)\n"
372 " a.shift();\n");
373
374 HEAP->CollectAllGarbage(i::Heap::kNoGCFlags);
375
376 const v8::HeapSnapshot* snapshot2 =
377 v8::HeapProfiler::TakeSnapshot(v8_str("s2"));
378
379 const v8::HeapGraphNode* global1 = GetGlobalObject(snapshot1);
380 const v8::HeapGraphNode* global2 = GetGlobalObject(snapshot2);
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000381 CHECK_NE_SNAPSHOT_OBJECT_ID(0, global1->GetId());
382 CHECK_EQ_SNAPSHOT_OBJECT_ID(global1->GetId(), global2->GetId());
ulan@chromium.org6ff65142012-03-21 09:52:17 +0000383
384 const v8::HeapGraphNode* a1 =
385 GetProperty(global1, v8::HeapGraphEdge::kProperty, "a");
386 CHECK_NE(NULL, a1);
387 const v8::HeapGraphNode* e1 =
388 GetProperty(a1, v8::HeapGraphEdge::kHidden, "1");
389 CHECK_NE(NULL, e1);
390 const v8::HeapGraphNode* k1 =
391 GetProperty(e1, v8::HeapGraphEdge::kInternal, "elements");
392 CHECK_NE(NULL, k1);
393 const v8::HeapGraphNode* a2 =
394 GetProperty(global2, v8::HeapGraphEdge::kProperty, "a");
395 CHECK_NE(NULL, a2);
396 const v8::HeapGraphNode* e2 =
397 GetProperty(a2, v8::HeapGraphEdge::kHidden, "1");
398 CHECK_NE(NULL, e2);
399 const v8::HeapGraphNode* k2 =
400 GetProperty(e2, v8::HeapGraphEdge::kInternal, "elements");
401 CHECK_NE(NULL, k2);
402
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000403 CHECK_EQ_SNAPSHOT_OBJECT_ID(a1->GetId(), a2->GetId());
404 CHECK_EQ_SNAPSHOT_OBJECT_ID(e1->GetId(), e2->GetId());
405 CHECK_EQ_SNAPSHOT_OBJECT_ID(k1->GetId(), k2->GetId());
ulan@chromium.org6ff65142012-03-21 09:52:17 +0000406}
407
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000408TEST(HeapEntryIdsAndGC) {
409 v8::HandleScope scope;
410 LocalContext env;
411
vegorov@chromium.org42841962010-10-18 11:18:59 +0000412 CompileRun(
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000413 "function A() {}\n"
414 "function B(x) { this.x = x; }\n"
415 "var a = new A();\n"
416 "var b = new B(a);");
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000417 v8::Local<v8::String> s1_str = v8_str("s1");
418 v8::Local<v8::String> s2_str = v8_str("s2");
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000419 const v8::HeapSnapshot* snapshot1 =
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000420 v8::HeapProfiler::TakeSnapshot(s1_str);
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000421
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000422 HEAP->CollectAllGarbage(i::Heap::kNoGCFlags);
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000423
424 const v8::HeapSnapshot* snapshot2 =
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000425 v8::HeapProfiler::TakeSnapshot(s2_str);
426
427 CHECK_GT(snapshot1->GetMaxSnapshotJSObjectId(), 7000);
428 CHECK(snapshot1->GetMaxSnapshotJSObjectId() <=
429 snapshot2->GetMaxSnapshotJSObjectId());
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000430
431 const v8::HeapGraphNode* global1 = GetGlobalObject(snapshot1);
432 const v8::HeapGraphNode* global2 = GetGlobalObject(snapshot2);
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000433 CHECK_NE_SNAPSHOT_OBJECT_ID(0, global1->GetId());
434 CHECK_EQ_SNAPSHOT_OBJECT_ID(global1->GetId(), global2->GetId());
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000435 const v8::HeapGraphNode* A1 =
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000436 GetProperty(global1, v8::HeapGraphEdge::kProperty, "A");
437 CHECK_NE(NULL, A1);
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000438 const v8::HeapGraphNode* A2 =
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000439 GetProperty(global2, v8::HeapGraphEdge::kProperty, "A");
440 CHECK_NE(NULL, A2);
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000441 CHECK_NE_SNAPSHOT_OBJECT_ID(0, A1->GetId());
442 CHECK_EQ_SNAPSHOT_OBJECT_ID(A1->GetId(), A2->GetId());
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000443 const v8::HeapGraphNode* B1 =
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000444 GetProperty(global1, v8::HeapGraphEdge::kProperty, "B");
445 CHECK_NE(NULL, B1);
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000446 const v8::HeapGraphNode* B2 =
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000447 GetProperty(global2, v8::HeapGraphEdge::kProperty, "B");
448 CHECK_NE(NULL, B2);
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000449 CHECK_NE_SNAPSHOT_OBJECT_ID(0, B1->GetId());
450 CHECK_EQ_SNAPSHOT_OBJECT_ID(B1->GetId(), B2->GetId());
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000451 const v8::HeapGraphNode* a1 =
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000452 GetProperty(global1, v8::HeapGraphEdge::kProperty, "a");
453 CHECK_NE(NULL, a1);
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000454 const v8::HeapGraphNode* a2 =
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000455 GetProperty(global2, v8::HeapGraphEdge::kProperty, "a");
456 CHECK_NE(NULL, a2);
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000457 CHECK_NE_SNAPSHOT_OBJECT_ID(0, a1->GetId());
458 CHECK_EQ_SNAPSHOT_OBJECT_ID(a1->GetId(), a2->GetId());
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000459 const v8::HeapGraphNode* b1 =
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000460 GetProperty(global1, v8::HeapGraphEdge::kProperty, "b");
461 CHECK_NE(NULL, b1);
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000462 const v8::HeapGraphNode* b2 =
vegorov@chromium.org26c16f82010-08-11 13:41:03 +0000463 GetProperty(global2, v8::HeapGraphEdge::kProperty, "b");
464 CHECK_NE(NULL, b2);
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000465 CHECK_NE_SNAPSHOT_OBJECT_ID(0, b1->GetId());
466 CHECK_EQ_SNAPSHOT_OBJECT_ID(b1->GetId(), b2->GetId());
ricow@chromium.org4980dff2010-07-19 08:33:45 +0000467}
468
469
ricow@chromium.orgeb7c1442010-10-04 08:54:21 +0000470TEST(HeapSnapshotRootPreservedAfterSorting) {
471 v8::HandleScope scope;
472 LocalContext env;
473 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000474 v8::HeapProfiler::TakeSnapshot(v8_str("s"));
ricow@chromium.orgeb7c1442010-10-04 08:54:21 +0000475 const v8::HeapGraphNode* root1 = snapshot->GetRoot();
476 const_cast<i::HeapSnapshot*>(reinterpret_cast<const i::HeapSnapshot*>(
477 snapshot))->GetSortedEntriesList();
478 const v8::HeapGraphNode* root2 = snapshot->GetRoot();
479 CHECK_EQ(root1, root2);
480}
481
482
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000483TEST(HeapEntryDominator) {
484 // The graph looks like this:
485 //
486 // -> node1
487 // a |^
488 // -> node5 ba
489 // a v|
490 // node6 -> node2
491 // b a |^
492 // -> node4 ba
493 // b v|
494 // -> node3
495 //
496 // The dominator for all nodes is node6.
497
498 v8::HandleScope scope;
499 LocalContext env;
500
501 CompileRun(
502 "function X(a, b) { this.a = a; this.b = b; }\n"
503 "node6 = new X(new X(new X()), new X(new X(),new X()));\n"
504 "(function(){\n"
505 "node6.a.a.b = node6.b.a; // node1 -> node2\n"
506 "node6.b.a.a = node6.a.a; // node2 -> node1\n"
507 "node6.b.a.b = node6.b.b; // node2 -> node3\n"
508 "node6.b.b.a = node6.b.a; // node3 -> node2\n"
509 "})();");
510
511 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000512 v8::HeapProfiler::TakeSnapshot(v8_str("dominators"));
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000513
514 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
515 CHECK_NE(NULL, global);
516 const v8::HeapGraphNode* node6 =
517 GetProperty(global, v8::HeapGraphEdge::kShortcut, "node6");
518 CHECK_NE(NULL, node6);
519 const v8::HeapGraphNode* node5 =
520 GetProperty(node6, v8::HeapGraphEdge::kProperty, "a");
521 CHECK_NE(NULL, node5);
522 const v8::HeapGraphNode* node4 =
523 GetProperty(node6, v8::HeapGraphEdge::kProperty, "b");
524 CHECK_NE(NULL, node4);
525 const v8::HeapGraphNode* node3 =
526 GetProperty(node4, v8::HeapGraphEdge::kProperty, "b");
527 CHECK_NE(NULL, node3);
528 const v8::HeapGraphNode* node2 =
529 GetProperty(node4, v8::HeapGraphEdge::kProperty, "a");
530 CHECK_NE(NULL, node2);
531 const v8::HeapGraphNode* node1 =
532 GetProperty(node5, v8::HeapGraphEdge::kProperty, "a");
533 CHECK_NE(NULL, node1);
534
535 CHECK_EQ(node6, node1->GetDominatorNode());
536 CHECK_EQ(node6, node2->GetDominatorNode());
537 CHECK_EQ(node6, node3->GetDominatorNode());
538 CHECK_EQ(node6, node4->GetDominatorNode());
539 CHECK_EQ(node6, node5->GetDominatorNode());
540}
541
542
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000543namespace {
544
545class TestJSONStream : public v8::OutputStream {
546 public:
547 TestJSONStream() : eos_signaled_(0), abort_countdown_(-1) {}
548 explicit TestJSONStream(int abort_countdown)
549 : eos_signaled_(0), abort_countdown_(abort_countdown) {}
550 virtual ~TestJSONStream() {}
551 virtual void EndOfStream() { ++eos_signaled_; }
552 virtual WriteResult WriteAsciiChunk(char* buffer, int chars_written) {
553 if (abort_countdown_ > 0) --abort_countdown_;
554 if (abort_countdown_ == 0) return kAbort;
555 CHECK_GT(chars_written, 0);
556 i::Vector<char> chunk = buffer_.AddBlock(chars_written, '\0');
557 memcpy(chunk.start(), buffer, chars_written);
558 return kContinue;
559 }
jkummerow@chromium.org28faa982012-04-13 09:58:30 +0000560 virtual WriteResult WriteUint32Chunk(uint32_t* buffer, int chars_written) {
561 ASSERT(false);
562 return kAbort;
563 }
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000564 void WriteTo(i::Vector<char> dest) { buffer_.WriteTo(dest); }
565 int eos_signaled() { return eos_signaled_; }
566 int size() { return buffer_.size(); }
jkummerow@chromium.org28faa982012-04-13 09:58:30 +0000567
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000568 private:
569 i::Collector<char> buffer_;
570 int eos_signaled_;
571 int abort_countdown_;
572};
573
574class AsciiResource: public v8::String::ExternalAsciiStringResource {
575 public:
576 explicit AsciiResource(i::Vector<char> string): data_(string.start()) {
577 length_ = string.length();
578 }
579 virtual const char* data() const { return data_; }
580 virtual size_t length() const { return length_; }
581 private:
582 const char* data_;
583 size_t length_;
584};
585
586} // namespace
587
588TEST(HeapSnapshotJSONSerialization) {
589 v8::HandleScope scope;
590 LocalContext env;
591
592#define STRING_LITERAL_FOR_TEST \
593 "\"String \\n\\r\\u0008\\u0081\\u0101\\u0801\\u8001\""
vegorov@chromium.org42841962010-10-18 11:18:59 +0000594 CompileRun(
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000595 "function A(s) { this.s = s; }\n"
596 "function B(x) { this.x = x; }\n"
597 "var a = new A(" STRING_LITERAL_FOR_TEST ");\n"
598 "var b = new B(a);");
599 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000600 v8::HeapProfiler::TakeSnapshot(v8_str("json"));
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000601 TestJSONStream stream;
602 snapshot->Serialize(&stream, v8::HeapSnapshot::kJSON);
603 CHECK_GT(stream.size(), 0);
604 CHECK_EQ(1, stream.eos_signaled());
605 i::ScopedVector<char> json(stream.size());
606 stream.WriteTo(json);
607
608 // Verify that snapshot string is valid JSON.
609 AsciiResource json_res(json);
610 v8::Local<v8::String> json_string = v8::String::NewExternal(&json_res);
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000611 env->Global()->Set(v8_str("json_snapshot"), json_string);
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000612 v8::Local<v8::Value> snapshot_parse_result = CompileRun(
613 "var parsed = JSON.parse(json_snapshot); true;");
614 CHECK(!snapshot_parse_result.IsEmpty());
615
616 // Verify that snapshot object has required fields.
617 v8::Local<v8::Object> parsed_snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000618 env->Global()->Get(v8_str("parsed"))->ToObject();
619 CHECK(parsed_snapshot->Has(v8_str("snapshot")));
620 CHECK(parsed_snapshot->Has(v8_str("nodes")));
621 CHECK(parsed_snapshot->Has(v8_str("strings")));
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000622
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000623 // Get node and edge "member" offsets.
624 v8::Local<v8::Value> meta_analysis_result = CompileRun(
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000625 "var parsed_meta = parsed.nodes[0];\n"
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000626 "var children_count_offset ="
627 " parsed_meta.fields.indexOf('children_count');\n"
628 "var children_offset ="
629 " parsed_meta.fields.indexOf('children');\n"
630 "var children_meta ="
631 " parsed_meta.types[children_offset];\n"
632 "var child_fields_count = children_meta.fields.length;\n"
633 "var child_type_offset ="
634 " children_meta.fields.indexOf('type');\n"
635 "var child_name_offset ="
636 " children_meta.fields.indexOf('name_or_index');\n"
637 "var child_to_node_offset ="
638 " children_meta.fields.indexOf('to_node');\n"
639 "var property_type ="
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000640 " children_meta.types[child_type_offset].indexOf('property');\n"
641 "var shortcut_type ="
642 " children_meta.types[child_type_offset].indexOf('shortcut');");
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000643 CHECK(!meta_analysis_result.IsEmpty());
644
645 // A helper function for processing encoded nodes.
646 CompileRun(
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000647 "function GetChildPosByProperty(pos, prop_name, prop_type) {\n"
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000648 " var nodes = parsed.nodes;\n"
649 " var strings = parsed.strings;\n"
650 " for (var i = 0,\n"
651 " count = nodes[pos + children_count_offset] * child_fields_count;\n"
652 " i < count; i += child_fields_count) {\n"
653 " var child_pos = pos + children_offset + i;\n"
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000654 " if (nodes[child_pos + child_type_offset] === prop_type\n"
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000655 " && strings[nodes[child_pos + child_name_offset]] === prop_name)\n"
656 " return nodes[child_pos + child_to_node_offset];\n"
657 " }\n"
658 " return null;\n"
659 "}\n");
660 // Get the string index using the path: <root> -> <global>.b.x.s
661 v8::Local<v8::Value> string_obj_pos_val = CompileRun(
662 "GetChildPosByProperty(\n"
663 " GetChildPosByProperty(\n"
664 " GetChildPosByProperty("
vegorov@chromium.org21b5e952010-11-23 10:24:40 +0000665 " parsed.nodes[1 + children_offset + child_to_node_offset],"
666 " \"b\",shortcut_type),\n"
667 " \"x\", property_type),"
668 " \"s\", property_type)");
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000669 CHECK(!string_obj_pos_val.IsEmpty());
670 int string_obj_pos =
671 static_cast<int>(string_obj_pos_val->ToNumber()->Value());
672 v8::Local<v8::Object> nodes_array =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000673 parsed_snapshot->Get(v8_str("nodes"))->ToObject();
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000674 int string_index = static_cast<int>(
675 nodes_array->Get(string_obj_pos + 1)->ToNumber()->Value());
676 CHECK_GT(string_index, 0);
677 v8::Local<v8::Object> strings_array =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000678 parsed_snapshot->Get(v8_str("strings"))->ToObject();
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000679 v8::Local<v8::String> string = strings_array->Get(string_index)->ToString();
680 v8::Local<v8::String> ref_string =
681 CompileRun(STRING_LITERAL_FOR_TEST)->ToString();
682#undef STRING_LITERAL_FOR_TEST
683 CHECK_EQ(*v8::String::Utf8Value(ref_string),
684 *v8::String::Utf8Value(string));
685}
686
687
688TEST(HeapSnapshotJSONSerializationAborting) {
689 v8::HandleScope scope;
690 LocalContext env;
691 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000692 v8::HeapProfiler::TakeSnapshot(v8_str("abort"));
erik.corry@gmail.comd88afa22010-09-15 12:33:05 +0000693 TestJSONStream stream(5);
694 snapshot->Serialize(&stream, v8::HeapSnapshot::kJSON);
695 CHECK_GT(stream.size(), 0);
696 CHECK_EQ(0, stream.eos_signaled());
697}
698
jkummerow@chromium.org28faa982012-04-13 09:58:30 +0000699namespace {
700
701class TestStatsStream : public v8::OutputStream {
702 public:
703 TestStatsStream()
704 : eos_signaled_(0),
705 numbers_written_(0),
706 entries_count_(0),
707 intervals_count_(0),
708 first_interval_index_(-1) { }
709 TestStatsStream(const TestStatsStream& stream)
710 : v8::OutputStream(stream),
711 eos_signaled_(stream.eos_signaled_),
712 numbers_written_(stream.numbers_written_),
713 entries_count_(stream.entries_count_),
714 intervals_count_(stream.intervals_count_),
715 first_interval_index_(stream.first_interval_index_) { }
716 virtual ~TestStatsStream() {}
717 virtual void EndOfStream() { ++eos_signaled_; }
718 virtual WriteResult WriteAsciiChunk(char* buffer, int chars_written) {
719 ASSERT(false);
720 return kAbort;
721 }
722 virtual WriteResult WriteUint32Chunk(uint32_t* buffer, int numbers_written) {
723 ++intervals_count_;
724 ASSERT(numbers_written);
725 numbers_written_ += numbers_written;
726 entries_count_ = 0;
727 if (first_interval_index_ == -1 && numbers_written != 0)
728 first_interval_index_ = buffer[0];
729 for (int i = 1; i < numbers_written; i += 2)
730 entries_count_ += buffer[i];
731
732 return kContinue;
733 }
734 int eos_signaled() { return eos_signaled_; }
735 int numbers_written() { return numbers_written_; }
736 uint32_t entries_count() const { return entries_count_; }
737 int intervals_count() const { return intervals_count_; }
738 int first_interval_index() const { return first_interval_index_; }
739
740 private:
741 int eos_signaled_;
742 int numbers_written_;
743 uint32_t entries_count_;
744 int intervals_count_;
745 int first_interval_index_;
746};
747
748} // namespace
749
750static TestStatsStream GetHeapStatsUpdate() {
751 TestStatsStream stream;
752 v8::HeapProfiler::PushHeapObjectsStats(&stream);
753 CHECK_EQ(1, stream.eos_signaled());
754 return stream;
755}
756
757
758TEST(HeapSnapshotObjectsStats) {
759 v8::HandleScope scope;
760 LocalContext env;
761
762 v8::HeapProfiler::StartHeapObjectsTracking();
763 // We have to call GC 5 times. In other case the garbage will be
764 // the reason of flakiness.
765 for (int i = 0; i < 5; ++i) {
766 HEAP->CollectAllGarbage(i::Heap::kNoGCFlags);
767 }
768
769 {
770 // Single chunk of data expected in update. Initial data.
771 TestStatsStream stats_update = GetHeapStatsUpdate();
772 CHECK_EQ(1, stats_update.intervals_count());
773 CHECK_EQ(2, stats_update.numbers_written());
774 CHECK_EQ(0, stats_update.first_interval_index());
775 }
776
777 // No data expected in update because nothing has happened.
778 CHECK_EQ(0, GetHeapStatsUpdate().numbers_written());
779 {
780 v8::HandleScope inner_scope_1;
781 v8::Local<v8::String> string1 = v8_str("string1");
782 {
783 // Single chunk of data with one new entry expected in update.
784 TestStatsStream stats_update = GetHeapStatsUpdate();
785 CHECK_EQ(1, stats_update.intervals_count());
786 CHECK_EQ(2, stats_update.numbers_written());
787 CHECK_EQ(1, stats_update.entries_count());
788 CHECK_EQ(2, stats_update.first_interval_index());
789 }
790
791 // No data expected in update because nothing happened.
792 CHECK_EQ(0, GetHeapStatsUpdate().numbers_written());
793
794 {
795 v8::HandleScope inner_scope_2;
796 v8::Local<v8::String> string2 = v8_str("string2");
797
798 {
799 v8::HandleScope inner_scope_3;
800 v8::Handle<v8::String> string3 = v8::String::New("string3");
801 v8::Handle<v8::String> string4 = v8::String::New("string4");
802
803 {
804 // Single chunk of data with three new entries expected in update.
805 TestStatsStream stats_update = GetHeapStatsUpdate();
806 CHECK_EQ(1, stats_update.intervals_count());
807 CHECK_EQ(2, stats_update.numbers_written());
808 CHECK_EQ(3, stats_update.entries_count());
809 CHECK_EQ(4, stats_update.first_interval_index());
810 }
811 }
812
813 {
814 // Single chunk of data with two left entries expected in update.
815 TestStatsStream stats_update = GetHeapStatsUpdate();
816 CHECK_EQ(1, stats_update.intervals_count());
817 CHECK_EQ(2, stats_update.numbers_written());
818 CHECK_EQ(1, stats_update.entries_count());
819 // Two strings from forth interval were released.
820 CHECK_EQ(4, stats_update.first_interval_index());
821 }
822 }
823
824 {
825 // Single chunk of data with 0 left entries expected in update.
826 TestStatsStream stats_update = GetHeapStatsUpdate();
827 CHECK_EQ(1, stats_update.intervals_count());
828 CHECK_EQ(2, stats_update.numbers_written());
829 CHECK_EQ(0, stats_update.entries_count());
830 // The last string from forth interval was released.
831 CHECK_EQ(4, stats_update.first_interval_index());
832 }
833 }
834 {
835 // Single chunk of data with 0 left entries expected in update.
836 TestStatsStream stats_update = GetHeapStatsUpdate();
837 CHECK_EQ(1, stats_update.intervals_count());
838 CHECK_EQ(2, stats_update.numbers_written());
839 CHECK_EQ(0, stats_update.entries_count());
840 // The only string from the second interval was released.
841 CHECK_EQ(2, stats_update.first_interval_index());
842 }
843
844 v8::HeapProfiler::StopHeapObjectsTracking();
845}
846
ager@chromium.orgbeb25712010-11-29 08:02:25 +0000847
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000848static void CheckChildrenIds(const v8::HeapSnapshot* snapshot,
849 const v8::HeapGraphNode* node,
850 int level, int max_level) {
851 if (level > max_level) return;
852 CHECK_EQ(node, snapshot->GetNodeById(node->GetId()));
853 for (int i = 0, count = node->GetChildrenCount(); i < count; ++i) {
854 const v8::HeapGraphEdge* prop = node->GetChild(i);
855 const v8::HeapGraphNode* child =
856 snapshot->GetNodeById(prop->GetToNode()->GetId());
jkummerow@chromium.org1456e702012-03-30 08:38:13 +0000857 CHECK_EQ_SNAPSHOT_OBJECT_ID(prop->GetToNode()->GetId(), child->GetId());
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000858 CHECK_EQ(prop->GetToNode(), child);
859 CheckChildrenIds(snapshot, child, level + 1, max_level);
860 }
861}
862
863
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000864TEST(HeapSnapshotGetNodeById) {
865 v8::HandleScope scope;
866 LocalContext env;
867
868 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000869 v8::HeapProfiler::TakeSnapshot(v8_str("id"));
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000870 const v8::HeapGraphNode* root = snapshot->GetRoot();
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +0000871 CheckChildrenIds(snapshot, root, 0, 3);
kasperl@chromium.orga5551262010-12-07 12:49:48 +0000872 // Check a big id, which should not exist yet.
873 CHECK_EQ(NULL, snapshot->GetNodeById(0x1000000UL));
874}
875
ager@chromium.org5f0c45f2010-12-17 08:51:21 +0000876
877namespace {
878
879class TestActivityControl : public v8::ActivityControl {
880 public:
881 explicit TestActivityControl(int abort_count)
882 : done_(0), total_(0), abort_count_(abort_count) {}
883 ControlOption ReportProgressValue(int done, int total) {
884 done_ = done;
885 total_ = total;
886 return --abort_count_ != 0 ? kContinue : kAbort;
887 }
888 int done() { return done_; }
889 int total() { return total_; }
890
891 private:
892 int done_;
893 int total_;
894 int abort_count_;
895};
896}
897
898TEST(TakeHeapSnapshotAborting) {
899 v8::HandleScope scope;
900 LocalContext env;
901
902 const int snapshots_count = v8::HeapProfiler::GetSnapshotsCount();
yangguo@chromium.orga7d3df92012-02-27 11:46:55 +0000903 TestActivityControl aborting_control(1);
ager@chromium.org5f0c45f2010-12-17 08:51:21 +0000904 const v8::HeapSnapshot* no_snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000905 v8::HeapProfiler::TakeSnapshot(v8_str("abort"),
ager@chromium.org5f0c45f2010-12-17 08:51:21 +0000906 v8::HeapSnapshot::kFull,
907 &aborting_control);
908 CHECK_EQ(NULL, no_snapshot);
909 CHECK_EQ(snapshots_count, v8::HeapProfiler::GetSnapshotsCount());
910 CHECK_GT(aborting_control.total(), aborting_control.done());
911
912 TestActivityControl control(-1); // Don't abort.
913 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +0000914 v8::HeapProfiler::TakeSnapshot(v8_str("full"),
ager@chromium.org5f0c45f2010-12-17 08:51:21 +0000915 v8::HeapSnapshot::kFull,
916 &control);
917 CHECK_NE(NULL, snapshot);
918 CHECK_EQ(snapshots_count + 1, v8::HeapProfiler::GetSnapshotsCount());
919 CHECK_EQ(control.total(), control.done());
920 CHECK_GT(control.total(), 0);
921}
922
whesse@chromium.orgb08986c2011-03-14 16:13:42 +0000923
924namespace {
925
926class TestRetainedObjectInfo : public v8::RetainedObjectInfo {
927 public:
928 TestRetainedObjectInfo(int hash,
danno@chromium.orgfa458e42012-02-01 10:48:36 +0000929 const char* group_label,
whesse@chromium.orgb08986c2011-03-14 16:13:42 +0000930 const char* label,
931 intptr_t element_count = -1,
932 intptr_t size = -1)
933 : disposed_(false),
934 hash_(hash),
danno@chromium.orgfa458e42012-02-01 10:48:36 +0000935 group_label_(group_label),
whesse@chromium.orgb08986c2011-03-14 16:13:42 +0000936 label_(label),
937 element_count_(element_count),
938 size_(size) {
939 instances.Add(this);
940 }
941 virtual ~TestRetainedObjectInfo() {}
942 virtual void Dispose() {
943 CHECK(!disposed_);
944 disposed_ = true;
945 }
946 virtual bool IsEquivalent(RetainedObjectInfo* other) {
947 return GetHash() == other->GetHash();
948 }
949 virtual intptr_t GetHash() { return hash_; }
danno@chromium.orgfa458e42012-02-01 10:48:36 +0000950 virtual const char* GetGroupLabel() { return group_label_; }
whesse@chromium.orgb08986c2011-03-14 16:13:42 +0000951 virtual const char* GetLabel() { return label_; }
952 virtual intptr_t GetElementCount() { return element_count_; }
953 virtual intptr_t GetSizeInBytes() { return size_; }
954 bool disposed() { return disposed_; }
955
956 static v8::RetainedObjectInfo* WrapperInfoCallback(
957 uint16_t class_id, v8::Handle<v8::Value> wrapper) {
958 if (class_id == 1) {
959 if (wrapper->IsString()) {
960 v8::String::AsciiValue ascii(wrapper);
961 if (strcmp(*ascii, "AAA") == 0)
danno@chromium.orgfa458e42012-02-01 10:48:36 +0000962 return new TestRetainedObjectInfo(1, "aaa-group", "aaa", 100);
whesse@chromium.orgb08986c2011-03-14 16:13:42 +0000963 else if (strcmp(*ascii, "BBB") == 0)
danno@chromium.orgfa458e42012-02-01 10:48:36 +0000964 return new TestRetainedObjectInfo(1, "aaa-group", "aaa", 100);
whesse@chromium.orgb08986c2011-03-14 16:13:42 +0000965 }
966 } else if (class_id == 2) {
967 if (wrapper->IsString()) {
968 v8::String::AsciiValue ascii(wrapper);
969 if (strcmp(*ascii, "CCC") == 0)
danno@chromium.orgfa458e42012-02-01 10:48:36 +0000970 return new TestRetainedObjectInfo(2, "ccc-group", "ccc");
whesse@chromium.orgb08986c2011-03-14 16:13:42 +0000971 }
972 }
973 CHECK(false);
974 return NULL;
975 }
976
977 static i::List<TestRetainedObjectInfo*> instances;
978
979 private:
980 bool disposed_;
981 int category_;
982 int hash_;
danno@chromium.orgfa458e42012-02-01 10:48:36 +0000983 const char* group_label_;
whesse@chromium.orgb08986c2011-03-14 16:13:42 +0000984 const char* label_;
985 intptr_t element_count_;
986 intptr_t size_;
987};
988
989
990i::List<TestRetainedObjectInfo*> TestRetainedObjectInfo::instances;
991}
992
993
994static const v8::HeapGraphNode* GetNode(const v8::HeapGraphNode* parent,
995 v8::HeapGraphNode::Type type,
996 const char* name) {
997 for (int i = 0, count = parent->GetChildrenCount(); i < count; ++i) {
998 const v8::HeapGraphNode* node = parent->GetChild(i)->GetToNode();
999 if (node->GetType() == type && strcmp(name,
1000 const_cast<i::HeapEntry*>(
1001 reinterpret_cast<const i::HeapEntry*>(node))->name()) == 0) {
1002 return node;
1003 }
1004 }
1005 return NULL;
1006}
1007
1008
1009TEST(HeapSnapshotRetainedObjectInfo) {
1010 v8::HandleScope scope;
1011 LocalContext env;
1012
1013 v8::HeapProfiler::DefineWrapperClass(
1014 1, TestRetainedObjectInfo::WrapperInfoCallback);
1015 v8::HeapProfiler::DefineWrapperClass(
1016 2, TestRetainedObjectInfo::WrapperInfoCallback);
1017 v8::Persistent<v8::String> p_AAA =
1018 v8::Persistent<v8::String>::New(v8_str("AAA"));
1019 p_AAA.SetWrapperClassId(1);
1020 v8::Persistent<v8::String> p_BBB =
1021 v8::Persistent<v8::String>::New(v8_str("BBB"));
1022 p_BBB.SetWrapperClassId(1);
1023 v8::Persistent<v8::String> p_CCC =
1024 v8::Persistent<v8::String>::New(v8_str("CCC"));
1025 p_CCC.SetWrapperClassId(2);
1026 CHECK_EQ(0, TestRetainedObjectInfo::instances.length());
1027 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001028 v8::HeapProfiler::TakeSnapshot(v8_str("retained"));
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001029
1030 CHECK_EQ(3, TestRetainedObjectInfo::instances.length());
1031 for (int i = 0; i < TestRetainedObjectInfo::instances.length(); ++i) {
1032 CHECK(TestRetainedObjectInfo::instances[i]->disposed());
1033 delete TestRetainedObjectInfo::instances[i];
1034 }
1035
danno@chromium.orgfa458e42012-02-01 10:48:36 +00001036 const v8::HeapGraphNode* native_group_aaa = GetNode(
jkummerow@chromium.orgab7dad42012-02-07 12:07:34 +00001037 snapshot->GetRoot(), v8::HeapGraphNode::kSynthetic, "aaa-group");
danno@chromium.orgfa458e42012-02-01 10:48:36 +00001038 CHECK_NE(NULL, native_group_aaa);
1039 CHECK_EQ(1, native_group_aaa->GetChildrenCount());
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001040 const v8::HeapGraphNode* aaa = GetNode(
danno@chromium.orgfa458e42012-02-01 10:48:36 +00001041 native_group_aaa, v8::HeapGraphNode::kNative, "aaa / 100 entries");
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001042 CHECK_NE(NULL, aaa);
danno@chromium.orgfa458e42012-02-01 10:48:36 +00001043 CHECK_EQ(2, aaa->GetChildrenCount());
1044
1045 const v8::HeapGraphNode* native_group_ccc = GetNode(
jkummerow@chromium.orgab7dad42012-02-07 12:07:34 +00001046 snapshot->GetRoot(), v8::HeapGraphNode::kSynthetic, "ccc-group");
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001047 const v8::HeapGraphNode* ccc = GetNode(
danno@chromium.orgfa458e42012-02-01 10:48:36 +00001048 native_group_ccc, v8::HeapGraphNode::kNative, "ccc");
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001049 CHECK_NE(NULL, ccc);
1050
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001051 const v8::HeapGraphNode* n_AAA = GetNode(
1052 aaa, v8::HeapGraphNode::kString, "AAA");
1053 CHECK_NE(NULL, n_AAA);
1054 const v8::HeapGraphNode* n_BBB = GetNode(
1055 aaa, v8::HeapGraphNode::kString, "BBB");
1056 CHECK_NE(NULL, n_BBB);
1057 CHECK_EQ(1, ccc->GetChildrenCount());
1058 const v8::HeapGraphNode* n_CCC = GetNode(
1059 ccc, v8::HeapGraphNode::kString, "CCC");
1060 CHECK_NE(NULL, n_CCC);
1061
kmillikin@chromium.orgc36ce6e2011-04-04 08:25:31 +00001062 CHECK_EQ(aaa, GetProperty(n_AAA, v8::HeapGraphEdge::kInternal, "native"));
1063 CHECK_EQ(aaa, GetProperty(n_BBB, v8::HeapGraphEdge::kInternal, "native"));
1064 CHECK_EQ(ccc, GetProperty(n_CCC, v8::HeapGraphEdge::kInternal, "native"));
whesse@chromium.orgb08986c2011-03-14 16:13:42 +00001065}
1066
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00001067
rossberg@chromium.org994edf62012-02-06 10:12:55 +00001068class GraphWithImplicitRefs {
1069 public:
1070 static const int kObjectsCount = 4;
1071 explicit GraphWithImplicitRefs(LocalContext* env) {
1072 CHECK_EQ(NULL, instance_);
1073 instance_ = this;
1074 for (int i = 0; i < kObjectsCount; i++) {
1075 objects_[i] = v8::Persistent<v8::Object>::New(v8::Object::New());
1076 }
1077 (*env)->Global()->Set(v8_str("root_object"), objects_[0]);
1078 }
1079 ~GraphWithImplicitRefs() {
1080 instance_ = NULL;
1081 }
1082
1083 static void gcPrologue() {
1084 instance_->AddImplicitReferences();
1085 }
1086
1087 private:
1088 void AddImplicitReferences() {
1089 // 0 -> 1
1090 v8::V8::AddImplicitReferences(
1091 v8::Persistent<v8::Object>::Cast(objects_[0]), &objects_[1], 1);
1092 // Adding two more references(note length=2 in params): 1 -> 2, 1 -> 3
1093 v8::V8::AddImplicitReferences(
1094 v8::Persistent<v8::Object>::Cast(objects_[1]), &objects_[2], 2);
1095 }
1096
1097 v8::Persistent<v8::Value> objects_[kObjectsCount];
1098 static GraphWithImplicitRefs* instance_;
1099};
1100
1101GraphWithImplicitRefs* GraphWithImplicitRefs::instance_ = NULL;
1102
1103
1104TEST(HeapSnapshotImplicitReferences) {
1105 v8::HandleScope scope;
1106 LocalContext env;
1107
1108 GraphWithImplicitRefs graph(&env);
1109 v8::V8::SetGlobalGCPrologueCallback(&GraphWithImplicitRefs::gcPrologue);
1110
1111 const v8::HeapSnapshot* snapshot =
1112 v8::HeapProfiler::TakeSnapshot(v8_str("implicit_refs"));
1113
1114 const v8::HeapGraphNode* global_object = GetGlobalObject(snapshot);
1115 // Use kShortcut type to skip intermediate JSGlobalPropertyCell
1116 const v8::HeapGraphNode* obj0 = GetProperty(
1117 global_object, v8::HeapGraphEdge::kShortcut, "root_object");
1118 CHECK(obj0);
1119 CHECK_EQ(v8::HeapGraphNode::kObject, obj0->GetType());
1120 const v8::HeapGraphNode* obj1 = GetProperty(
1121 obj0, v8::HeapGraphEdge::kInternal, "native");
1122 CHECK(obj1);
1123 int implicit_targets_count = 0;
1124 for (int i = 0, count = obj1->GetChildrenCount(); i < count; ++i) {
1125 const v8::HeapGraphEdge* prop = obj1->GetChild(i);
1126 v8::String::AsciiValue prop_name(prop->GetName());
1127 if (prop->GetType() == v8::HeapGraphEdge::kInternal &&
1128 strcmp("native", *prop_name) == 0) {
1129 ++implicit_targets_count;
1130 }
1131 }
1132 CHECK_EQ(2, implicit_targets_count);
1133 v8::V8::SetGlobalGCPrologueCallback(NULL);
1134}
1135
1136
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00001137TEST(DeleteAllHeapSnapshots) {
1138 v8::HandleScope scope;
1139 LocalContext env;
1140
1141 CHECK_EQ(0, v8::HeapProfiler::GetSnapshotsCount());
1142 v8::HeapProfiler::DeleteAllSnapshots();
1143 CHECK_EQ(0, v8::HeapProfiler::GetSnapshotsCount());
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001144 CHECK_NE(NULL, v8::HeapProfiler::TakeSnapshot(v8_str("1")));
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00001145 CHECK_EQ(1, v8::HeapProfiler::GetSnapshotsCount());
1146 v8::HeapProfiler::DeleteAllSnapshots();
1147 CHECK_EQ(0, v8::HeapProfiler::GetSnapshotsCount());
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001148 CHECK_NE(NULL, v8::HeapProfiler::TakeSnapshot(v8_str("1")));
1149 CHECK_NE(NULL, v8::HeapProfiler::TakeSnapshot(v8_str("2")));
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00001150 CHECK_EQ(2, v8::HeapProfiler::GetSnapshotsCount());
1151 v8::HeapProfiler::DeleteAllSnapshots();
1152 CHECK_EQ(0, v8::HeapProfiler::GetSnapshotsCount());
1153}
1154
1155
1156TEST(DeleteHeapSnapshot) {
1157 v8::HandleScope scope;
1158 LocalContext env;
1159
1160 CHECK_EQ(0, v8::HeapProfiler::GetSnapshotsCount());
1161 const v8::HeapSnapshot* s1 =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001162 v8::HeapProfiler::TakeSnapshot(v8_str("1"));
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00001163 CHECK_NE(NULL, s1);
1164 CHECK_EQ(1, v8::HeapProfiler::GetSnapshotsCount());
1165 unsigned uid1 = s1->GetUid();
1166 CHECK_EQ(s1, v8::HeapProfiler::FindSnapshot(uid1));
1167 const_cast<v8::HeapSnapshot*>(s1)->Delete();
1168 CHECK_EQ(0, v8::HeapProfiler::GetSnapshotsCount());
1169 CHECK_EQ(NULL, v8::HeapProfiler::FindSnapshot(uid1));
1170
1171 const v8::HeapSnapshot* s2 =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001172 v8::HeapProfiler::TakeSnapshot(v8_str("2"));
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00001173 CHECK_NE(NULL, s2);
1174 CHECK_EQ(1, v8::HeapProfiler::GetSnapshotsCount());
1175 unsigned uid2 = s2->GetUid();
1176 CHECK_NE(static_cast<int>(uid1), static_cast<int>(uid2));
1177 CHECK_EQ(s2, v8::HeapProfiler::FindSnapshot(uid2));
1178 const v8::HeapSnapshot* s3 =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001179 v8::HeapProfiler::TakeSnapshot(v8_str("3"));
sgjesse@chromium.orgea88ce92011-03-23 11:19:56 +00001180 CHECK_NE(NULL, s3);
1181 CHECK_EQ(2, v8::HeapProfiler::GetSnapshotsCount());
1182 unsigned uid3 = s3->GetUid();
1183 CHECK_NE(static_cast<int>(uid1), static_cast<int>(uid3));
1184 CHECK_EQ(s3, v8::HeapProfiler::FindSnapshot(uid3));
1185 const_cast<v8::HeapSnapshot*>(s2)->Delete();
1186 CHECK_EQ(1, v8::HeapProfiler::GetSnapshotsCount());
1187 CHECK_EQ(NULL, v8::HeapProfiler::FindSnapshot(uid2));
1188 CHECK_EQ(s3, v8::HeapProfiler::FindSnapshot(uid3));
1189 const_cast<v8::HeapSnapshot*>(s3)->Delete();
1190 CHECK_EQ(0, v8::HeapProfiler::GetSnapshotsCount());
1191 CHECK_EQ(NULL, v8::HeapProfiler::FindSnapshot(uid3));
1192}
1193
ricow@chromium.orgd2be9012011-06-01 06:00:58 +00001194
1195TEST(DocumentURL) {
1196 v8::HandleScope scope;
1197 LocalContext env;
1198
1199 CompileRun("document = { URL:\"abcdefgh\" };");
1200
1201 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001202 v8::HeapProfiler::TakeSnapshot(v8_str("document"));
ricow@chromium.orgd2be9012011-06-01 06:00:58 +00001203 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
1204 CHECK_NE(NULL, global);
1205 CHECK_EQ("Object / abcdefgh",
1206 const_cast<i::HeapEntry*>(
1207 reinterpret_cast<const i::HeapEntry*>(global))->name());
1208}
1209
1210
1211TEST(DocumentWithException) {
1212 v8::HandleScope scope;
1213 LocalContext env;
1214
1215 CompileRun(
1216 "this.__defineGetter__(\"document\", function() { throw new Error(); })");
1217 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001218 v8::HeapProfiler::TakeSnapshot(v8_str("document"));
ricow@chromium.orgd2be9012011-06-01 06:00:58 +00001219 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
1220 CHECK_NE(NULL, global);
1221 CHECK_EQ("Object",
1222 const_cast<i::HeapEntry*>(
1223 reinterpret_cast<const i::HeapEntry*>(global))->name());
1224}
1225
1226
1227TEST(DocumentURLWithException) {
1228 v8::HandleScope scope;
1229 LocalContext env;
1230
1231 CompileRun(
1232 "function URLWithException() {}\n"
1233 "URLWithException.prototype = { get URL() { throw new Error(); } };\n"
1234 "document = { URL: new URLWithException() };");
1235 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001236 v8::HeapProfiler::TakeSnapshot(v8_str("document"));
ricow@chromium.orgd2be9012011-06-01 06:00:58 +00001237 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
1238 CHECK_NE(NULL, global);
1239 CHECK_EQ("Object",
1240 const_cast<i::HeapEntry*>(
1241 reinterpret_cast<const i::HeapEntry*>(global))->name());
1242}
1243
lrn@chromium.orgac2828d2011-06-23 06:29:21 +00001244
erikcorry0ad885c2011-11-21 13:51:57 +00001245TEST(NoHandleLeaks) {
1246 v8::HandleScope scope;
1247 LocalContext env;
1248
1249 CompileRun("document = { URL:\"abcdefgh\" };");
1250
1251 v8::Handle<v8::String> name(v8_str("leakz"));
1252 int count_before = i::HandleScope::NumberOfHandles();
1253 v8::HeapProfiler::TakeSnapshot(name);
1254 int count_after = i::HandleScope::NumberOfHandles();
1255 CHECK_EQ(count_before, count_after);
1256}
1257
1258
lrn@chromium.orgac2828d2011-06-23 06:29:21 +00001259TEST(NodesIteration) {
1260 v8::HandleScope scope;
1261 LocalContext env;
1262 const v8::HeapSnapshot* snapshot =
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001263 v8::HeapProfiler::TakeSnapshot(v8_str("iteration"));
lrn@chromium.orgac2828d2011-06-23 06:29:21 +00001264 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
1265 CHECK_NE(NULL, global);
1266 // Verify that we can find this object by iteration.
1267 const int nodes_count = snapshot->GetNodesCount();
1268 int count = 0;
1269 for (int i = 0; i < nodes_count; ++i) {
1270 if (snapshot->GetNode(i) == global)
1271 ++count;
1272 }
1273 CHECK_EQ(1, count);
1274}
ricow@chromium.orgddd545c2011-08-24 12:02:41 +00001275
1276
erik.corry@gmail.comc3b670f2011-10-05 21:44:48 +00001277TEST(GetHeapValue) {
1278 v8::HandleScope scope;
1279 LocalContext env;
1280
1281 CompileRun("a = { s_prop: \'value\', n_prop: 0.1 };");
1282 const v8::HeapSnapshot* snapshot =
1283 v8::HeapProfiler::TakeSnapshot(v8_str("value"));
1284 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
1285 CHECK(global->GetHeapValue()->IsObject());
1286 v8::Local<v8::Object> js_global =
1287 env->Global()->GetPrototype().As<v8::Object>();
1288 CHECK(js_global == global->GetHeapValue());
1289 const v8::HeapGraphNode* obj = GetProperty(
1290 global, v8::HeapGraphEdge::kShortcut, "a");
1291 CHECK(obj->GetHeapValue()->IsObject());
1292 v8::Local<v8::Object> js_obj = js_global->Get(v8_str("a")).As<v8::Object>();
1293 CHECK(js_obj == obj->GetHeapValue());
1294 const v8::HeapGraphNode* s_prop =
1295 GetProperty(obj, v8::HeapGraphEdge::kProperty, "s_prop");
1296 v8::Local<v8::String> js_s_prop =
1297 js_obj->Get(v8_str("s_prop")).As<v8::String>();
1298 CHECK(js_s_prop == s_prop->GetHeapValue());
1299 const v8::HeapGraphNode* n_prop =
1300 GetProperty(obj, v8::HeapGraphEdge::kProperty, "n_prop");
1301 v8::Local<v8::Number> js_n_prop =
1302 js_obj->Get(v8_str("n_prop")).As<v8::Number>();
1303 CHECK(js_n_prop == n_prop->GetHeapValue());
1304}
1305
1306
1307TEST(GetHeapValueForDeletedObject) {
1308 v8::HandleScope scope;
1309 LocalContext env;
1310
1311 // It is impossible to delete a global property, so we are about to delete a
1312 // property of the "a" object. Also, the "p" object can't be an empty one
1313 // because the empty object is static and isn't actually deleted.
1314 CompileRun("a = { p: { r: {} } };");
1315 const v8::HeapSnapshot* snapshot =
1316 v8::HeapProfiler::TakeSnapshot(v8_str("snapshot"));
1317 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
1318 const v8::HeapGraphNode* obj = GetProperty(
1319 global, v8::HeapGraphEdge::kShortcut, "a");
1320 const v8::HeapGraphNode* prop = GetProperty(
1321 obj, v8::HeapGraphEdge::kProperty, "p");
1322 {
1323 // Perform the check inside a nested local scope to avoid creating a
1324 // reference to the object we are deleting.
1325 v8::HandleScope scope;
1326 CHECK(prop->GetHeapValue()->IsObject());
1327 }
1328 CompileRun("delete a.p;");
1329 CHECK(prop->GetHeapValue()->IsUndefined());
1330}
1331
1332
ricow@chromium.orgddd545c2011-08-24 12:02:41 +00001333static int StringCmp(const char* ref, i::String* act) {
kmillikin@chromium.org83e16822011-09-13 08:21:47 +00001334 i::SmartArrayPointer<char> s_act = act->ToCString();
ricow@chromium.orgddd545c2011-08-24 12:02:41 +00001335 int result = strcmp(ref, *s_act);
1336 if (result != 0)
1337 fprintf(stderr, "Expected: \"%s\", Actual: \"%s\"\n", ref, *s_act);
1338 return result;
1339}
1340
1341
1342TEST(GetConstructorName) {
1343 v8::HandleScope scope;
1344 LocalContext env;
1345
1346 CompileRun(
1347 "function Constructor1() {};\n"
1348 "var obj1 = new Constructor1();\n"
1349 "var Constructor2 = function() {};\n"
1350 "var obj2 = new Constructor2();\n"
1351 "var obj3 = {};\n"
1352 "obj3.constructor = function Constructor3() {};\n"
1353 "var obj4 = {};\n"
1354 "// Slow properties\n"
1355 "for (var i=0; i<2000; ++i) obj4[\"p\" + i] = i;\n"
1356 "obj4.constructor = function Constructor4() {};\n"
1357 "var obj5 = {};\n"
1358 "var obj6 = {};\n"
1359 "obj6.constructor = 6;");
1360 v8::Local<v8::Object> js_global =
1361 env->Global()->GetPrototype().As<v8::Object>();
1362 v8::Local<v8::Object> obj1 = js_global->Get(v8_str("obj1")).As<v8::Object>();
1363 i::Handle<i::JSObject> js_obj1 = v8::Utils::OpenHandle(*obj1);
1364 CHECK_EQ(0, StringCmp(
1365 "Constructor1", i::V8HeapExplorer::GetConstructorName(*js_obj1)));
1366 v8::Local<v8::Object> obj2 = js_global->Get(v8_str("obj2")).As<v8::Object>();
1367 i::Handle<i::JSObject> js_obj2 = v8::Utils::OpenHandle(*obj2);
1368 CHECK_EQ(0, StringCmp(
1369 "Constructor2", i::V8HeapExplorer::GetConstructorName(*js_obj2)));
1370 v8::Local<v8::Object> obj3 = js_global->Get(v8_str("obj3")).As<v8::Object>();
1371 i::Handle<i::JSObject> js_obj3 = v8::Utils::OpenHandle(*obj3);
1372 CHECK_EQ(0, StringCmp(
1373 "Constructor3", i::V8HeapExplorer::GetConstructorName(*js_obj3)));
1374 v8::Local<v8::Object> obj4 = js_global->Get(v8_str("obj4")).As<v8::Object>();
1375 i::Handle<i::JSObject> js_obj4 = v8::Utils::OpenHandle(*obj4);
1376 CHECK_EQ(0, StringCmp(
1377 "Constructor4", i::V8HeapExplorer::GetConstructorName(*js_obj4)));
1378 v8::Local<v8::Object> obj5 = js_global->Get(v8_str("obj5")).As<v8::Object>();
1379 i::Handle<i::JSObject> js_obj5 = v8::Utils::OpenHandle(*obj5);
1380 CHECK_EQ(0, StringCmp(
1381 "Object", i::V8HeapExplorer::GetConstructorName(*js_obj5)));
1382 v8::Local<v8::Object> obj6 = js_global->Get(v8_str("obj6")).As<v8::Object>();
1383 i::Handle<i::JSObject> js_obj6 = v8::Utils::OpenHandle(*obj6);
1384 CHECK_EQ(0, StringCmp(
1385 "Object", i::V8HeapExplorer::GetConstructorName(*js_obj6)));
1386}
jkummerow@chromium.org04e4f1e2011-11-14 13:36:17 +00001387
erikcorry0ad885c2011-11-21 13:51:57 +00001388
jkummerow@chromium.org04e4f1e2011-11-14 13:36:17 +00001389TEST(FastCaseGetter) {
1390 v8::HandleScope scope;
1391 LocalContext env;
1392
1393 CompileRun("var obj1 = {};\n"
1394 "obj1.__defineGetter__('propWithGetter', function Y() {\n"
1395 " return 42;\n"
1396 "});\n"
1397 "obj1.__defineSetter__('propWithSetter', function Z(value) {\n"
1398 " return this.value_ = value;\n"
1399 "});\n");
1400 const v8::HeapSnapshot* snapshot =
1401 v8::HeapProfiler::TakeSnapshot(v8_str("fastCaseGetter"));
1402
1403 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
1404 CHECK_NE(NULL, global);
1405 const v8::HeapGraphNode* obj1 =
1406 GetProperty(global, v8::HeapGraphEdge::kShortcut, "obj1");
1407 CHECK_NE(NULL, obj1);
1408 const v8::HeapGraphNode* getterFunction =
1409 GetProperty(obj1, v8::HeapGraphEdge::kProperty, "get-propWithGetter");
1410 CHECK_NE(NULL, getterFunction);
1411 const v8::HeapGraphNode* setterFunction =
1412 GetProperty(obj1, v8::HeapGraphEdge::kProperty, "set-propWithSetter");
1413 CHECK_NE(NULL, setterFunction);
1414}
ricow@chromium.org64e3a4b2011-12-13 08:07:27 +00001415
1416
1417bool HasWeakEdge(const v8::HeapGraphNode* node) {
1418 for (int i = 0; i < node->GetChildrenCount(); ++i) {
1419 const v8::HeapGraphEdge* handle_edge = node->GetChild(i);
1420 if (handle_edge->GetType() == v8::HeapGraphEdge::kWeak) return true;
1421 }
1422 return false;
1423}
1424
1425
1426bool HasWeakGlobalHandle() {
1427 const v8::HeapSnapshot* snapshot =
1428 v8::HeapProfiler::TakeSnapshot(v8_str("weaks"));
1429 const v8::HeapGraphNode* gc_roots = GetNode(
1430 snapshot->GetRoot(), v8::HeapGraphNode::kObject, "(GC roots)");
1431 CHECK_NE(NULL, gc_roots);
1432 const v8::HeapGraphNode* global_handles = GetNode(
1433 gc_roots, v8::HeapGraphNode::kObject, "(Global handles)");
1434 CHECK_NE(NULL, global_handles);
1435 return HasWeakEdge(global_handles);
1436}
1437
1438
1439static void PersistentHandleCallback(v8::Persistent<v8::Value> handle, void*) {
1440 handle.Dispose();
1441}
1442
1443
1444TEST(WeakGlobalHandle) {
1445 v8::HandleScope scope;
1446 LocalContext env;
1447
1448 CHECK(!HasWeakGlobalHandle());
1449
1450 v8::Persistent<v8::Object> handle =
1451 v8::Persistent<v8::Object>::New(v8::Object::New());
1452 handle.MakeWeak(NULL, PersistentHandleCallback);
1453
1454 CHECK(HasWeakGlobalHandle());
1455}
1456
1457
1458TEST(WeakGlobalContextRefs) {
1459 v8::HandleScope scope;
1460 LocalContext env;
1461
1462 const v8::HeapSnapshot* snapshot =
1463 v8::HeapProfiler::TakeSnapshot(v8_str("weaks"));
1464 const v8::HeapGraphNode* gc_roots = GetNode(
1465 snapshot->GetRoot(), v8::HeapGraphNode::kObject, "(GC roots)");
1466 CHECK_NE(NULL, gc_roots);
1467 const v8::HeapGraphNode* global_handles = GetNode(
1468 gc_roots, v8::HeapGraphNode::kObject, "(Global handles)");
1469 CHECK_NE(NULL, global_handles);
1470 const v8::HeapGraphNode* global_context = GetNode(
1471 global_handles, v8::HeapGraphNode::kHidden, "system / GlobalContext");
1472 CHECK_NE(NULL, global_context);
1473 CHECK(HasWeakEdge(global_context));
1474}
1475
1476
1477TEST(SfiAndJsFunctionWeakRefs) {
1478 v8::HandleScope scope;
1479 LocalContext env;
1480
1481 CompileRun(
1482 "fun = (function (x) { return function () { return x + 1; } })(1);");
1483 const v8::HeapSnapshot* snapshot =
1484 v8::HeapProfiler::TakeSnapshot(v8_str("fun"));
1485 const v8::HeapGraphNode* global = GetGlobalObject(snapshot);
1486 CHECK_NE(NULL, global);
1487 const v8::HeapGraphNode* fun =
1488 GetProperty(global, v8::HeapGraphEdge::kShortcut, "fun");
1489 CHECK(HasWeakEdge(fun));
1490 const v8::HeapGraphNode* shared =
1491 GetProperty(fun, v8::HeapGraphEdge::kInternal, "shared");
1492 CHECK(HasWeakEdge(shared));
1493}
erik.corry@gmail.combbceb572012-03-09 10:52:05 +00001494
1495
1496TEST(PersistentHandleCount) {
1497 v8::HandleScope scope;
1498 LocalContext env;
1499
1500 // V8 also uses global handles internally, so we can't test for an absolute
1501 // number.
1502 int global_handle_count = v8::HeapProfiler::GetPersistentHandleCount();
1503
1504 // Create some persistent handles.
1505 v8::Persistent<v8::String> p_AAA =
1506 v8::Persistent<v8::String>::New(v8_str("AAA"));
1507 CHECK_EQ(global_handle_count + 1,
1508 v8::HeapProfiler::GetPersistentHandleCount());
1509 v8::Persistent<v8::String> p_BBB =
1510 v8::Persistent<v8::String>::New(v8_str("BBB"));
1511 CHECK_EQ(global_handle_count + 2,
1512 v8::HeapProfiler::GetPersistentHandleCount());
1513 v8::Persistent<v8::String> p_CCC =
1514 v8::Persistent<v8::String>::New(v8_str("CCC"));
1515 CHECK_EQ(global_handle_count + 3,
1516 v8::HeapProfiler::GetPersistentHandleCount());
1517
1518 // Dipose the persistent handles in a different order.
1519 p_AAA.Dispose();
1520 CHECK_EQ(global_handle_count + 2,
1521 v8::HeapProfiler::GetPersistentHandleCount());
1522 p_CCC.Dispose();
1523 CHECK_EQ(global_handle_count + 1,
1524 v8::HeapProfiler::GetPersistentHandleCount());
1525 p_BBB.Dispose();
1526 CHECK_EQ(global_handle_count, v8::HeapProfiler::GetPersistentHandleCount());
1527}