From b042072df4fcf8fd036965d2c9f5d17c34e0d2c1 Mon Sep 17 00:00:00 2001 From: Paul Davis Date: Mon, 11 Apr 2016 09:46:45 -0400 Subject: [PATCH] fix logic for removing an AutomationControl from a ControlGroup --- libs/ardour/ardour/automation_control.h | 15 +++++++-- libs/ardour/ardour/control_group_member.h | 40 +++++++++++++++++++++++ libs/ardour/automation_control.cc | 8 +++-- libs/ardour/control_group.cc | 14 ++++++-- 4 files changed, 69 insertions(+), 8 deletions(-) create mode 100644 libs/ardour/ardour/control_group_member.h diff --git a/libs/ardour/ardour/automation_control.h b/libs/ardour/ardour/automation_control.h index d84f0a091f..a2eff17d9a 100644 --- a/libs/ardour/ardour/automation_control.h +++ b/libs/ardour/ardour/automation_control.h @@ -33,10 +33,12 @@ #include "evoral/types.hpp" #include "evoral/Control.hpp" -#include "ardour/libardour_visibility.h" #include "ardour/automation_list.h" +#include "ardour/control_group_member.h" #include "ardour/parameter_descriptor.h" +#include "ardour/libardour_visibility.h" + namespace ARDOUR { class Session; @@ -49,6 +51,7 @@ class LIBARDOUR_API AutomationControl : public PBD::Controllable , public Evoral::Control , public boost::enable_shared_from_this + , public ControlGroupMember { public: AutomationControl(ARDOUR::Session&, @@ -116,8 +119,6 @@ class LIBARDOUR_API AutomationControl const ARDOUR::Session& session() const { return _session; } void commit_transaction (bool did_write); - void set_group (boost::shared_ptr); - protected: ARDOUR::Session& _session; boost::shared_ptr _group; @@ -132,6 +133,14 @@ class LIBARDOUR_API AutomationControl */ virtual void actually_set_value (double value, PBD::Controllable::GroupControlDisposition); + + private: + /* I am unclear on why we have to make ControlGroup a friend in order + to get access to the ::set_group() method when it is already + declared to be a friend in ControlGroupMember. Oh well. + */ + friend class ControlGroup; + void set_group (boost::shared_ptr); }; class SlavableAutomationControl : public AutomationControl diff --git a/libs/ardour/ardour/control_group_member.h b/libs/ardour/ardour/control_group_member.h new file mode 100644 index 0000000000..385de9c157 --- /dev/null +++ b/libs/ardour/ardour/control_group_member.h @@ -0,0 +1,40 @@ +/* + Copyright (C) 2016 Paul Davis + + This program 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 2 of the License, or (at your option) + any later version. + + This program 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 + for more details. + + You should have received a copy of the GNU General Public License along + with this program; if not, write to the Free Software Foundation, Inc., + 675 Mass Ave, Cambridge, MA 02139, USA. +*/ + +#ifndef __libardour_control_group_member_h__ +#define __libardour_control_group_member_h__ + +namespace ARDOUR { + +class ControlGroup; + +class LIBARDOUR_API ControlGroupMember +{ + public: + virtual ~ControlGroupMember () {}; + private: + friend class ControlGroup; + /* Only a ControlGroup can call this; all membership changes must be + mediated by the ControlGroup, not by operating on the member. + */ + virtual void set_group (boost::shared_ptr) = 0; +}; + +} /* namespace */ + +#endif /* __libardour_control_group_member_h__ */ diff --git a/libs/ardour/automation_control.cc b/libs/ardour/automation_control.cc index 7efaa07f23..99e2f54165 100644 --- a/libs/ardour/automation_control.cc +++ b/libs/ardour/automation_control.cc @@ -268,9 +268,11 @@ AutomationControl::interface_to_internal (double val) const void AutomationControl::set_group (boost::shared_ptr cg) { - if (_group) { - _group->remove_control (shared_from_this()); - } + /* this method can only be called by a ControlGroup. We do not need + to ensure consistency by calling ControlGroup::remove_control(), + since we are guaranteed that the ControlGroup will take care of that + for us. + */ _group = cg; } diff --git a/libs/ardour/control_group.cc b/libs/ardour/control_group.cc index 6752940d27..2fe6af26ef 100644 --- a/libs/ardour/control_group.cc +++ b/libs/ardour/control_group.cc @@ -104,9 +104,19 @@ ControlGroup::control_going_away (boost::weak_ptr wac) int ControlGroup::remove_control (boost::shared_ptr ac) { - Glib::Threads::RWLock::WriterLock lm (controls_lock); + int erased; + + { + Glib::Threads::RWLock::WriterLock lm (controls_lock); + erased = _controls.erase (ac->id()); + } + + if (erased) { + ac->set_group (boost::shared_ptr()); + } + /* return zero if erased, non-zero otherwise */ - return !(_controls.erase (ac->id()) > 0); + return !erased; } int -- 2.30.2