diff options
author | Michael Stahl <michael.stahl@allotropia.de> | 2021-08-17 17:15:18 +0200 |
---|---|---|
committer | Michael Stahl <michael.stahl@allotropia.de> | 2021-08-19 12:13:32 +0200 |
commit | 71ebb55b4e888c8d1642d52742fc6d0d11ba3b9e (patch) | |
tree | cf64463af6cf4785c1d4fabb975a2694eec33e0d /toolkit/source | |
parent | 074790ad6f4e02e7aed3f9c954f0b4005b40fa21 (diff) |
toolkit: fix deadlock in MutableTreeNode
Broadcasting may call back into this, from
TreeControlPeer::updateNode().
Release maMutex before broadcasting to fix it.
(regression f7ee6baa1b14410fa79aac5dd9d6877df44a823a)
Change-Id: I91a8e043aaed02422601e5d4d5253379abab5d7b
Reviewed-on: https://gerrit.libreoffice.org/c/core/+/120616
Tested-by: Jenkins
Reviewed-by: Michael Stahl <michael.stahl@allotropia.de>
Diffstat (limited to 'toolkit/source')
-rw-r--r-- | toolkit/source/controls/tree/treedatamodel.cxx | 24 |
1 files changed, 14 insertions, 10 deletions
diff --git a/toolkit/source/controls/tree/treedatamodel.cxx b/toolkit/source/controls/tree/treedatamodel.cxx index b482d854349c..935fc39d46ff 100644 --- a/toolkit/source/controls/tree/treedatamodel.cxx +++ b/toolkit/source/controls/tree/treedatamodel.cxx @@ -85,7 +85,8 @@ public: void setParent( MutableTreeNode* pParent ); void broadcast_changes(); - void broadcast_changes(const Reference< XTreeNode >& xNode, bool bNew); + void broadcast_changes(std::unique_lock<std::mutex> & rLock, + const Reference< XTreeNode >& xNode, bool bNew); // XMutableTreeNode virtual css::uno::Any SAL_CALL getDataValue() override; @@ -273,11 +274,14 @@ void MutableTreeNode::broadcast_changes() } } -void MutableTreeNode::broadcast_changes(const Reference< XTreeNode >& xNode, bool bNew) +void MutableTreeNode::broadcast_changes(std::unique_lock<std::mutex> & rLock, + const Reference< XTreeNode >& xNode, bool const bNew) { - if( mxModel.is() ) + auto const xModel(mxModel); + rLock.unlock(); + if (xModel.is()) { - mxModel->broadcast( bNew ? nodes_inserted : nodes_removed, this, xNode ); + xModel->broadcast(bNew ? nodes_inserted : nodes_removed, this, xNode); } } @@ -295,7 +299,7 @@ void SAL_CALL MutableTreeNode::setDataValue( const Any& _datavalue ) void SAL_CALL MutableTreeNode::appendChild( const Reference< XMutableTreeNode >& xChildNode ) { - std::scoped_lock aGuard( maMutex ); + std::unique_lock aGuard( maMutex ); rtl::Reference< MutableTreeNode > xImpl( dynamic_cast< MutableTreeNode* >( xChildNode.get() ) ); if( !xImpl.is() || xImpl->mbIsInserted || (this == xImpl.get()) ) @@ -305,12 +309,12 @@ void SAL_CALL MutableTreeNode::appendChild( const Reference< XMutableTreeNode >& xImpl->setParent(this); xImpl->mbIsInserted = true; - broadcast_changes( xChildNode, true ); + broadcast_changes(aGuard, xChildNode, true); } void SAL_CALL MutableTreeNode::insertChildByIndex( sal_Int32 nChildIndex, const Reference< XMutableTreeNode >& xChildNode ) { - std::scoped_lock aGuard( maMutex ); + std::unique_lock aGuard( maMutex ); if( (nChildIndex < 0) || (nChildIndex > static_cast<sal_Int32>(maChildren.size())) ) throw IndexOutOfBoundsException(); @@ -327,12 +331,12 @@ void SAL_CALL MutableTreeNode::insertChildByIndex( sal_Int32 nChildIndex, const maChildren.insert( aIter, xImpl ); xImpl->setParent( this ); - broadcast_changes( xChildNode, true ); + broadcast_changes(aGuard, xChildNode, true); } void SAL_CALL MutableTreeNode::removeChildByIndex( sal_Int32 nChildIndex ) { - std::scoped_lock aGuard( maMutex ); + std::unique_lock aGuard( maMutex ); if( (nChildIndex < 0) || (nChildIndex >= static_cast<sal_Int32>(maChildren.size())) ) throw IndexOutOfBoundsException(); @@ -351,7 +355,7 @@ void SAL_CALL MutableTreeNode::removeChildByIndex( sal_Int32 nChildIndex ) xImpl->setParent(nullptr); xImpl->mbIsInserted = false; - broadcast_changes( xImpl, false ); + broadcast_changes(aGuard, xImpl, false); } void SAL_CALL MutableTreeNode::setHasChildrenOnDemand( sal_Bool bChildrenOnDemand ) |