/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */ /* ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is mozilla.org code. * * The Initial Developer of the Original Code is Mozilla Foundation. * Portions created by the Initial Developer are Copyright (C) 2007 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Jonas Sicking (Original Author) * * Alternatively, the contents of this file may be used under the terms of * either of the GNU General Public License Version 2 or later (the "GPL"), * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include "nsCCUncollectableMarker.h" #include "nsIObserverService.h" #include "nsIDocShell.h" #include "nsIDocShellTreeItem.h" #include "nsServiceManagerUtils.h" #include "nsIDOMDocument.h" #include "nsIContentViewer.h" #include "nsIDocument.h" #include "nsIWindowMediator.h" #include "nsPIDOMWindow.h" #include "nsIWebNavigation.h" #include "nsISHistory.h" #include "nsISHEntry.h" #include "nsISHContainer.h" #include "nsIWindowWatcher.h" static PRBool sInited = 0; PRUint32 nsCCUncollectableMarker::sGeneration = 0; NS_IMPL_ISUPPORTS1(nsCCUncollectableMarker, nsIObserver) /* static */ nsresult nsCCUncollectableMarker::Init() { if (sInited) { return NS_OK; } nsCOMPtr marker = new nsCCUncollectableMarker; NS_ENSURE_TRUE(marker, NS_ERROR_OUT_OF_MEMORY); nsresult rv; nsCOMPtr obs = do_GetService("@mozilla.org/observer-service;1", &rv); NS_ENSURE_SUCCESS(rv, rv); // This makes the observer service hold an owning reference to the marker rv = obs->AddObserver(marker, "xpcom-shutdown", PR_FALSE); NS_ENSURE_SUCCESS(rv, rv); rv = obs->AddObserver(marker, "cycle-collector-begin", PR_FALSE); NS_ENSURE_SUCCESS(rv, rv); sInited = PR_TRUE; return NS_OK; } void MarkContentViewer(nsIContentViewer* aViewer) { if (!aViewer) { return; } nsCOMPtr domDoc; aViewer->GetDOMDocument(getter_AddRefs(domDoc)); nsCOMPtr doc = do_QueryInterface(domDoc); if (doc) { doc->MarkUncollectableForCCGeneration(nsCCUncollectableMarker::sGeneration); } } void MarkDocShell(nsIDocShellTreeNode* aNode); void MarkSHEntry(nsISHEntry* aSHEntry) { if (!aSHEntry) { return; } nsCOMPtr cview; aSHEntry->GetContentViewer(getter_AddRefs(cview)); MarkContentViewer(cview); nsCOMPtr child; PRInt32 i = 0; while (NS_SUCCEEDED(aSHEntry->ChildShellAt(i++, getter_AddRefs(child))) && child) { MarkDocShell(child); } nsCOMPtr shCont = do_QueryInterface(aSHEntry); PRInt32 count; shCont->GetChildCount(&count); for (i = 0; i < count; ++i) { nsCOMPtr childEntry; shCont->GetChildAt(i, getter_AddRefs(childEntry)); MarkSHEntry(childEntry); } } void MarkDocShell(nsIDocShellTreeNode* aNode) { nsCOMPtr shell = do_QueryInterface(aNode); if (!shell) { return; } nsCOMPtr cview; shell->GetContentViewer(getter_AddRefs(cview)); MarkContentViewer(cview); nsCOMPtr webNav = do_QueryInterface(shell); nsCOMPtr history; webNav->GetSessionHistory(getter_AddRefs(history)); if (history) { PRInt32 i, historyCount; history->GetCount(&historyCount); for (i = 0; i < historyCount; ++i) { nsCOMPtr historyEntry; history->GetEntryAtIndex(i, PR_FALSE, getter_AddRefs(historyEntry)); nsCOMPtr shEntry = do_QueryInterface(historyEntry); MarkSHEntry(shEntry); } } PRInt32 i, childCount; aNode->GetChildCount(&childCount); for (i = 0; i < childCount; ++i) { nsCOMPtr child; aNode->GetChildAt(i, getter_AddRefs(child)); MarkDocShell(child); } } void MarkWindowList(nsISimpleEnumerator* aWindowList) { nsCOMPtr iter; while (NS_SUCCEEDED(aWindowList->GetNext(getter_AddRefs(iter))) && iter) { nsCOMPtr window = do_QueryInterface(iter); if (window) { nsCOMPtr rootDocShell = do_QueryInterface(window->GetDocShell()); MarkDocShell(rootDocShell); } } } nsresult nsCCUncollectableMarker::Observe(nsISupports* aSubject, const char* aTopic, const PRUnichar* aData) { nsresult rv; if (!strcmp(aTopic, "xpcom-shutdown")) { nsCOMPtr obs = do_GetService("@mozilla.org/observer-service;1", &rv); NS_ENSURE_SUCCESS(rv, rv); // No need for kungFuDeathGrip here, yay observerservice! obs->RemoveObserver(this, "xpcom-shutdown"); obs->RemoveObserver(this, "cycle-collector-begin"); sGeneration = 0; return NS_OK; } NS_ASSERTION(!strcmp(aTopic, "cycle-collector-begin"), "wrong topic"); // Increase generation to effectivly unmark all current objects if (!++sGeneration) { ++sGeneration; } // Iterate all toplevel windows nsCOMPtr windowList; nsCOMPtr med = do_GetService(NS_WINDOWMEDIATOR_CONTRACTID); if (med) { rv = med->GetEnumerator(nsnull, getter_AddRefs(windowList)); NS_ENSURE_SUCCESS(rv, rv); MarkWindowList(windowList); } nsCOMPtr ww = do_GetService(NS_WINDOWWATCHER_CONTRACTID); if (ww) { rv = ww->GetWindowEnumerator(getter_AddRefs(windowList)); NS_ENSURE_SUCCESS(rv, rv); MarkWindowList(windowList); } return NS_OK; }