2 Copyright (C) 1999-2009 Paul Davis
4 This program is free software; you can redistribute it and/or modify
5 it under the terms of the GNU General Public License as published by
6 the Free Software Foundation; either version 2 of the License, or
7 (at your option) any later version.
9 This program is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 GNU General Public License for more details.
14 You should have received a copy of the GNU General Public License
15 along with this program; if not, write to the Free Software
16 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
19 #include <boost/bind.hpp>
21 #include "pbd/error.h"
22 #include "pbd/compose.h"
24 #include "ardour/session.h"
25 #include "ardour/route.h"
26 #include "ardour/track.h"
32 using namespace ARDOUR;
36 Session::set_monitoring (boost::shared_ptr<RouteList> rl, MonitorChoice mc, SessionEvent::RTeventCallback after, bool group_override)
38 queue_event (get_rt_event (rl, mc, after, group_override, &Session::rt_set_monitoring));
42 Session::rt_set_monitoring (boost::shared_ptr<RouteList> rl, MonitorChoice mc, bool /* group_override */)
44 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
45 if (!(*i)->is_auditioner()) {
46 boost::shared_ptr<Track> t = boost::dynamic_pointer_cast<Track> (*i);
48 t->set_monitoring (mc);
57 Session::set_solo (boost::shared_ptr<RouteList> rl, bool yn, SessionEvent::RTeventCallback after, bool group_override)
59 queue_event (get_rt_event (rl, yn, after, group_override, &Session::rt_set_solo));
63 Session::rt_set_solo (boost::shared_ptr<RouteList> rl, bool yn, bool /* group_override */)
65 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
66 if (!(*i)->is_auditioner()) {
67 (*i)->set_solo (yn, this);
75 Session::cancel_solo_after_disconnect (boost::shared_ptr<Route> r, bool upstream, SessionEvent::RTeventCallback after)
77 boost::shared_ptr<RouteList> rl (new RouteList);
80 queue_event (get_rt_event (rl, upstream, after, false, &Session::rt_cancel_solo_after_disconnect));
84 Session::rt_cancel_solo_after_disconnect (boost::shared_ptr<RouteList> rl, bool upstream, bool /* group_override */)
86 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
87 if (!(*i)->is_auditioner()) {
88 (*i)->cancel_solo_after_disconnect (upstream);
91 /* no need to call set-dirty - the disconnect will already have done that */
95 Session::set_just_one_solo (boost::shared_ptr<Route> r, bool yn, SessionEvent::RTeventCallback after)
97 /* its a bit silly to have to do this, but it keeps the API for this public method sane (we're
98 only going to solo one route) and keeps our ability to use get_rt_event() for the internal
102 boost::shared_ptr<RouteList> rl (new RouteList);
105 queue_event (get_rt_event (rl, yn, after, false, &Session::rt_set_just_one_solo));
109 Session::rt_set_just_one_solo (boost::shared_ptr<RouteList> just_one, bool yn, bool /*ignored*/)
111 boost::shared_ptr<RouteList> rl = routes.reader ();
112 boost::shared_ptr<Route> r = just_one->front();
114 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
115 if (!(*i)->is_auditioner() && r != *i) {
116 (*i)->set_solo (!yn, (*i)->route_group());
120 r->set_solo (yn, r->route_group());
126 Session::set_listen (boost::shared_ptr<RouteList> rl, bool yn, SessionEvent::RTeventCallback after, bool group_override)
128 queue_event (get_rt_event (rl, yn, after, group_override, &Session::rt_set_listen));
132 Session::rt_set_listen (boost::shared_ptr<RouteList> rl, bool yn, bool /*group_override*/ )
134 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
135 if (!(*i)->is_auditioner()) {
136 (*i)->set_listen (yn, this);
144 Session::set_mute (boost::shared_ptr<RouteList> rl, bool yn, SessionEvent::RTeventCallback after, bool group_override)
146 /* Set superficial value of mute controls for automation. */
147 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
148 boost::shared_ptr<Route::MuteControllable> mc = (*i)->mute_control();
149 mc->set_superficial_value(yn);
152 queue_event (get_rt_event (rl, yn, after, group_override, &Session::rt_set_mute));
156 Session::rt_set_mute (boost::shared_ptr<RouteList> rl, bool yn, bool /*group_override*/)
158 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
159 if (!(*i)->is_monitor() && !(*i)->is_auditioner()) {
160 (*i)->set_mute (yn, this);
168 Session::set_solo_isolated (boost::shared_ptr<RouteList> rl, bool yn, SessionEvent::RTeventCallback after, bool group_override)
170 queue_event (get_rt_event (rl, yn, after, group_override, &Session::rt_set_solo_isolated));
174 Session::rt_set_solo_isolated (boost::shared_ptr<RouteList> rl, bool yn, bool /*group_override*/)
176 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
177 if (!(*i)->is_master() && !(*i)->is_monitor() && !(*i)->is_auditioner()) {
178 (*i)->set_solo_isolated (yn, this);
186 Session::set_record_enabled (boost::shared_ptr<RouteList> rl, bool yn, SessionEvent::RTeventCallback after, bool group_override)
192 /* do the non-RT part of rec-enabling first - the RT part will be done
193 * on the next process cycle. This does mean that theoretically we are
194 * doing things provisionally on the assumption that the rec-enable
195 * change will work, but this had better be a solid assumption for
199 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
200 if ((*i)->is_auditioner() || (*i)->record_safe ()) {
204 boost::shared_ptr<Track> t;
206 if ((t = boost::dynamic_pointer_cast<Track>(*i)) != 0) {
207 t->prep_record_enabled (yn, (group_override ? (void*) t->route_group() : (void *) this));
211 queue_event (get_rt_event (rl, yn, after, group_override, &Session::rt_set_record_enabled));
215 Session::rt_set_record_enabled (boost::shared_ptr<RouteList> rl, bool yn, bool group_override)
217 for (RouteList::iterator i = rl->begin(); i != rl->end(); ++i) {
218 if ((*i)->is_auditioner() || (*i)->record_safe ()) {
222 boost::shared_ptr<Track> t;
224 if ((t = boost::dynamic_pointer_cast<Track>(*i)) != 0) {
225 t->set_record_enabled (yn, (group_override ? (void*) t->route_group() : (void *) this));
234 Session::set_record_safe (boost::shared_ptr<RouteList> rl, bool yn, SessionEvent::RTeventCallback after, bool group_override)
236 set_record_enabled (rl, false, after, group_override);
237 queue_event (get_rt_event (rl, yn, after, group_override, &Session::rt_set_record_safe));
241 Session::rt_set_record_safe (boost::shared_ptr<RouteList> rl, bool yn, bool group_override)
243 for (RouteList::iterator i = rl->begin (); i != rl->end (); ++i) {
244 if ((*i)->is_auditioner ()) { // REQUIRES REVIEW Can audiotioner be in Record Safe mode?
248 boost::shared_ptr<Track> t;
250 if ((t = boost::dynamic_pointer_cast<Track>(*i)) != 0) {
251 t->set_record_safe (yn, (group_override ? (void*) t->route_group () : (void *) this));
259 Session::process_rtop (SessionEvent* ev)
263 if (ev->event_loop) {
264 ev->event_loop->call_slot (MISSING_INVALIDATOR, boost::bind (ev->rt_return, ev));
266 warning << string_compose ("programming error: %1", X_("Session RT event queued from thread without a UI - cleanup in RT thread!")) << endmsg;