diff options
author | Eike Rathke <erack@redhat.com> | 2012-12-12 22:54:28 +0100 |
---|---|---|
committer | Eike Rathke <erack@redhat.com> | 2012-12-12 22:56:21 +0100 |
commit | a60b149eede0c79dc08f00fd76d648a9cce0e660 (patch) | |
tree | 97e16d42cc580a47bddc987aed1de4b7e08427e2 | |
parent | 2316fe0dfdac3a295ec982a4f34648d0de367951 (diff) |
reworked solution for i#118012 crash during deletion of rows
commit 16155fdc39c004dc924a3b6919eb7c86da23c119 introduced a bottleneck
in area broadcasts with the change from
http://svn.apache.org/viewvc?view=revision&revision=1297916
That for each broadcast copied all areas in the affected slot(s) before
broadcasting, just in case that during Notify() an area would be removed
from the same slot invalidating the iterator, and attempted to find the
area again in the original set. For documents with lots of areas in a
slot and/or lots of slots that would be a major performance penalty.
Reworked such that to-be-erased area entries are marked and remembered
instead and cleaned up after iteration.
Change-Id: Ia485941746f435f8410d2084868263e84f25ffcb
-rw-r--r-- | sc/source/core/data/bcaslot.cxx | 183 | ||||
-rw-r--r-- | sc/source/core/inc/bcaslot.hxx | 53 |
2 files changed, 145 insertions, 91 deletions
diff --git a/sc/source/core/data/bcaslot.cxx b/sc/source/core/data/bcaslot.cxx index 602f6a7e8519..2a0ca67b0f2d 100644 --- a/sc/source/core/data/bcaslot.cxx +++ b/sc/source/core/data/bcaslot.cxx @@ -21,8 +21,6 @@ #include <svl/listener.hxx> #include <svl/listeneriter.hxx> -#include <boost/mem_fn.hpp> - #include "document.hxx" #include "brdcst.hxx" #include "bcaslot.hxx" @@ -114,7 +112,8 @@ ScBroadcastAreaSlot::ScBroadcastAreaSlot( ScDocument* pDocument, ScBroadcastAreaSlotMachine* pBASMa ) : aTmpSeekBroadcastArea( ScRange()), pDoc( pDocument ), - pBASM( pBASMa ) + pBASM( pBASMa ), + mbInBroadcastIteration( false) { } @@ -126,7 +125,7 @@ ScBroadcastAreaSlot::~ScBroadcastAreaSlot() { // Prevent hash from accessing dangling pointer in case area is // deleted. - ScBroadcastArea* pArea = *aIter; + ScBroadcastArea* pArea = (*aIter).mpArea; // Erase all so no hash will be accessed upon destruction of the // boost::unordered_map. aBroadcastAreaTbl.erase( aIter++); @@ -174,7 +173,7 @@ bool ScBroadcastAreaSlot::StartListeningArea( const ScRange& rRange, // would add quite some penalty for all but the first formula cell. ScBroadcastAreas::const_iterator aIter( FindBroadcastArea( rRange)); if (aIter != aBroadcastAreaTbl.end()) - rpArea = *aIter; + rpArea = (*aIter).mpArea; else { rpArea = new ScBroadcastArea( rRange); @@ -221,18 +220,15 @@ void ScBroadcastAreaSlot::EndListeningArea( const ScRange& rRange, if ( !rpArea ) { ScBroadcastAreas::iterator aIter( FindBroadcastArea( rRange)); - if (aIter == aBroadcastAreaTbl.end()) + if (aIter == aBroadcastAreaTbl.end() || isMarkedErased( aIter)) return; - rpArea = *aIter; + rpArea = (*aIter).mpArea; pListener->EndListening( rpArea->GetBroadcaster() ); if ( !rpArea->GetBroadcaster().HasListeners() ) { // if nobody is listening we can dispose it - aBroadcastAreaTbl.erase( aIter); - if ( !rpArea->DecRef() ) - { - delete rpArea; - rpArea = NULL; - } + if (rpArea->GetRef() == 1) + rpArea = NULL; // will be deleted by erase + EraseArea( aIter); } } else @@ -240,15 +236,12 @@ void ScBroadcastAreaSlot::EndListeningArea( const ScRange& rRange, if ( !rpArea->GetBroadcaster().HasListeners() ) { ScBroadcastAreas::iterator aIter( FindBroadcastArea( rRange)); - if (aIter == aBroadcastAreaTbl.end()) + if (aIter == aBroadcastAreaTbl.end() || isMarkedErased( aIter)) return; - OSL_ENSURE( *aIter == rpArea, "EndListeningArea: area pointer mismatch"); - aBroadcastAreaTbl.erase( aIter); - if ( !rpArea->DecRef() ) - { - delete rpArea; - rpArea = NULL; - } + OSL_ENSURE( (*aIter).mpArea == rpArea, "EndListeningArea: area pointer mismatch"); + if (rpArea->GetRef() == 1) + rpArea = NULL; // will be deleted by erase + EraseArea( aIter); } } } @@ -262,30 +255,20 @@ ScBroadcastAreas::iterator ScBroadcastAreaSlot::FindBroadcastArea( } -sal_Bool ScBroadcastAreaSlot::AreaBroadcast( const ScHint& rHint) const +sal_Bool ScBroadcastAreaSlot::AreaBroadcast( const ScHint& rHint) { if (aBroadcastAreaTbl.empty()) return false; + bool bInBroadcast = mbInBroadcastIteration; + mbInBroadcastIteration = true; sal_Bool bIsBroadcasted = false; - - // issue 118012 - // do not iterate on <aBoardcastAreaTbl> as its reveals that its iterators - // are destroyed during notification. - std::vector< ScBroadcastArea* > aCopyForIteration( aBroadcastAreaTbl.begin(), aBroadcastAreaTbl.end() ); - std::for_each( aCopyForIteration.begin(), aCopyForIteration.end(), boost::mem_fn( &ScBroadcastArea::IncRef ) ); - const ScAddress& rAddress = rHint.GetAddress(); - const std::vector< ScBroadcastArea* >::const_iterator aEnd( aCopyForIteration.end() ); - std::vector< ScBroadcastArea* >::const_iterator aIter; - for ( aIter = aCopyForIteration.begin(); aIter != aEnd; ++aIter ) + for (ScBroadcastAreas::const_iterator aIter( aBroadcastAreaTbl.begin()), + aIterEnd( aBroadcastAreaTbl.end()); aIter != aIterEnd; ++aIter ) { - ScBroadcastArea* pArea = *aIter; - // check, if copied item has been already removed from <aBroadcastAreaTbl> - if ( aBroadcastAreaTbl.find( pArea ) == aBroadcastAreaTbl.end() ) - { + if (isMarkedErased( aIter)) continue; - } - + ScBroadcastArea* pArea = (*aIter).mpArea; const ScRange& rAreaRange = pArea->GetRange(); if (rAreaRange.In( rAddress)) { @@ -296,45 +279,29 @@ sal_Bool ScBroadcastAreaSlot::AreaBroadcast( const ScHint& rHint) const } } } - - // delete no longer referenced <ScBroadcastArea> instances - for ( aIter = aCopyForIteration.begin(); aIter != aEnd; ++aIter ) - { - ScBroadcastArea* pArea = *aIter; - if ( !pArea->DecRef() ) - { - delete pArea; - } - } - + mbInBroadcastIteration = bInBroadcast; + // A Notify() during broadcast may call EndListeningArea() and thus dispose + // an area if it was the last listener, which would invalidate an iterator + // pointing to it, hence the real erase is done afterwards. + FinallyEraseAreas(); return bIsBroadcasted; } sal_Bool ScBroadcastAreaSlot::AreaBroadcastInRange( const ScRange& rRange, - const ScHint& rHint) const + const ScHint& rHint) { if (aBroadcastAreaTbl.empty()) return false; + bool bInBroadcast = mbInBroadcastIteration; + mbInBroadcastIteration = true; sal_Bool bIsBroadcasted = false; - - // issue 118012 - // do not iterate on <aBoardcastAreaTbl> as its reveals that its iterators - // are destroyed during notification. - std::vector< ScBroadcastArea* > aCopyForIteration( aBroadcastAreaTbl.begin(), aBroadcastAreaTbl.end() ); - std::for_each( aCopyForIteration.begin(), aCopyForIteration.end(), boost::mem_fn( &ScBroadcastArea::IncRef ) ); - - const std::vector< ScBroadcastArea* >::const_iterator aEnd( aCopyForIteration.end() ); - std::vector< ScBroadcastArea* >::const_iterator aIter; - for ( aIter = aCopyForIteration.begin(); aIter != aEnd; ++aIter ) + for (ScBroadcastAreas::const_iterator aIter( aBroadcastAreaTbl.begin()), + aIterEnd( aBroadcastAreaTbl.end()); aIter != aIterEnd; ++aIter ) { - ScBroadcastArea* pArea = *aIter; - // check, if copied item has been already removed from <aBroadcastAreaTbl> - if ( aBroadcastAreaTbl.find( pArea ) == aBroadcastAreaTbl.end() ) - { + if (isMarkedErased( aIter)) continue; - } - + ScBroadcastArea* pArea = (*aIter).mpArea; const ScRange& rAreaRange = pArea->GetRange(); if (rAreaRange.Intersects( rRange )) { @@ -345,17 +312,11 @@ sal_Bool ScBroadcastAreaSlot::AreaBroadcastInRange( const ScRange& rRange, } } } - - // delete no longer referenced <ScBroadcastArea> instances - for ( aIter = aCopyForIteration.begin(); aIter != aEnd; ++aIter ) - { - ScBroadcastArea* pArea = *aIter; - if ( !pArea->DecRef() ) - { - delete pArea; - } - } - + mbInBroadcastIteration = bInBroadcast; + // A Notify() during broadcast may call EndListeningArea() and thus dispose + // an area if it was the last listener, which would invalidate an iterator + // pointing to it, hence the real erase is done afterwards. + FinallyEraseAreas(); return bIsBroadcasted; } @@ -367,10 +328,10 @@ void ScBroadcastAreaSlot::DelBroadcastAreasInRange( const ScRange& rRange ) for (ScBroadcastAreas::iterator aIter( aBroadcastAreaTbl.begin()); aIter != aBroadcastAreaTbl.end(); /* increment in body */ ) { - const ScRange& rAreaRange = (*aIter)->GetRange(); + const ScRange& rAreaRange = (*aIter).mpArea->GetRange(); if (rRange.In( rAreaRange)) { - ScBroadcastArea* pArea = *aIter; + ScBroadcastArea* pArea = (*aIter).mpArea; aBroadcastAreaTbl.erase( aIter++); // erase before modifying if (!pArea->DecRef()) { @@ -398,7 +359,7 @@ void ScBroadcastAreaSlot::UpdateRemove( UpdateRefMode eUpdateRefMode, for ( ScBroadcastAreas::iterator aIter( aBroadcastAreaTbl.begin()); aIter != aBroadcastAreaTbl.end(); /* increment in body */ ) { - ScBroadcastArea* pArea = *aIter; + ScBroadcastArea* pArea = (*aIter).mpArea; if ( pArea->IsInUpdateChain() ) { aBroadcastAreaTbl.erase( aIter++); @@ -435,7 +396,7 @@ void ScBroadcastAreaSlot::UpdateRemoveArea( ScBroadcastArea* pArea ) ScBroadcastAreas::iterator aIter( aBroadcastAreaTbl.find( pArea)); if (aIter == aBroadcastAreaTbl.end()) return; - if (*aIter != pArea) + if ((*aIter).mpArea != pArea) OSL_FAIL( "UpdateRemoveArea: area pointer mismatch"); else { @@ -448,13 +409,13 @@ void ScBroadcastAreaSlot::UpdateRemoveArea( ScBroadcastArea* pArea ) void ScBroadcastAreaSlot::UpdateInsert( ScBroadcastArea* pArea ) { ::std::pair< ScBroadcastAreas::iterator, bool > aPair = - aBroadcastAreaTbl.insert( pArea ); + aBroadcastAreaTbl.insert( pArea); if (aPair.second) pArea->IncRef(); else { // Identical area already exists, add listeners. - ScBroadcastArea* pTarget = *(aPair.first); + ScBroadcastArea* pTarget = (*(aPair.first)).mpArea; if (pArea != pTarget) { SvtBroadcaster& rTarget = pTarget->GetBroadcaster(); @@ -469,6 +430,29 @@ void ScBroadcastAreaSlot::UpdateInsert( ScBroadcastArea* pArea ) } +void ScBroadcastAreaSlot::EraseArea( ScBroadcastAreas::iterator& rIter ) +{ + if (mbInBroadcastIteration) + { + (*rIter).mbErasure = true; // mark for erasure + pBASM->PushAreaToBeErased( this, rIter); + } + else + { + ScBroadcastArea* pArea = (*rIter).mpArea; + aBroadcastAreaTbl.erase( rIter); + if (!pArea->DecRef()) + delete pArea; + } +} + + +void ScBroadcastAreaSlot::FinallyEraseAreas() +{ + pBASM->FinallyEraseAreas( this); +} + + // --- ScBroadcastAreaSlotMachine ------------------------------------- ScBroadcastAreaSlotMachine::TableSlots::TableSlots() @@ -508,6 +492,9 @@ ScBroadcastAreaSlotMachine::~ScBroadcastAreaSlotMachine() delete (*iTab).second; } delete pBCAlways; + // Areas to-be-erased still present is a serious error in handling, but at + // this stage there's nothing we can do anymore. + SAL_WARN_IF( !maAreasToBeErased.empty(), "sc", "ScBroadcastAreaSlotMachine::dtor: maAreasToBeErased not empty"); } @@ -959,4 +946,34 @@ size_t ScBroadcastAreaSlotMachine::RemoveBulkArea( const ScBroadcastArea* pArea return aBulkBroadcastAreas.erase( pArea ); } + +void ScBroadcastAreaSlotMachine::PushAreaToBeErased( ScBroadcastAreaSlot* pSlot, + ScBroadcastAreas::iterator& rIter ) +{ + maAreasToBeErased.push_back( ::std::make_pair( pSlot, rIter)); +} + + +void ScBroadcastAreaSlotMachine::FinallyEraseAreas( ScBroadcastAreaSlot* pSlot ) +{ + SAL_WARN_IF( pSlot->IsInBroadcastIteration(), "sc", + "ScBroadcastAreaSlotMachine::FinallyEraseAreas: during iteration? NO!"); + if (pSlot->IsInBroadcastIteration()) + return; + + // maAreasToBeErased is a simple vector so erasing an element may + // invalidate iterators and would be inefficient anyway. Instead, copy + // elements to be preserved (usually none!) to temporary vector and swap. + AreasToBeErased aCopy; + for (AreasToBeErased::iterator aIt( maAreasToBeErased.begin()); + aIt != maAreasToBeErased.end(); ++aIt) + { + if ((*aIt).first == pSlot) + pSlot->EraseArea( (*aIt).second); + else + aCopy.push_back( *aIt); + } + maAreasToBeErased.swap( aCopy); +} + /* vim:set shiftwidth=4 softtabstop=4 expandtab: */ diff --git a/sc/source/core/inc/bcaslot.hxx b/sc/source/core/inc/bcaslot.hxx index eefc04d5f126..4fbbb850b15a 100644 --- a/sc/source/core/inc/bcaslot.hxx +++ b/sc/source/core/inc/bcaslot.hxx @@ -71,23 +71,31 @@ inline bool ScBroadcastArea::operator==( const ScBroadcastArea & rArea ) const //============================================================================= +struct ScBroadcastAreaEntry +{ + ScBroadcastArea* mpArea; + mutable bool mbErasure; ///< TRUE if marked for erasure in this set + + ScBroadcastAreaEntry( ScBroadcastArea* p ) : mpArea( p), mbErasure( false) {} +}; + struct ScBroadcastAreaHash { - size_t operator()( const ScBroadcastArea* p ) const + size_t operator()( const ScBroadcastAreaEntry& rEntry ) const { - return p->GetRange().hashArea(); + return rEntry.mpArea->GetRange().hashArea(); } }; struct ScBroadcastAreaEqual { - bool operator()( const ScBroadcastArea* p1, const ScBroadcastArea* p2) const + bool operator()( const ScBroadcastAreaEntry& rEntry1, const ScBroadcastAreaEntry& rEntry2) const { - return *p1 == *p2; + return *rEntry1.mpArea == *rEntry2.mpArea; } }; -typedef ::boost::unordered_set< ScBroadcastArea*, ScBroadcastAreaHash, ScBroadcastAreaEqual > ScBroadcastAreas; +typedef ::boost::unordered_set< ScBroadcastAreaEntry, ScBroadcastAreaHash, ScBroadcastAreaEqual > ScBroadcastAreas; //============================================================================= @@ -122,6 +130,7 @@ private: mutable ScBroadcastArea aTmpSeekBroadcastArea; // for FindBroadcastArea() ScDocument* pDoc; ScBroadcastAreaSlotMachine* pBASM; + bool mbInBroadcastIteration; ScBroadcastAreas::iterator FindBroadcastArea( const ScRange& rRange ) const; @@ -135,6 +144,14 @@ private: */ bool CheckHardRecalcStateCondition() const; + /** Finally erase all areas pushed as to-be-erased. */ + void FinallyEraseAreas(); + + bool isMarkedErased( const ScBroadcastAreas::iterator& rIter ) + { + return (*rIter).mbErasure; + } + public: ScBroadcastAreaSlot( ScDocument* pDoc, ScBroadcastAreaSlotMachine* pBASM ); @@ -172,16 +189,27 @@ public: void EndListeningArea( const ScRange& rRange, SvtListener* pListener, ScBroadcastArea*& rpArea ); - sal_Bool AreaBroadcast( const ScHint& rHint ) const; + sal_Bool AreaBroadcast( const ScHint& rHint ); /// @return sal_True if at least one broadcast occurred. sal_Bool AreaBroadcastInRange( const ScRange& rRange, - const ScHint& rHint ) const; + const ScHint& rHint ); void DelBroadcastAreasInRange( const ScRange& rRange ); void UpdateRemove( UpdateRefMode eUpdateRefMode, const ScRange& rRange, SCsCOL nDx, SCsROW nDy, SCsTAB nDz ); void UpdateRemoveArea( ScBroadcastArea* pArea ); void UpdateInsert( ScBroadcastArea* pArea ); + + + bool IsInBroadcastIteration() const { return mbInBroadcastIteration; } + + /** Erase an area from set and delete it if last reference, or if + mbInBroadcastIteration is set push it to the vector of to-be-erased + areas instead. + + Meant to be used internally and from ScBroadcastAreaSlotMachine only. + */ + void EraseArea( ScBroadcastAreas::iterator& rIter ); }; @@ -229,14 +257,17 @@ private: typedef ::std::map< SCTAB, TableSlots* > TableSlotsMap; + typedef ::std::vector< ::std::pair< ScBroadcastAreaSlot*, ScBroadcastAreas::iterator > > AreasToBeErased; + private: ScBroadcastAreasBulk aBulkBroadcastAreas; TableSlotsMap aTableSlotsMap; + AreasToBeErased maAreasToBeErased; SvtBroadcaster *pBCAlways; // for the RC_ALWAYS special range ScDocument *pDoc; ScBroadcastArea *pUpdateChain; ScBroadcastArea *pEOUpdateChain; - sal_uLong nInBulkBroadcast; + sal_uLong nInBulkBroadcast; inline SCSIZE ComputeSlotOffset( const ScAddress& rAddress ) const; void ComputeAreaPoints( const ScRange& rRange, @@ -267,6 +298,12 @@ public: inline ScBroadcastArea* GetEOUpdateChain() const { return pEOUpdateChain; } inline void SetEOUpdateChain( ScBroadcastArea* p ) { pEOUpdateChain = p; } inline bool IsInBulkBroadcast() const { return nInBulkBroadcast > 0; } + + // only for ScBroadcastAreaSlot + void PushAreaToBeErased( ScBroadcastAreaSlot* pSlot, + ScBroadcastAreas::iterator& rIter ); + // only for ScBroadcastAreaSlot + void FinallyEraseAreas( ScBroadcastAreaSlot* pSlot ); }; |