Try to fix intermittent refcount assertions in the presence of more than one thread...
[mozilla-central.git] / xpcom / ds / nsObserverList.cpp
blob43080c93dd8893709f1eb64ac89774573bad4b28
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /* ***** BEGIN LICENSE BLOCK *****
3 * Version: MPL 1.1/GPL 2.0/LGPL 2.1
5 * The contents of this file are subject to the Mozilla Public License Version
6 * 1.1 (the "License"); you may not use this file except in compliance with
7 * the License. You may obtain a copy of the License at
8 * http://www.mozilla.org/MPL/
10 * Software distributed under the License is distributed on an "AS IS" basis,
11 * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
12 * for the specific language governing rights and limitations under the
13 * License.
15 * The Original Code is mozilla.org code.
17 * The Initial Developer of the Original Code is
18 * Netscape Communications Corporation.
19 * Portions created by the Initial Developer are Copyright (C) 1998
20 * the Initial Developer. All Rights Reserved.
22 * Contributor(s):
24 * Alternatively, the contents of this file may be used under the terms of
25 * either of the GNU General Public License Version 2 or later (the "GPL"),
26 * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
27 * in which case the provisions of the GPL or the LGPL are applicable instead
28 * of those above. If you wish to allow use of your version of this file only
29 * under the terms of either the GPL or the LGPL, and not to allow others to
30 * use your version of this file under the terms of the MPL, indicate your
31 * decision by deleting the provisions above and replace them with the notice
32 * and other provisions required by the GPL or the LGPL. If you do not delete
33 * the provisions above, a recipient may use your version of this file under
34 * the terms of any one of the MPL, the GPL or the LGPL.
36 * ***** END LICENSE BLOCK ***** */
38 #include "nsObserverList.h"
40 #include "nsAutoPtr.h"
41 #include "nsCOMArray.h"
42 #include "nsISimpleEnumerator.h"
44 nsresult
45 nsObserverList::AddObserver(nsIObserver* anObserver, PRBool ownsWeak)
47 NS_ASSERTION(anObserver, "Null input");
49 if (!ownsWeak) {
50 ObserverRef* o = mObservers.AppendElement(anObserver);
51 if (!o)
52 return NS_ERROR_OUT_OF_MEMORY;
54 return NS_OK;
57 nsCOMPtr<nsIWeakReference> weak = do_GetWeakReference(anObserver);
58 if (!weak)
59 return NS_NOINTERFACE;
61 ObserverRef *o = mObservers.AppendElement(weak);
62 if (!o)
63 return NS_ERROR_OUT_OF_MEMORY;
65 return NS_OK;
68 nsresult
69 nsObserverList::RemoveObserver(nsIObserver* anObserver)
71 NS_ASSERTION(anObserver, "Null input");
73 if (mObservers.RemoveElement(static_cast<nsISupports*>(anObserver)))
74 return NS_OK;
76 nsCOMPtr<nsIWeakReference> observerRef = do_GetWeakReference(anObserver);
77 if (!observerRef)
78 return NS_ERROR_FAILURE;
80 if (!mObservers.RemoveElement(observerRef))
81 return NS_ERROR_FAILURE;
83 return NS_OK;
86 nsresult
87 nsObserverList::GetObserverList(nsISimpleEnumerator** anEnumerator)
89 nsRefPtr<nsObserverEnumerator> e(new nsObserverEnumerator(this));
90 if (!e)
91 return NS_ERROR_OUT_OF_MEMORY;
93 NS_ADDREF(*anEnumerator = e);
94 return NS_OK;
97 void
98 nsObserverList::FillObserverArray(nsCOMArray<nsIObserver> &aArray)
100 aArray.SetCapacity(mObservers.Length());
102 nsTArray<ObserverRef> observers(mObservers);
104 for (PRInt32 i = observers.Length() - 1; i >= 0; --i) {
105 if (observers[i].isWeakRef) {
106 nsCOMPtr<nsIObserver> o(do_QueryReferent(observers[i].asWeak()));
107 if (o) {
108 aArray.AppendObject(o);
110 else {
111 // the object has gone away, remove the weakref
112 mObservers.RemoveElement(observers[i].asWeak());
115 else {
116 aArray.AppendObject(observers[i].asObserver());
121 void
122 nsObserverList::NotifyObservers(nsISupports *aSubject,
123 const char *aTopic,
124 const PRUnichar *someData)
126 nsCOMArray<nsIObserver> observers;
127 FillObserverArray(observers);
129 for (PRInt32 i = 0; i < observers.Count(); ++i) {
130 observers[i]->Observe(aSubject, aTopic, someData);
134 NS_IMPL_ISUPPORTS1(nsObserverEnumerator, nsISimpleEnumerator)
136 nsObserverEnumerator::nsObserverEnumerator(nsObserverList* aObserverList)
137 : mIndex(0)
139 aObserverList->FillObserverArray(mObservers);
142 NS_IMETHODIMP
143 nsObserverEnumerator::HasMoreElements(PRBool *aResult)
145 *aResult = (mIndex < mObservers.Count());
146 return NS_OK;
149 NS_IMETHODIMP
150 nsObserverEnumerator::GetNext(nsISupports* *aResult)
152 if (mIndex == mObservers.Count()) {
153 NS_ERROR("Enumerating after HasMoreElements returned false.");
154 return NS_ERROR_UNEXPECTED;
157 NS_ADDREF(*aResult = mObservers[mIndex]);
158 ++mIndex;
159 return NS_OK;