MCP: Mixbus32C: Restore missing filter controls to the Dyn page.
[ardour.git] / libs / ardour / route.cc
index 043b11f33092a68ae408a58a99e6ca5c0d90afcd..dba71584e2c4390abbd183da182682148df8b901 100644 (file)
@@ -32,7 +32,7 @@
 #include "pbd/enumwriter.h"
 #include "pbd/memento_command.h"
 #include "pbd/stacktrace.h"
-#include "pbd/convert.h"
+#include "pbd/types_convert.h"
 #include "pbd/unwind.h"
 
 #include "ardour/amp.h"
@@ -71,6 +71,7 @@
 #include "ardour/session.h"
 #include "ardour/solo_control.h"
 #include "ardour/solo_isolate_control.h"
+#include "ardour/types_convert.h"
 #include "ardour/unknown_processor.h"
 #include "ardour/utils.h"
 #include "ardour/vca.h"
@@ -88,7 +89,6 @@ Route::Route (Session& sess, string name, PresentationInfo::Flag flag, DataType
        : Stripable (sess, name, PresentationInfo (flag))
        , GraphNode (sess._process_graph)
        , Muteable (sess, name)
-       , Automatable (sess)
        , _active (true)
        , _signal_latency (0)
        , _signal_latency_at_amp_position (0)
@@ -120,7 +120,7 @@ Route::Route (Session& sess, string name, PresentationInfo::Flag flag, DataType
 
 boost::weak_ptr<Route>
 Route::weakroute () {
-       return boost::weak_ptr<Route> (shared_from_this ());
+       return boost::weak_ptr<Route> (boost::dynamic_pointer_cast<Route> (shared_from_this ()));
 }
 
 int
@@ -876,7 +876,7 @@ Route::add_processor_from_xml_2X (const XMLNode& node, int version)
                //A2 uses the "active" flag in the toplevel redirect node, not in the child plugin/IO
                if (i != children.end()) {
                        if ((prop = (*i)->property (X_("active"))) != 0) {
-                               if ( string_is_affirmative (prop->value()) && (!_session.get_bypass_all_loaded_plugins () || !processor->display_to_user () ) )
+                               if ( string_to<bool> (prop->value()) && (!_session.get_bypass_all_loaded_plugins () || !processor->display_to_user () ) )
                                        processor->activate();
                                else
                                        processor->deactivate();
@@ -969,7 +969,7 @@ Route::add_processors (const ProcessorList& others, boost::shared_ptr<Processor>
                flags &= mask;
 
                if (flags != None) {
-                       boost::optional<int> rv = PluginSetup (shared_from_this (), pi, flags);  /* EMIT SIGNAL */
+                       boost::optional<int> rv = PluginSetup (boost::dynamic_pointer_cast<Route>(shared_from_this ()), pi, flags);  /* EMIT SIGNAL */
                        int mode = rv.get_value_or (0);
                        switch (mode & 3) {
                                case 1:
@@ -2316,25 +2316,22 @@ Route::state(bool full_state)
 
        XMLNode *node = new XMLNode("Route");
        ProcessorList::iterator i;
-       char buf[32];
 
-       id().print (buf, sizeof (buf));
-       node->add_property("id", buf);
-       node->add_property ("name", _name);
-       node->add_property("default-type", _default_type.to_string());
-       node->add_property ("strict-io", _strict_io);
+       node->set_property ("id", id ());
+       node->set_property ("name", name());
+       node->set_property ("default-type", _default_type);
+       node->set_property ("strict-io", _strict_io);
 
        node->add_child_nocopy (_presentation_info.get_state());
 
-       node->add_property("active", _active?"yes":"no");
-       string p;
-       node->add_property("denormal-protection", _denormal_protection?"yes":"no");
-       node->add_property("meter-point", enum_2_string (_meter_point));
+       node->set_property ("active", _active);
+       node->set_property ("denormal-protection", _denormal_protection);
+       node->set_property ("meter-point", _meter_point);
 
-       node->add_property("meter-type", enum_2_string (_meter_type));
+       node->set_property ("meter-type", _meter_type);
 
        if (_route_group) {
-               node->add_property("route-group", _route_group->name());
+               node->set_property ("route-group", _route_group->name());
        }
 
        node->add_child_nocopy (_solo_control->get_state ());
@@ -2393,8 +2390,7 @@ Route::state(bool full_state)
        if (_custom_meter_position_noted) {
                boost::shared_ptr<Processor> after = _processor_after_last_custom_meter.lock ();
                if (after) {
-                       after->id().print (buf, sizeof (buf));
-                       node->add_property (X_("processor-after-last-custom-meter"), buf);
+                       node->set_property (X_("processor-after-last-custom-meter"), after->id());
                }
        }
 
@@ -2417,15 +2413,15 @@ Route::set_state (const XMLNode& node, int version)
        XMLNodeList nlist;
        XMLNodeConstIterator niter;
        XMLNode *child;
-       XMLProperty const * prop;
 
        if (node.name() != "Route"){
                error << string_compose(_("Bad node sent to Route::set_state() [%1]"), node.name()) << endmsg;
                return -1;
        }
 
-       if ((prop = node.property (X_("name"))) != 0) {
-               Route::set_name (prop->value());
+       std::string route_name;
+       if (node.get_property (X_("name"), route_name)) {
+               Route::set_name (route_name);
        }
 
        set_id (node);
@@ -2433,9 +2429,7 @@ Route::set_state (const XMLNode& node, int version)
 
        Stripable::set_state (node, version);
 
-       if ((prop = node.property (X_("strict-io"))) != 0) {
-               _strict_io = string_is_affirmative (prop->value());
-       }
+       node.get_property (X_("strict-io"), _strict_io);
 
        if (is_monitor()) {
                /* monitor bus does not get a panner, but if (re)created
@@ -2457,13 +2451,14 @@ Route::set_state (const XMLNode& node, int version)
                child = *niter;
 
                if (child->name() == IO::state_node_name) {
-                       if ((prop = child->property (X_("direction"))) == 0) {
+                       std::string direction;
+                       if (!child->get_property (X_("direction"), direction)) {
                                continue;
                        }
 
-                       if (prop->value() == "Input") {
+                       if (direction == "Input") {
                                _input->set_state (*child, version);
-                       } else if (prop->value() == "Output") {
+                       } else if (direction == "Output") {
                                _output->set_state (*child, version);
                        }
 
@@ -2480,17 +2475,15 @@ Route::set_state (const XMLNode& node, int version)
                }
        }
 
-       if ((prop = node.property (X_("meter-point"))) != 0) {
-               MeterPoint mp = MeterPoint (string_2_enum (prop->value (), _meter_point));
+       MeterPoint mp;
+       if (node.get_property (X_("meter-point"), mp)) {
                set_meter_point (mp, true);
                if (_meter) {
                        _meter->set_display_to_user (_meter_point == MeterCustom);
                }
        }
 
-       if ((prop = node.property (X_("meter-type"))) != 0) {
-               _meter_type = MeterType (string_2_enum (prop->value (), _meter_type));
-       }
+       node.get_property (X_("meter-type"), _meter_type);
 
        _initial_io_setup = false;
 
@@ -2499,22 +2492,25 @@ Route::set_state (const XMLNode& node, int version)
        // this looks up the internal instrument in processors
        reset_instrument_info();
 
-       if ((prop = node.property (X_("denormal-protection"))) != 0) {
-               set_denormal_protection (string_is_affirmative (prop->value()));
+       bool denormal_protection;
+       if (node.get_property (X_("denormal-protection"), denormal_protection)) {
+               set_denormal_protection (denormal_protection);
        }
 
        /* convert old 3001 state */
-       if ((prop = node.property (X_("phase-invert"))) != 0) {
-               _phase_control->set_phase_invert (boost::dynamic_bitset<> (prop->value ()));
+       std::string phase_invert_str;
+       if (node.get_property (X_("phase-invert"), phase_invert_str)) {
+               _phase_control->set_phase_invert (boost::dynamic_bitset<> (phase_invert_str));
        }
 
-       if ((prop = node.property (X_("active"))) != 0) {
-               bool yn = string_is_affirmative (prop->value());
-               set_active (yn, this);
+       bool is_active;
+       if (node.get_property (X_("active"), is_active)) {
+               set_active (is_active, this);
        }
 
-       if ((prop = node.property (X_("processor-after-last-custom-meter"))) != 0) {
-               PBD::ID id (prop->value ());
+       std::string id_string;
+       if (node.get_property (X_("processor-after-last-custom-meter"), id_string)) {
+               PBD::ID id (id_string);
                Glib::Threads::RWLock::ReaderLock lm (_processor_lock);
                ProcessorList::const_iterator i = _processors.begin ();
                while (i != _processors.end() && (*i)->id() != id) {
@@ -2538,24 +2534,25 @@ Route::set_state (const XMLNode& node, int version)
                        _comment = cmt->content();
 
                }  else if (child->name() == Controllable::xml_node_name) {
-                       if ((prop = child->property (X_("name"))) == 0) {
+                       std::string control_name;
+                       if (!child->get_property (X_("name"), control_name)) {
                                continue;
                        }
 
-                       if (prop->value() == _gain_control->name()) {
+                       if (control_name == _gain_control->name()) {
                                _gain_control->set_state (*child, version);
-                       } else if (prop->value() == _solo_control->name()) {
+                       } else if (control_name == _solo_control->name()) {
                                _solo_control->set_state (*child, version);
-                       } else if (prop->value() == _solo_safe_control->name()) {
+                       } else if (control_name == _solo_safe_control->name()) {
                                _solo_safe_control->set_state (*child, version);
-                       } else if (prop->value() == _solo_isolate_control->name()) {
+                       } else if (control_name == _solo_isolate_control->name()) {
                                _solo_isolate_control->set_state (*child, version);
-                       } else if (prop->value() == _mute_control->name()) {
+                       } else if (control_name == _mute_control->name()) {
                                _mute_control->set_state (*child, version);
-                       } else if (prop->value() == _phase_control->name()) {
+                       } else if (control_name == _phase_control->name()) {
                                _phase_control->set_state (*child, version);
                        } else {
-                               Evoral::Parameter p = EventTypeMap::instance().from_symbol (prop->value());
+                               Evoral::Parameter p = EventTypeMap::instance().from_symbol (control_name);
                                if (p.type () >= MidiCCAutomation && p.type () < MidiSystemExclusiveAutomation) {
                                        boost::shared_ptr<AutomationControl> ac = automation_control (p, true);
                                        if (ac) {
@@ -2597,13 +2594,13 @@ Route::set_state_2X (const XMLNode& node, int version)
        Stripable::set_state (node, version);
 
        if ((prop = node.property (X_("denormal-protection"))) != 0) {
-               set_denormal_protection (string_is_affirmative (prop->value()));
+               set_denormal_protection (string_to<bool> (prop->value()));
        }
 
        if ((prop = node.property (X_("muted"))) != 0) {
 
                bool first = true;
-               bool muted = string_is_affirmative (prop->value());
+               bool muted = string_to<bool> (prop->value());
 
                if (muted) {
 
@@ -2611,7 +2608,7 @@ Route::set_state_2X (const XMLNode& node, int version)
 
                        if ((prop = node.property (X_("mute-affects-pre-fader"))) != 0) {
 
-                               if (string_is_affirmative (prop->value())){
+                               if (string_to<bool> (prop->value())){
                                        mute_point = mute_point + "PreFader";
                                        first = false;
                                }
@@ -2619,7 +2616,7 @@ Route::set_state_2X (const XMLNode& node, int version)
 
                        if ((prop = node.property (X_("mute-affects-post-fader"))) != 0) {
 
-                               if (string_is_affirmative (prop->value())){
+                               if (string_to<bool> (prop->value())){
 
                                        if (!first) {
                                                mute_point = mute_point + ",";
@@ -2632,7 +2629,7 @@ Route::set_state_2X (const XMLNode& node, int version)
 
                        if ((prop = node.property (X_("mute-affects-control-outs"))) != 0) {
 
-                               if (string_is_affirmative (prop->value())){
+                               if (string_to<bool> (prop->value())){
 
                                        if (!first) {
                                                mute_point = mute_point + ",";
@@ -2645,7 +2642,7 @@ Route::set_state_2X (const XMLNode& node, int version)
 
                        if ((prop = node.property (X_("mute-affects-main-outs"))) != 0) {
 
-                               if (string_is_affirmative (prop->value())){
+                               if (string_to<bool> (prop->value())){
 
                                        if (!first) {
                                                mute_point = mute_point + ",";
@@ -2687,7 +2684,7 @@ Route::set_state_2X (const XMLNode& node, int version)
                        set_id (*child);
 
                        if ((prop = child->property (X_("active"))) != 0) {
-                               bool yn = string_is_affirmative (prop->value());
+                               bool yn = string_to<bool> (prop->value());
                                _active = !yn; // force switch
                                set_active (yn, this);
                        }
@@ -3285,13 +3282,13 @@ Route::direct_feeds_according_to_reality (boost::shared_ptr<Route> other, bool*
 bool
 Route::direct_feeds_according_to_graph (boost::shared_ptr<Route> other, bool* via_send_only)
 {
-       return _session._current_route_graph.has (shared_from_this (), other, via_send_only);
+       return _session._current_route_graph.has (boost::dynamic_pointer_cast<Route> (shared_from_this ()), other, via_send_only);
 }
 
 bool
 Route::feeds_according_to_graph (boost::shared_ptr<Route> other)
 {
-       return _session._current_route_graph.feeds (shared_from_this (), other);
+       return _session._current_route_graph.feeds (boost::dynamic_pointer_cast<Route> (shared_from_this ()), other);
 }
 
 /** Called from the (non-realtime) butler thread when the transport is stopped */
@@ -3340,7 +3337,7 @@ Route::input_change_handler (IOChange change, void * /*src*/)
                                        continue;
                                }
                                bool sends_only;
-                               bool does_feed = (*i)->direct_feeds_according_to_reality (shared_from_this(), &sends_only);
+                               bool does_feed = (*i)->direct_feeds_according_to_reality (boost::dynamic_pointer_cast<Route> (shared_from_this()), &sends_only);
                                if (does_feed && !sends_only) {
                                        if ((*i)->soloed()) {
                                                ++sbou;
@@ -3454,7 +3451,7 @@ Route::output_change_handler (IOChange change, void * /*src*/)
                                _solo_control->mod_solo_by_others_downstream (delta);
                                // Session::route_solo_changed() does not propagate indirect solo-changes
                                // propagate upstream to tracks
-                               boost::shared_ptr<Route> shared_this = shared_from_this();
+                               boost::shared_ptr<Route> shared_this = boost::dynamic_pointer_cast<Route> (shared_from_this());
                                for (RouteList::iterator i = routes->begin(); i != routes->end(); ++i) {
                                        if ((*i).get() == this || !can_solo()) {
                                                continue;
@@ -3516,11 +3513,7 @@ Route::no_roll (pframes_t nframes, framepos_t start_frame, framepos_t end_frame,
                return 0;
        }
 
-       if (n_outputs().n_total() == 0) {
-               return 0;
-       }
-
-       if (!_active || n_inputs() == ChanCount::ZERO)  {
+       if (!_active) {
                silence_unlocked (nframes);
                return 0;
        }
@@ -3565,11 +3558,7 @@ Route::roll (pframes_t nframes, framepos_t start_frame, framepos_t end_frame, in
                return 0;
        }
 
-       if (n_outputs().n_total() == 0) {
-               return 0;
-       }
-
-       if (!_active || n_inputs().n_total() == 0) {
+       if (!_active) {
                silence_unlocked (nframes);
                return 0;
        }
@@ -3608,10 +3597,6 @@ Route::silent_roll (pframes_t nframes, framepos_t /*start_frame*/, framepos_t /*
 void
 Route::flush_processors ()
 {
-       /* XXX shouldn't really try to take this lock, since
-          this is called from the RT audio thread.
-       */
-
        Glib::Threads::RWLock::ReaderLock lm (_processor_lock);
 
        for (ProcessorList::iterator i = _processors.begin(); i != _processors.end(); ++i) {
@@ -4085,7 +4070,7 @@ Route::set_name (const string& str)
 void
 Route::set_name_in_state (XMLNode& node, string const & name, bool rename_playlist)
 {
-       node.add_property (X_("name"), name);
+       node.set_property (X_("name"), name);
 
        XMLNodeList children = node.children();
        for (XMLNodeIterator i = children.begin(); i != children.end(); ++i) {
@@ -4096,17 +4081,17 @@ Route::set_name_in_state (XMLNode& node, string const & name, bool rename_playli
 
                } else if ((*i)->name() == X_("Processor")) {
 
-                       XMLProperty const * role = (*i)->property (X_("role"));
-                       if (role && role->value() == X_("Main")) {
-                               (*i)->add_property (X_("name"), name);
+                       std::string str;
+                       if ((*i)->get_property (X_("role"), str) && str == X_("Main")) {
+                               (*i)->set_property (X_("name"), name);
                        }
 
                } else if ((*i)->name() == X_("Diskstream")) {
 
                        if (rename_playlist) {
-                               (*i)->add_property (X_("playlist"), string_compose ("%1.1", name).c_str());
+                               (*i)->set_property (X_("playlist"), name + ".1");
                        }
-                       (*i)->add_property (X_("name"), name);
+                       (*i)->set_property (X_("name"), name);
 
                }
        }
@@ -4161,6 +4146,7 @@ Route::set_active (bool yn, void* src)
                _active = yn;
                _input->set_active (yn);
                _output->set_active (yn);
+               flush_processors ();
                active_changed (); // EMIT SIGNAL
                _session.set_dirty ();
        }
@@ -5127,6 +5113,38 @@ Route::eq_hpf_controllable () const
 #endif
 }
 
+boost::shared_ptr<AutomationControl>
+Route::eq_lpf_controllable () const
+{
+#ifdef MIXBUS32C
+       boost::shared_ptr<PluginInsert> eq = ch_eq();
+
+       if (!eq) {
+               return boost::shared_ptr<AutomationControl>();
+       }
+
+       return boost::dynamic_pointer_cast<ARDOUR::AutomationControl> (eq->control (Evoral::Parameter (ARDOUR::PluginAutomation, 0, 6)));
+#else
+       return boost::shared_ptr<AutomationControl>();
+#endif
+}
+
+boost::shared_ptr<AutomationControl>
+Route::filter_enable_controllable () const
+{
+#ifdef MIXBUS32C
+       boost::shared_ptr<PluginInsert> eq = ch_eq();
+
+       if (!eq) {
+               return boost::shared_ptr<AutomationControl>();
+       }
+
+       return boost::dynamic_pointer_cast<ARDOUR::AutomationControl> (eq->control (Evoral::Parameter (ARDOUR::PluginAutomation, 0, 2)));
+#else
+       return boost::shared_ptr<AutomationControl>();
+#endif
+}
+
 string
 Route::eq_band_name (uint32_t band) const
 {
@@ -5464,3 +5482,23 @@ Route::clear_all_solo_state ()
 {
        _solo_control->clear_all_solo_state ();
 }
+
+boost::shared_ptr<AutomationControl>
+Route::automation_control_recurse (PBD::ID const & id) const
+{
+       boost::shared_ptr<AutomationControl> ac = Automatable::automation_control (id);
+
+       if (ac) {
+               return ac;
+       }
+
+       Glib::Threads::RWLock::ReaderLock lm (_processor_lock);
+
+       for (ProcessorList::const_iterator i = _processors.begin(); i != _processors.end(); ++i) {
+               if ((ac = (*i)->automation_control (id))) {
+                       return ac;
+               }
+       }
+
+       return boost::shared_ptr<AutomationControl> ();
+}