X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/8cd6a9ad50c00a94e62558a3b55e814028d81100..e0e2cbbe946954db58b28aab3d9cc207987a6a5a:/src/common/event.cpp diff --git a/src/common/event.cpp b/src/common/event.cpp index 69abfcc2bb..9e8fbdf508 100644 --- a/src/common/event.cpp +++ b/src/common/event.cpp @@ -1032,6 +1032,8 @@ void wxEventHashTable::GrowEventTypeTable() wxEvtHandler::wxEvtHandler() { + m_beingDeleted = false; + m_nextHandler = (wxEvtHandler *) NULL; m_previousHandler = (wxEvtHandler *) NULL; m_enabled = true; @@ -1045,6 +1047,8 @@ wxEvtHandler::wxEvtHandler() wxEvtHandler::~wxEvtHandler() { + m_beingDeleted = true; + // Takes itself out of the list of handlers if (m_previousHandler) m_previousHandler->m_nextHandler = m_nextHandler; @@ -1061,9 +1065,7 @@ wxEvtHandler::~wxEvtHandler() { wxDynamicEventTableEntry *entry = (wxDynamicEventTableEntry*)*it; -#if wxUSE_WEAKREF // Remove ourselves from sink destructor notifications - // (this has usually been been done, in wxTrackable destructor) wxEvtHandler *eventSink = entry->m_eventSink; if ( eventSink ) { @@ -1075,7 +1077,6 @@ wxEvtHandler::~wxEvtHandler() delete evtConnRef; } } -#endif // wxUSE_WEAKREF if (entry->m_callbackUserData) delete entry->m_callbackUserData; @@ -1085,8 +1086,17 @@ wxEvtHandler::~wxEvtHandler() }; if (m_pendingEvents) + { + // At this time, we could still be used from other threads. + // Continue to use sync objects. + wxENTER_CRIT_SECT( m_pendingEventsLock ); + m_pendingEvents->DeleteContents(true); - delete m_pendingEvents; + delete m_pendingEvents; + m_pendingEvents = NULL; + + wxLEAVE_CRIT_SECT( m_pendingEventsLock ); + } // Remove us from wxPendingEvents if necessary. if ( wxPendingEvents ) @@ -1130,23 +1140,20 @@ bool wxEvtHandler::ProcessThreadEvent(const wxEvent& event) #endif // wxUSE_THREADS -void wxEvtHandler::AddPendingEvent(const wxEvent& event) +void wxEvtHandler::QueueEvent(wxEvent *event) { - // 1) Add event to list of pending events of this event handler - - wxEvent *eventCopy = event.Clone(); + wxCHECK_RET( event, "NULL event can't be posted" ); - // we must be able to copy the events here so the event class must - // implement Clone() properly instead of just providing a NULL stab for it - wxCHECK_RET( eventCopy, - _T("events of this type aren't supposed to be posted") ); + // Catch the situation where destructor is already invoked (in another thread) + if( m_beingDeleted ) return; + // 1) Add this event to our list of pending events wxENTER_CRIT_SECT( m_pendingEventsLock ); if ( !m_pendingEvents ) m_pendingEvents = new wxList; - m_pendingEvents->Append(eventCopy); + m_pendingEvents->Append(event); wxLEAVE_CRIT_SECT( m_pendingEventsLock ); @@ -1171,10 +1178,21 @@ void wxEvtHandler::ProcessPendingEvents() { wxENTER_CRIT_SECT( m_pendingEventsLock ); - // this method is only called by wxApp if this handler does have - // pending events - wxCHECK_RET( m_pendingEvents && !m_pendingEvents->IsEmpty(), - "should have pending events if called" ); + // This method is only called by wxApp if this handler does have + // pending events, but it happens occasionally when using multi- + // threading and we don't want a crash due to that. + if( !m_pendingEvents ) + { + wxLEAVE_CRIT_SECT( m_pendingEventsLock ); + return; + } + + if( m_pendingEvents->IsEmpty() ) + { + wxPendingEvents->DeleteObject(this); + wxLEAVE_CRIT_SECT( m_pendingEventsLock ); + return; + } wxList::compatibility_iterator node = m_pendingEvents->GetFirst(); wxEventPtr event(wx_static_cast(wxEvent *, node->GetData())); @@ -1187,7 +1205,17 @@ void wxEvtHandler::ProcessPendingEvents() // if there are no more pending events left, we don't need to stay in this // list if ( m_pendingEvents->IsEmpty() ) + { +#if wxUSE_THREADS + if (wxPendingEventsLocker) + wxENTER_CRIT_SECT(*wxPendingEventsLocker); +#endif wxPendingEvents->DeleteObject(this); +#if wxUSE_THREADS + if (wxPendingEventsLocker) + wxLEAVE_CRIT_SECT(*wxPendingEventsLocker); +#endif + } wxLEAVE_CRIT_SECT( m_pendingEventsLock ); @@ -1375,7 +1403,6 @@ void wxEvtHandler::Connect( int id, int lastId, // Insert at the front of the list so most recent additions are found first m_dynamicEvents->Insert( (wxObject*) entry ); -#if wxUSE_WEAKREF // Make sure we get to know when a sink is destroyed if ( eventSink && eventSink != this ) { @@ -1385,7 +1412,6 @@ void wxEvtHandler::Connect( int id, int lastId, else evtConnRef = new wxEventConnectionRef(this, eventSink); } -#endif // wxUSE_WEAKREF } bool wxEvtHandler::Disconnect( int id, int lastId, wxEventType eventType, @@ -1396,7 +1422,6 @@ bool wxEvtHandler::Disconnect( int id, int lastId, wxEventType eventType, if (!m_dynamicEvents) return false; -#if wxUSE_WEAKREF // Remove connection from tracker node (wxEventConnectionRef) if ( eventSink && eventSink != this ) { @@ -1404,7 +1429,6 @@ bool wxEvtHandler::Disconnect( int id, int lastId, wxEventType eventType, if ( evtConnRef ) evtConnRef->DecRef(); } -#endif // wxUSE_WEAKREF wxList::compatibility_iterator node = m_dynamicEvents->GetFirst(); while (node) @@ -1496,7 +1520,6 @@ void *wxEvtHandler::DoGetClientData() const return m_clientData; } -#if wxUSE_WEAKREF // A helper to find an wxEventConnectionRef object wxEventConnectionRef * wxEvtHandler::FindRefInTrackerList(wxEvtHandler *eventSink) @@ -1536,7 +1559,6 @@ void wxEvtHandler::OnSinkDestroyed( wxEvtHandler *sink ) node = node_nxt; } } -#endif // wxUSE_WEAKREF #endif // wxUSE_BASE