mirror of
https://github.com/OPM/ResInsight.git
synced 2025-02-25 18:55:39 -06:00
#6437 Fix Signal crash and improve signal code
* The crash fix is ensuring we don't loop through m_disconnectCallbacks in DeleteSignal::send() * This is because the callback will end up calling disconnect which in turn erases the entry from m_disconnectCallbacks, invalidating the iterators used in the loop.
This commit is contained in:
@@ -50,6 +50,7 @@ set( PROJECT_FILES
|
|||||||
cafNotificationCenter.h
|
cafNotificationCenter.h
|
||||||
|
|
||||||
cafSignal.h
|
cafSignal.h
|
||||||
|
cafSignal.cpp
|
||||||
|
|
||||||
cafTristate.cpp
|
cafTristate.cpp
|
||||||
cafTristate.h
|
cafTristate.h
|
||||||
|
|||||||
@@ -119,6 +119,8 @@ void PdmObjectHandle::objectsWithReferringPtrFields( std::vector<PdmObjectHandle
|
|||||||
//--------------------------------------------------------------------------------------------------
|
//--------------------------------------------------------------------------------------------------
|
||||||
void PdmObjectHandle::prepareForDelete()
|
void PdmObjectHandle::prepareForDelete()
|
||||||
{
|
{
|
||||||
|
this->sendDeleteSignal();
|
||||||
|
|
||||||
m_parentField = nullptr;
|
m_parentField = nullptr;
|
||||||
|
|
||||||
for ( size_t i = 0; i < m_capabilities.size(); ++i )
|
for ( size_t i = 0; i < m_capabilities.size(); ++i )
|
||||||
|
|||||||
96
Fwk/AppFwk/cafProjectDataModel/cafPdmCore/cafSignal.cpp
Normal file
96
Fwk/AppFwk/cafProjectDataModel/cafPdmCore/cafSignal.cpp
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
//##################################################################################################
|
||||||
|
//
|
||||||
|
// Custom Visualization Core library
|
||||||
|
// Copyright (C) 2020- Ceetron Solutions AS
|
||||||
|
//
|
||||||
|
// This library may be used under the terms of either the GNU General Public License or
|
||||||
|
// the GNU Lesser General Public License as follows:
|
||||||
|
//
|
||||||
|
// GNU General Public License Usage
|
||||||
|
// This library is free software: you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU General Public License as published by
|
||||||
|
// the Free Software Foundation, either version 3 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This library is distributed in the hope that it will be useful, but WITHOUT ANY
|
||||||
|
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
// FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
//
|
||||||
|
// See the GNU General Public License at <<http://www.gnu.org/licenses/gpl.html>>
|
||||||
|
// for more details.
|
||||||
|
//
|
||||||
|
// GNU Lesser General Public License Usage
|
||||||
|
// This library is free software; you can redistribute it and/or modify
|
||||||
|
// it under the terms of the GNU Lesser General Public License as published by
|
||||||
|
// the Free Software Foundation; either version 2.1 of the License, or
|
||||||
|
// (at your option) any later version.
|
||||||
|
//
|
||||||
|
// This library is distributed in the hope that it will be useful, but WITHOUT ANY
|
||||||
|
// WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
// FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
//
|
||||||
|
// See the GNU Lesser General Public License at <<http://www.gnu.org/licenses/lgpl-2.1.html>>
|
||||||
|
// for more details.
|
||||||
|
//
|
||||||
|
//##################################################################################################
|
||||||
|
#include "cafSignal.h"
|
||||||
|
|
||||||
|
using namespace caf;
|
||||||
|
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
///
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
DeleteSignal::DeleteSignal( SignalObserver* observer )
|
||||||
|
: m_observer( observer )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
///
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
void DeleteSignal::disconnect( SignalObserver* observer )
|
||||||
|
{
|
||||||
|
// Remove the observer from the call back list
|
||||||
|
m_disconnectCallbacks.erase( observer );
|
||||||
|
}
|
||||||
|
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
///
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
void DeleteSignal::send()
|
||||||
|
{
|
||||||
|
// Make sure we swap out the disconnect signals to avoid
|
||||||
|
// erasing from the signal-map while looping through it.
|
||||||
|
std::map<SignalObserver*, DisconnectCallback> disconnectCallbacks;
|
||||||
|
disconnectCallbacks.swap( m_disconnectCallbacks );
|
||||||
|
|
||||||
|
for ( auto signalCallbackPair : disconnectCallbacks )
|
||||||
|
{
|
||||||
|
signalCallbackPair.second( m_observer );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
///
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
SignalObserver::SignalObserver()
|
||||||
|
: beingDeleted( this )
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
///
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
void SignalObserver::sendDeleteSignal()
|
||||||
|
{
|
||||||
|
// Make sure we
|
||||||
|
beingDeleted.send();
|
||||||
|
}
|
||||||
|
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
///
|
||||||
|
//--------------------------------------------------------------------------------------------------
|
||||||
|
SignalObserver::~SignalObserver()
|
||||||
|
{
|
||||||
|
sendDeleteSignal();
|
||||||
|
}
|
||||||
@@ -56,10 +56,7 @@ class DeleteSignal
|
|||||||
public:
|
public:
|
||||||
using DisconnectCallback = std::function<void( SignalObserver* )>;
|
using DisconnectCallback = std::function<void( SignalObserver* )>;
|
||||||
|
|
||||||
DeleteSignal( SignalObserver* observer )
|
DeleteSignal( SignalObserver* observer );
|
||||||
: m_observer( observer )
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
template <typename ClassType>
|
template <typename ClassType>
|
||||||
void connect( ClassType* signal )
|
void connect( ClassType* signal )
|
||||||
@@ -72,14 +69,8 @@ public:
|
|||||||
|
|
||||||
if ( signalCasted ) m_disconnectCallbacks[signalCasted] = disconnectCallback;
|
if ( signalCasted ) m_disconnectCallbacks[signalCasted] = disconnectCallback;
|
||||||
}
|
}
|
||||||
void disconnect( SignalObserver* observer ) { m_disconnectCallbacks.erase( observer ); }
|
void disconnect( SignalObserver* observer );
|
||||||
void send()
|
void send();
|
||||||
{
|
|
||||||
for ( auto signalCallbackPair : m_disconnectCallbacks )
|
|
||||||
{
|
|
||||||
signalCallbackPair.second( m_observer );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::map<SignalObserver*, DisconnectCallback> m_disconnectCallbacks;
|
std::map<SignalObserver*, DisconnectCallback> m_disconnectCallbacks;
|
||||||
@@ -107,11 +98,9 @@ public:
|
|||||||
DeleteSignal beingDeleted;
|
DeleteSignal beingDeleted;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
SignalObserver()
|
SignalObserver();
|
||||||
: beingDeleted( this )
|
void sendDeleteSignal();
|
||||||
{
|
virtual ~SignalObserver();
|
||||||
}
|
|
||||||
virtual ~SignalObserver() { beingDeleted.send(); }
|
|
||||||
|
|
||||||
}; // namespace caf
|
}; // namespace caf
|
||||||
|
|
||||||
@@ -170,6 +159,7 @@ public:
|
|||||||
};
|
};
|
||||||
connect( observer, lambda );
|
connect( observer, lambda );
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename ClassType>
|
template <typename ClassType>
|
||||||
void connect( ClassType* observer, const MemberCallback& callback )
|
void connect( ClassType* observer, const MemberCallback& callback )
|
||||||
{
|
{
|
||||||
@@ -187,6 +177,7 @@ public:
|
|||||||
m_observerCallbacks.erase( observer );
|
m_observerCallbacks.erase( observer );
|
||||||
observer->beingDeleted.disconnect( this );
|
observer->beingDeleted.disconnect( this );
|
||||||
}
|
}
|
||||||
|
|
||||||
void send( Args... args )
|
void send( Args... args )
|
||||||
{
|
{
|
||||||
for ( auto observerCallbackPair : m_observerCallbacks )
|
for ( auto observerCallbackPair : m_observerCallbacks )
|
||||||
@@ -197,12 +188,14 @@ public:
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void block( SignalObserver* observer )
|
void block( SignalObserver* observer )
|
||||||
{
|
{
|
||||||
auto it = m_observerCallbacks.find( observer );
|
auto it = m_observerCallbacks.find( observer );
|
||||||
CAF_ASSERT( it != m_observerCallbacks.end() );
|
CAF_ASSERT( it != m_observerCallbacks.end() );
|
||||||
it->second.second = false;
|
it->second.second = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void unblock( SignalObserver* observer )
|
void unblock( SignalObserver* observer )
|
||||||
{
|
{
|
||||||
auto it = m_observerCallbacks.find( observer );
|
auto it = m_observerCallbacks.find( observer );
|
||||||
|
|||||||
Reference in New Issue
Block a user