Fix leak of keybinding table and keybinding documents. r=waterson b=17390, 24645

This commit is contained in:
dbaron%fas.harvard.edu 2000-07-27 23:31:09 +00:00
parent 1dfb7c70a6
commit 70057bf9f7
2 changed files with 104 additions and 34 deletions

View File

@ -64,6 +64,9 @@
#include "nsIPresContext.h"
#include "nsIDOMEventTarget.h"
#include "nsIObserver.h"
#include "nsIObserverService.h"
enum {
VK_CANCEL = 3,
VK_BACK = 8,
@ -199,13 +202,25 @@ enum eEventType {
eKeyUp
};
class nsXULKeyBindingDeleter : nsIObserver {
public:
nsXULKeyBindingDeleter();
virtual ~nsXULKeyBindingDeleter() { };
// nsISupports
NS_DECL_ISUPPORTS
// nsIObserver
NS_DECL_NSIOBSERVER
};
////////////////////////////////////////////////////////////////////////
// KeyListenerImpl
//
// This is the key listener implementation for keybinding
//
class nsXULKeyListenerImpl : public nsIXULKeyListener,
public nsIDOMKeyListener
public nsIDOMKeyListener
{
public:
nsXULKeyListenerImpl(void);
@ -220,6 +235,9 @@ public:
nsIDOMElement * aElement,
nsIDOMDocument * aDocument);
// to release the keybindings table from module shutdown
static void Shutdown(void);
// nsIDOMKeyListener
/**
@ -288,7 +306,6 @@ private:
nsIDOMElement* element; // Weak reference. The element will go away first.
nsIDOMXULDocument* mDOMDocument; // Weak reference.
static PRUint32 gRefCnt;
static nsSupportsHashtable* mKeyBindingTable;
// The "xul key" modifier can be any of the known modifiers:
@ -297,7 +314,6 @@ private:
} mXULKeyModifier;
};
PRUint32 nsXULKeyListenerImpl::gRefCnt = 0;
nsSupportsHashtable* nsXULKeyListenerImpl::mKeyBindingTable = nsnull;
class nsProxyStream : public nsIInputStream
@ -355,29 +371,51 @@ NS_IMPL_ISUPPORTS(nsProxyStream, NS_GET_IID(nsIInputStream));
////////////////////////////////////////////////////////////////////////
nsXULKeyBindingDeleter::nsXULKeyBindingDeleter(void)
{
NS_INIT_ISUPPORTS();
nsresult rv;
NS_WITH_SERVICE (nsIObserverService, observerService, NS_OBSERVERSERVICE_PROGID, &rv);
if (NS_FAILED(rv)) { return; }
nsAutoString topic;
topic.AssignWithConversion(NS_XPCOM_SHUTDOWN_OBSERVER_ID);
observerService->AddObserver(this,topic.GetUnicode());
}
NS_IMPL_ISUPPORTS1(nsXULKeyBindingDeleter, nsIObserver);
NS_IMETHODIMP
nsXULKeyBindingDeleter::Observe(nsISupports *aSubject, const PRUnichar *aTopic, const PRUnichar *someData)
{
#ifdef DEBUG
nsAutoString topic;
topic.AssignWithConversion(NS_XPCOM_SHUTDOWN_OBSERVER_ID);
NS_ASSERTION(topic.EqualsWithConversion(aTopic), "not shutdown");
#endif
nsXULKeyListenerImpl::Shutdown();
NS_RELEASE_THIS(); // we don't need to exist anymore
return NS_OK;
}
////////////////////////////////////////////////////////////////////////
nsXULKeyListenerImpl::nsXULKeyListenerImpl(void)
{
NS_INIT_REFCNT();
gRefCnt++;
if (gRefCnt == 1) {
if (!mKeyBindingTable) {
mKeyBindingTable = new nsSupportsHashtable();
#define LEAK_KEY_BINDINGS_TABLE_BUG_27739
#ifdef LEAK_KEY_BINDINGS_TABLE_BUG_27739
// XXX This is a total hack to deal with bug 27739. At some point,
// the keybindings table will go away, and we won't need to worry
// about this leak anymore.
++gRefCnt;
#endif
// the keybindings table will go away.
nsXULKeyBindingDeleter *deleter = new nsXULKeyBindingDeleter();
NS_IF_ADDREF(deleter); // this will delete itself, and the
// keybindings table, on XPCOM shutdown
}
}
nsXULKeyListenerImpl::~nsXULKeyListenerImpl(void)
{
gRefCnt--;
if (gRefCnt == 0)
delete mKeyBindingTable;
}
NS_IMPL_ADDREF(nsXULKeyListenerImpl)
@ -434,6 +472,50 @@ nsXULKeyListenerImpl::Init(
return NS_OK;
}
static PRBool PR_CALLBACK BreakScriptObjectCycle(nsHashKey *aKey, void *aData, void* closure)
{
nsIDOMXULDocument* domdoc = dont_AddRef(NS_STATIC_CAST(nsIDOMXULDocument*, aData));
nsCOMPtr<nsIDocument> document( do_QueryInterface(domdoc) );
nsCOMPtr<nsIScriptGlobalObject> globalObject;
document->GetScriptGlobalObject(getter_AddRefs(globalObject));
nsCOMPtr<nsIScriptContext> context;
if (globalObject) {
// get the context for GC below
globalObject->GetContext(getter_AddRefs(context));
// break circular references
globalObject->SetNewDocument(nsnull);
}
// break circular references (could this be in the if block above?)
document->SetScriptGlobalObject(nsnull);
if (globalObject) {
// break circular references
// this must be after SetScriptGlobalObject(nsnull)
// XXX Should this be SetDocShell instead?
globalObject->SetDocShell(nsnull);
}
if (context) {
// we must call GC on this context to clear newborn objects
context->GC();
}
return PR_TRUE;
}
//static
void
nsXULKeyListenerImpl::Shutdown(void)
{
if (mKeyBindingTable) {
mKeyBindingTable->Enumerate(BreakScriptObjectCycle);
delete mKeyBindingTable;
mKeyBindingTable = nsnull;
}
}
////////////////////////////////////////////////////////////////
// nsIDOMKeyListener
@ -516,7 +598,6 @@ nsresult nsXULKeyListenerImpl::DoKey(nsIDOMEvent* aKeyEvent, eEventType aEventTy
nsCAutoString editorPlatformFile = "chrome://communicator/content/platformEditorBindings.xul";
nsresult result;
nsCOMPtr<nsIDOMXULDocument> document;
if (!handled) {
while (piWindow && !handled) {
@ -543,6 +624,7 @@ nsresult nsXULKeyListenerImpl::DoKey(nsIDOMEvent* aKeyEvent, eEventType aEventTy
docShell->GetPresShell(getter_AddRefs(presShell));
PRBool editorHasBindings = PR_FALSE;
nsCOMPtr<nsIDOMXULDocument> document;
nsCOMPtr<nsIDOMXULDocument> platformDoc;
if (presShell) {
PRBool isEditor;
@ -1053,6 +1135,9 @@ PRBool nsXULKeyListenerImpl::IsMatchingCharCode(const nsString &theChar, const n
NS_IMETHODIMP nsXULKeyListenerImpl::GetKeyBindingDocument(nsCAutoString& aURLStr, nsIDOMXULDocument** aResult)
{
nsCOMPtr<nsIDOMXULDocument> document;
// This could only happen after shutdown. Is this right??
NS_ASSERTION(mKeyBindingTable, "Doing stuff after module shutdown!");
if (!mKeyBindingTable) return NS_ERROR_NULL_POINTER;
if (!aURLStr.IsEmpty()) {
nsCOMPtr<nsIURL> uri;
nsComponentManager::CreateInstance("component://netscape/network/standard-url",

View File

@ -20,24 +20,9 @@
* Contributor(s):
*/
#ifndef nsIXULKeyListener_h__
#define nsIXULKeyListener_h__
#ifndef nsXULKeyListener_h__
#define nsXULKeyListener_h__
// Generate this!
// {2C453161-0942-11d3-BF87-00105A1B0627}
#define NS_IXULKEYLISTENER_IID \
{ 0x2c453161, 0x942, 0x11d3, { 0xbf, 0x87, 0x0, 0x10, 0x5a, 0x1b, 0x6, 0x27 } }
#include "nsIXULKeyListener.h"
class nsIDOMElement;
class nsIXULKeyListener: public nsISupports {
public:
static const nsIID& GetIID() { static nsIID iid = NS_IXULKEYLISTENER_IID; return iid; }
NS_IMETHOD Init(nsIDOMElement* anElement, nsIDOMDocument* aDocument) = 0;
};
extern nsresult
NS_NewXULKeyListener(nsIXULKeyListener** result);
#endif // nsIXULKeyListener_h__
#endif // nsXULKeyListener_h__