Version 6.4.0.0.beta1, tag libreoffice-6.4.0.0.beta1
[LibreOffice.git] / cppuhelper / source / weak.cxx
bloba7ef2410eeb8d946386e09cfc738672e762d6b0c
1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
2 /*
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #include <sal/config.h>
21 #include <sal/log.hxx>
23 #include <osl/diagnose.h>
24 #include <osl/mutex.hxx>
25 #include <cppuhelper/weakagg.hxx>
26 #include <cppuhelper/exc_hlp.hxx>
27 #include <cppuhelper/queryinterface.hxx>
29 #include <com/sun/star/lang/DisposedException.hpp>
31 #include <algorithm>
32 #include <vector>
34 using namespace osl;
35 using namespace com::sun::star::uno;
37 /** */ //for docpp
38 namespace cppu
41 // due to static Reflection destruction from usr, there must be a mutex leak (#73272#)
42 // this is used to lock all instances of OWeakConnectionPoint and OWeakRefListener as well as OWeakObject::m_pWeakConnectionPoint
43 static Mutex & getWeakMutex()
45 static Mutex * s_pMutex = new Mutex();
46 return *s_pMutex;
50 //-- OWeakConnectionPoint ----------------------------------------------------
52 class OWeakConnectionPoint: public XAdapter
54 public:
55 /**
56 Hold the weak object without an acquire (only the pointer).
58 explicit OWeakConnectionPoint( OWeakObject* pObj )
59 : m_aRefCount( 0 )
60 , m_pObject(pObj)
63 // noncopyable
64 OWeakConnectionPoint(const OWeakConnectionPoint&) = delete;
65 const OWeakConnectionPoint& operator=(const OWeakConnectionPoint&) = delete;
67 // XInterface
68 Any SAL_CALL queryInterface( const Type & rType ) override;
69 void SAL_CALL acquire() throw() override;
70 void SAL_CALL release() throw() override;
72 // XAdapter
73 css::uno::Reference< css::uno::XInterface > SAL_CALL queryAdapted() override;
74 void SAL_CALL addReference( const css::uno::Reference< css::uno::XReference >& xRef ) override;
75 void SAL_CALL removeReference( const css::uno::Reference< css::uno::XReference >& xRef ) override;
77 /// Called from the weak object if the reference count goes to zero.
78 ///
79 /// @throws css::uno::RuntimeException
80 void dispose();
82 private:
83 virtual ~OWeakConnectionPoint() {}
85 /// The reference counter.
86 oslInterlockedCount m_aRefCount;
87 /// The weak object
88 OWeakObject* m_pObject;
89 /// The container to hold the weak references
90 std::vector<Reference<XReference>> m_aReferences;
93 // XInterface
94 Any SAL_CALL OWeakConnectionPoint::queryInterface( const Type & rType )
96 return ::cppu::queryInterface(
97 rType, static_cast< XAdapter * >( this ), static_cast< XInterface * >( this ) );
100 // XInterface
101 void SAL_CALL OWeakConnectionPoint::acquire() throw()
103 osl_atomic_increment( &m_aRefCount );
106 // XInterface
107 void SAL_CALL OWeakConnectionPoint::release() throw()
109 if (! osl_atomic_decrement( &m_aRefCount ))
110 delete this;
113 void OWeakConnectionPoint::dispose()
115 std::vector<Reference<XReference>> aCopy;
116 { // only hold the mutex while we access the field
117 MutexGuard aGuard(getWeakMutex());
118 // OWeakObject is not the only owner of this, so clear m_pObject
119 // so that queryAdapted() won't use it now that it's dead
120 m_pObject = nullptr;
121 // other code is going to call removeReference while we are doing this, so we need a
122 // copy, but since we are disposing and going away, we can just take the original data
123 aCopy.swap(m_aReferences);
125 Any ex;
126 for (const Reference<XReference> & i : aCopy )
130 i->dispose();
132 catch (css::lang::DisposedException &) {}
133 catch (RuntimeException &)
135 ex = cppu::getCaughtException();
138 if (ex.hasValue())
140 cppu::throwException(ex);
144 // XInterface
145 Reference< XInterface > SAL_CALL OWeakConnectionPoint::queryAdapted()
147 Reference< XInterface > ret;
149 ClearableMutexGuard guard(getWeakMutex());
151 if (m_pObject)
153 oslInterlockedCount n = osl_atomic_increment( &m_pObject->m_refCount );
155 if (n > 1)
157 // The reference is incremented. The object cannot be destroyed.
158 // Release the guard at the earliest point.
159 guard.clear();
160 // WeakObject has a (XInterface *) cast operator
161 ret = *m_pObject;
162 osl_atomic_decrement( &m_pObject->m_refCount );
164 else
165 // Another thread wait in the dispose method at the guard
166 osl_atomic_decrement( &m_pObject->m_refCount );
169 return ret;
172 // XInterface
173 void SAL_CALL OWeakConnectionPoint::addReference(const Reference< XReference >& rRef)
175 MutexGuard aGuard(getWeakMutex());
176 m_aReferences.push_back( rRef );
179 // XInterface
180 void SAL_CALL OWeakConnectionPoint::removeReference(const Reference< XReference >& rRef)
182 MutexGuard aGuard(getWeakMutex());
183 // Search from end because the thing that last added a ref is most likely to be the
184 // first to remove a ref.
185 // It's not really valid to compare the pointer directly, but it's faster.
186 auto it = std::find_if(m_aReferences.rbegin(), m_aReferences.rend(),
187 [&rRef](const Reference<XReference>& rxRef) { return rxRef.get() == rRef.get(); });
188 if (it != m_aReferences.rend()) {
189 m_aReferences.erase( it.base()-1 );
190 return;
192 // interface not found, use the correct compare method
193 it = std::find(m_aReferences.rbegin(), m_aReferences.rend(), rRef);
194 if ( it != m_aReferences.rend() )
195 m_aReferences.erase( it.base()-1 );
199 //-- OWeakObject -------------------------------------------------------
202 #ifdef _MSC_VER
203 // Accidentally occurs in msvc mapfile = > had to be outlined.
204 OWeakObject::OWeakObject()
205 : m_refCount( 0 ),
206 m_pWeakConnectionPoint( nullptr )
209 #endif
211 // XInterface
212 Any SAL_CALL OWeakObject::queryInterface( const Type & rType )
214 return ::cppu::queryInterface(
215 rType,
216 static_cast< XWeak * >( this ), static_cast< XInterface * >( this ) );
219 // XInterface
220 void SAL_CALL OWeakObject::acquire() throw()
222 osl_atomic_increment( &m_refCount );
225 // XInterface
226 void SAL_CALL OWeakObject::release() throw()
228 if (osl_atomic_decrement( &m_refCount ) == 0) {
229 // notify/clear all weak-refs before object's dtor is executed
230 // (which may check weak-refs to this object):
231 disposeWeakConnectionPoint();
232 // destroy object:
233 delete this;
237 void OWeakObject::disposeWeakConnectionPoint()
239 OSL_PRECOND( m_refCount == 0, "OWeakObject::disposeWeakConnectionPoint: only to be called with a ref count of 0!" );
240 if (m_pWeakConnectionPoint != nullptr) {
241 OWeakConnectionPoint * const p = m_pWeakConnectionPoint;
242 m_pWeakConnectionPoint = nullptr;
243 try {
244 p->dispose();
246 catch (RuntimeException const& exc) {
247 SAL_WARN( "cppuhelper", exc );
249 p->release();
253 OWeakObject::~OWeakObject() COVERITY_NOEXCEPT_FALSE
257 // XWeak
258 Reference< XAdapter > SAL_CALL OWeakObject::queryAdapter()
260 if (!m_pWeakConnectionPoint)
262 // only acquire mutex if member is not created
263 MutexGuard aGuard( getWeakMutex() );
264 if( !m_pWeakConnectionPoint )
266 OWeakConnectionPoint * p = new OWeakConnectionPoint(this);
267 p->acquire();
268 m_pWeakConnectionPoint = p;
272 return m_pWeakConnectionPoint;
276 //-- OWeakAggObject ----------------------------------------------------
278 OWeakAggObject::~OWeakAggObject()
282 // XInterface
283 void OWeakAggObject::acquire() throw()
285 Reference<XInterface > x( xDelegator );
286 if (x.is())
287 x->acquire();
288 else
289 OWeakObject::acquire();
292 // XInterface
293 void OWeakAggObject::release() throw()
295 Reference<XInterface > x( xDelegator );
296 if (x.is())
297 x->release();
298 else
299 OWeakObject::release();
302 // XInterface
303 Any OWeakAggObject::queryInterface( const Type & rType )
305 Reference< XInterface > x( xDelegator ); // harden ref
306 return (x.is() ? x->queryInterface( rType ) : queryAggregation( rType ));
309 // XAggregation
310 Any OWeakAggObject::queryAggregation( const Type & rType )
312 return ::cppu::queryInterface(
313 rType,
314 static_cast< XInterface * >( static_cast< OWeakObject * >( this ) ),
315 static_cast< XAggregation * >( this ),
316 static_cast< XWeak * >( this ) );
319 // XAggregation
320 void OWeakAggObject::setDelegator( const Reference<XInterface > & rDelegator )
322 xDelegator = rDelegator;
327 /** */ //for docpp
328 namespace com
330 /** */ //for docpp
331 namespace sun
333 /** */ //for docpp
334 namespace star
336 /** */ //for docpp
337 namespace uno
341 //-- OWeakRefListener -----------------------------------------------------
343 class OWeakRefListener final : public XReference
345 public:
346 explicit OWeakRefListener(const Reference< XInterface >& xInt);
347 virtual ~OWeakRefListener();
349 // noncopyable
350 OWeakRefListener(const OWeakRefListener&) = delete;
351 const OWeakRefListener& operator=(const OWeakRefListener&) = delete;
353 // XInterface
354 Any SAL_CALL queryInterface( const Type & rType ) override;
355 void SAL_CALL acquire() throw() override;
356 void SAL_CALL release() throw() override;
358 // XReference
359 void SAL_CALL dispose() override;
361 /// The reference counter.
362 oslInterlockedCount m_aRefCount;
363 /// The connection point of the weak object, guarded by getWeakMutex()
364 Reference< XAdapter > m_XWeakConnectionPoint;
367 OWeakRefListener::OWeakRefListener(const Reference< XInterface >& xInt)
368 : m_aRefCount( 1 )
372 Reference< XWeak > xWeak( Reference< XWeak >::query( xInt ) );
374 if (xWeak.is())
376 m_XWeakConnectionPoint = xWeak->queryAdapter();
378 if (m_XWeakConnectionPoint.is())
380 m_XWeakConnectionPoint->addReference(static_cast<XReference*>(this));
384 catch (RuntimeException &) { OSL_ASSERT( false ); } // assert here, but no unexpected()
385 osl_atomic_decrement( &m_aRefCount );
388 OWeakRefListener::~OWeakRefListener()
392 if (m_XWeakConnectionPoint.is())
394 acquire(); // don't die again
395 m_XWeakConnectionPoint->removeReference(static_cast<XReference*>(this));
398 catch (RuntimeException &) { OSL_ASSERT( false ); } // assert here, but no unexpected()
401 // XInterface
402 Any SAL_CALL OWeakRefListener::queryInterface( const Type & rType )
404 return ::cppu::queryInterface(
405 rType, static_cast< XReference * >( this ), static_cast< XInterface * >( this ) );
408 // XInterface
409 void SAL_CALL OWeakRefListener::acquire() throw()
411 osl_atomic_increment( &m_aRefCount );
414 // XInterface
415 void SAL_CALL OWeakRefListener::release() throw()
417 if( ! osl_atomic_decrement( &m_aRefCount ) )
418 delete this;
421 void SAL_CALL OWeakRefListener::dispose()
423 Reference< XAdapter > xAdp;
425 MutexGuard guard(cppu::getWeakMutex());
426 if( m_XWeakConnectionPoint.is() )
428 xAdp = m_XWeakConnectionPoint;
429 m_XWeakConnectionPoint.clear();
433 if( xAdp.is() )
434 xAdp->removeReference(static_cast<XReference*>(this));
438 //-- WeakReferenceHelper ----------------------------------------------------------
440 WeakReferenceHelper::WeakReferenceHelper(const Reference< XInterface >& xInt)
441 : m_pImpl( nullptr )
443 if (xInt.is())
445 m_pImpl = new OWeakRefListener(xInt);
446 m_pImpl->acquire();
450 WeakReferenceHelper::WeakReferenceHelper(const WeakReferenceHelper& rWeakRef)
451 : m_pImpl( nullptr )
453 Reference< XInterface > xInt( rWeakRef.get() );
454 if (xInt.is())
456 m_pImpl = new OWeakRefListener(xInt);
457 m_pImpl->acquire();
461 void WeakReferenceHelper::clear()
465 if (m_pImpl)
467 m_pImpl->dispose();
468 m_pImpl->release();
469 m_pImpl = nullptr;
472 catch (RuntimeException &) { OSL_ASSERT( false ); } // assert here, but no unexpected()
475 WeakReferenceHelper& WeakReferenceHelper::operator=(const WeakReferenceHelper& rWeakRef)
477 if (this == &rWeakRef)
479 return *this;
481 Reference< XInterface > xInt( rWeakRef.get() );
482 return operator = ( xInt );
485 WeakReferenceHelper & WeakReferenceHelper::operator =(
486 WeakReferenceHelper && other)
488 clear();
489 std::swap(m_pImpl, other.m_pImpl);
490 return *this;
493 WeakReferenceHelper & SAL_CALL
494 WeakReferenceHelper::operator= (const Reference< XInterface > & xInt)
498 clear();
499 if (xInt.is())
501 m_pImpl = new OWeakRefListener(xInt);
502 m_pImpl->acquire();
505 catch (RuntimeException &) { OSL_ASSERT( false ); } // assert here, but no unexpected()
506 return *this;
509 WeakReferenceHelper::~WeakReferenceHelper()
511 clear();
514 Reference< XInterface > WeakReferenceHelper::get() const
518 Reference< XAdapter > xAdp;
520 // must lock to access m_XWeakConnectionPoint
521 MutexGuard guard(cppu::getWeakMutex());
522 if( m_pImpl && m_pImpl->m_XWeakConnectionPoint.is() )
523 xAdp = m_pImpl->m_XWeakConnectionPoint;
526 if (xAdp.is())
527 return xAdp->queryAdapted();
529 catch (RuntimeException &)
531 OSL_ASSERT( false );
532 } // assert here, but no unexpected()
534 return Reference< XInterface >();
542 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */