From 14b0ca31bcb62e5b7e9e77634ef9cd2e8cf65800 Mon Sep 17 00:00:00 2001 From: Paul Davis Date: Tue, 30 Mar 2010 15:18:43 +0000 Subject: handle deletion of UI objects between the time that a callback is queued with the UI event loop and the execution of the callback (intrusive, big) git-svn-id: svn://localhost/ardour2/branches/3.0@6807 d708f5d6-7413-0410-9779-e7cbd77b26cf --- libs/pbd/pbd/abstract_ui.cc | 58 +++++++++++++++++++++++++++++++++------------ 1 file changed, 43 insertions(+), 15 deletions(-) (limited to 'libs/pbd/pbd/abstract_ui.cc') diff --git a/libs/pbd/pbd/abstract_ui.cc b/libs/pbd/pbd/abstract_ui.cc index cc7010a415..71371fe4d9 100644 --- a/libs/pbd/pbd/abstract_ui.cc +++ b/libs/pbd/pbd/abstract_ui.cc @@ -61,11 +61,13 @@ AbstractUI::get_request (RequestType rt) } vec.buf[0]->type = rt; + vec.buf[0]->valid = true; return vec.buf[0]; } RequestObject* req = new RequestObject; req->type = rt; + return req; } @@ -98,10 +100,15 @@ AbstractUI::handle_ui_requests () if (vec.len[0] == 0) { break; } else { - request_buffer_map_lock.unlock (); - do_request (vec.buf[0]); - request_buffer_map_lock.lock (); - i->second->increment_read_ptr (1); + if (vec.buf[0]->valid) { + request_buffer_map_lock.unlock (); + do_request (vec.buf[0]); + request_buffer_map_lock.lock (); + if (vec.buf[0]->invalidation) { + vec.buf[0]->invalidation->request = 0; + } + i->second->increment_read_ptr (1); + } } } } @@ -115,6 +122,30 @@ AbstractUI::handle_ui_requests () while (!request_list.empty()) { RequestObject* req = request_list.front (); request_list.pop_front (); + + /* We need to use this lock, because its the one + returned by slot_invalidation_mutex() and protects + against request invalidation. + */ + + request_buffer_map_lock.lock (); + if (!req->valid) { + delete req; + request_buffer_map_lock.unlock (); + continue; + } + + /* we're about to execute this request, so its + too late for any invalidation. mark + the request as "done" before we start. + */ + + if (req->invalidation) { + req->invalidation->request = 0; + } + + request_buffer_map_lock.unlock (); + lm.release (); do_request (req); @@ -152,14 +183,9 @@ AbstractUI::send_request (RequestObject *req) } template void -AbstractUI::call_slot (const boost::function& f) +AbstractUI::call_slot (InvalidationRecord* invalidation, const boost::function& f) { if (caller_is_self()) { -#ifndef NDEBUG - if (getenv ("DEBUG_THREADED_SIGNALS")) { - std::cerr << "functor called in correct thread for " << name() << " , execute ...\n"; - } -#endif f (); return; } @@ -171,11 +197,13 @@ AbstractUI::call_slot (const boost::function& f) } req->the_slot = f; -#ifndef NDEBUG - if (getenv ("DEBUG_THREADED_SIGNALS")) { - std::cerr << "functor called in wrong thread for " << name() << " (from " << pthread_name() << ") send request ...\n"; - } -#endif + req->invalidation = invalidation; + + if (invalidation) { + invalidation->request = req; + invalidation->event_loop = this; + } + send_request (req); } -- cgit v1.2.3