kocienda | 66a6d36 | 2001-08-24 14:24:45 +0000 | [diff] [blame] | 1 | /* |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 2 | * Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008, 2009, 2011, 2013 Apple Inc. All rights reserved. |
eseidel | c504bad | 2007-10-24 21:31:27 +0000 | [diff] [blame] | 3 | * Copyright (C) 2007 Eric Seidel <eric@webkit.org> |
kocienda | 66a6d36 | 2001-08-24 14:24:45 +0000 | [diff] [blame] | 4 | * |
| 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 |
mjs | cdff33b | 2006-01-23 21:41:36 +0000 | [diff] [blame] | 17 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA |
mjs | 6f821c8 | 2002-03-22 00:31:57 +0000 | [diff] [blame] | 18 | * |
kocienda | 66a6d36 | 2001-08-24 14:24:45 +0000 | [diff] [blame] | 19 | */ |
| 20 | |
mjs | b64c50a | 2005-10-03 21:13:12 +0000 | [diff] [blame] | 21 | #include "config.h" |
ggaren@apple.com | a8b3854 | 2011-01-10 23:43:56 +0000 | [diff] [blame] | 22 | #include "Heap.h" |
darin | ff399e0 | 2002-11-23 07:49:05 +0000 | [diff] [blame] | 23 | |
ggaren@apple.com | 79ab2a9 | 2011-02-02 05:05:55 +0000 | [diff] [blame] | 24 | #include "CodeBlock.h" |
ggaren@apple.com | 0b32d09 | 2011-04-11 21:57:31 +0000 | [diff] [blame] | 25 | #include "ConservativeRoots.h" |
mhahnenberg@apple.com | 02e39c7 | 2012-10-12 19:38:35 +0000 | [diff] [blame] | 26 | #include "CopiedSpace.h" |
mark.lam@apple.com | a4fe7ab | 2012-11-09 03:03:44 +0000 | [diff] [blame] | 27 | #include "CopiedSpaceInlines.h" |
| 28 | #include "CopyVisitorInlines.h" |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 29 | #include "DFGWorklist.h" |
mhahnenberg@apple.com | ac6f1fd | 2013-11-15 19:53:30 +0000 | [diff] [blame] | 30 | #include "DelayedReleaseScope.h" |
ggaren@apple.com | 2ccf7a9 | 2010-08-03 20:34:17 +0000 | [diff] [blame] | 31 | #include "GCActivityCallback.h" |
fpizlo@apple.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 32 | #include "GCIncomingRefCountedSetInlines.h" |
mhahnenberg@apple.com | bee96a3 | 2013-09-16 19:48:48 +0000 | [diff] [blame] | 33 | #include "HeapIterationScope.h" |
ggaren@apple.com | d2b624b | 2011-05-24 23:46:35 +0000 | [diff] [blame] | 34 | #include "HeapRootVisitor.h" |
mhahnenberg@apple.com | 3a2fb03 | 2012-10-05 17:35:49 +0000 | [diff] [blame] | 35 | #include "HeapStatistics.h" |
mhahnenberg@apple.com | eb39abc | 2012-05-31 03:04:00 +0000 | [diff] [blame] | 36 | #include "IncrementalSweeper.h" |
weinig@apple.com | f5b90cd | 2009-01-16 00:24:24 +0000 | [diff] [blame] | 37 | #include "Interpreter.h" |
ggaren@apple.com | f086387 | 2007-12-03 07:53:03 +0000 | [diff] [blame] | 38 | #include "JSGlobalObject.h" |
ap@webkit.org | 01aff70 | 2008-08-20 07:23:06 +0000 | [diff] [blame] | 39 | #include "JSLock.h" |
darin@apple.com | c46cfaf | 2009-06-18 19:18:12 +0000 | [diff] [blame] | 40 | #include "JSONObject.h" |
fpizlo@apple.com | a4b4cbe | 2013-01-12 04:47:03 +0000 | [diff] [blame] | 41 | #include "Operations.h" |
mhahnenberg@apple.com | 7a22bab | 2013-12-10 19:37:29 +0000 | [diff] [blame] | 42 | #include "RecursiveAllocationScope.h" |
ggaren@apple.com | c2c8028 | 2011-01-15 01:52:28 +0000 | [diff] [blame] | 43 | #include "Tracing.h" |
oliver@apple.com | f0c01b8 | 2012-11-07 00:13:54 +0000 | [diff] [blame] | 44 | #include "UnlinkedCodeBlock.h" |
mhahnenberg@apple.com | 7a22bab | 2013-12-10 19:37:29 +0000 | [diff] [blame] | 45 | #include "VM.h" |
ggaren@apple.com | ab71e97 | 2012-04-28 05:57:46 +0000 | [diff] [blame] | 46 | #include "WeakSetInlines.h" |
ggaren@apple.com | f266349 | 2011-02-05 00:20:16 +0000 | [diff] [blame] | 47 | #include <algorithm> |
ggaren@apple.com | 9a07093 | 2012-05-22 19:17:57 +0000 | [diff] [blame] | 48 | #include <wtf/RAMSize.h> |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 49 | #include <wtf/CurrentTime.h> |
darin | d589360 | 2005-08-01 05:02:13 +0000 | [diff] [blame] | 50 | |
ggaren@apple.com | f266349 | 2011-02-05 00:20:16 +0000 | [diff] [blame] | 51 | using namespace std; |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 52 | using namespace JSC; |
ggaren@apple.com | f266349 | 2011-02-05 00:20:16 +0000 | [diff] [blame] | 53 | |
cwzwarich@webkit.org | 3f782f6 | 2008-09-08 01:28:33 +0000 | [diff] [blame] | 54 | namespace JSC { |
kocienda | 66a6d36 | 2001-08-24 14:24:45 +0000 | [diff] [blame] | 55 | |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 56 | namespace { |
| 57 | |
ggaren@apple.com | 9a07093 | 2012-05-22 19:17:57 +0000 | [diff] [blame] | 58 | static const size_t largeHeapSize = 32 * MB; // About 1.5X the average webpage. |
| 59 | static const size_t smallHeapSize = 1 * MB; // Matches the FastMalloc per-thread cache. |
oliver@apple.com | 3eb8037 | 2011-10-12 01:11:04 +0000 | [diff] [blame] | 60 | |
fpizlo@apple.com | ebccac0 | 2013-09-10 03:16:31 +0000 | [diff] [blame] | 61 | #define ENABLE_GC_LOGGING 0 |
mhahnenberg@apple.com | 8e937f8 | 2013-09-09 22:41:00 +0000 | [diff] [blame] | 62 | |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 63 | #if ENABLE(GC_LOGGING) |
oliver@apple.com | 7379261 | 2011-10-12 00:24:12 +0000 | [diff] [blame] | 64 | #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\"") \ |
| 69 | static type name arguments; \ |
| 70 | _Pragma("clang diagnostic pop") |
| 71 | #else |
| 72 | #define DEFINE_GC_LOGGING_GLOBAL(type, name, arguments) \ |
| 73 | static type name arguments; |
| 74 | #endif // COMPILER(CLANG) |
| 75 | |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 76 | struct 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.com | 01902c8 | 2012-11-22 04:23:36 +0000 | [diff] [blame] | 87 | 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.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 88 | } |
| 89 | double m_time; |
| 90 | double m_min; |
| 91 | double m_max; |
| 92 | size_t m_count; |
| 93 | const char* m_name; |
| 94 | }; |
| 95 | |
| 96 | struct GCTimerScope { |
| 97 | GCTimerScope(GCTimer* timer) |
| 98 | : m_timer(timer) |
commit-queue@webkit.org | f15eb3d | 2013-08-14 00:51:25 +0000 | [diff] [blame] | 99 | , m_start(WTF::monotonicallyIncreasingTime()) |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 100 | { |
| 101 | } |
| 102 | ~GCTimerScope() |
| 103 | { |
commit-queue@webkit.org | f15eb3d | 2013-08-14 00:51:25 +0000 | [diff] [blame] | 104 | double delta = WTF::monotonicallyIncreasingTime() - m_start; |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 105 | 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.com | 7379261 | 2011-10-12 00:24:12 +0000 | [diff] [blame] | 116 | struct 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.com | 01902c8 | 2012-11-22 04:23:36 +0000 | [diff] [blame] | 137 | 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.com | 7379261 | 2011-10-12 00:24:12 +0000 | [diff] [blame] | 138 | } |
| 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.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 145 | |
oliver@apple.com | 7379261 | 2011-10-12 00:24:12 +0000 | [diff] [blame] | 146 | #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.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 150 | #else |
| 151 | |
| 152 | #define GCPHASE(name) do { } while (false) |
| 153 | #define COND_GCPHASE(cond, name1, name2) do { } while (false) |
oliver@apple.com | 7379261 | 2011-10-12 00:24:12 +0000 | [diff] [blame] | 154 | #define GCCOUNTER(name, value) do { } while (false) |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 155 | #endif |
| 156 | |
ggaren@apple.com | 9a07093 | 2012-05-22 19:17:57 +0000 | [diff] [blame] | 157 | static inline size_t minHeapSize(HeapType heapType, size_t ramSize) |
fpizlo@apple.com | f49ce5c | 2011-08-02 20:40:17 +0000 | [diff] [blame] | 158 | { |
ggaren@apple.com | 9a07093 | 2012-05-22 19:17:57 +0000 | [diff] [blame] | 159 | if (heapType == LargeHeap) |
| 160 | return min(largeHeapSize, ramSize / 4); |
fpizlo@apple.com | c7b4721 | 2011-08-03 23:35:03 +0000 | [diff] [blame] | 161 | return smallHeapSize; |
fpizlo@apple.com | f49ce5c | 2011-08-02 20:40:17 +0000 | [diff] [blame] | 162 | } |
ggaren@apple.com | f266349 | 2011-02-05 00:20:16 +0000 | [diff] [blame] | 163 | |
ggaren@apple.com | 9a07093 | 2012-05-22 19:17:57 +0000 | [diff] [blame] | 164 | static 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.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 174 | static inline bool isValidSharedInstanceThreadState(VM* vm) |
ggaren@apple.com | 99cc9d9 | 2011-05-26 21:46:09 +0000 | [diff] [blame] | 175 | { |
andersca@apple.com | b987aae | 2013-07-26 00:13:13 +0000 | [diff] [blame] | 176 | return vm->currentThreadIsHoldingAPILock(); |
ggaren@apple.com | 99cc9d9 | 2011-05-26 21:46:09 +0000 | [diff] [blame] | 177 | } |
| 178 | |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 179 | static inline bool isValidThreadState(VM* vm) |
ggaren@apple.com | 99cc9d9 | 2011-05-26 21:46:09 +0000 | [diff] [blame] | 180 | { |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 181 | if (vm->identifierTable != wtfThreadData().currentIdentifierTable()) |
ggaren@apple.com | 99cc9d9 | 2011-05-26 21:46:09 +0000 | [diff] [blame] | 182 | return false; |
| 183 | |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 184 | if (vm->isSharedInstance() && !isValidSharedInstanceThreadState(vm)) |
ggaren@apple.com | 99cc9d9 | 2011-05-26 21:46:09 +0000 | [diff] [blame] | 185 | return false; |
| 186 | |
| 187 | return true; |
| 188 | } |
| 189 | |
ggaren@apple.com | def139e | 2012-09-26 02:27:54 +0000 | [diff] [blame] | 190 | struct 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.com | 041d0a2 | 2012-05-18 00:40:30 +0000 | [diff] [blame] | 199 | struct Count : public MarkedBlock::CountFunctor { |
| 200 | void operator()(JSCell*) { count(1); } |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 201 | }; |
| 202 | |
ggaren@apple.com | 041d0a2 | 2012-05-18 00:40:30 +0000 | [diff] [blame] | 203 | struct 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.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 211 | }; |
| 212 | |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 213 | class RecordType { |
| 214 | public: |
| 215 | typedef PassOwnPtr<TypeCountSet> ReturnType; |
| 216 | |
| 217 | RecordType(); |
| 218 | void operator()(JSCell*); |
| 219 | ReturnType returnValue(); |
| 220 | |
| 221 | private: |
| 222 | const char* typeName(JSCell*); |
| 223 | OwnPtr<TypeCountSet> m_typeCountSet; |
| 224 | }; |
| 225 | |
| 226 | inline RecordType::RecordType() |
| 227 | : m_typeCountSet(adoptPtr(new TypeCountSet)) |
| 228 | { |
| 229 | } |
| 230 | |
| 231 | inline 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 | |
| 239 | inline void RecordType::operator()(JSCell* cell) |
| 240 | { |
| 241 | m_typeCountSet->add(typeName(cell)); |
| 242 | } |
| 243 | |
| 244 | inline PassOwnPtr<TypeCountSet> RecordType::returnValue() |
| 245 | { |
| 246 | return m_typeCountSet.release(); |
| 247 | } |
| 248 | |
| 249 | } // anonymous namespace |
| 250 | |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 251 | Heap::Heap(VM* vm, HeapType heapType) |
ggaren@apple.com | 9a07093 | 2012-05-22 19:17:57 +0000 | [diff] [blame] | 252 | : m_heapType(heapType) |
| 253 | , m_ramSize(ramSize()) |
| 254 | , m_minBytesPerCycle(minHeapSize(m_heapType, m_ramSize)) |
mhahnenberg@apple.com | 98daed0 | 2012-04-25 01:29:42 +0000 | [diff] [blame] | 255 | , m_sizeAfterLastCollect(0) |
mhahnenberg@apple.com | 8b5cfd3 | 2012-04-20 00:05:37 +0000 | [diff] [blame] | 256 | , m_bytesAllocatedLimit(m_minBytesPerCycle) |
| 257 | , m_bytesAllocated(0) |
mhahnenberg@apple.com | 376a417 | 2012-05-03 22:31:43 +0000 | [diff] [blame] | 258 | , m_bytesAbandoned(0) |
mhahnenberg@apple.com | e2789cf | 2013-09-09 00:11:57 +0000 | [diff] [blame] | 259 | , m_totalBytesVisited(0) |
| 260 | , m_totalBytesCopied(0) |
fpizlo@apple.com | f49ce5c | 2011-08-02 20:40:17 +0000 | [diff] [blame] | 261 | , m_operationInProgress(NoOperation) |
mhahnenberg@apple.com | 123e67f | 2013-04-01 16:01:01 +0000 | [diff] [blame] | 262 | , m_blockAllocator() |
oliver@apple.com | 44d8954 | 2011-09-20 18:31:37 +0000 | [diff] [blame] | 263 | , m_objectSpace(this) |
mhahnenberg@apple.com | 5d0b30a | 2012-01-19 21:49:56 +0000 | [diff] [blame] | 264 | , m_storageSpace(this) |
oliver@apple.com | 8cebf37 | 2013-07-25 04:03:46 +0000 | [diff] [blame] | 265 | , m_extraMemoryUsage(0) |
ggaren@apple.com | 123f685 | 2011-03-14 04:56:46 +0000 | [diff] [blame] | 266 | , m_machineThreads(this) |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 267 | , m_sharedData(vm) |
mhahnenberg@apple.com | c58d54d | 2011-12-16 19:06:44 +0000 | [diff] [blame] | 268 | , m_slotVisitor(m_sharedData) |
mhahnenberg@apple.com | 02e39c7 | 2012-10-12 19:38:35 +0000 | [diff] [blame] | 269 | , m_copyVisitor(m_sharedData) |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 270 | , m_handleSet(vm) |
commit-queue@webkit.org | f42601f | 2011-07-12 22:35:39 +0000 | [diff] [blame] | 271 | , m_isSafeToCollect(false) |
mhahnenberg@apple.com | 4968e1a | 2013-12-18 22:50:40 +0000 | [diff] [blame^] | 272 | #if ENABLE(GGC) |
| 273 | , m_writeBarrierBuffer(128) |
| 274 | #endif |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 275 | , m_vm(vm) |
mhahnenberg@apple.com | f9133a1 | 2012-03-06 21:32:24 +0000 | [diff] [blame] | 276 | , m_lastGCLength(0) |
commit-queue@webkit.org | f15eb3d | 2013-08-14 00:51:25 +0000 | [diff] [blame] | 277 | , m_lastCodeDiscardTime(WTF::monotonicallyIncreasingTime()) |
mhahnenberg@apple.com | a7ec41b | 2012-06-19 19:17:31 +0000 | [diff] [blame] | 278 | , m_activityCallback(DefaultGCActivityCallback::create(this)) |
| 279 | , m_sweeper(IncrementalSweeper::create(this)) |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 280 | , m_deferralDepth(0) |
ap@webkit.org | 960c28e | 2008-06-19 17:29:29 +0000 | [diff] [blame] | 281 | { |
mhahnenberg@apple.com | 5d0b30a | 2012-01-19 21:49:56 +0000 | [diff] [blame] | 282 | m_storageSpace.init(); |
ap@webkit.org | 960c28e | 2008-06-19 17:29:29 +0000 | [diff] [blame] | 283 | } |
darin | 070530c | 2002-10-07 21:06:29 +0000 | [diff] [blame] | 284 | |
ap@webkit.org | 0b7e63b | 2008-06-26 18:59:26 +0000 | [diff] [blame] | 285 | Heap::~Heap() |
| 286 | { |
darin@apple.com | b6295d4 | 2008-10-02 23:48:47 +0000 | [diff] [blame] | 287 | } |
| 288 | |
mhahnenberg@apple.com | 2e132e4 | 2012-05-03 00:14:05 +0000 | [diff] [blame] | 289 | bool Heap::isPagedOut(double deadline) |
| 290 | { |
| 291 | return m_objectSpace.isPagedOut(deadline) || m_storageSpace.isPagedOut(deadline); |
| 292 | } |
| 293 | |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 294 | // The VM is being destroyed and the collector will never run again. |
ggaren@apple.com | c143e90 | 2012-04-28 20:51:27 +0000 | [diff] [blame] | 295 | // Run all pending finalizers now because we won't get another chance. |
| 296 | void Heap::lastChanceToFinalize() |
darin@apple.com | b6295d4 | 2008-10-02 23:48:47 +0000 | [diff] [blame] | 297 | { |
mark.lam@apple.com | 3072125 | 2013-11-21 05:29:42 +0000 | [diff] [blame] | 298 | RELEASE_ASSERT(!m_vm->entryScope); |
oliver@apple.com | 903b0c0 | 2013-01-24 01:40:37 +0000 | [diff] [blame] | 299 | RELEASE_ASSERT(m_operationInProgress == NoOperation); |
ap@webkit.org | 8672bcf | 2008-08-11 12:01:26 +0000 | [diff] [blame] | 300 | |
ggaren@apple.com | 96fa0e7 | 2012-05-23 20:47:46 +0000 | [diff] [blame] | 301 | m_objectSpace.lastChanceToFinalize(); |
ap@webkit.org | 0b7e63b | 2008-06-26 18:59:26 +0000 | [diff] [blame] | 302 | } |
| 303 | |
ggaren@apple.com | 0705026 | 2011-01-30 03:32:52 +0000 | [diff] [blame] | 304 | void Heap::reportExtraMemoryCostSlowCase(size_t cost) |
mjs | 06ed466 | 2007-07-25 21:50:00 +0000 | [diff] [blame] | 305 | { |
| 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.com | 8b5cfd3 | 2012-04-20 00:05:37 +0000 | [diff] [blame] | 317 | didAllocate(cost); |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 318 | collectIfNecessaryOrDefer(); |
mjs | 06ed466 | 2007-07-25 21:50:00 +0000 | [diff] [blame] | 319 | } |
| 320 | |
mhahnenberg@apple.com | 98daed0 | 2012-04-25 01:29:42 +0000 | [diff] [blame] | 321 | void 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.com | 376a417 | 2012-05-03 22:31:43 +0000 | [diff] [blame] | 331 | didAbandon(abandonedBytes); |
| 332 | } |
| 333 | |
| 334 | void Heap::didAbandon(size_t bytes) |
| 335 | { |
dbates@webkit.org | 6122a1a | 2013-10-16 23:36:25 +0000 | [diff] [blame] | 336 | if (m_activityCallback) |
| 337 | m_activityCallback->didAllocate(m_bytesAllocated + m_bytesAbandoned); |
mhahnenberg@apple.com | 376a417 | 2012-05-03 22:31:43 +0000 | [diff] [blame] | 338 | m_bytesAbandoned += bytes; |
mhahnenberg@apple.com | 98daed0 | 2012-04-25 01:29:42 +0000 | [diff] [blame] | 339 | } |
| 340 | |
ggaren@apple.com | dc067b6 | 2009-05-01 22:43:39 +0000 | [diff] [blame] | 341 | void Heap::protect(JSValue k) |
mjs | c4bb295 | 2005-12-13 11:06:10 +0000 | [diff] [blame] | 342 | { |
mjs | 9f1f97a | 2007-03-20 23:57:01 +0000 | [diff] [blame] | 343 | ASSERT(k); |
andersca@apple.com | b987aae | 2013-07-26 00:13:13 +0000 | [diff] [blame] | 344 | ASSERT(m_vm->currentThreadIsHoldingAPILock()); |
mjs | c4bb295 | 2005-12-13 11:06:10 +0000 | [diff] [blame] | 345 | |
weinig@apple.com | 92fdaef | 2009-01-20 00:54:18 +0000 | [diff] [blame] | 346 | if (!k.isCell()) |
ap@webkit.org | 84ea6b9 | 2008-06-04 16:29:49 +0000 | [diff] [blame] | 347 | return; |
mjs | c4bb295 | 2005-12-13 11:06:10 +0000 | [diff] [blame] | 348 | |
weinig@apple.com | 92fdaef | 2009-01-20 00:54:18 +0000 | [diff] [blame] | 349 | m_protectedValues.add(k.asCell()); |
mjs | c4bb295 | 2005-12-13 11:06:10 +0000 | [diff] [blame] | 350 | } |
| 351 | |
barraclough@apple.com | 88bb311 | 2010-03-24 07:11:51 +0000 | [diff] [blame] | 352 | bool Heap::unprotect(JSValue k) |
mjs | c4bb295 | 2005-12-13 11:06:10 +0000 | [diff] [blame] | 353 | { |
mjs | 9f1f97a | 2007-03-20 23:57:01 +0000 | [diff] [blame] | 354 | ASSERT(k); |
andersca@apple.com | b987aae | 2013-07-26 00:13:13 +0000 | [diff] [blame] | 355 | ASSERT(m_vm->currentThreadIsHoldingAPILock()); |
mjs | c4bb295 | 2005-12-13 11:06:10 +0000 | [diff] [blame] | 356 | |
weinig@apple.com | 92fdaef | 2009-01-20 00:54:18 +0000 | [diff] [blame] | 357 | if (!k.isCell()) |
barraclough@apple.com | 88bb311 | 2010-03-24 07:11:51 +0000 | [diff] [blame] | 358 | return false; |
mjs | c4bb295 | 2005-12-13 11:06:10 +0000 | [diff] [blame] | 359 | |
barraclough@apple.com | 88bb311 | 2010-03-24 07:11:51 +0000 | [diff] [blame] | 360 | return m_protectedValues.remove(k.asCell()); |
mjs | c4bb295 | 2005-12-13 11:06:10 +0000 | [diff] [blame] | 361 | } |
| 362 | |
fpizlo@apple.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 363 | void 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.com | e178918 | 2011-05-26 23:47:40 +0000 | [diff] [blame] | 371 | void Heap::markProtectedObjects(HeapRootVisitor& heapRootVisitor) |
ap@webkit.org | 960c28e | 2008-06-19 17:29:29 +0000 | [diff] [blame] | 372 | { |
ap@webkit.org | 32dab94 | 2008-07-02 07:00:53 +0000 | [diff] [blame] | 373 | ProtectCountSet::iterator end = m_protectedValues.end(); |
ggaren@apple.com | 7c9ff4d | 2011-01-21 04:07:38 +0000 | [diff] [blame] | 374 | for (ProtectCountSet::iterator it = m_protectedValues.begin(); it != end; ++it) |
benjamin@webkit.org | ee55405 | 2012-10-07 23:12:07 +0000 | [diff] [blame] | 375 | heapRootVisitor.visit(&it->key); |
mjs | 99f74d9 | 2004-04-23 22:40:31 +0000 | [diff] [blame] | 376 | } |
| 377 | |
oliver@apple.com | 4e3f965 | 2013-04-08 04:14:50 +0000 | [diff] [blame] | 378 | void Heap::pushTempSortVector(Vector<ValueStringPair, 0, UnsafeVectorOverflow>* tempVector) |
msaboff@apple.com | 86ea014 | 2010-12-09 18:27:13 +0000 | [diff] [blame] | 379 | { |
| 380 | m_tempSortingVectors.append(tempVector); |
| 381 | } |
| 382 | |
oliver@apple.com | 4e3f965 | 2013-04-08 04:14:50 +0000 | [diff] [blame] | 383 | void Heap::popTempSortVector(Vector<ValueStringPair, 0, UnsafeVectorOverflow>* tempVector) |
msaboff@apple.com | 86ea014 | 2010-12-09 18:27:13 +0000 | [diff] [blame] | 384 | { |
| 385 | ASSERT_UNUSED(tempVector, tempVector == m_tempSortingVectors.last()); |
| 386 | m_tempSortingVectors.removeLast(); |
| 387 | } |
mhahnenberg@apple.com | 5d0b30a | 2012-01-19 21:49:56 +0000 | [diff] [blame] | 388 | |
ggaren@apple.com | e178918 | 2011-05-26 23:47:40 +0000 | [diff] [blame] | 389 | void Heap::markTempSortVectors(HeapRootVisitor& heapRootVisitor) |
msaboff@apple.com | 86ea014 | 2010-12-09 18:27:13 +0000 | [diff] [blame] | 390 | { |
oliver@apple.com | 4e3f965 | 2013-04-08 04:14:50 +0000 | [diff] [blame] | 391 | typedef Vector<Vector<ValueStringPair, 0, UnsafeVectorOverflow>* > VectorOfValueStringVectors; |
msaboff@apple.com | 86ea014 | 2010-12-09 18:27:13 +0000 | [diff] [blame] | 392 | |
| 393 | VectorOfValueStringVectors::iterator end = m_tempSortingVectors.end(); |
| 394 | for (VectorOfValueStringVectors::iterator it = m_tempSortingVectors.begin(); it != end; ++it) { |
oliver@apple.com | 4e3f965 | 2013-04-08 04:14:50 +0000 | [diff] [blame] | 395 | Vector<ValueStringPair, 0, UnsafeVectorOverflow>* tempSortingVector = *it; |
msaboff@apple.com | 86ea014 | 2010-12-09 18:27:13 +0000 | [diff] [blame] | 396 | |
| 397 | Vector<ValueStringPair>::iterator vectorEnd = tempSortingVector->end(); |
ggaren@apple.com | 7c9ff4d | 2011-01-21 04:07:38 +0000 | [diff] [blame] | 398 | for (Vector<ValueStringPair>::iterator vectorIt = tempSortingVector->begin(); vectorIt != vectorEnd; ++vectorIt) { |
msaboff@apple.com | 86ea014 | 2010-12-09 18:27:13 +0000 | [diff] [blame] | 399 | if (vectorIt->first) |
ggaren@apple.com | a683389 | 2011-06-08 04:57:08 +0000 | [diff] [blame] | 400 | heapRootVisitor.visit(&vectorIt->first); |
ggaren@apple.com | 7c9ff4d | 2011-01-21 04:07:38 +0000 | [diff] [blame] | 401 | } |
msaboff@apple.com | 86ea014 | 2010-12-09 18:27:13 +0000 | [diff] [blame] | 402 | } |
| 403 | } |
ggaren@apple.com | 86f2195 | 2011-01-22 04:27:18 +0000 | [diff] [blame] | 404 | |
fpizlo@apple.com | d13163d | 2011-09-03 05:14:04 +0000 | [diff] [blame] | 405 | void Heap::harvestWeakReferences() |
| 406 | { |
| 407 | m_slotVisitor.harvestWeakReferences(); |
| 408 | } |
| 409 | |
fpizlo@apple.com | f5e1fe1 | 2011-11-10 21:59:39 +0000 | [diff] [blame] | 410 | void Heap::finalizeUnconditionalFinalizers() |
| 411 | { |
| 412 | m_slotVisitor.finalizeUnconditionalFinalizers(); |
| 413 | } |
| 414 | |
mark.lam@apple.com | 4fbb9c3 | 2012-10-09 07:12:56 +0000 | [diff] [blame] | 415 | inline JSStack& Heap::stack() |
ggaren@apple.com | 86f2195 | 2011-01-22 04:27:18 +0000 | [diff] [blame] | 416 | { |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 417 | return m_vm->interpreter->stack(); |
ggaren@apple.com | 86f2195 | 2011-01-22 04:27:18 +0000 | [diff] [blame] | 418 | } |
| 419 | |
mhahnenberg@apple.com | bee96a3 | 2013-09-16 19:48:48 +0000 | [diff] [blame] | 420 | void Heap::willStartIterating() |
mark.lam@apple.com | d4eb0d3 | 2013-04-17 16:32:44 +0000 | [diff] [blame] | 421 | { |
mhahnenberg@apple.com | bee96a3 | 2013-09-16 19:48:48 +0000 | [diff] [blame] | 422 | m_objectSpace.willStartIterating(); |
| 423 | } |
| 424 | |
| 425 | void Heap::didFinishIterating() |
| 426 | { |
| 427 | m_objectSpace.didFinishIterating(); |
mark.lam@apple.com | d4eb0d3 | 2013-04-17 16:32:44 +0000 | [diff] [blame] | 428 | } |
| 429 | |
oliver@apple.com | 1db480d | 2011-06-28 01:32:01 +0000 | [diff] [blame] | 430 | void Heap::getConservativeRegisterRoots(HashSet<JSCell*>& roots) |
| 431 | { |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 432 | ASSERT(isValidThreadState(m_vm)); |
mark.lam@apple.com | 4fbb9c3 | 2012-10-09 07:12:56 +0000 | [diff] [blame] | 433 | 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.com | 1db480d | 2011-06-28 01:32:01 +0000 | [diff] [blame] | 438 | setMarked(registerRoots[i]); |
| 439 | roots.add(registerRoots[i]); |
| 440 | } |
oliver@apple.com | 1db480d | 2011-06-28 01:32:01 +0000 | [diff] [blame] | 441 | } |
| 442 | |
ggaren@apple.com | bd41f7e | 2013-01-30 00:36:50 +0000 | [diff] [blame] | 443 | void Heap::markRoots() |
kocienda | 66a6d36 | 2001-08-24 14:24:45 +0000 | [diff] [blame] | 444 | { |
fpizlo@apple.com | 8ce62fd | 2011-11-06 11:39:12 +0000 | [diff] [blame] | 445 | SamplingRegion samplingRegion("Garbage Collection: Tracing"); |
| 446 | |
ggaren@apple.com | bd41f7e | 2013-01-30 00:36:50 +0000 | [diff] [blame] | 447 | GCPHASE(MarkRoots); |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 448 | ASSERT(isValidThreadState(m_vm)); |
ap@webkit.org | 01aff70 | 2008-08-20 07:23:06 +0000 | [diff] [blame] | 449 | |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 450 | #if ENABLE(OBJECT_MARK_LOGGING) |
commit-queue@webkit.org | f15eb3d | 2013-08-14 00:51:25 +0000 | [diff] [blame] | 451 | double gcStartTime = WTF::monotonicallyIncreasingTime(); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 452 | #endif |
| 453 | |
ggaren@apple.com | 02bf040 | 2011-03-16 18:35:49 +0000 | [diff] [blame] | 454 | void* dummy; |
fpizlo@apple.com | 706f5f3 | 2011-09-21 23:36:35 +0000 | [diff] [blame] | 455 | |
ggaren@apple.com | 99cc9d9 | 2011-05-26 21:46:09 +0000 | [diff] [blame] | 456 | // 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.com | 5d0b30a | 2012-01-19 21:49:56 +0000 | [diff] [blame] | 458 | ConservativeRoots machineThreadRoots(&m_objectSpace.blocks(), &m_storageSpace); |
fpizlo@apple.com | 746b8c5 | 2012-07-09 23:26:54 +0000 | [diff] [blame] | 459 | m_jitStubRoutines.clearMarks(); |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 460 | { |
| 461 | GCPHASE(GatherConservativeRoots); |
| 462 | m_machineThreads.gatherConservativeRoots(machineThreadRoots, &dummy); |
| 463 | } |
ggaren@apple.com | 02bf040 | 2011-03-16 18:35:49 +0000 | [diff] [blame] | 464 | |
mark.lam@apple.com | 4fbb9c3 | 2012-10-09 07:12:56 +0000 | [diff] [blame] | 465 | ConservativeRoots stackRoots(&m_objectSpace.blocks(), &m_storageSpace); |
fpizlo@apple.com | 195d7b8 | 2013-09-03 16:39:29 +0000 | [diff] [blame] | 466 | m_codeBlocks.clearMarks(); |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 467 | { |
mark.lam@apple.com | 4fbb9c3 | 2012-10-09 07:12:56 +0000 | [diff] [blame] | 468 | GCPHASE(GatherStackRoots); |
fpizlo@apple.com | 195d7b8 | 2013-09-03 16:39:29 +0000 | [diff] [blame] | 469 | stack().gatherConservativeRoots(stackRoots, m_jitStubRoutines, m_codeBlocks); |
fpizlo@apple.com | 7969ed7 | 2013-11-28 00:22:43 +0000 | [diff] [blame] | 470 | stack().sanitizeStack(); |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 471 | } |
msaboff@apple.com | 1283039 | 2012-05-21 05:42:56 +0000 | [diff] [blame] | 472 | |
| 473 | #if ENABLE(DFG_JIT) |
| 474 | ConservativeRoots scratchBufferRoots(&m_objectSpace.blocks(), &m_storageSpace); |
| 475 | { |
| 476 | GCPHASE(GatherScratchBufferRoots); |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 477 | m_vm->gatherConservativeRoots(scratchBufferRoots); |
msaboff@apple.com | 1283039 | 2012-05-21 05:42:56 +0000 | [diff] [blame] | 478 | } |
| 479 | #endif |
| 480 | |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 481 | { |
mhahnenberg@apple.com | 23421aa | 2013-09-08 23:36:40 +0000 | [diff] [blame] | 482 | GCPHASE(ClearLivenessData); |
| 483 | m_objectSpace.clearNewlyAllocated(); |
ggaren@apple.com | 041d0a2 | 2012-05-18 00:40:30 +0000 | [diff] [blame] | 484 | m_objectSpace.clearMarks(); |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 485 | } |
ggaren@apple.com | 2029c2a | 2011-01-15 03:28:16 +0000 | [diff] [blame] | 486 | |
mhahnenberg@apple.com | 02e39c7 | 2012-10-12 19:38:35 +0000 | [diff] [blame] | 487 | m_sharedData.didStartMarking(); |
ggaren@apple.com | 6e1f8c1 | 2011-06-16 22:01:43 +0000 | [diff] [blame] | 488 | SlotVisitor& visitor = m_slotVisitor; |
msaboff@apple.com | d08f350 | 2012-07-03 22:57:00 +0000 | [diff] [blame] | 489 | visitor.setup(); |
ggaren@apple.com | 28cfb35 | 2011-06-10 00:02:23 +0000 | [diff] [blame] | 490 | HeapRootVisitor heapRootVisitor(visitor); |
oliver@apple.com | c094933 | 2011-09-29 22:52:45 +0000 | [diff] [blame] | 491 | |
oliver@apple.com | 7379261 | 2011-10-12 00:24:12 +0000 | [diff] [blame] | 492 | { |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 493 | ParallelModeEnabler enabler(visitor); |
oliver@apple.com | f0c01b8 | 2012-11-07 00:13:54 +0000 | [diff] [blame] | 494 | |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 495 | m_vm->smallStrings.visitStrongReferences(visitor); |
oliver@apple.com | e722ad0 | 2013-01-09 02:37:29 +0000 | [diff] [blame] | 496 | |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 497 | { |
| 498 | GCPHASE(VisitMachineRoots); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 499 | MARK_LOG_ROOT(visitor, "C++ Stack"); |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 500 | visitor.append(machineThreadRoots); |
| 501 | visitor.donateAndDrain(); |
| 502 | } |
| 503 | { |
mark.lam@apple.com | 4fbb9c3 | 2012-10-09 07:12:56 +0000 | [diff] [blame] | 504 | GCPHASE(VisitStackRoots); |
| 505 | MARK_LOG_ROOT(visitor, "Stack"); |
| 506 | visitor.append(stackRoots); |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 507 | visitor.donateAndDrain(); |
| 508 | } |
msaboff@apple.com | 1283039 | 2012-05-21 05:42:56 +0000 | [diff] [blame] | 509 | #if ENABLE(DFG_JIT) |
| 510 | { |
| 511 | GCPHASE(VisitScratchBufferRoots); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 512 | MARK_LOG_ROOT(visitor, "Scratch Buffers"); |
msaboff@apple.com | 1283039 | 2012-05-21 05:42:56 +0000 | [diff] [blame] | 513 | visitor.append(scratchBufferRoots); |
| 514 | visitor.donateAndDrain(); |
| 515 | } |
| 516 | #endif |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 517 | { |
| 518 | GCPHASE(VisitProtectedObjects); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 519 | MARK_LOG_ROOT(visitor, "Protected Objects"); |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 520 | markProtectedObjects(heapRootVisitor); |
| 521 | visitor.donateAndDrain(); |
| 522 | } |
| 523 | { |
| 524 | GCPHASE(VisitTempSortVectors); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 525 | MARK_LOG_ROOT(visitor, "Temp Sort Vectors"); |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 526 | markTempSortVectors(heapRootVisitor); |
| 527 | visitor.donateAndDrain(); |
| 528 | } |
| 529 | |
| 530 | { |
| 531 | GCPHASE(MarkingArgumentBuffers); |
| 532 | if (m_markListSet && m_markListSet->size()) { |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 533 | MARK_LOG_ROOT(visitor, "Argument Buffers"); |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 534 | MarkedArgumentBuffer::markLists(heapRootVisitor, *m_markListSet); |
| 535 | visitor.donateAndDrain(); |
| 536 | } |
| 537 | } |
commit-queue@webkit.org | 3f922f9 | 2013-08-29 00:28:42 +0000 | [diff] [blame] | 538 | if (m_vm->exception()) { |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 539 | GCPHASE(MarkingException); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 540 | MARK_LOG_ROOT(visitor, "Exceptions"); |
commit-queue@webkit.org | 3f922f9 | 2013-08-29 00:28:42 +0000 | [diff] [blame] | 541 | heapRootVisitor.visit(m_vm->addressOfException()); |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 542 | visitor.donateAndDrain(); |
| 543 | } |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 544 | |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 545 | { |
| 546 | GCPHASE(VisitStrongHandles); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 547 | MARK_LOG_ROOT(visitor, "Strong Handles"); |
ggaren@apple.com | 61b9700 | 2012-04-06 22:35:55 +0000 | [diff] [blame] | 548 | m_handleSet.visitStrongHandles(heapRootVisitor); |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 549 | visitor.donateAndDrain(); |
| 550 | } |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 551 | |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 552 | { |
| 553 | GCPHASE(HandleStack); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 554 | MARK_LOG_ROOT(visitor, "Handle Stack"); |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 555 | m_handleStack.visit(heapRootVisitor); |
| 556 | visitor.donateAndDrain(); |
| 557 | } |
| 558 | |
| 559 | { |
fpizlo@apple.com | 746b8c5 | 2012-07-09 23:26:54 +0000 | [diff] [blame] | 560 | GCPHASE(TraceCodeBlocksAndJITStubRoutines); |
| 561 | MARK_LOG_ROOT(visitor, "Trace Code Blocks and JIT Stub Routines"); |
fpizlo@apple.com | 195d7b8 | 2013-09-03 16:39:29 +0000 | [diff] [blame] | 562 | m_codeBlocks.traceMarked(visitor); |
fpizlo@apple.com | 746b8c5 | 2012-07-09 23:26:54 +0000 | [diff] [blame] | 563 | m_jitStubRoutines.traceMarkedStubRoutines(visitor); |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 564 | visitor.donateAndDrain(); |
| 565 | } |
| 566 | |
| 567 | #if ENABLE(PARALLEL_GC) |
| 568 | { |
| 569 | GCPHASE(Convergence); |
| 570 | visitor.drainFromShared(SlotVisitor::MasterDrain); |
| 571 | } |
| 572 | #endif |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 573 | } |
ggaren@apple.com | 6d8758c | 2011-04-14 23:20:25 +0000 | [diff] [blame] | 574 | |
ggaren@apple.com | 7621581 | 2012-04-04 05:28:13 +0000 | [diff] [blame] | 575 | // Weak references must be marked last because their liveness depends on |
| 576 | // the liveness of the rest of the object graph. |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 577 | { |
ggaren@apple.com | 7621581 | 2012-04-04 05:28:13 +0000 | [diff] [blame] | 578 | GCPHASE(VisitingLiveWeakHandles); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 579 | MARK_LOG_ROOT(visitor, "Live Weak Handles"); |
fpizlo@apple.com | f8e2ef8 | 2011-11-15 06:58:19 +0000 | [diff] [blame] | 580 | while (true) { |
ggaren@apple.com | 39281e2 | 2012-05-24 21:18:10 +0000 | [diff] [blame] | 581 | m_objectSpace.visitWeakSets(heapRootVisitor); |
fpizlo@apple.com | f8e2ef8 | 2011-11-15 06:58:19 +0000 | [diff] [blame] | 582 | harvestWeakReferences(); |
| 583 | if (visitor.isEmpty()) |
| 584 | break; |
fpizlo@apple.com | 622ead7 | 2011-11-01 06:43:37 +0000 | [diff] [blame] | 585 | { |
| 586 | ParallelModeEnabler enabler(visitor); |
| 587 | visitor.donateAndDrain(); |
| 588 | #if ENABLE(PARALLEL_GC) |
| 589 | visitor.drainFromShared(SlotVisitor::MasterDrain); |
| 590 | #endif |
| 591 | } |
fpizlo@apple.com | f8e2ef8 | 2011-11-15 06:58:19 +0000 | [diff] [blame] | 592 | } |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 593 | } |
ggaren@apple.com | 7621581 | 2012-04-04 05:28:13 +0000 | [diff] [blame] | 594 | |
oliver@apple.com | 7379261 | 2011-10-12 00:24:12 +0000 | [diff] [blame] | 595 | GCCOUNTER(VisitedValueCount, visitor.visitCount()); |
fpizlo@apple.com | f5e1fe1 | 2011-11-10 21:59:39 +0000 | [diff] [blame] | 596 | |
mhahnenberg@apple.com | 02e39c7 | 2012-10-12 19:38:35 +0000 | [diff] [blame] | 597 | m_sharedData.didFinishMarking(); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 598 | #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.org | f15eb3d | 2013-08-14 00:51:25 +0000 | [diff] [blame] | 603 | MARK_LOG_MESSAGE2("\nNumber of live Objects after full GC %lu, took %.6f secs\n", visitCount, WTF::monotonicallyIncreasingTime() - gcStartTime); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 604 | #endif |
| 605 | |
mhahnenberg@apple.com | e2789cf | 2013-09-09 00:11:57 +0000 | [diff] [blame] | 606 | 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.com | 433d02f | 2011-04-21 23:08:15 +0000 | [diff] [blame] | 613 | visitor.reset(); |
msaboff@apple.com | 9d9eab6 | 2012-06-06 23:11:09 +0000 | [diff] [blame] | 614 | #if ENABLE(PARALLEL_GC) |
| 615 | m_sharedData.resetChildren(); |
| 616 | #endif |
msaboff@apple.com | d08f350 | 2012-07-03 22:57:00 +0000 | [diff] [blame] | 617 | m_sharedData.reset(); |
mhahnenberg@apple.com | 02e39c7 | 2012-10-12 19:38:35 +0000 | [diff] [blame] | 618 | } |
| 619 | |
| 620 | void Heap::copyBackingStores() |
| 621 | { |
| 622 | m_storageSpace.startedCopying(); |
| 623 | if (m_storageSpace.shouldDoCopyPhase()) { |
| 624 | m_sharedData.didStartCopying(); |
mhahnenberg@apple.com | 3ea8406 | 2012-10-18 19:44:25 +0000 | [diff] [blame] | 625 | m_copyVisitor.startCopying(); |
| 626 | m_copyVisitor.copyFromShared(); |
| 627 | m_copyVisitor.doneCopying(); |
mhahnenberg@apple.com | c061f5d | 2012-10-13 01:11:04 +0000 | [diff] [blame] | 628 | // 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.com | 02e39c7 | 2012-10-12 19:38:35 +0000 | [diff] [blame] | 631 | m_sharedData.didFinishCopying(); |
mhahnenberg@apple.com | c061f5d | 2012-10-13 01:11:04 +0000 | [diff] [blame] | 632 | } else |
| 633 | m_storageSpace.doneCopying(); |
mjs | 6f821c8 | 2002-03-22 00:31:57 +0000 | [diff] [blame] | 634 | } |
kocienda | 66a6d36 | 2001-08-24 14:24:45 +0000 | [diff] [blame] | 635 | |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 636 | size_t Heap::objectCount() |
mjs | 6d32b07 | 2002-11-20 09:34:02 +0000 | [diff] [blame] | 637 | { |
ggaren@apple.com | 041d0a2 | 2012-05-18 00:40:30 +0000 | [diff] [blame] | 638 | return m_objectSpace.objectCount(); |
mjs | 6d32b07 | 2002-11-20 09:34:02 +0000 | [diff] [blame] | 639 | } |
| 640 | |
fpizlo@apple.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 641 | size_t Heap::extraSize() |
| 642 | { |
| 643 | return m_extraMemoryUsage + m_arrayBuffers.size(); |
| 644 | } |
| 645 | |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 646 | size_t Heap::size() |
eric@webkit.org | 30dc2a3 | 2010-05-15 08:46:23 +0000 | [diff] [blame] | 647 | { |
fpizlo@apple.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 648 | return m_objectSpace.size() + m_storageSpace.size() + extraSize(); |
eric@webkit.org | 30dc2a3 | 2010-05-15 08:46:23 +0000 | [diff] [blame] | 649 | } |
| 650 | |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 651 | size_t Heap::capacity() |
ggaren@apple.com | 0705026 | 2011-01-30 03:32:52 +0000 | [diff] [blame] | 652 | { |
fpizlo@apple.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 653 | return m_objectSpace.capacity() + m_storageSpace.capacity() + extraSize(); |
darin | 52b4943 | 2002-05-09 00:11:06 +0000 | [diff] [blame] | 654 | } |
| 655 | |
mhahnenberg@apple.com | e2789cf | 2013-09-09 00:11:57 +0000 | [diff] [blame] | 656 | size_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.org | 960c28e | 2008-06-19 17:29:29 +0000 | [diff] [blame] | 666 | size_t Heap::protectedGlobalObjectCount() |
ggaren@apple.com | 6e4309e | 2008-01-11 09:52:27 +0000 | [diff] [blame] | 667 | { |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 668 | return forEachProtectedCell<CountIfGlobalObject>(); |
| 669 | } |
ggaren@apple.com | 79ab2a9 | 2011-02-02 05:05:55 +0000 | [diff] [blame] | 670 | |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 671 | size_t Heap::globalObjectCount() |
| 672 | { |
mhahnenberg@apple.com | bee96a3 | 2013-09-16 19:48:48 +0000 | [diff] [blame] | 673 | HeapIterationScope iterationScope(*this); |
| 674 | return m_objectSpace.forEachLiveCell<CountIfGlobalObject>(iterationScope); |
ggaren@apple.com | 6e4309e | 2008-01-11 09:52:27 +0000 | [diff] [blame] | 675 | } |
| 676 | |
ap@webkit.org | 960c28e | 2008-06-19 17:29:29 +0000 | [diff] [blame] | 677 | size_t Heap::protectedObjectCount() |
darin | 52b4943 | 2002-05-09 00:11:06 +0000 | [diff] [blame] | 678 | { |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 679 | return forEachProtectedCell<Count>(); |
ggaren@apple.com | b843ba8 | 2011-02-11 23:31:04 +0000 | [diff] [blame] | 680 | } |
| 681 | |
| 682 | PassOwnPtr<TypeCountSet> Heap::protectedObjectTypeCounts() |
| 683 | { |
ggaren@apple.com | 8a23d6a | 2011-06-09 19:44:37 +0000 | [diff] [blame] | 684 | return forEachProtectedCell<RecordType>(); |
oliver@apple.com | bf9b7f8 | 2011-04-14 20:17:11 +0000 | [diff] [blame] | 685 | } |
| 686 | |
ggaren@apple.com | b843ba8 | 2011-02-11 23:31:04 +0000 | [diff] [blame] | 687 | PassOwnPtr<TypeCountSet> Heap::objectTypeCounts() |
ggaren@apple.com | 0fabb55 | 2010-02-11 21:28:52 +0000 | [diff] [blame] | 688 | { |
mhahnenberg@apple.com | bee96a3 | 2013-09-16 19:48:48 +0000 | [diff] [blame] | 689 | HeapIterationScope iterationScope(*this); |
| 690 | return m_objectSpace.forEachLiveCell<RecordType>(iterationScope); |
ggaren@apple.com | 0fabb55 | 2010-02-11 21:28:52 +0000 | [diff] [blame] | 691 | } |
| 692 | |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 693 | void Heap::deleteAllCompiledCode() |
mhahnenberg@apple.com | cab4752 | 2012-03-30 00:36:37 +0000 | [diff] [blame] | 694 | { |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 695 | // 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.com | 3072125 | 2013-11-21 05:29:42 +0000 | [diff] [blame] | 697 | if (m_vm->entryScope) |
msaboff@apple.com | 03e811f | 2012-04-06 16:09:22 +0000 | [diff] [blame] | 698 | return; |
| 699 | |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 700 | 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.com | 195d7b8 | 2013-09-03 16:39:29 +0000 | [diff] [blame] | 706 | m_codeBlocks.clearMarks(); |
| 707 | m_codeBlocks.deleteUnmarkedAndUnreferenced(); |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 708 | } |
| 709 | |
| 710 | void 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.com | 195d7b8 | 2013-09-03 16:39:29 +0000 | [diff] [blame] | 724 | m_codeBlocks.deleteUnmarkedAndUnreferenced(); |
fpizlo@apple.com | 746b8c5 | 2012-07-09 23:26:54 +0000 | [diff] [blame] | 725 | m_jitStubRoutines.deleteUnmarkedJettisonedStubRoutines(); |
mhahnenberg@apple.com | cab4752 | 2012-03-30 00:36:37 +0000 | [diff] [blame] | 726 | } |
| 727 | |
ggaren@apple.com | 6a42981 | 2009-12-14 08:13:24 +0000 | [diff] [blame] | 728 | void Heap::collectAllGarbage() |
ggaren@apple.com | 9f98034 | 2008-10-15 23:33:07 +0000 | [diff] [blame] | 729 | { |
commit-queue@webkit.org | f42601f | 2011-07-12 22:35:39 +0000 | [diff] [blame] | 730 | if (!m_isSafeToCollect) |
| 731 | return; |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 732 | |
mhahnenberg@apple.com | ac6f1fd | 2013-11-15 19:53:30 +0000 | [diff] [blame] | 733 | DelayedReleaseScope delayedReleaseScope(m_objectSpace); |
ggaren@apple.com | 4f1e836 | 2011-06-08 06:16:32 +0000 | [diff] [blame] | 734 | collect(DoSweep); |
ggaren@apple.com | 7ddf61b | 2011-01-30 23:07:11 +0000 | [diff] [blame] | 735 | } |
| 736 | |
mhahnenberg@apple.com | 2b64eec0 | 2012-04-18 16:18:32 +0000 | [diff] [blame] | 737 | static double minute = 60.0; |
| 738 | |
ggaren@apple.com | 4f1e836 | 2011-06-08 06:16:32 +0000 | [diff] [blame] | 739 | void Heap::collect(SweepToggle sweepToggle) |
ggaren@apple.com | 7ddf61b | 2011-01-30 23:07:11 +0000 | [diff] [blame] | 740 | { |
oliver@apple.com | a03796a | 2013-07-25 04:01:20 +0000 | [diff] [blame] | 741 | #if ENABLE(ALLOCATION_LOGGING) |
| 742 | dataLogF("JSC GC starting collection.\n"); |
| 743 | #endif |
| 744 | |
fpizlo@apple.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 745 | double before = 0; |
| 746 | if (Options::logGC()) { |
| 747 | dataLog("[GC", sweepToggle == DoSweep ? " (eager sweep)" : "", ": "); |
| 748 | before = currentTimeMS(); |
| 749 | } |
| 750 | |
fpizlo@apple.com | 8ce62fd | 2011-11-06 11:39:12 +0000 | [diff] [blame] | 751 | SamplingRegion samplingRegion("Garbage Collection"); |
| 752 | |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 753 | RELEASE_ASSERT(!m_deferralDepth); |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 754 | GCPHASE(Collect); |
andersca@apple.com | b987aae | 2013-07-26 00:13:13 +0000 | [diff] [blame] | 755 | ASSERT(vm()->currentThreadIsHoldingAPILock()); |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 756 | RELEASE_ASSERT(vm()->identifierTable == wtfThreadData().currentIdentifierTable()); |
commit-queue@webkit.org | f42601f | 2011-07-12 22:35:39 +0000 | [diff] [blame] | 757 | ASSERT(m_isSafeToCollect); |
ggaren@apple.com | 6a42981 | 2009-12-14 08:13:24 +0000 | [diff] [blame] | 758 | JAVASCRIPTCORE_GC_BEGIN(); |
oliver@apple.com | 51b41da | 2013-01-23 21:44:29 +0000 | [diff] [blame] | 759 | RELEASE_ASSERT(m_operationInProgress == NoOperation); |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 760 | |
mhahnenberg@apple.com | 7a22bab | 2013-12-10 19:37:29 +0000 | [diff] [blame] | 761 | { |
| 762 | RecursiveAllocationScope scope(*this); |
| 763 | m_vm->prepareToDiscardCode(); |
| 764 | } |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 765 | |
mhahnenberg@apple.com | 47c9c53 | 2012-06-05 20:38:21 +0000 | [diff] [blame] | 766 | m_operationInProgress = Collection; |
oliver@apple.com | 8cebf37 | 2013-07-25 04:03:46 +0000 | [diff] [blame] | 767 | m_extraMemoryUsage = 0; |
mhahnenberg@apple.com | 2b64eec0 | 2012-04-18 16:18:32 +0000 | [diff] [blame] | 768 | |
dbates@webkit.org | 6122a1a | 2013-10-16 23:36:25 +0000 | [diff] [blame] | 769 | if (m_activityCallback) |
| 770 | m_activityCallback->willCollect(); |
mhahnenberg@apple.com | 0eb606f | 2012-04-20 19:55:21 +0000 | [diff] [blame] | 771 | |
commit-queue@webkit.org | f15eb3d | 2013-08-14 00:51:25 +0000 | [diff] [blame] | 772 | double lastGCStartTime = WTF::monotonicallyIncreasingTime(); |
mhahnenberg@apple.com | 2b64eec0 | 2012-04-18 16:18:32 +0000 | [diff] [blame] | 773 | if (lastGCStartTime - m_lastCodeDiscardTime > minute) { |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 774 | deleteAllCompiledCode(); |
commit-queue@webkit.org | f15eb3d | 2013-08-14 00:51:25 +0000 | [diff] [blame] | 775 | m_lastCodeDiscardTime = WTF::monotonicallyIncreasingTime(); |
mhahnenberg@apple.com | 2b64eec0 | 2012-04-18 16:18:32 +0000 | [diff] [blame] | 776 | } |
| 777 | |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 778 | { |
mhahnenberg@apple.com | bee96a3 | 2013-09-16 19:48:48 +0000 | [diff] [blame] | 779 | GCPHASE(StopAllocation); |
| 780 | m_objectSpace.stopAllocating(); |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 781 | } |
oliver@apple.com | fa9733f | 2011-09-30 22:23:33 +0000 | [diff] [blame] | 782 | |
ggaren@apple.com | bd41f7e | 2013-01-30 00:36:50 +0000 | [diff] [blame] | 783 | markRoots(); |
fpizlo@apple.com | f5e1fe1 | 2011-11-10 21:59:39 +0000 | [diff] [blame] | 784 | |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 785 | { |
ggaren@apple.com | 39281e2 | 2012-05-24 21:18:10 +0000 | [diff] [blame] | 786 | GCPHASE(ReapingWeakHandles); |
| 787 | m_objectSpace.reapWeakSets(); |
| 788 | } |
| 789 | |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 790 | JAVASCRIPTCORE_GC_MARKED(); |
fpizlo@apple.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 791 | |
| 792 | { |
| 793 | GCPHASE(SweepingArrayBuffers); |
| 794 | m_arrayBuffers.sweep(); |
| 795 | } |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 796 | |
ggaren@apple.com | 39281e2 | 2012-05-24 21:18:10 +0000 | [diff] [blame] | 797 | { |
mhahnenberg@apple.com | 02e39c7 | 2012-10-12 19:38:35 +0000 | [diff] [blame] | 798 | 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.com | f5e1fe1 | 2011-11-10 21:59:39 +0000 | [diff] [blame] | 806 | GCPHASE(FinalizeUnconditionalFinalizers); |
| 807 | finalizeUnconditionalFinalizers(); |
| 808 | } |
ggaren@apple.com | 39281e2 | 2012-05-24 21:18:10 +0000 | [diff] [blame] | 809 | |
fpizlo@apple.com | f5e1fe1 | 2011-11-10 21:59:39 +0000 | [diff] [blame] | 810 | { |
fpizlo@apple.com | 45928f8 | 2011-11-17 03:58:48 +0000 | [diff] [blame] | 811 | GCPHASE(DeleteCodeBlocks); |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 812 | deleteUnmarkedCompiledCode(); |
fpizlo@apple.com | 45928f8 | 2011-11-17 03:58:48 +0000 | [diff] [blame] | 813 | } |
ggaren@apple.com | 6a42981 | 2009-12-14 08:13:24 +0000 | [diff] [blame] | 814 | |
ggaren@apple.com | d17e17e | 2013-02-19 01:39:12 +0000 | [diff] [blame] | 815 | { |
| 816 | GCPHASE(DeleteSourceProviderCaches); |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 817 | m_vm->clearSourceProviderCaches(); |
ggaren@apple.com | d17e17e | 2013-02-19 01:39:12 +0000 | [diff] [blame] | 818 | } |
| 819 | |
ggaren@apple.com | 03c4f9d | 2011-02-11 00:20:17 +0000 | [diff] [blame] | 820 | if (sweepToggle == DoSweep) { |
fpizlo@apple.com | 8ce62fd | 2011-11-06 11:39:12 +0000 | [diff] [blame] | 821 | SamplingRegion samplingRegion("Garbage Collection: Sweeping"); |
oliver@apple.com | 32b720a | 2011-10-05 19:07:18 +0000 | [diff] [blame] | 822 | GCPHASE(Sweeping); |
ggaren@apple.com | 041d0a2 | 2012-05-18 00:40:30 +0000 | [diff] [blame] | 823 | m_objectSpace.sweep(); |
ggaren@apple.com | 7573617 | 2012-04-24 19:11:12 +0000 | [diff] [blame] | 824 | m_objectSpace.shrink(); |
ggaren@apple.com | 03c4f9d | 2011-02-11 00:20:17 +0000 | [diff] [blame] | 825 | } |
ggaren@apple.com | 7ddf61b | 2011-01-30 23:07:11 +0000 | [diff] [blame] | 826 | |
mhahnenberg@apple.com | 02e39c7 | 2012-10-12 19:38:35 +0000 | [diff] [blame] | 827 | m_sweeper->startSweeping(m_blockSnapshot); |
mhahnenberg@apple.com | eb39abc | 2012-05-31 03:04:00 +0000 | [diff] [blame] | 828 | m_bytesAbandoned = 0; |
| 829 | |
ggaren@apple.com | 39281e2 | 2012-05-24 21:18:10 +0000 | [diff] [blame] | 830 | { |
| 831 | GCPHASE(ResetAllocators); |
| 832 | m_objectSpace.resetAllocators(); |
| 833 | } |
| 834 | |
mhahnenberg@apple.com | e2789cf | 2013-09-09 00:11:57 +0000 | [diff] [blame] | 835 | size_t currentHeapSize = sizeAfterCollect(); |
mhahnenberg@apple.com | 3a2fb03 | 2012-10-05 17:35:49 +0000 | [diff] [blame] | 836 | if (Options::gcMaxHeapSize() && currentHeapSize > Options::gcMaxHeapSize()) |
| 837 | HeapStatistics::exitWithFailure(); |
| 838 | |
ggaren@apple.com | bd41f7e | 2013-01-30 00:36:50 +0000 | [diff] [blame] | 839 | m_sizeAfterLastCollect = currentHeapSize; |
ggaren@apple.com | 9a07093 | 2012-05-22 19:17:57 +0000 | [diff] [blame] | 840 | |
ggaren@apple.com | bd41f7e | 2013-01-30 00:36:50 +0000 | [diff] [blame] | 841 | // 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.com | 8b5cfd3 | 2012-04-20 00:05:37 +0000 | [diff] [blame] | 847 | m_bytesAllocated = 0; |
commit-queue@webkit.org | f15eb3d | 2013-08-14 00:51:25 +0000 | [diff] [blame] | 848 | double lastGCEndTime = WTF::monotonicallyIncreasingTime(); |
mhahnenberg@apple.com | f9133a1 | 2012-03-06 21:32:24 +0000 | [diff] [blame] | 849 | m_lastGCLength = lastGCEndTime - lastGCStartTime; |
ggaren@apple.com | def139e | 2012-09-26 02:27:54 +0000 | [diff] [blame] | 850 | |
mhahnenberg@apple.com | 3a2fb03 | 2012-10-05 17:35:49 +0000 | [diff] [blame] | 851 | if (Options::recordGCPauseTimes()) |
| 852 | HeapStatistics::recordGCPauseTime(lastGCStartTime, lastGCEndTime); |
oliver@apple.com | 51b41da | 2013-01-23 21:44:29 +0000 | [diff] [blame] | 853 | RELEASE_ASSERT(m_operationInProgress == Collection); |
| 854 | |
mhahnenberg@apple.com | 47c9c53 | 2012-06-05 20:38:21 +0000 | [diff] [blame] | 855 | m_operationInProgress = NoOperation; |
ggaren@apple.com | 6a42981 | 2009-12-14 08:13:24 +0000 | [diff] [blame] | 856 | JAVASCRIPTCORE_GC_END(); |
ggaren@apple.com | def139e | 2012-09-26 02:27:54 +0000 | [diff] [blame] | 857 | |
| 858 | if (Options::useZombieMode()) |
| 859 | zombifyDeadObjects(); |
| 860 | |
| 861 | if (Options::objectsAreImmortal()) |
| 862 | markDeadObjects(); |
| 863 | |
mhahnenberg@apple.com | 3a2fb03 | 2012-10-05 17:35:49 +0000 | [diff] [blame] | 864 | if (Options::showObjectStatistics()) |
| 865 | HeapStatistics::showObjectStatistics(this); |
fpizlo@apple.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 866 | |
| 867 | if (Options::logGC()) { |
| 868 | double after = currentTimeMS(); |
| 869 | dataLog(after - before, " ms, ", currentHeapSize / 1024, " kb]\n"); |
| 870 | } |
oliver@apple.com | a03796a | 2013-07-25 04:01:20 +0000 | [diff] [blame] | 871 | |
| 872 | #if ENABLE(ALLOCATION_LOGGING) |
| 873 | dataLogF("JSC GC finishing collection.\n"); |
| 874 | #endif |
ggaren@apple.com | def139e | 2012-09-26 02:27:54 +0000 | [diff] [blame] | 875 | } |
| 876 | |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 877 | bool 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.com | def139e | 2012-09-26 02:27:54 +0000 | [diff] [blame] | 889 | void Heap::markDeadObjects() |
| 890 | { |
mhahnenberg@apple.com | bee96a3 | 2013-09-16 19:48:48 +0000 | [diff] [blame] | 891 | HeapIterationScope iterationScope(*this); |
| 892 | m_objectSpace.forEachDeadCell<MarkObject>(iterationScope); |
ggaren@apple.com | 6a42981 | 2009-12-14 08:13:24 +0000 | [diff] [blame] | 893 | } |
| 894 | |
mhahnenberg@apple.com | 765a7de | 2013-04-15 23:17:51 +0000 | [diff] [blame] | 895 | void Heap::setActivityCallback(PassOwnPtr<GCActivityCallback> activityCallback) |
ggaren@apple.com | 2ccf7a9 | 2010-08-03 20:34:17 +0000 | [diff] [blame] | 896 | { |
| 897 | m_activityCallback = activityCallback; |
| 898 | } |
| 899 | |
ggaren@apple.com | aa64ff3 | 2010-12-03 01:16:11 +0000 | [diff] [blame] | 900 | GCActivityCallback* Heap::activityCallback() |
| 901 | { |
mhahnenberg@apple.com | 765a7de | 2013-04-15 23:17:51 +0000 | [diff] [blame] | 902 | return m_activityCallback.get(); |
ggaren@apple.com | aa64ff3 | 2010-12-03 01:16:11 +0000 | [diff] [blame] | 903 | } |
| 904 | |
dbates@webkit.org | 0ba2704 | 2013-10-18 23:57:24 +0000 | [diff] [blame] | 905 | void Heap::setIncrementalSweeper(PassOwnPtr<IncrementalSweeper> sweeper) |
| 906 | { |
| 907 | m_sweeper = sweeper; |
| 908 | } |
| 909 | |
mhahnenberg@apple.com | eb39abc | 2012-05-31 03:04:00 +0000 | [diff] [blame] | 910 | IncrementalSweeper* Heap::sweeper() |
| 911 | { |
mhahnenberg@apple.com | 765a7de | 2013-04-15 23:17:51 +0000 | [diff] [blame] | 912 | return m_sweeper.get(); |
mhahnenberg@apple.com | eb39abc | 2012-05-31 03:04:00 +0000 | [diff] [blame] | 913 | } |
| 914 | |
msaboff@apple.com | fb5f4cc | 2012-05-14 23:47:53 +0000 | [diff] [blame] | 915 | void Heap::setGarbageCollectionTimerEnabled(bool enable) |
| 916 | { |
dbates@webkit.org | 156dd1a | 2013-10-18 17:22:55 +0000 | [diff] [blame] | 917 | if (m_activityCallback) |
| 918 | m_activityCallback->setEnabled(enable); |
msaboff@apple.com | fb5f4cc | 2012-05-14 23:47:53 +0000 | [diff] [blame] | 919 | } |
| 920 | |
mhahnenberg@apple.com | 8b5cfd3 | 2012-04-20 00:05:37 +0000 | [diff] [blame] | 921 | void Heap::didAllocate(size_t bytes) |
| 922 | { |
dbates@webkit.org | 6122a1a | 2013-10-16 23:36:25 +0000 | [diff] [blame] | 923 | if (m_activityCallback) |
| 924 | m_activityCallback->didAllocate(m_bytesAllocated + m_bytesAbandoned); |
mhahnenberg@apple.com | 8b5cfd3 | 2012-04-20 00:05:37 +0000 | [diff] [blame] | 925 | m_bytesAllocated += bytes; |
| 926 | } |
| 927 | |
ggaren@apple.com | 06a8bb6 | 2012-09-12 04:08:18 +0000 | [diff] [blame] | 928 | bool Heap::isValidAllocation(size_t) |
ggaren@apple.com | 99cc9d9 | 2011-05-26 21:46:09 +0000 | [diff] [blame] | 929 | { |
ggaren@apple.com | 9a9a4b5 | 2013-04-18 19:32:17 +0000 | [diff] [blame] | 930 | if (!isValidThreadState(m_vm)) |
ggaren@apple.com | 99cc9d9 | 2011-05-26 21:46:09 +0000 | [diff] [blame] | 931 | return false; |
| 932 | |
ggaren@apple.com | 99cc9d9 | 2011-05-26 21:46:09 +0000 | [diff] [blame] | 933 | if (m_operationInProgress != NoOperation) |
| 934 | return false; |
| 935 | |
| 936 | return true; |
| 937 | } |
| 938 | |
ggaren@apple.com | 75bf76c | 2011-10-05 22:44:36 +0000 | [diff] [blame] | 939 | void Heap::addFinalizer(JSCell* cell, Finalizer finalizer) |
| 940 | { |
ggaren@apple.com | ab71e97 | 2012-04-28 05:57:46 +0000 | [diff] [blame] | 941 | WeakSet::allocate(cell, &m_finalizerOwner, reinterpret_cast<void*>(finalizer)); // Balanced by FinalizerOwner::finalize(). |
ggaren@apple.com | 75bf76c | 2011-10-05 22:44:36 +0000 | [diff] [blame] | 942 | } |
| 943 | |
| 944 | void Heap::FinalizerOwner::finalize(Handle<Unknown> handle, void* context) |
| 945 | { |
ggaren@apple.com | 7621581 | 2012-04-04 05:28:13 +0000 | [diff] [blame] | 946 | HandleSlot slot = handle.slot(); |
ggaren@apple.com | 75bf76c | 2011-10-05 22:44:36 +0000 | [diff] [blame] | 947 | Finalizer finalizer = reinterpret_cast<Finalizer>(context); |
ggaren@apple.com | 7621581 | 2012-04-04 05:28:13 +0000 | [diff] [blame] | 948 | finalizer(slot->asCell()); |
ggaren@apple.com | 61b9700 | 2012-04-06 22:35:55 +0000 | [diff] [blame] | 949 | WeakSet::deallocate(WeakImpl::asWeakImpl(slot)); |
ggaren@apple.com | 75bf76c | 2011-10-05 22:44:36 +0000 | [diff] [blame] | 950 | } |
| 951 | |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 952 | void Heap::addCompiledCode(ExecutableBase* executable) |
mhahnenberg@apple.com | cab4752 | 2012-03-30 00:36:37 +0000 | [diff] [blame] | 953 | { |
ggaren@apple.com | d714757 | 2012-06-08 18:17:16 +0000 | [diff] [blame] | 954 | m_compiledCode.append(executable); |
mhahnenberg@apple.com | cab4752 | 2012-03-30 00:36:37 +0000 | [diff] [blame] | 955 | } |
| 956 | |
ggaren@apple.com | def139e | 2012-09-26 02:27:54 +0000 | [diff] [blame] | 957 | class Zombify : public MarkedBlock::VoidFunctor { |
mhahnenberg@apple.com | 66b96f2 | 2012-09-07 05:53:25 +0000 | [diff] [blame] | 958 | public: |
mhahnenberg@apple.com | 66b96f2 | 2012-09-07 05:53:25 +0000 | [diff] [blame] | 959 | void operator()(JSCell* cell) |
| 960 | { |
mhahnenberg@apple.com | 66b96f2 | 2012-09-07 05:53:25 +0000 | [diff] [blame] | 961 | 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.com | def139e | 2012-09-26 02:27:54 +0000 | [diff] [blame] | 968 | void* limit = static_cast<void*>(reinterpret_cast<char*>(cell) + MarkedBlock::blockFor(cell)->cellSize()); |
mhahnenberg@apple.com | 66b96f2 | 2012-09-07 05:53:25 +0000 | [diff] [blame] | 969 | for (; current < limit; current++) |
| 970 | *current = reinterpret_cast<void*>(0xbbadbeef); |
| 971 | } |
mhahnenberg@apple.com | 66b96f2 | 2012-09-07 05:53:25 +0000 | [diff] [blame] | 972 | }; |
| 973 | |
| 974 | void Heap::zombifyDeadObjects() |
| 975 | { |
ggaren@apple.com | def139e | 2012-09-26 02:27:54 +0000 | [diff] [blame] | 976 | // Sweep now because destructors will crash once we're zombified. |
mhahnenberg@apple.com | 66b96f2 | 2012-09-07 05:53:25 +0000 | [diff] [blame] | 977 | m_objectSpace.sweep(); |
mhahnenberg@apple.com | bee96a3 | 2013-09-16 19:48:48 +0000 | [diff] [blame] | 978 | HeapIterationScope iterationScope(*this); |
| 979 | m_objectSpace.forEachDeadCell<Zombify>(iterationScope); |
mhahnenberg@apple.com | 66b96f2 | 2012-09-07 05:53:25 +0000 | [diff] [blame] | 980 | } |
| 981 | |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 982 | void 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.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 989 | void Heap::decrementDeferralDepth() |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 990 | { |
| 991 | RELEASE_ASSERT(m_deferralDepth >= 1); |
| 992 | |
| 993 | m_deferralDepth--; |
fpizlo@apple.com | 0e0d931 | 2013-08-15 20:43:06 +0000 | [diff] [blame] | 994 | } |
| 995 | |
| 996 | void Heap::decrementDeferralDepthAndGCIfNeeded() |
| 997 | { |
| 998 | decrementDeferralDepth(); |
oliver@apple.com | 284cc3d | 2013-07-25 04:00:33 +0000 | [diff] [blame] | 999 | collectIfNecessaryOrDefer(); |
| 1000 | } |
| 1001 | |
mhahnenberg@apple.com | 4968e1a | 2013-12-18 22:50:40 +0000 | [diff] [blame^] | 1002 | void 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.org | 3f782f6 | 2008-09-08 01:28:33 +0000 | [diff] [blame] | 1012 | } // namespace JSC |