/* * Copyright (C) 2007, 2014 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "GCController.h" #include "JSDOMWindow.h" #include #include #include #include #include #include using namespace JSC; namespace WebCore { static void collect(void*) { JSLockHolder lock(JSDOMWindow::commonVM()); JSDOMWindow::commonVM().heap.collectAllGarbage(); } GCController& GCController::singleton() { static NeverDestroyed controller; return controller; } GCController::GCController() : m_GCTimer(*this, &GCController::gcTimerFired) { } void GCController::garbageCollectSoon() { // We only use reportAbandonedObjectGraph for systems for which there's an implementation // of the garbage collector timers in JavaScriptCore. We wouldn't need this if JavaScriptCore // used a timer implementation from WTF like RunLoop::Timer. #if USE(CF) || USE(GLIB) || PLATFORM(QT) JSLockHolder lock(JSDOMWindow::commonVM()); JSDOMWindow::commonVM().heap.reportAbandonedObjectGraph(); #else garbageCollectOnNextRunLoop(); #endif } void GCController::garbageCollectOnNextRunLoop() { if (!m_GCTimer.isActive()) m_GCTimer.startOneShot(0); } void GCController::gcTimerFired() { collect(nullptr); } void GCController::garbageCollectNow() { JSLockHolder lock(JSDOMWindow::commonVM()); if (!JSDOMWindow::commonVM().heap.isBusy()) { JSDOMWindow::commonVM().heap.collectAllGarbage(); WTF::releaseFastMallocFreeMemory(); } } void GCController::garbageCollectNowIfNotDoneRecently() { #if USE(CF) || USE(GLIB) || PLATFORM(QT) JSLockHolder lock(JSDOMWindow::commonVM()); if (!JSDOMWindow::commonVM().heap.isBusy()) JSDOMWindow::commonVM().heap.collectAllGarbageIfNotDoneRecently(); #else garbageCollectSoon(); #endif } void GCController::garbageCollectOnAlternateThreadForDebugging(bool waitUntilDone) { ThreadIdentifier threadID = createThread(collect, 0, "WebCore: GCController"); if (waitUntilDone) { waitForThreadCompletion(threadID); return; } detachThread(threadID); } void GCController::setJavaScriptGarbageCollectorTimerEnabled(bool enable) { JSDOMWindow::commonVM().heap.setGarbageCollectionTimerEnabled(enable); } void GCController::deleteAllCode() { JSLockHolder lock(JSDOMWindow::commonVM()); JSDOMWindow::commonVM().deleteAllCode(); } void GCController::deleteAllLinkedCode() { JSLockHolder lock(JSDOMWindow::commonVM()); JSDOMWindow::commonVM().deleteAllLinkedCode(); } } // namespace WebCore