OLD | NEW |
1 /* | 1 /* |
2 * Copyright (C) 2014 Google Inc. All rights reserved. | 2 * Copyright (C) 2014 Google Inc. All rights reserved. |
3 * | 3 * |
4 * Redistribution and use in source and binary forms, with or without | 4 * Redistribution and use in source and binary forms, with or without |
5 * modification, are permitted provided that the following conditions are | 5 * modification, are permitted provided that the following conditions are |
6 * met: | 6 * met: |
7 * | 7 * |
8 * * Redistributions of source code must retain the above copyright | 8 * * Redistributions of source code must retain the above copyright |
9 * notice, this list of conditions and the following disclaimer. | 9 * notice, this list of conditions and the following disclaimer. |
10 * * Redistributions in binary form must reproduce the above | 10 * * Redistributions in binary form must reproduce the above |
(...skipping 16 matching lines...) Expand all Loading... |
27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 27 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 28 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
29 */ | 29 */ |
30 | 30 |
31 #include "public/web/WebLeakDetector.h" | 31 #include "public/web/WebLeakDetector.h" |
32 | 32 |
33 #include "bindings/core/v8/V8GCController.h" | 33 #include "bindings/core/v8/V8GCController.h" |
34 #include "core/editing/spellcheck/SpellChecker.h" | 34 #include "core/editing/spellcheck/SpellChecker.h" |
35 #include "core/fetch/MemoryCache.h" | 35 #include "core/fetch/MemoryCache.h" |
36 #include "core/inspector/InstanceCounters.h" | 36 #include "core/inspector/InstanceCounters.h" |
37 #include "core/workers/InProcessWorkerMessagingProxy.h" | |
38 #include "core/workers/WorkerThread.h" | 37 #include "core/workers/WorkerThread.h" |
39 #include "platform/Timer.h" | 38 #include "platform/Timer.h" |
40 #include "public/web/WebFrame.h" | 39 #include "public/web/WebFrame.h" |
41 #include "web/WebLocalFrameImpl.h" | 40 #include "web/WebLocalFrameImpl.h" |
42 | 41 |
43 namespace blink { | 42 namespace blink { |
44 | 43 |
45 namespace { | 44 namespace { |
46 | 45 |
47 class WebLeakDetectorImpl final : public WebLeakDetector { | 46 class WebLeakDetectorImpl final : public WebLeakDetector { |
48 WTF_MAKE_NONCOPYABLE(WebLeakDetectorImpl); | 47 WTF_MAKE_NONCOPYABLE(WebLeakDetectorImpl); |
49 public: | 48 public: |
50 explicit WebLeakDetectorImpl(WebLeakDetectorClient* client) | 49 explicit WebLeakDetectorImpl(WebLeakDetectorClient* client) |
51 : m_client(client) | 50 : m_client(client) |
52 , m_delayedGCAndReportTimer(this, &WebLeakDetectorImpl::delayedGCAndRepo
rt) | 51 , m_delayedGCAndReportTimer(this, &WebLeakDetectorImpl::delayedGCAndRepo
rt) |
53 , m_delayedReportTimer(this, &WebLeakDetectorImpl::delayedReport) | 52 , m_delayedReportTimer(this, &WebLeakDetectorImpl::delayedReport) |
54 , m_numberOfWorkerGCs(0) | 53 , m_numberOfGCNeeded(0) |
55 { | 54 { |
56 DCHECK(m_client); | 55 DCHECK(m_client); |
57 } | 56 } |
58 | 57 |
59 ~WebLeakDetectorImpl() override {} | 58 ~WebLeakDetectorImpl() override {} |
60 | 59 |
61 void prepareForLeakDetection(WebFrame*) override; | 60 void prepareForLeakDetection(WebFrame*) override; |
62 void collectGarbageAndReport() override; | 61 void collectGarbageAndReport() override; |
63 | 62 |
64 private: | 63 private: |
65 void delayedGCAndReport(Timer<WebLeakDetectorImpl>*); | 64 void delayedGCAndReport(Timer<WebLeakDetectorImpl>*); |
66 void delayedReport(Timer<WebLeakDetectorImpl>*); | 65 void delayedReport(Timer<WebLeakDetectorImpl>*); |
67 | 66 |
68 WebLeakDetectorClient* m_client; | 67 WebLeakDetectorClient* m_client; |
69 Timer<WebLeakDetectorImpl> m_delayedGCAndReportTimer; | 68 Timer<WebLeakDetectorImpl> m_delayedGCAndReportTimer; |
70 Timer<WebLeakDetectorImpl> m_delayedReportTimer; | 69 Timer<WebLeakDetectorImpl> m_delayedReportTimer; |
71 int m_numberOfWorkerGCs; | 70 int m_numberOfGCNeeded; |
72 }; | 71 }; |
73 | 72 |
74 void WebLeakDetectorImpl::prepareForLeakDetection(WebFrame* frame) | 73 void WebLeakDetectorImpl::prepareForLeakDetection(WebFrame* frame) |
75 { | 74 { |
76 v8::Isolate* isolate = v8::Isolate::GetCurrent(); | 75 v8::Isolate* isolate = v8::Isolate::GetCurrent(); |
77 v8::HandleScope handleScope(isolate); | 76 v8::HandleScope handleScope(isolate); |
78 | 77 |
79 // For example, calling isValidEmailAddress in EmailInputType.cpp with a | 78 // For example, calling isValidEmailAddress in EmailInputType.cpp with a |
80 // non-empty string creates a static ScriptRegexp value which holds a | 79 // non-empty string creates a static ScriptRegexp value which holds a |
81 // V8PerContextData indirectly. This affects the number of V8PerContextData. | 80 // V8PerContextData indirectly. This affects the number of V8PerContextData. |
(...skipping 22 matching lines...) Expand all Loading... |
104 | 103 |
105 void WebLeakDetectorImpl::collectGarbageAndReport() | 104 void WebLeakDetectorImpl::collectGarbageAndReport() |
106 { | 105 { |
107 V8GCController::collectAllGarbageForTesting(V8PerIsolateData::mainThreadIsol
ate()); | 106 V8GCController::collectAllGarbageForTesting(V8PerIsolateData::mainThreadIsol
ate()); |
108 // Note: Oilpan precise GC is scheduled at the end of the event loop. | 107 // Note: Oilpan precise GC is scheduled at the end of the event loop. |
109 | 108 |
110 // Task queue may contain delayed object destruction tasks. | 109 // Task queue may contain delayed object destruction tasks. |
111 // This method is called from navigation hook inside FrameLoader, | 110 // This method is called from navigation hook inside FrameLoader, |
112 // so previous document is still held by the loader until the next event loo
p. | 111 // so previous document is still held by the loader until the next event loo
p. |
113 // Complete all pending tasks before proceeding to gc. | 112 // Complete all pending tasks before proceeding to gc. |
114 m_numberOfWorkerGCs = 0; | 113 m_numberOfGCNeeded = 2; |
115 m_delayedGCAndReportTimer.startOneShot(0, BLINK_FROM_HERE); | 114 m_delayedGCAndReportTimer.startOneShot(0, BLINK_FROM_HERE); |
116 } | 115 } |
117 | 116 |
118 void WebLeakDetectorImpl::delayedGCAndReport(Timer<WebLeakDetectorImpl>*) | 117 void WebLeakDetectorImpl::delayedGCAndReport(Timer<WebLeakDetectorImpl>*) |
119 { | 118 { |
120 // The initial GC will have swept out Resource objects, but their finalizers | 119 // We do a second and third GC here to address flakiness |
121 // will let go of yet more objects, needing another GC after having been bac
k | 120 // The second GC is necessary as Resource GC may have postponed clean-up tas
ks to next event loop. |
122 // to the event loop to run any clean-up tasks. | 121 // The third GC is necessary for cleaning up Document after worker object di
ed. |
| 122 |
123 V8GCController::collectAllGarbageForTesting(V8PerIsolateData::mainThreadIsol
ate()); | 123 V8GCController::collectAllGarbageForTesting(V8PerIsolateData::mainThreadIsol
ate()); |
124 // Note: Oilpan precise GC is scheduled at the end of the event loop. | 124 // Note: Oilpan precise GC is scheduled at the end of the event loop. |
125 | 125 |
126 // If there are any in-process worker proxy objects still around waiting for | 126 // Inspect counters on the next event loop. |
127 // the worker to notify of destruction, we need to ensure that these have al
l | 127 if (--m_numberOfGCNeeded) |
128 // been destructed and their garbage collected (as they hold a Document | |
129 // reference, and would generate a leak if not.) | |
130 // | |
131 // Do that by monitoring the remaining proxy count, iterating at most two ti
mes. | |
132 if (InProcessWorkerMessagingProxy::proxyCount() && m_numberOfWorkerGCs < 2)
{ | |
133 m_numberOfWorkerGCs++; | |
134 m_delayedGCAndReportTimer.startOneShot(0, BLINK_FROM_HERE); | 128 m_delayedGCAndReportTimer.startOneShot(0, BLINK_FROM_HERE); |
135 } else { | 129 else |
136 m_delayedReportTimer.startOneShot(0, BLINK_FROM_HERE); | 130 m_delayedReportTimer.startOneShot(0, BLINK_FROM_HERE); |
137 } | |
138 } | 131 } |
139 | 132 |
140 void WebLeakDetectorImpl::delayedReport(Timer<WebLeakDetectorImpl>*) | 133 void WebLeakDetectorImpl::delayedReport(Timer<WebLeakDetectorImpl>*) |
141 { | 134 { |
142 DCHECK(m_client); | 135 DCHECK(m_client); |
143 | 136 |
144 WebLeakDetectorClient::Result result; | 137 WebLeakDetectorClient::Result result; |
145 result.numberOfLiveAudioNodes = InstanceCounters::counterValue(InstanceCount
ers::AudioHandlerCounter); | 138 result.numberOfLiveAudioNodes = InstanceCounters::counterValue(InstanceCount
ers::AudioHandlerCounter); |
146 result.numberOfLiveDocuments = InstanceCounters::counterValue(InstanceCounte
rs::DocumentCounter); | 139 result.numberOfLiveDocuments = InstanceCounters::counterValue(InstanceCounte
rs::DocumentCounter); |
147 result.numberOfLiveNodes = InstanceCounters::counterValue(InstanceCounters::
NodeCounter); | 140 result.numberOfLiveNodes = InstanceCounters::counterValue(InstanceCounters::
NodeCounter); |
(...skipping 12 matching lines...) Expand all Loading... |
160 } | 153 } |
161 | 154 |
162 } // namespace | 155 } // namespace |
163 | 156 |
164 WebLeakDetector* WebLeakDetector::create(WebLeakDetectorClient* client) | 157 WebLeakDetector* WebLeakDetector::create(WebLeakDetectorClient* client) |
165 { | 158 { |
166 return new WebLeakDetectorImpl(client); | 159 return new WebLeakDetectorImpl(client); |
167 } | 160 } |
168 | 161 |
169 } // namespace blink | 162 } // namespace blink |
OLD | NEW |