From 854472d550b23732f2897bc52002ebe4e07a4faf Mon Sep 17 00:00:00 2001 From: Seth Hillbrand Date: Tue, 5 Oct 2021 20:10:20 -0700 Subject: [PATCH] Prevent automatic RTree recaching Calling UpdateItem() may be performed by in a common loop, e.g. for( SCH_ITEM* item : GetScreen()->Items() ) or similar. We cannot call GetScreen()->Update( SCH_ITEM* ) in this routine as it will remove and re-add the item to the RTree, invalidating iterators. If needed, the items need to be cached to an external container before updating Fixes https://gitlab.com/kicad/code/kicad/issues/9318 --- eeschema/bus-wire-junction.cpp | 2 +- eeschema/dialogs/dialog_field_properties.cpp | 2 +- eeschema/dialogs/dialog_line_wire_bus_properties.cpp | 2 +- eeschema/dialogs/dialog_sheet_pin_properties.cpp | 2 +- eeschema/dialogs/dialog_symbol_properties.cpp | 4 ++-- eeschema/dialogs/dialog_text_and_label_properties.cpp | 2 +- eeschema/getpart.cpp | 4 ++-- eeschema/sch_base_frame.cpp | 9 +++++++-- eeschema/sch_base_frame.h | 2 +- eeschema/tools/sch_edit_tool.cpp | 10 +++++----- eeschema/tools/sch_editor_control.cpp | 4 ++-- eeschema/tools/sch_line_wire_bus_tool.cpp | 2 +- eeschema/tools/symbol_editor_edit_tool.cpp | 4 ++-- eeschema/tools/symbol_editor_pin_tool.cpp | 2 +- 14 files changed, 28 insertions(+), 23 deletions(-) diff --git a/eeschema/bus-wire-junction.cpp b/eeschema/bus-wire-junction.cpp index d218c17a74..69cae1cd8b 100644 --- a/eeschema/bus-wire-junction.cpp +++ b/eeschema/bus-wire-junction.cpp @@ -306,7 +306,7 @@ bool SCH_EDIT_FRAME::BreakSegment( SCH_LINE* aSegment, const wxPoint& aPoint, SaveCopyInUndoList( aScreen, newSegment, UNDO_REDO::NEWITEM, true ); SaveCopyInUndoList( aScreen, aSegment, UNDO_REDO::CHANGED, true ); - UpdateItem( aSegment ); + UpdateItem( aSegment, false, true ); aSegment->SetEndPoint( aPoint ); if( aNewSegment ) diff --git a/eeschema/dialogs/dialog_field_properties.cpp b/eeschema/dialogs/dialog_field_properties.cpp index 81f282a718..3c9ba1194f 100644 --- a/eeschema/dialogs/dialog_field_properties.cpp +++ b/eeschema/dialogs/dialog_field_properties.cpp @@ -542,7 +542,7 @@ void DIALOG_SCH_FIELD_PROPERTIES::UpdateField( SCH_FIELD* aField, SCH_SHEET_PATH else otherUnit->GetField( DATASHEET_FIELD )->SetText( m_text ); - editFrame->UpdateItem( otherUnit ); + editFrame->UpdateItem( otherUnit, false, true ); } } } diff --git a/eeschema/dialogs/dialog_line_wire_bus_properties.cpp b/eeschema/dialogs/dialog_line_wire_bus_properties.cpp index 2a10445f11..f85051d7ca 100644 --- a/eeschema/dialogs/dialog_line_wire_bus_properties.cpp +++ b/eeschema/dialogs/dialog_line_wire_bus_properties.cpp @@ -175,7 +175,7 @@ bool DIALOG_LINE_WIRE_BUS_PROPERTIES::TransferDataFromWindow() stroke.SetColor( m_colorSwatch->GetSwatchColor() ); strokeItem->SetStroke( stroke ); - m_frame->UpdateItem( strokeItem ); + m_frame->UpdateItem( strokeItem, false, true ); } m_frame->GetCanvas()->Refresh(); diff --git a/eeschema/dialogs/dialog_sheet_pin_properties.cpp b/eeschema/dialogs/dialog_sheet_pin_properties.cpp index c1e824ec07..c2d8672525 100644 --- a/eeschema/dialogs/dialog_sheet_pin_properties.cpp +++ b/eeschema/dialogs/dialog_sheet_pin_properties.cpp @@ -125,7 +125,7 @@ bool DIALOG_SHEET_PIN_PROPERTIES::TransferDataFromWindow() auto shape = static_cast( m_choiceConnectionType->GetCurrentSelection() ); m_sheetPin->SetShape( shape ); - m_frame->UpdateItem( m_sheetPin ); + m_frame->UpdateItem( m_sheetPin, false, true ); m_frame->GetCanvas()->Refresh(); m_frame->OnModify(); diff --git a/eeschema/dialogs/dialog_symbol_properties.cpp b/eeschema/dialogs/dialog_symbol_properties.cpp index 0865456f39..d95fa79db7 100644 --- a/eeschema/dialogs/dialog_symbol_properties.cpp +++ b/eeschema/dialogs/dialog_symbol_properties.cpp @@ -730,7 +730,7 @@ bool DIALOG_SYMBOL_PROPERTIES::TransferDataFromWindow() otherUnit->GetField( DATASHEET_FIELD )->SetText( m_fields->at( DATASHEET_FIELD ).GetText() ); otherUnit->SetIncludeInBom( !m_cbExcludeFromBom->IsChecked() ); otherUnit->SetIncludeOnBoard( !m_cbExcludeFromBoard->IsChecked() ); - GetParent()->UpdateItem( otherUnit ); + GetParent()->UpdateItem( otherUnit, false, true ); } } } @@ -748,7 +748,7 @@ bool DIALOG_SYMBOL_PROPERTIES::TransferDataFromWindow() currentScreen->Append( m_symbol ); GetParent()->TestDanglingEnds(); - GetParent()->UpdateItem( m_symbol ); + GetParent()->UpdateItem( m_symbol, false, true ); GetParent()->OnModify(); // This must go after OnModify() so that the connectivity graph will have been updated. diff --git a/eeschema/dialogs/dialog_text_and_label_properties.cpp b/eeschema/dialogs/dialog_text_and_label_properties.cpp index e60f2951cd..0dceb317da 100644 --- a/eeschema/dialogs/dialog_text_and_label_properties.cpp +++ b/eeschema/dialogs/dialog_text_and_label_properties.cpp @@ -380,7 +380,7 @@ bool DIALOG_TEXT_AND_LABEL_PROPERTIES::TransferDataFromWindow() } } - m_Parent->UpdateItem( m_CurrentText ); + m_Parent->UpdateItem( m_CurrentText, false, true ); m_Parent->GetCanvas()->Refresh(); m_Parent->OnModify(); diff --git a/eeschema/getpart.cpp b/eeschema/getpart.cpp index f3ccf10f9d..0cfc47aac8 100644 --- a/eeschema/getpart.cpp +++ b/eeschema/getpart.cpp @@ -226,7 +226,7 @@ void SCH_EDIT_FRAME::SelectUnit( SCH_SYMBOL* aSymbol, int aUnit ) TestDanglingEnds(); - UpdateItem( aSymbol ); + UpdateItem( aSymbol, false, true ); OnModify(); } } @@ -270,6 +270,6 @@ void SCH_EDIT_FRAME::ConvertPart( SCH_SYMBOL* aSymbol ) if( aSymbol->IsSelected() ) m_toolManager->RunAction( EE_ACTIONS::addItemToSel, true, aSymbol ); - UpdateItem( aSymbol ); + UpdateItem( aSymbol, false, true ); OnModify(); } diff --git a/eeschema/sch_base_frame.cpp b/eeschema/sch_base_frame.cpp index c53cb6d3de..5d7a736b42 100644 --- a/eeschema/sch_base_frame.cpp +++ b/eeschema/sch_base_frame.cpp @@ -299,7 +299,7 @@ void SCH_BASE_FRAME::createCanvas() } -void SCH_BASE_FRAME::UpdateItem( EDA_ITEM* aItem, bool isAddOrDelete ) +void SCH_BASE_FRAME::UpdateItem( EDA_ITEM* aItem, bool isAddOrDelete, bool aUpdateRtree ) { EDA_ITEM* parent = aItem->GetParent(); @@ -321,7 +321,12 @@ void SCH_BASE_FRAME::UpdateItem( EDA_ITEM* aItem, bool isAddOrDelete ) GetCanvas()->GetView()->Update( parent, KIGFX::REPAINT ); } - GetScreen()->Update( static_cast( aItem ) ); + /** + * Be careful when calling this. Update will invalidate RTree iterators, so you cannot call this + * while doing things like `for( SCH_ITEM* item : screen->Items() )` + */ + if( aUpdateRtree) + GetScreen()->Update( static_cast( aItem ) ); // Calling Refresh() here introduces a bi-stable state: when doing operations on a // large number of items if at some point the refresh timer times out and does a diff --git a/eeschema/sch_base_frame.h b/eeschema/sch_base_frame.h index 5f07a52a72..bc0d56f4e4 100644 --- a/eeschema/sch_base_frame.h +++ b/eeschema/sch_base_frame.h @@ -213,7 +213,7 @@ public: /** * Mark an item for refresh. */ - void UpdateItem( EDA_ITEM* aItem, bool isAddOrDelete = false ); + void UpdateItem( EDA_ITEM* aItem, bool isAddOrDelete = false, bool aUpdateRtree = false ); /** * Mark selected items for refresh. diff --git a/eeschema/tools/sch_edit_tool.cpp b/eeschema/tools/sch_edit_tool.cpp index 50b640139e..c430615918 100644 --- a/eeschema/tools/sch_edit_tool.cpp +++ b/eeschema/tools/sch_edit_tool.cpp @@ -551,7 +551,7 @@ int SCH_EDIT_TOOL::Rotate( const TOOL_EVENT& aEvent ) } connections = head->IsConnectable(); - m_frame->UpdateItem( head ); + m_frame->UpdateItem( head, false, true ); } else { @@ -627,7 +627,7 @@ int SCH_EDIT_TOOL::Rotate( const TOOL_EVENT& aEvent ) } connections |= item->IsConnectable(); - m_frame->UpdateItem( item ); + m_frame->UpdateItem( item, false, true ); updateItem( item, true ); } @@ -760,7 +760,7 @@ int SCH_EDIT_TOOL::Mirror( const TOOL_EVENT& aEvent ) } connections = item->IsConnectable(); - m_frame->UpdateItem( item ); + m_frame->UpdateItem( item, false, true ); } else if( selection.GetSize() > 1 ) { @@ -815,7 +815,7 @@ int SCH_EDIT_TOOL::Mirror( const TOOL_EVENT& aEvent ) } connections |= item->IsConnectable(); - m_frame->UpdateItem( item ); + m_frame->UpdateItem( item, false, true ); } } @@ -1134,7 +1134,7 @@ void SCH_EDIT_TOOL::editFieldText( SCH_FIELD* aField ) if( m_frame->eeconfig()->m_AutoplaceFields.enable || parentType == SCH_SHEET_T ) static_cast( aField->GetParent() )->AutoAutoplaceFields( m_frame->GetScreen() ); - m_frame->UpdateItem( aField ); + m_frame->UpdateItem( aField, false, true ); m_frame->OnModify(); // This must go after OnModify() so that the connectivity graph will have been updated. diff --git a/eeschema/tools/sch_editor_control.cpp b/eeschema/tools/sch_editor_control.cpp index eac4073cd5..62e508a765 100644 --- a/eeschema/tools/sch_editor_control.cpp +++ b/eeschema/tools/sch_editor_control.cpp @@ -544,7 +544,7 @@ int SCH_EDITOR_CONTROL::ReplaceAndFindNext( const TOOL_EVENT& aEvent ) { if( item->Replace( data, sheet ) ) { - m_frame->UpdateItem( item ); + m_frame->UpdateItem( item, false, true ); m_frame->GetCurrentSheet().UpdateAllScreenReferences(); m_frame->OnModify(); } @@ -575,7 +575,7 @@ int SCH_EDITOR_CONTROL::ReplaceAll( const TOOL_EVENT& aEvent ) { if( item->Replace( data, sheet ) ) { - m_frame->UpdateItem( item ); + m_frame->UpdateItem( item, false, true ); modified = true; } diff --git a/eeschema/tools/sch_line_wire_bus_tool.cpp b/eeschema/tools/sch_line_wire_bus_tool.cpp index 4b751b89ac..ec9498de38 100644 --- a/eeschema/tools/sch_line_wire_bus_tool.cpp +++ b/eeschema/tools/sch_line_wire_bus_tool.cpp @@ -705,7 +705,7 @@ int SCH_LINE_WIRE_BUS_TOOL::doDrawSegments( const std::string& aTool, int aType, m_busUnfold.flipY = flipY; m_busUnfold.flipX = flipX; - m_frame->UpdateItem( entry ); + m_frame->UpdateItem( entry, false, true ); m_wires.front()->SetStartPoint( entry->GetEnd() ); } diff --git a/eeschema/tools/symbol_editor_edit_tool.cpp b/eeschema/tools/symbol_editor_edit_tool.cpp index 1e793d881e..891f9fba50 100644 --- a/eeschema/tools/symbol_editor_edit_tool.cpp +++ b/eeschema/tools/symbol_editor_edit_tool.cpp @@ -167,7 +167,7 @@ int SYMBOL_EDITOR_EDIT_TOOL::Rotate( const TOOL_EVENT& aEvent ) { item = static_cast( selection.GetItem( ii ) ); item->Rotate( rotPoint, ccw ); - m_frame->UpdateItem( item ); + m_frame->UpdateItem( item, false, true ); } if( item->IsMoving() ) @@ -216,7 +216,7 @@ int SYMBOL_EDITOR_EDIT_TOOL::Mirror( const TOOL_EVENT& aEvent ) else item->MirrorHorizontal( mirrorPoint ); - m_frame->UpdateItem( item ); + m_frame->UpdateItem( item, false, true ); } m_toolMgr->PostEvent( EVENTS::SelectedItemsModified ); diff --git a/eeschema/tools/symbol_editor_pin_tool.cpp b/eeschema/tools/symbol_editor_pin_tool.cpp index f6f142caa0..38c47715eb 100644 --- a/eeschema/tools/symbol_editor_pin_tool.cpp +++ b/eeschema/tools/symbol_editor_pin_tool.cpp @@ -194,7 +194,7 @@ bool SYMBOL_EDITOR_PIN_TOOL::EditPinProperties( LIB_PIN* aPin ) } } - m_frame->UpdateItem( aPin ); + m_frame->UpdateItem( aPin, false, true ); m_frame->OnModify( ); std::vector items;