blob: 5a1ca2a15e8ac6b8c1f21c3b3efad45b1f86ca99 [file] [log] [blame]
kocienda66a6d362001-08-24 14:24:45 +00001/*
oliver@apple.com284cc3d2013-07-25 04:00:33 +00002 * Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2011, 2013 Apple Inc. All rights reserved.
eseidelc504bad2007-10-24 21:31:27 +00003 * Copyright (C) 2007 Eric Seidel <eric@webkit.org>
kocienda66a6d362001-08-24 14:24:45 +00004 *
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2 of the License, or (at your option) any later version.
9 *
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
14 *
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
mjscdff33b2006-01-23 21:41:36 +000017 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
mjs6f821c82002-03-22 00:31:57 +000018 *
kocienda66a6d362001-08-24 14:24:45 +000019 */
20
mjsb64c50a2005-10-03 21:13:12 +000021#include "config.h"
ggaren@apple.coma8b38542011-01-10 23:43:56 +000022#include "Heap.h"
darinff399e02002-11-23 07:49:05 +000023
ggaren@apple.com79ab2a92011-02-02 05:05:55 +000024#include "CodeBlock.h"
ggaren@apple.com0b32d092011-04-11 21:57:31 +000025#include "ConservativeRoots.h"
mhahnenberg@apple.com02e39c72012-10-12 19:38:35 +000026#include "CopiedSpace.h"
mark.lam@apple.coma4fe7ab2012-11-09 03:03:44 +000027#include "CopiedSpaceInlines.h"
28#include "CopyVisitorInlines.h"
oliver@apple.com284cc3d2013-07-25 04:00:33 +000029#include "DFGWorklist.h"
mhahnenberg@apple.comac6f1fd2013-11-15 19:53:30 +000030#include "DelayedReleaseScope.h"
ggaren@apple.com2ccf7a92010-08-03 20:34:17 +000031#include "GCActivityCallback.h"
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +000032#include "GCIncomingRefCountedSetInlines.h"
mhahnenberg@apple.combee96a32013-09-16 19:48:48 +000033#include "HeapIterationScope.h"
ggaren@apple.comd2b624b2011-05-24 23:46:35 +000034#include "HeapRootVisitor.h"
mhahnenberg@apple.com3a2fb032012-10-05 17:35:49 +000035#include "HeapStatistics.h"
mhahnenberg@apple.comeb39abc2012-05-31 03:04:00 +000036#include "IncrementalSweeper.h"
weinig@apple.comf5b90cd2009-01-16 00:24:24 +000037#include "Interpreter.h"
ggaren@apple.comf0863872007-12-03 07:53:03 +000038#include "JSGlobalObject.h"
ap@webkit.org01aff702008-08-20 07:23:06 +000039#include "JSLock.h"
darin@apple.comc46cfaf2009-06-18 19:18:12 +000040#include "JSONObject.h"
fpizlo@apple.coma4b4cbe2013-01-12 04:47:03 +000041#include "Operations.h"
mhahnenberg@apple.com7a22bab2013-12-10 19:37:29 +000042#include "RecursiveAllocationScope.h"
ggaren@apple.comc2c80282011-01-15 01:52:28 +000043#include "Tracing.h"
oliver@apple.comf0c01b82012-11-07 00:13:54 +000044#include "UnlinkedCodeBlock.h"
mhahnenberg@apple.com7a22bab2013-12-10 19:37:29 +000045#include "VM.h"
ggaren@apple.comab71e972012-04-28 05:57:46 +000046#include "WeakSetInlines.h"
ggaren@apple.comf2663492011-02-05 00:20:16 +000047#include <algorithm>
ggaren@apple.com9a070932012-05-22 19:17:57 +000048#include <wtf/RAMSize.h>
fpizlo@apple.com622ead72011-11-01 06:43:37 +000049#include <wtf/CurrentTime.h>
darind5893602005-08-01 05:02:13 +000050
ggaren@apple.comf2663492011-02-05 00:20:16 +000051using namespace std;
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +000052using namespace JSC;
ggaren@apple.comf2663492011-02-05 00:20:16 +000053
cwzwarich@webkit.org3f782f62008-09-08 01:28:33 +000054namespace JSC {
kocienda66a6d362001-08-24 14:24:45 +000055
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +000056namespace {
57
ggaren@apple.com9a070932012-05-22 19:17:57 +000058static const size_t largeHeapSize = 32 * MB; // About 1.5X the average webpage.
59static const size_t smallHeapSize = 1 * MB; // Matches the FastMalloc per-thread cache.
oliver@apple.com3eb80372011-10-12 01:11:04 +000060
fpizlo@apple.comebccac02013-09-10 03:16:31 +000061#define ENABLE_GC_LOGGING 0
mhahnenberg@apple.com8e937f82013-09-09 22:41:00 +000062
oliver@apple.com32b720a2011-10-05 19:07:18 +000063#if ENABLE(GC_LOGGING)
oliver@apple.com73792612011-10-12 00:24:12 +000064#if COMPILER(CLANG)
65#define DEFINE_GC_LOGGING_GLOBAL(type, name, arguments) \
66_Pragma("clang diagnostic push") \
67_Pragma("clang diagnostic ignored \"-Wglobal-constructors\"") \
68_Pragma("clang diagnostic ignored \"-Wexit-time-destructors\"") \
69static type name arguments; \
70_Pragma("clang diagnostic pop")
71#else
72#define DEFINE_GC_LOGGING_GLOBAL(type, name, arguments) \
73static type name arguments;
74#endif // COMPILER(CLANG)
75
oliver@apple.com32b720a2011-10-05 19:07:18 +000076struct GCTimer {
77 GCTimer(const char* name)
78 : m_time(0)
79 , m_min(100000000)
80 , m_max(0)
81 , m_count(0)
82 , m_name(name)
83 {
84 }
85 ~GCTimer()
86 {
fpizlo@apple.com01902c82012-11-22 04:23:36 +000087 dataLogF("%s: %.2lfms (avg. %.2lf, min. %.2lf, max. %.2lf)\n", m_name, m_time * 1000, m_time * 1000 / m_count, m_min*1000, m_max*1000);
oliver@apple.com32b720a2011-10-05 19:07:18 +000088 }
89 double m_time;
90 double m_min;
91 double m_max;
92 size_t m_count;
93 const char* m_name;
94};
95
96struct GCTimerScope {
97 GCTimerScope(GCTimer* timer)
98 : m_timer(timer)
commit-queue@webkit.orgf15eb3d2013-08-14 00:51:25 +000099 , m_start(WTF::monotonicallyIncreasingTime())
oliver@apple.com32b720a2011-10-05 19:07:18 +0000100 {
101 }
102 ~GCTimerScope()
103 {
commit-queue@webkit.orgf15eb3d2013-08-14 00:51:25 +0000104 double delta = WTF::monotonicallyIncreasingTime() - m_start;
oliver@apple.com32b720a2011-10-05 19:07:18 +0000105 if (delta < m_timer->m_min)
106 m_timer->m_min = delta;
107 if (delta > m_timer->m_max)
108 m_timer->m_max = delta;
109 m_timer->m_count++;
110 m_timer->m_time += delta;
111 }
112 GCTimer* m_timer;
113 double m_start;
114};
115
oliver@apple.com73792612011-10-12 00:24:12 +0000116struct GCCounter {
117 GCCounter(const char* name)
118 : m_name(name)
119 , m_count(0)
120 , m_total(0)
121 , m_min(10000000)
122 , m_max(0)
123 {
124 }
125
126 void count(size_t amount)
127 {
128 m_count++;
129 m_total += amount;
130 if (amount < m_min)
131 m_min = amount;
132 if (amount > m_max)
133 m_max = amount;
134 }
135 ~GCCounter()
136 {
fpizlo@apple.com01902c82012-11-22 04:23:36 +0000137 dataLogF("%s: %zu values (avg. %zu, min. %zu, max. %zu)\n", m_name, m_total, m_total / m_count, m_min, m_max);
oliver@apple.com73792612011-10-12 00:24:12 +0000138 }
139 const char* m_name;
140 size_t m_count;
141 size_t m_total;
142 size_t m_min;
143 size_t m_max;
144};
oliver@apple.com32b720a2011-10-05 19:07:18 +0000145
oliver@apple.com73792612011-10-12 00:24:12 +0000146#define GCPHASE(name) DEFINE_GC_LOGGING_GLOBAL(GCTimer, name##Timer, (#name)); GCTimerScope name##TimerScope(&name##Timer)
147#define COND_GCPHASE(cond, name1, name2) DEFINE_GC_LOGGING_GLOBAL(GCTimer, name1##Timer, (#name1)); DEFINE_GC_LOGGING_GLOBAL(GCTimer, name2##Timer, (#name2)); GCTimerScope name1##CondTimerScope(cond ? &name1##Timer : &name2##Timer)
148#define GCCOUNTER(name, value) do { DEFINE_GC_LOGGING_GLOBAL(GCCounter, name##Counter, (#name)); name##Counter.count(value); } while (false)
149
oliver@apple.com32b720a2011-10-05 19:07:18 +0000150#else
151
152#define GCPHASE(name) do { } while (false)
153#define COND_GCPHASE(cond, name1, name2) do { } while (false)
oliver@apple.com73792612011-10-12 00:24:12 +0000154#define GCCOUNTER(name, value) do { } while (false)
oliver@apple.com32b720a2011-10-05 19:07:18 +0000155#endif
156
ggaren@apple.com9a070932012-05-22 19:17:57 +0000157static inline size_t minHeapSize(HeapType heapType, size_t ramSize)
fpizlo@apple.comf49ce5c2011-08-02 20:40:17 +0000158{
ggaren@apple.com9a070932012-05-22 19:17:57 +0000159 if (heapType == LargeHeap)
160 return min(largeHeapSize, ramSize / 4);
fpizlo@apple.comc7b47212011-08-03 23:35:03 +0000161 return smallHeapSize;
fpizlo@apple.comf49ce5c2011-08-02 20:40:17 +0000162}
ggaren@apple.comf2663492011-02-05 00:20:16 +0000163
ggaren@apple.com9a070932012-05-22 19:17:57 +0000164static inline size_t proportionalHeapSize(size_t heapSize, size_t ramSize)
165{
166 // Try to stay under 1/2 RAM size to leave room for the DOM, rendering, networking, etc.
167 if (heapSize < ramSize / 4)
168 return 2 * heapSize;
169 if (heapSize < ramSize / 2)
170 return 1.5 * heapSize;
171 return 1.25 * heapSize;
172}
173
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000174static inline bool isValidSharedInstanceThreadState(VM* vm)
ggaren@apple.com99cc9d92011-05-26 21:46:09 +0000175{
andersca@apple.comb987aae2013-07-26 00:13:13 +0000176 return vm->currentThreadIsHoldingAPILock();
ggaren@apple.com99cc9d92011-05-26 21:46:09 +0000177}
178
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000179static inline bool isValidThreadState(VM* vm)
ggaren@apple.com99cc9d92011-05-26 21:46:09 +0000180{
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000181 if (vm->identifierTable != wtfThreadData().currentIdentifierTable())
ggaren@apple.com99cc9d92011-05-26 21:46:09 +0000182 return false;
183
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000184 if (vm->isSharedInstance() && !isValidSharedInstanceThreadState(vm))
ggaren@apple.com99cc9d92011-05-26 21:46:09 +0000185 return false;
186
187 return true;
188}
189
ggaren@apple.comdef139e2012-09-26 02:27:54 +0000190struct MarkObject : public MarkedBlock::VoidFunctor {
191 void operator()(JSCell* cell)
192 {
193 if (cell->isZapped())
194 return;
195 Heap::heap(cell)->setMarked(cell);
196 }
197};
198
ggaren@apple.com041d0a22012-05-18 00:40:30 +0000199struct Count : public MarkedBlock::CountFunctor {
200 void operator()(JSCell*) { count(1); }
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000201};
202
ggaren@apple.com041d0a22012-05-18 00:40:30 +0000203struct CountIfGlobalObject : MarkedBlock::CountFunctor {
204 void operator()(JSCell* cell) {
205 if (!cell->isObject())
206 return;
207 if (!asObject(cell)->isGlobalObject())
208 return;
209 count(1);
210 }
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000211};
212
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000213class RecordType {
214public:
215 typedef PassOwnPtr<TypeCountSet> ReturnType;
216
217 RecordType();
218 void operator()(JSCell*);
219 ReturnType returnValue();
220
221private:
222 const char* typeName(JSCell*);
223 OwnPtr<TypeCountSet> m_typeCountSet;
224};
225
226inline RecordType::RecordType()
227 : m_typeCountSet(adoptPtr(new TypeCountSet))
228{
229}
230
231inline const char* RecordType::typeName(JSCell* cell)
232{
233 const ClassInfo* info = cell->classInfo();
234 if (!info || !info->className)
235 return "[unknown]";
236 return info->className;
237}
238
239inline void RecordType::operator()(JSCell* cell)
240{
241 m_typeCountSet->add(typeName(cell));
242}
243
244inline PassOwnPtr<TypeCountSet> RecordType::returnValue()
245{
246 return m_typeCountSet.release();
247}
248
249} // anonymous namespace
250
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000251Heap::Heap(VM* vm, HeapType heapType)
ggaren@apple.com9a070932012-05-22 19:17:57 +0000252 : m_heapType(heapType)
253 , m_ramSize(ramSize())
254 , m_minBytesPerCycle(minHeapSize(m_heapType, m_ramSize))
mhahnenberg@apple.com98daed02012-04-25 01:29:42 +0000255 , m_sizeAfterLastCollect(0)
mhahnenberg@apple.com8b5cfd32012-04-20 00:05:37 +0000256 , m_bytesAllocatedLimit(m_minBytesPerCycle)
257 , m_bytesAllocated(0)
mhahnenberg@apple.com376a4172012-05-03 22:31:43 +0000258 , m_bytesAbandoned(0)
mhahnenberg@apple.come2789cf2013-09-09 00:11:57 +0000259 , m_totalBytesVisited(0)
260 , m_totalBytesCopied(0)
fpizlo@apple.comf49ce5c2011-08-02 20:40:17 +0000261 , m_operationInProgress(NoOperation)
mhahnenberg@apple.com123e67f2013-04-01 16:01:01 +0000262 , m_blockAllocator()
oliver@apple.com44d89542011-09-20 18:31:37 +0000263 , m_objectSpace(this)
mhahnenberg@apple.com5d0b30a2012-01-19 21:49:56 +0000264 , m_storageSpace(this)
oliver@apple.com8cebf372013-07-25 04:03:46 +0000265 , m_extraMemoryUsage(0)
ggaren@apple.com123f6852011-03-14 04:56:46 +0000266 , m_machineThreads(this)
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000267 , m_sharedData(vm)
mhahnenberg@apple.comc58d54d2011-12-16 19:06:44 +0000268 , m_slotVisitor(m_sharedData)
mhahnenberg@apple.com02e39c72012-10-12 19:38:35 +0000269 , m_copyVisitor(m_sharedData)
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000270 , m_handleSet(vm)
commit-queue@webkit.orgf42601f2011-07-12 22:35:39 +0000271 , m_isSafeToCollect(false)
mhahnenberg@apple.com4968e1a2013-12-18 22:50:40 +0000272#if ENABLE(GGC)
273 , m_writeBarrierBuffer(128)
274#endif
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000275 , m_vm(vm)
mhahnenberg@apple.comf9133a12012-03-06 21:32:24 +0000276 , m_lastGCLength(0)
commit-queue@webkit.orgf15eb3d2013-08-14 00:51:25 +0000277 , m_lastCodeDiscardTime(WTF::monotonicallyIncreasingTime())
mhahnenberg@apple.coma7ec41b2012-06-19 19:17:31 +0000278 , m_activityCallback(DefaultGCActivityCallback::create(this))
279 , m_sweeper(IncrementalSweeper::create(this))
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000280 , m_deferralDepth(0)
ap@webkit.org960c28e2008-06-19 17:29:29 +0000281{
mhahnenberg@apple.com5d0b30a2012-01-19 21:49:56 +0000282 m_storageSpace.init();
ap@webkit.org960c28e2008-06-19 17:29:29 +0000283}
darin070530c2002-10-07 21:06:29 +0000284
ap@webkit.org0b7e63b2008-06-26 18:59:26 +0000285Heap::~Heap()
286{
darin@apple.comb6295d42008-10-02 23:48:47 +0000287}
288
mhahnenberg@apple.com2e132e42012-05-03 00:14:05 +0000289bool Heap::isPagedOut(double deadline)
290{
291 return m_objectSpace.isPagedOut(deadline) || m_storageSpace.isPagedOut(deadline);
292}
293
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000294// The VM is being destroyed and the collector will never run again.
ggaren@apple.comc143e902012-04-28 20:51:27 +0000295// Run all pending finalizers now because we won't get another chance.
296void Heap::lastChanceToFinalize()
darin@apple.comb6295d42008-10-02 23:48:47 +0000297{
mark.lam@apple.com30721252013-11-21 05:29:42 +0000298 RELEASE_ASSERT(!m_vm->entryScope);
oliver@apple.com903b0c02013-01-24 01:40:37 +0000299 RELEASE_ASSERT(m_operationInProgress == NoOperation);
ap@webkit.org8672bcf2008-08-11 12:01:26 +0000300
ggaren@apple.com96fa0e72012-05-23 20:47:46 +0000301 m_objectSpace.lastChanceToFinalize();
ap@webkit.org0b7e63b2008-06-26 18:59:26 +0000302}
303
ggaren@apple.com07050262011-01-30 03:32:52 +0000304void Heap::reportExtraMemoryCostSlowCase(size_t cost)
mjs06ed4662007-07-25 21:50:00 +0000305{
306 // Our frequency of garbage collection tries to balance memory use against speed
307 // by collecting based on the number of newly created values. However, for values
308 // that hold on to a great deal of memory that's not in the form of other JS values,
309 // that is not good enough - in some cases a lot of those objects can pile up and
310 // use crazy amounts of memory without a GC happening. So we track these extra
311 // memory costs. Only unusually large objects are noted, and we only keep track
312 // of this extra cost until the next GC. In garbage collected languages, most values
313 // are either very short lived temporaries, or have extremely long lifetimes. So
314 // if a large value survives one garbage collection, there is not much point to
315 // collecting more frequently as long as it stays alive.
316
mhahnenberg@apple.com8b5cfd32012-04-20 00:05:37 +0000317 didAllocate(cost);
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000318 collectIfNecessaryOrDefer();
mjs06ed4662007-07-25 21:50:00 +0000319}
320
mhahnenberg@apple.com98daed02012-04-25 01:29:42 +0000321void Heap::reportAbandonedObjectGraph()
322{
323 // Our clients don't know exactly how much memory they
324 // are abandoning so we just guess for them.
325 double abandonedBytes = 0.10 * m_sizeAfterLastCollect;
326
327 // We want to accelerate the next collection. Because memory has just
328 // been abandoned, the next collection has the potential to
329 // be more profitable. Since allocation is the trigger for collection,
330 // we hasten the next collection by pretending that we've allocated more memory.
mhahnenberg@apple.com376a4172012-05-03 22:31:43 +0000331 didAbandon(abandonedBytes);
332}
333
334void Heap::didAbandon(size_t bytes)
335{
dbates@webkit.org6122a1a2013-10-16 23:36:25 +0000336 if (m_activityCallback)
337 m_activityCallback->didAllocate(m_bytesAllocated + m_bytesAbandoned);
mhahnenberg@apple.com376a4172012-05-03 22:31:43 +0000338 m_bytesAbandoned += bytes;
mhahnenberg@apple.com98daed02012-04-25 01:29:42 +0000339}
340
ggaren@apple.comdc067b62009-05-01 22:43:39 +0000341void Heap::protect(JSValue k)
mjsc4bb2952005-12-13 11:06:10 +0000342{
mjs9f1f97a2007-03-20 23:57:01 +0000343 ASSERT(k);
andersca@apple.comb987aae2013-07-26 00:13:13 +0000344 ASSERT(m_vm->currentThreadIsHoldingAPILock());
mjsc4bb2952005-12-13 11:06:10 +0000345
weinig@apple.com92fdaef2009-01-20 00:54:18 +0000346 if (!k.isCell())
ap@webkit.org84ea6b92008-06-04 16:29:49 +0000347 return;
mjsc4bb2952005-12-13 11:06:10 +0000348
weinig@apple.com92fdaef2009-01-20 00:54:18 +0000349 m_protectedValues.add(k.asCell());
mjsc4bb2952005-12-13 11:06:10 +0000350}
351
barraclough@apple.com88bb3112010-03-24 07:11:51 +0000352bool Heap::unprotect(JSValue k)
mjsc4bb2952005-12-13 11:06:10 +0000353{
mjs9f1f97a2007-03-20 23:57:01 +0000354 ASSERT(k);
andersca@apple.comb987aae2013-07-26 00:13:13 +0000355 ASSERT(m_vm->currentThreadIsHoldingAPILock());
mjsc4bb2952005-12-13 11:06:10 +0000356
weinig@apple.com92fdaef2009-01-20 00:54:18 +0000357 if (!k.isCell())
barraclough@apple.com88bb3112010-03-24 07:11:51 +0000358 return false;
mjsc4bb2952005-12-13 11:06:10 +0000359
barraclough@apple.com88bb3112010-03-24 07:11:51 +0000360 return m_protectedValues.remove(k.asCell());
mjsc4bb2952005-12-13 11:06:10 +0000361}
362
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +0000363void Heap::addReference(JSCell* cell, ArrayBuffer* buffer)
364{
365 if (m_arrayBuffers.addReference(cell, buffer)) {
366 collectIfNecessaryOrDefer();
367 didAllocate(buffer->gcSizeEstimateInBytes());
368 }
369}
370
ggaren@apple.come1789182011-05-26 23:47:40 +0000371void Heap::markProtectedObjects(HeapRootVisitor& heapRootVisitor)
ap@webkit.org960c28e2008-06-19 17:29:29 +0000372{
ap@webkit.org32dab942008-07-02 07:00:53 +0000373 ProtectCountSet::iterator end = m_protectedValues.end();
ggaren@apple.com7c9ff4d2011-01-21 04:07:38 +0000374 for (ProtectCountSet::iterator it = m_protectedValues.begin(); it != end; ++it)
benjamin@webkit.orgee554052012-10-07 23:12:07 +0000375 heapRootVisitor.visit(&it->key);
mjs99f74d92004-04-23 22:40:31 +0000376}
377
oliver@apple.com4e3f9652013-04-08 04:14:50 +0000378void Heap::pushTempSortVector(Vector<ValueStringPair, 0, UnsafeVectorOverflow>* tempVector)
msaboff@apple.com86ea0142010-12-09 18:27:13 +0000379{
380 m_tempSortingVectors.append(tempVector);
381}
382
oliver@apple.com4e3f9652013-04-08 04:14:50 +0000383void Heap::popTempSortVector(Vector<ValueStringPair, 0, UnsafeVectorOverflow>* tempVector)
msaboff@apple.com86ea0142010-12-09 18:27:13 +0000384{
385 ASSERT_UNUSED(tempVector, tempVector == m_tempSortingVectors.last());
386 m_tempSortingVectors.removeLast();
387}
mhahnenberg@apple.com5d0b30a2012-01-19 21:49:56 +0000388
ggaren@apple.come1789182011-05-26 23:47:40 +0000389void Heap::markTempSortVectors(HeapRootVisitor& heapRootVisitor)
msaboff@apple.com86ea0142010-12-09 18:27:13 +0000390{
oliver@apple.com4e3f9652013-04-08 04:14:50 +0000391 typedef Vector<Vector<ValueStringPair, 0, UnsafeVectorOverflow>* > VectorOfValueStringVectors;
msaboff@apple.com86ea0142010-12-09 18:27:13 +0000392
393 VectorOfValueStringVectors::iterator end = m_tempSortingVectors.end();
394 for (VectorOfValueStringVectors::iterator it = m_tempSortingVectors.begin(); it != end; ++it) {
oliver@apple.com4e3f9652013-04-08 04:14:50 +0000395 Vector<ValueStringPair, 0, UnsafeVectorOverflow>* tempSortingVector = *it;
msaboff@apple.com86ea0142010-12-09 18:27:13 +0000396
397 Vector<ValueStringPair>::iterator vectorEnd = tempSortingVector->end();
ggaren@apple.com7c9ff4d2011-01-21 04:07:38 +0000398 for (Vector<ValueStringPair>::iterator vectorIt = tempSortingVector->begin(); vectorIt != vectorEnd; ++vectorIt) {
msaboff@apple.com86ea0142010-12-09 18:27:13 +0000399 if (vectorIt->first)
ggaren@apple.coma6833892011-06-08 04:57:08 +0000400 heapRootVisitor.visit(&vectorIt->first);
ggaren@apple.com7c9ff4d2011-01-21 04:07:38 +0000401 }
msaboff@apple.com86ea0142010-12-09 18:27:13 +0000402 }
403}
ggaren@apple.com86f21952011-01-22 04:27:18 +0000404
fpizlo@apple.comd13163d2011-09-03 05:14:04 +0000405void Heap::harvestWeakReferences()
406{
407 m_slotVisitor.harvestWeakReferences();
408}
409
fpizlo@apple.comf5e1fe12011-11-10 21:59:39 +0000410void Heap::finalizeUnconditionalFinalizers()
411{
412 m_slotVisitor.finalizeUnconditionalFinalizers();
413}
414
mark.lam@apple.com4fbb9c32012-10-09 07:12:56 +0000415inline JSStack& Heap::stack()
ggaren@apple.com86f21952011-01-22 04:27:18 +0000416{
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000417 return m_vm->interpreter->stack();
ggaren@apple.com86f21952011-01-22 04:27:18 +0000418}
419
mhahnenberg@apple.combee96a32013-09-16 19:48:48 +0000420void Heap::willStartIterating()
mark.lam@apple.comd4eb0d32013-04-17 16:32:44 +0000421{
mhahnenberg@apple.combee96a32013-09-16 19:48:48 +0000422 m_objectSpace.willStartIterating();
423}
424
425void Heap::didFinishIterating()
426{
427 m_objectSpace.didFinishIterating();
mark.lam@apple.comd4eb0d32013-04-17 16:32:44 +0000428}
429
oliver@apple.com1db480d2011-06-28 01:32:01 +0000430void Heap::getConservativeRegisterRoots(HashSet<JSCell*>& roots)
431{
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000432 ASSERT(isValidThreadState(m_vm));
mark.lam@apple.com4fbb9c32012-10-09 07:12:56 +0000433 ConservativeRoots stackRoots(&m_objectSpace.blocks(), &m_storageSpace);
434 stack().gatherConservativeRoots(stackRoots);
435 size_t stackRootCount = stackRoots.size();
436 JSCell** registerRoots = stackRoots.roots();
437 for (size_t i = 0; i < stackRootCount; i++) {
oliver@apple.com1db480d2011-06-28 01:32:01 +0000438 setMarked(registerRoots[i]);
439 roots.add(registerRoots[i]);
440 }
oliver@apple.com1db480d2011-06-28 01:32:01 +0000441}
442
ggaren@apple.combd41f7e2013-01-30 00:36:50 +0000443void Heap::markRoots()
kocienda66a6d362001-08-24 14:24:45 +0000444{
fpizlo@apple.com8ce62fd2011-11-06 11:39:12 +0000445 SamplingRegion samplingRegion("Garbage Collection: Tracing");
446
ggaren@apple.combd41f7e2013-01-30 00:36:50 +0000447 GCPHASE(MarkRoots);
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000448 ASSERT(isValidThreadState(m_vm));
ap@webkit.org01aff702008-08-20 07:23:06 +0000449
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000450#if ENABLE(OBJECT_MARK_LOGGING)
commit-queue@webkit.orgf15eb3d2013-08-14 00:51:25 +0000451 double gcStartTime = WTF::monotonicallyIncreasingTime();
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000452#endif
453
ggaren@apple.com02bf0402011-03-16 18:35:49 +0000454 void* dummy;
fpizlo@apple.com706f5f32011-09-21 23:36:35 +0000455
ggaren@apple.com99cc9d92011-05-26 21:46:09 +0000456 // We gather conservative roots before clearing mark bits because conservative
457 // gathering uses the mark bits to determine whether a reference is valid.
mhahnenberg@apple.com5d0b30a2012-01-19 21:49:56 +0000458 ConservativeRoots machineThreadRoots(&m_objectSpace.blocks(), &m_storageSpace);
fpizlo@apple.com746b8c52012-07-09 23:26:54 +0000459 m_jitStubRoutines.clearMarks();
oliver@apple.com32b720a2011-10-05 19:07:18 +0000460 {
461 GCPHASE(GatherConservativeRoots);
462 m_machineThreads.gatherConservativeRoots(machineThreadRoots, &dummy);
463 }
ggaren@apple.com02bf0402011-03-16 18:35:49 +0000464
mark.lam@apple.com4fbb9c32012-10-09 07:12:56 +0000465 ConservativeRoots stackRoots(&m_objectSpace.blocks(), &m_storageSpace);
fpizlo@apple.com195d7b82013-09-03 16:39:29 +0000466 m_codeBlocks.clearMarks();
oliver@apple.com32b720a2011-10-05 19:07:18 +0000467 {
mark.lam@apple.com4fbb9c32012-10-09 07:12:56 +0000468 GCPHASE(GatherStackRoots);
fpizlo@apple.com195d7b82013-09-03 16:39:29 +0000469 stack().gatherConservativeRoots(stackRoots, m_jitStubRoutines, m_codeBlocks);
fpizlo@apple.com7969ed72013-11-28 00:22:43 +0000470 stack().sanitizeStack();
oliver@apple.com32b720a2011-10-05 19:07:18 +0000471 }
msaboff@apple.com12830392012-05-21 05:42:56 +0000472
473#if ENABLE(DFG_JIT)
474 ConservativeRoots scratchBufferRoots(&m_objectSpace.blocks(), &m_storageSpace);
475 {
476 GCPHASE(GatherScratchBufferRoots);
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000477 m_vm->gatherConservativeRoots(scratchBufferRoots);
msaboff@apple.com12830392012-05-21 05:42:56 +0000478 }
479#endif
480
oliver@apple.com32b720a2011-10-05 19:07:18 +0000481 {
mhahnenberg@apple.com23421aa2013-09-08 23:36:40 +0000482 GCPHASE(ClearLivenessData);
483 m_objectSpace.clearNewlyAllocated();
ggaren@apple.com041d0a22012-05-18 00:40:30 +0000484 m_objectSpace.clearMarks();
oliver@apple.com32b720a2011-10-05 19:07:18 +0000485 }
ggaren@apple.com2029c2a2011-01-15 03:28:16 +0000486
mhahnenberg@apple.com02e39c72012-10-12 19:38:35 +0000487 m_sharedData.didStartMarking();
ggaren@apple.com6e1f8c12011-06-16 22:01:43 +0000488 SlotVisitor& visitor = m_slotVisitor;
msaboff@apple.comd08f3502012-07-03 22:57:00 +0000489 visitor.setup();
ggaren@apple.com28cfb352011-06-10 00:02:23 +0000490 HeapRootVisitor heapRootVisitor(visitor);
oliver@apple.comc0949332011-09-29 22:52:45 +0000491
oliver@apple.com73792612011-10-12 00:24:12 +0000492 {
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000493 ParallelModeEnabler enabler(visitor);
oliver@apple.comf0c01b82012-11-07 00:13:54 +0000494
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000495 m_vm->smallStrings.visitStrongReferences(visitor);
oliver@apple.come722ad02013-01-09 02:37:29 +0000496
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000497 {
498 GCPHASE(VisitMachineRoots);
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000499 MARK_LOG_ROOT(visitor, "C++ Stack");
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000500 visitor.append(machineThreadRoots);
501 visitor.donateAndDrain();
502 }
503 {
mark.lam@apple.com4fbb9c32012-10-09 07:12:56 +0000504 GCPHASE(VisitStackRoots);
505 MARK_LOG_ROOT(visitor, "Stack");
506 visitor.append(stackRoots);
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000507 visitor.donateAndDrain();
508 }
msaboff@apple.com12830392012-05-21 05:42:56 +0000509#if ENABLE(DFG_JIT)
510 {
511 GCPHASE(VisitScratchBufferRoots);
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000512 MARK_LOG_ROOT(visitor, "Scratch Buffers");
msaboff@apple.com12830392012-05-21 05:42:56 +0000513 visitor.append(scratchBufferRoots);
514 visitor.donateAndDrain();
515 }
516#endif
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000517 {
518 GCPHASE(VisitProtectedObjects);
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000519 MARK_LOG_ROOT(visitor, "Protected Objects");
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000520 markProtectedObjects(heapRootVisitor);
521 visitor.donateAndDrain();
522 }
523 {
524 GCPHASE(VisitTempSortVectors);
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000525 MARK_LOG_ROOT(visitor, "Temp Sort Vectors");
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000526 markTempSortVectors(heapRootVisitor);
527 visitor.donateAndDrain();
528 }
529
530 {
531 GCPHASE(MarkingArgumentBuffers);
532 if (m_markListSet && m_markListSet->size()) {
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000533 MARK_LOG_ROOT(visitor, "Argument Buffers");
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000534 MarkedArgumentBuffer::markLists(heapRootVisitor, *m_markListSet);
535 visitor.donateAndDrain();
536 }
537 }
commit-queue@webkit.org3f922f92013-08-29 00:28:42 +0000538 if (m_vm->exception()) {
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000539 GCPHASE(MarkingException);
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000540 MARK_LOG_ROOT(visitor, "Exceptions");
commit-queue@webkit.org3f922f92013-08-29 00:28:42 +0000541 heapRootVisitor.visit(m_vm->addressOfException());
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000542 visitor.donateAndDrain();
543 }
oliver@apple.com32b720a2011-10-05 19:07:18 +0000544
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000545 {
546 GCPHASE(VisitStrongHandles);
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000547 MARK_LOG_ROOT(visitor, "Strong Handles");
ggaren@apple.com61b97002012-04-06 22:35:55 +0000548 m_handleSet.visitStrongHandles(heapRootVisitor);
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000549 visitor.donateAndDrain();
550 }
oliver@apple.com32b720a2011-10-05 19:07:18 +0000551
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000552 {
553 GCPHASE(HandleStack);
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000554 MARK_LOG_ROOT(visitor, "Handle Stack");
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000555 m_handleStack.visit(heapRootVisitor);
556 visitor.donateAndDrain();
557 }
558
559 {
fpizlo@apple.com746b8c52012-07-09 23:26:54 +0000560 GCPHASE(TraceCodeBlocksAndJITStubRoutines);
561 MARK_LOG_ROOT(visitor, "Trace Code Blocks and JIT Stub Routines");
fpizlo@apple.com195d7b82013-09-03 16:39:29 +0000562 m_codeBlocks.traceMarked(visitor);
fpizlo@apple.com746b8c52012-07-09 23:26:54 +0000563 m_jitStubRoutines.traceMarkedStubRoutines(visitor);
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000564 visitor.donateAndDrain();
565 }
566
567#if ENABLE(PARALLEL_GC)
568 {
569 GCPHASE(Convergence);
570 visitor.drainFromShared(SlotVisitor::MasterDrain);
571 }
572#endif
oliver@apple.com32b720a2011-10-05 19:07:18 +0000573 }
ggaren@apple.com6d8758c2011-04-14 23:20:25 +0000574
ggaren@apple.com76215812012-04-04 05:28:13 +0000575 // Weak references must be marked last because their liveness depends on
576 // the liveness of the rest of the object graph.
oliver@apple.com32b720a2011-10-05 19:07:18 +0000577 {
ggaren@apple.com76215812012-04-04 05:28:13 +0000578 GCPHASE(VisitingLiveWeakHandles);
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000579 MARK_LOG_ROOT(visitor, "Live Weak Handles");
fpizlo@apple.comf8e2ef82011-11-15 06:58:19 +0000580 while (true) {
ggaren@apple.com39281e22012-05-24 21:18:10 +0000581 m_objectSpace.visitWeakSets(heapRootVisitor);
fpizlo@apple.comf8e2ef82011-11-15 06:58:19 +0000582 harvestWeakReferences();
583 if (visitor.isEmpty())
584 break;
fpizlo@apple.com622ead72011-11-01 06:43:37 +0000585 {
586 ParallelModeEnabler enabler(visitor);
587 visitor.donateAndDrain();
588#if ENABLE(PARALLEL_GC)
589 visitor.drainFromShared(SlotVisitor::MasterDrain);
590#endif
591 }
fpizlo@apple.comf8e2ef82011-11-15 06:58:19 +0000592 }
oliver@apple.com32b720a2011-10-05 19:07:18 +0000593 }
ggaren@apple.com76215812012-04-04 05:28:13 +0000594
oliver@apple.com73792612011-10-12 00:24:12 +0000595 GCCOUNTER(VisitedValueCount, visitor.visitCount());
fpizlo@apple.comf5e1fe12011-11-10 21:59:39 +0000596
mhahnenberg@apple.com02e39c72012-10-12 19:38:35 +0000597 m_sharedData.didFinishMarking();
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000598#if ENABLE(OBJECT_MARK_LOGGING)
599 size_t visitCount = visitor.visitCount();
600#if ENABLE(PARALLEL_GC)
601 visitCount += m_sharedData.childVisitCount();
602#endif
commit-queue@webkit.orgf15eb3d2013-08-14 00:51:25 +0000603 MARK_LOG_MESSAGE2("\nNumber of live Objects after full GC %lu, took %.6f secs\n", visitCount, WTF::monotonicallyIncreasingTime() - gcStartTime);
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000604#endif
605
mhahnenberg@apple.come2789cf2013-09-09 00:11:57 +0000606 m_totalBytesVisited = visitor.bytesVisited();
607 m_totalBytesCopied = visitor.bytesCopied();
608#if ENABLE(PARALLEL_GC)
609 m_totalBytesVisited += m_sharedData.childBytesVisited();
610 m_totalBytesCopied += m_sharedData.childBytesCopied();
611#endif
612
oliver@apple.com433d02f2011-04-21 23:08:15 +0000613 visitor.reset();
msaboff@apple.com9d9eab62012-06-06 23:11:09 +0000614#if ENABLE(PARALLEL_GC)
615 m_sharedData.resetChildren();
616#endif
msaboff@apple.comd08f3502012-07-03 22:57:00 +0000617 m_sharedData.reset();
mhahnenberg@apple.com02e39c72012-10-12 19:38:35 +0000618}
619
620void Heap::copyBackingStores()
621{
622 m_storageSpace.startedCopying();
623 if (m_storageSpace.shouldDoCopyPhase()) {
624 m_sharedData.didStartCopying();
mhahnenberg@apple.com3ea84062012-10-18 19:44:25 +0000625 m_copyVisitor.startCopying();
626 m_copyVisitor.copyFromShared();
627 m_copyVisitor.doneCopying();
mhahnenberg@apple.comc061f5d2012-10-13 01:11:04 +0000628 // We need to wait for everybody to finish and return their CopiedBlocks
629 // before signaling that the phase is complete.
630 m_storageSpace.doneCopying();
mhahnenberg@apple.com02e39c72012-10-12 19:38:35 +0000631 m_sharedData.didFinishCopying();
mhahnenberg@apple.comc061f5d2012-10-13 01:11:04 +0000632 } else
633 m_storageSpace.doneCopying();
mjs6f821c82002-03-22 00:31:57 +0000634}
kocienda66a6d362001-08-24 14:24:45 +0000635
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000636size_t Heap::objectCount()
mjs6d32b072002-11-20 09:34:02 +0000637{
ggaren@apple.com041d0a22012-05-18 00:40:30 +0000638 return m_objectSpace.objectCount();
mjs6d32b072002-11-20 09:34:02 +0000639}
640
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +0000641size_t Heap::extraSize()
642{
643 return m_extraMemoryUsage + m_arrayBuffers.size();
644}
645
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000646size_t Heap::size()
eric@webkit.org30dc2a32010-05-15 08:46:23 +0000647{
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +0000648 return m_objectSpace.size() + m_storageSpace.size() + extraSize();
eric@webkit.org30dc2a32010-05-15 08:46:23 +0000649}
650
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000651size_t Heap::capacity()
ggaren@apple.com07050262011-01-30 03:32:52 +0000652{
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +0000653 return m_objectSpace.capacity() + m_storageSpace.capacity() + extraSize();
darin52b49432002-05-09 00:11:06 +0000654}
655
mhahnenberg@apple.come2789cf2013-09-09 00:11:57 +0000656size_t Heap::sizeAfterCollect()
657{
658 // The result here may not agree with the normal Heap::size().
659 // This is due to the fact that we only count live copied bytes
660 // rather than all used (including dead) copied bytes, thus it's
661 // always the case that m_totalBytesCopied <= m_storageSpace.size().
662 ASSERT(m_totalBytesCopied <= m_storageSpace.size());
663 return m_totalBytesVisited + m_totalBytesCopied + extraSize();
664}
665
ap@webkit.org960c28e2008-06-19 17:29:29 +0000666size_t Heap::protectedGlobalObjectCount()
ggaren@apple.com6e4309e2008-01-11 09:52:27 +0000667{
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000668 return forEachProtectedCell<CountIfGlobalObject>();
669}
ggaren@apple.com79ab2a92011-02-02 05:05:55 +0000670
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000671size_t Heap::globalObjectCount()
672{
mhahnenberg@apple.combee96a32013-09-16 19:48:48 +0000673 HeapIterationScope iterationScope(*this);
674 return m_objectSpace.forEachLiveCell<CountIfGlobalObject>(iterationScope);
ggaren@apple.com6e4309e2008-01-11 09:52:27 +0000675}
676
ap@webkit.org960c28e2008-06-19 17:29:29 +0000677size_t Heap::protectedObjectCount()
darin52b49432002-05-09 00:11:06 +0000678{
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000679 return forEachProtectedCell<Count>();
ggaren@apple.comb843ba82011-02-11 23:31:04 +0000680}
681
682PassOwnPtr<TypeCountSet> Heap::protectedObjectTypeCounts()
683{
ggaren@apple.com8a23d6a2011-06-09 19:44:37 +0000684 return forEachProtectedCell<RecordType>();
oliver@apple.combf9b7f82011-04-14 20:17:11 +0000685}
686
ggaren@apple.comb843ba82011-02-11 23:31:04 +0000687PassOwnPtr<TypeCountSet> Heap::objectTypeCounts()
ggaren@apple.com0fabb552010-02-11 21:28:52 +0000688{
mhahnenberg@apple.combee96a32013-09-16 19:48:48 +0000689 HeapIterationScope iterationScope(*this);
690 return m_objectSpace.forEachLiveCell<RecordType>(iterationScope);
ggaren@apple.com0fabb552010-02-11 21:28:52 +0000691}
692
ggaren@apple.comd7147572012-06-08 18:17:16 +0000693void Heap::deleteAllCompiledCode()
mhahnenberg@apple.comcab47522012-03-30 00:36:37 +0000694{
ggaren@apple.comd7147572012-06-08 18:17:16 +0000695 // If JavaScript is running, it's not safe to delete code, since we'll end
696 // up deleting code that is live on the stack.
mark.lam@apple.com30721252013-11-21 05:29:42 +0000697 if (m_vm->entryScope)
msaboff@apple.com03e811f2012-04-06 16:09:22 +0000698 return;
699
ggaren@apple.comd7147572012-06-08 18:17:16 +0000700 for (ExecutableBase* current = m_compiledCode.head(); current; current = current->next()) {
701 if (!current->isFunctionExecutable())
702 continue;
703 static_cast<FunctionExecutable*>(current)->clearCodeIfNotCompiling();
704 }
705
fpizlo@apple.com195d7b82013-09-03 16:39:29 +0000706 m_codeBlocks.clearMarks();
707 m_codeBlocks.deleteUnmarkedAndUnreferenced();
ggaren@apple.comd7147572012-06-08 18:17:16 +0000708}
709
710void Heap::deleteUnmarkedCompiledCode()
711{
712 ExecutableBase* next;
713 for (ExecutableBase* current = m_compiledCode.head(); current; current = next) {
714 next = current->next();
715 if (isMarked(current))
716 continue;
717
718 // We do this because executable memory is limited on some platforms and because
719 // CodeBlock requires eager finalization.
720 ExecutableBase::clearCodeVirtual(current);
721 m_compiledCode.remove(current);
722 }
723
fpizlo@apple.com195d7b82013-09-03 16:39:29 +0000724 m_codeBlocks.deleteUnmarkedAndUnreferenced();
fpizlo@apple.com746b8c52012-07-09 23:26:54 +0000725 m_jitStubRoutines.deleteUnmarkedJettisonedStubRoutines();
mhahnenberg@apple.comcab47522012-03-30 00:36:37 +0000726}
727
ggaren@apple.com6a429812009-12-14 08:13:24 +0000728void Heap::collectAllGarbage()
ggaren@apple.com9f980342008-10-15 23:33:07 +0000729{
commit-queue@webkit.orgf42601f2011-07-12 22:35:39 +0000730 if (!m_isSafeToCollect)
731 return;
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000732
mhahnenberg@apple.comac6f1fd2013-11-15 19:53:30 +0000733 DelayedReleaseScope delayedReleaseScope(m_objectSpace);
ggaren@apple.com4f1e8362011-06-08 06:16:32 +0000734 collect(DoSweep);
ggaren@apple.com7ddf61b2011-01-30 23:07:11 +0000735}
736
mhahnenberg@apple.com2b64eec02012-04-18 16:18:32 +0000737static double minute = 60.0;
738
ggaren@apple.com4f1e8362011-06-08 06:16:32 +0000739void Heap::collect(SweepToggle sweepToggle)
ggaren@apple.com7ddf61b2011-01-30 23:07:11 +0000740{
oliver@apple.coma03796a2013-07-25 04:01:20 +0000741#if ENABLE(ALLOCATION_LOGGING)
742 dataLogF("JSC GC starting collection.\n");
743#endif
744
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +0000745 double before = 0;
746 if (Options::logGC()) {
747 dataLog("[GC", sweepToggle == DoSweep ? " (eager sweep)" : "", ": ");
748 before = currentTimeMS();
749 }
750
fpizlo@apple.com8ce62fd2011-11-06 11:39:12 +0000751 SamplingRegion samplingRegion("Garbage Collection");
752
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000753 RELEASE_ASSERT(!m_deferralDepth);
oliver@apple.com32b720a2011-10-05 19:07:18 +0000754 GCPHASE(Collect);
andersca@apple.comb987aae2013-07-26 00:13:13 +0000755 ASSERT(vm()->currentThreadIsHoldingAPILock());
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000756 RELEASE_ASSERT(vm()->identifierTable == wtfThreadData().currentIdentifierTable());
commit-queue@webkit.orgf42601f2011-07-12 22:35:39 +0000757 ASSERT(m_isSafeToCollect);
ggaren@apple.com6a429812009-12-14 08:13:24 +0000758 JAVASCRIPTCORE_GC_BEGIN();
oliver@apple.com51b41da2013-01-23 21:44:29 +0000759 RELEASE_ASSERT(m_operationInProgress == NoOperation);
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000760
mhahnenberg@apple.com7a22bab2013-12-10 19:37:29 +0000761 {
762 RecursiveAllocationScope scope(*this);
763 m_vm->prepareToDiscardCode();
764 }
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000765
mhahnenberg@apple.com47c9c532012-06-05 20:38:21 +0000766 m_operationInProgress = Collection;
oliver@apple.com8cebf372013-07-25 04:03:46 +0000767 m_extraMemoryUsage = 0;
mhahnenberg@apple.com2b64eec02012-04-18 16:18:32 +0000768
dbates@webkit.org6122a1a2013-10-16 23:36:25 +0000769 if (m_activityCallback)
770 m_activityCallback->willCollect();
mhahnenberg@apple.com0eb606f2012-04-20 19:55:21 +0000771
commit-queue@webkit.orgf15eb3d2013-08-14 00:51:25 +0000772 double lastGCStartTime = WTF::monotonicallyIncreasingTime();
mhahnenberg@apple.com2b64eec02012-04-18 16:18:32 +0000773 if (lastGCStartTime - m_lastCodeDiscardTime > minute) {
ggaren@apple.comd7147572012-06-08 18:17:16 +0000774 deleteAllCompiledCode();
commit-queue@webkit.orgf15eb3d2013-08-14 00:51:25 +0000775 m_lastCodeDiscardTime = WTF::monotonicallyIncreasingTime();
mhahnenberg@apple.com2b64eec02012-04-18 16:18:32 +0000776 }
777
oliver@apple.com32b720a2011-10-05 19:07:18 +0000778 {
mhahnenberg@apple.combee96a32013-09-16 19:48:48 +0000779 GCPHASE(StopAllocation);
780 m_objectSpace.stopAllocating();
oliver@apple.com32b720a2011-10-05 19:07:18 +0000781 }
oliver@apple.comfa9733f2011-09-30 22:23:33 +0000782
ggaren@apple.combd41f7e2013-01-30 00:36:50 +0000783 markRoots();
fpizlo@apple.comf5e1fe12011-11-10 21:59:39 +0000784
oliver@apple.com32b720a2011-10-05 19:07:18 +0000785 {
ggaren@apple.com39281e22012-05-24 21:18:10 +0000786 GCPHASE(ReapingWeakHandles);
787 m_objectSpace.reapWeakSets();
788 }
789
ggaren@apple.comd7147572012-06-08 18:17:16 +0000790 JAVASCRIPTCORE_GC_MARKED();
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +0000791
792 {
793 GCPHASE(SweepingArrayBuffers);
794 m_arrayBuffers.sweep();
795 }
ggaren@apple.comd7147572012-06-08 18:17:16 +0000796
ggaren@apple.com39281e22012-05-24 21:18:10 +0000797 {
mhahnenberg@apple.com02e39c72012-10-12 19:38:35 +0000798 m_blockSnapshot.resize(m_objectSpace.blocks().set().size());
799 MarkedBlockSnapshotFunctor functor(m_blockSnapshot);
800 m_objectSpace.forEachBlock(functor);
801 }
802
803 copyBackingStores();
804
805 {
fpizlo@apple.comf5e1fe12011-11-10 21:59:39 +0000806 GCPHASE(FinalizeUnconditionalFinalizers);
807 finalizeUnconditionalFinalizers();
808 }
ggaren@apple.com39281e22012-05-24 21:18:10 +0000809
fpizlo@apple.comf5e1fe12011-11-10 21:59:39 +0000810 {
fpizlo@apple.com45928f82011-11-17 03:58:48 +0000811 GCPHASE(DeleteCodeBlocks);
ggaren@apple.comd7147572012-06-08 18:17:16 +0000812 deleteUnmarkedCompiledCode();
fpizlo@apple.com45928f82011-11-17 03:58:48 +0000813 }
ggaren@apple.com6a429812009-12-14 08:13:24 +0000814
ggaren@apple.comd17e17e2013-02-19 01:39:12 +0000815 {
816 GCPHASE(DeleteSourceProviderCaches);
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000817 m_vm->clearSourceProviderCaches();
ggaren@apple.comd17e17e2013-02-19 01:39:12 +0000818 }
819
ggaren@apple.com03c4f9d2011-02-11 00:20:17 +0000820 if (sweepToggle == DoSweep) {
fpizlo@apple.com8ce62fd2011-11-06 11:39:12 +0000821 SamplingRegion samplingRegion("Garbage Collection: Sweeping");
oliver@apple.com32b720a2011-10-05 19:07:18 +0000822 GCPHASE(Sweeping);
ggaren@apple.com041d0a22012-05-18 00:40:30 +0000823 m_objectSpace.sweep();
ggaren@apple.com75736172012-04-24 19:11:12 +0000824 m_objectSpace.shrink();
ggaren@apple.com03c4f9d2011-02-11 00:20:17 +0000825 }
ggaren@apple.com7ddf61b2011-01-30 23:07:11 +0000826
mhahnenberg@apple.com02e39c72012-10-12 19:38:35 +0000827 m_sweeper->startSweeping(m_blockSnapshot);
mhahnenberg@apple.comeb39abc2012-05-31 03:04:00 +0000828 m_bytesAbandoned = 0;
829
ggaren@apple.com39281e22012-05-24 21:18:10 +0000830 {
831 GCPHASE(ResetAllocators);
832 m_objectSpace.resetAllocators();
833 }
834
mhahnenberg@apple.come2789cf2013-09-09 00:11:57 +0000835 size_t currentHeapSize = sizeAfterCollect();
mhahnenberg@apple.com3a2fb032012-10-05 17:35:49 +0000836 if (Options::gcMaxHeapSize() && currentHeapSize > Options::gcMaxHeapSize())
837 HeapStatistics::exitWithFailure();
838
ggaren@apple.combd41f7e2013-01-30 00:36:50 +0000839 m_sizeAfterLastCollect = currentHeapSize;
ggaren@apple.com9a070932012-05-22 19:17:57 +0000840
ggaren@apple.combd41f7e2013-01-30 00:36:50 +0000841 // To avoid pathological GC churn in very small and very large heaps, we set
842 // the new allocation limit based on the current size of the heap, with a
843 // fixed minimum.
844 size_t maxHeapSize = max(minHeapSize(m_heapType, m_ramSize), proportionalHeapSize(currentHeapSize, m_ramSize));
845 m_bytesAllocatedLimit = maxHeapSize - currentHeapSize;
846
mhahnenberg@apple.com8b5cfd32012-04-20 00:05:37 +0000847 m_bytesAllocated = 0;
commit-queue@webkit.orgf15eb3d2013-08-14 00:51:25 +0000848 double lastGCEndTime = WTF::monotonicallyIncreasingTime();
mhahnenberg@apple.comf9133a12012-03-06 21:32:24 +0000849 m_lastGCLength = lastGCEndTime - lastGCStartTime;
ggaren@apple.comdef139e2012-09-26 02:27:54 +0000850
mhahnenberg@apple.com3a2fb032012-10-05 17:35:49 +0000851 if (Options::recordGCPauseTimes())
852 HeapStatistics::recordGCPauseTime(lastGCStartTime, lastGCEndTime);
oliver@apple.com51b41da2013-01-23 21:44:29 +0000853 RELEASE_ASSERT(m_operationInProgress == Collection);
854
mhahnenberg@apple.com47c9c532012-06-05 20:38:21 +0000855 m_operationInProgress = NoOperation;
ggaren@apple.com6a429812009-12-14 08:13:24 +0000856 JAVASCRIPTCORE_GC_END();
ggaren@apple.comdef139e2012-09-26 02:27:54 +0000857
858 if (Options::useZombieMode())
859 zombifyDeadObjects();
860
861 if (Options::objectsAreImmortal())
862 markDeadObjects();
863
mhahnenberg@apple.com3a2fb032012-10-05 17:35:49 +0000864 if (Options::showObjectStatistics())
865 HeapStatistics::showObjectStatistics(this);
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +0000866
867 if (Options::logGC()) {
868 double after = currentTimeMS();
869 dataLog(after - before, " ms, ", currentHeapSize / 1024, " kb]\n");
870 }
oliver@apple.coma03796a2013-07-25 04:01:20 +0000871
872#if ENABLE(ALLOCATION_LOGGING)
873 dataLogF("JSC GC finishing collection.\n");
874#endif
ggaren@apple.comdef139e2012-09-26 02:27:54 +0000875}
876
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000877bool Heap::collectIfNecessaryOrDefer()
878{
879 if (m_deferralDepth)
880 return false;
881
882 if (!shouldCollect())
883 return false;
884
885 collect(DoNotSweep);
886 return true;
887}
888
ggaren@apple.comdef139e2012-09-26 02:27:54 +0000889void Heap::markDeadObjects()
890{
mhahnenberg@apple.combee96a32013-09-16 19:48:48 +0000891 HeapIterationScope iterationScope(*this);
892 m_objectSpace.forEachDeadCell<MarkObject>(iterationScope);
ggaren@apple.com6a429812009-12-14 08:13:24 +0000893}
894
mhahnenberg@apple.com765a7de2013-04-15 23:17:51 +0000895void Heap::setActivityCallback(PassOwnPtr<GCActivityCallback> activityCallback)
ggaren@apple.com2ccf7a92010-08-03 20:34:17 +0000896{
897 m_activityCallback = activityCallback;
898}
899
ggaren@apple.comaa64ff32010-12-03 01:16:11 +0000900GCActivityCallback* Heap::activityCallback()
901{
mhahnenberg@apple.com765a7de2013-04-15 23:17:51 +0000902 return m_activityCallback.get();
ggaren@apple.comaa64ff32010-12-03 01:16:11 +0000903}
904
dbates@webkit.org0ba27042013-10-18 23:57:24 +0000905void Heap::setIncrementalSweeper(PassOwnPtr<IncrementalSweeper> sweeper)
906{
907 m_sweeper = sweeper;
908}
909
mhahnenberg@apple.comeb39abc2012-05-31 03:04:00 +0000910IncrementalSweeper* Heap::sweeper()
911{
mhahnenberg@apple.com765a7de2013-04-15 23:17:51 +0000912 return m_sweeper.get();
mhahnenberg@apple.comeb39abc2012-05-31 03:04:00 +0000913}
914
msaboff@apple.comfb5f4cc2012-05-14 23:47:53 +0000915void Heap::setGarbageCollectionTimerEnabled(bool enable)
916{
dbates@webkit.org156dd1a2013-10-18 17:22:55 +0000917 if (m_activityCallback)
918 m_activityCallback->setEnabled(enable);
msaboff@apple.comfb5f4cc2012-05-14 23:47:53 +0000919}
920
mhahnenberg@apple.com8b5cfd32012-04-20 00:05:37 +0000921void Heap::didAllocate(size_t bytes)
922{
dbates@webkit.org6122a1a2013-10-16 23:36:25 +0000923 if (m_activityCallback)
924 m_activityCallback->didAllocate(m_bytesAllocated + m_bytesAbandoned);
mhahnenberg@apple.com8b5cfd32012-04-20 00:05:37 +0000925 m_bytesAllocated += bytes;
926}
927
ggaren@apple.com06a8bb62012-09-12 04:08:18 +0000928bool Heap::isValidAllocation(size_t)
ggaren@apple.com99cc9d92011-05-26 21:46:09 +0000929{
ggaren@apple.com9a9a4b52013-04-18 19:32:17 +0000930 if (!isValidThreadState(m_vm))
ggaren@apple.com99cc9d92011-05-26 21:46:09 +0000931 return false;
932
ggaren@apple.com99cc9d92011-05-26 21:46:09 +0000933 if (m_operationInProgress != NoOperation)
934 return false;
935
936 return true;
937}
938
ggaren@apple.com75bf76c2011-10-05 22:44:36 +0000939void Heap::addFinalizer(JSCell* cell, Finalizer finalizer)
940{
ggaren@apple.comab71e972012-04-28 05:57:46 +0000941 WeakSet::allocate(cell, &m_finalizerOwner, reinterpret_cast<void*>(finalizer)); // Balanced by FinalizerOwner::finalize().
ggaren@apple.com75bf76c2011-10-05 22:44:36 +0000942}
943
944void Heap::FinalizerOwner::finalize(Handle<Unknown> handle, void* context)
945{
ggaren@apple.com76215812012-04-04 05:28:13 +0000946 HandleSlot slot = handle.slot();
ggaren@apple.com75bf76c2011-10-05 22:44:36 +0000947 Finalizer finalizer = reinterpret_cast<Finalizer>(context);
ggaren@apple.com76215812012-04-04 05:28:13 +0000948 finalizer(slot->asCell());
ggaren@apple.com61b97002012-04-06 22:35:55 +0000949 WeakSet::deallocate(WeakImpl::asWeakImpl(slot));
ggaren@apple.com75bf76c2011-10-05 22:44:36 +0000950}
951
ggaren@apple.comd7147572012-06-08 18:17:16 +0000952void Heap::addCompiledCode(ExecutableBase* executable)
mhahnenberg@apple.comcab47522012-03-30 00:36:37 +0000953{
ggaren@apple.comd7147572012-06-08 18:17:16 +0000954 m_compiledCode.append(executable);
mhahnenberg@apple.comcab47522012-03-30 00:36:37 +0000955}
956
ggaren@apple.comdef139e2012-09-26 02:27:54 +0000957class Zombify : public MarkedBlock::VoidFunctor {
mhahnenberg@apple.com66b96f22012-09-07 05:53:25 +0000958public:
mhahnenberg@apple.com66b96f22012-09-07 05:53:25 +0000959 void operator()(JSCell* cell)
960 {
mhahnenberg@apple.com66b96f22012-09-07 05:53:25 +0000961 void** current = reinterpret_cast<void**>(cell);
962
963 // We want to maintain zapped-ness because that's how we know if we've called
964 // the destructor.
965 if (cell->isZapped())
966 current++;
967
ggaren@apple.comdef139e2012-09-26 02:27:54 +0000968 void* limit = static_cast<void*>(reinterpret_cast<char*>(cell) + MarkedBlock::blockFor(cell)->cellSize());
mhahnenberg@apple.com66b96f22012-09-07 05:53:25 +0000969 for (; current < limit; current++)
970 *current = reinterpret_cast<void*>(0xbbadbeef);
971 }
mhahnenberg@apple.com66b96f22012-09-07 05:53:25 +0000972};
973
974void Heap::zombifyDeadObjects()
975{
ggaren@apple.comdef139e2012-09-26 02:27:54 +0000976 // Sweep now because destructors will crash once we're zombified.
mhahnenberg@apple.com66b96f22012-09-07 05:53:25 +0000977 m_objectSpace.sweep();
mhahnenberg@apple.combee96a32013-09-16 19:48:48 +0000978 HeapIterationScope iterationScope(*this);
979 m_objectSpace.forEachDeadCell<Zombify>(iterationScope);
mhahnenberg@apple.com66b96f22012-09-07 05:53:25 +0000980}
981
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000982void Heap::incrementDeferralDepth()
983{
984 RELEASE_ASSERT(m_deferralDepth < 100); // Sanity check to make sure this doesn't get ridiculous.
985
986 m_deferralDepth++;
987}
988
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +0000989void Heap::decrementDeferralDepth()
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000990{
991 RELEASE_ASSERT(m_deferralDepth >= 1);
992
993 m_deferralDepth--;
fpizlo@apple.com0e0d9312013-08-15 20:43:06 +0000994}
995
996void Heap::decrementDeferralDepthAndGCIfNeeded()
997{
998 decrementDeferralDepth();
oliver@apple.com284cc3d2013-07-25 04:00:33 +0000999 collectIfNecessaryOrDefer();
1000}
1001
mhahnenberg@apple.com4968e1a2013-12-18 22:50:40 +00001002void Heap::flushWriteBarrierBuffer(JSCell* cell)
1003{
1004#if ENABLE(GGC)
1005 m_writeBarrierBuffer.flush(*this);
1006 m_writeBarrierBuffer.add(cell);
1007#else
1008 UNUSED_PARAM(cell);
1009#endif
1010}
1011
cwzwarich@webkit.org3f782f62008-09-08 01:28:33 +00001012} // namespace JSC