Mercurial > hg > svcore
changeset 1727:8efce64dd85e
Fix potential deadlock when notifying a handler that more models are [not] available
author | Chris Cannam |
---|---|
date | Thu, 20 Jun 2019 11:09:36 +0100 (2019-06-20) |
parents | d7ae9bfb015e |
children | abd8b9673028 649ac57c5a2d |
files | transform/ModelTransformerFactory.cpp |
diffstat | 1 files changed, 26 insertions(+), 5 deletions(-) [+] |
line wrap: on
line diff
--- a/transform/ModelTransformerFactory.cpp Thu Jun 20 11:09:05 2019 +0100 +++ b/transform/ModelTransformerFactory.cpp Thu Jun 20 11:09:36 2019 +0100 @@ -32,11 +32,14 @@ #include <iostream> #include <set> +#include <map> #include <QRegExp> #include <QMutexLocker> using std::vector; +using std::set; +using std::map; ModelTransformerFactory * ModelTransformerFactory::m_instance = new ModelTransformerFactory; @@ -53,7 +56,7 @@ ModelTransformer::Input ModelTransformerFactory::getConfigurationForTransform(Transform &transform, - const std::vector<Model *> &candidateInputModels, + const vector<Model *> &candidateInputModels, Model *defaultInputModel, AudioPlaySource *source, sv_frame_t startFrame, @@ -260,8 +263,6 @@ void ModelTransformerFactory::transformerFinished() { - QMutexLocker locker(&m_mutex); - QObject *s = sender(); ModelTransformer *transformer = dynamic_cast<ModelTransformer *>(s); @@ -272,6 +273,8 @@ return; } + m_mutex.lock(); + if (m_runningTransformers.find(transformer) == m_runningTransformers.end()) { cerr << "WARNING: ModelTransformerFactory::transformerFinished(" << transformer @@ -281,16 +284,34 @@ m_runningTransformers.erase(transformer); + map<AdditionalModelHandler *, vector<Model *>> toNotifyOfMore; + vector<AdditionalModelHandler *> toNotifyOfNoMore; + if (m_handlers.find(transformer) != m_handlers.end()) { if (transformer->willHaveAdditionalOutputModels()) { vector<Model *> mm = transformer->detachAdditionalOutputModels(); - m_handlers[transformer]->moreModelsAvailable(mm); + toNotifyOfMore[m_handlers[transformer]] = mm; } else { - m_handlers[transformer]->noMoreModelsAvailable(); + toNotifyOfNoMore.push_back(m_handlers[transformer]); } m_handlers.erase(transformer); } + m_mutex.unlock(); + + // These calls have to be made without the mutex held, as they may + // ultimately call back on us (e.g. we have one baroque situation + // where this could trigger a command to create a layer, which + // triggers the command history to clip the stack, which deletes a + // spare old model, which calls back on our modelAboutToBeDeleted) + + for (const auto &i: toNotifyOfMore) { + i.first->moreModelsAvailable(i.second); + } + for (AdditionalModelHandler *handler: toNotifyOfNoMore) { + handler->noMoreModelsAvailable(); + } + if (transformer->isAbandoned()) { if (transformer->getMessage() != "") { emit transformFailed("", transformer->getMessage());