2008-06-02 17:41:35 -04:00
|
|
|
/*
|
2009-12-20 11:49:55 -05:00
|
|
|
Copyright (C) 2006,2007 John Anderson
|
|
|
|
|
|
|
|
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.
|
2008-06-02 17:41:35 -04:00
|
|
|
*/
|
2009-12-20 11:49:55 -05:00
|
|
|
|
2008-06-02 17:41:35 -04:00
|
|
|
#ifndef ardour_mackie_control_protocol_h
|
|
|
|
#define ardour_mackie_control_protocol_h
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
|
|
|
|
#include <sys/time.h>
|
|
|
|
#include <pthread.h>
|
|
|
|
|
|
|
|
#include <glibmm/thread.h>
|
|
|
|
|
2009-10-29 20:21:40 -04:00
|
|
|
#include "ardour/types.h"
|
2009-12-20 11:49:55 -05:00
|
|
|
#include "ardour/midi_ui.h"
|
2009-10-29 20:21:40 -04:00
|
|
|
#include "midi++/types.h"
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-10-29 20:21:40 -04:00
|
|
|
#include "control_protocol/control_protocol.h"
|
2008-06-02 17:41:35 -04:00
|
|
|
#include "midi_byte_array.h"
|
|
|
|
#include "controls.h"
|
2008-12-12 17:55:03 -05:00
|
|
|
#include "dummy_port.h"
|
2008-06-02 17:41:35 -04:00
|
|
|
#include "route_signal.h"
|
|
|
|
#include "mackie_button_handler.h"
|
|
|
|
#include "mackie_port.h"
|
2008-12-12 17:55:03 -05:00
|
|
|
#include "mackie_jog_wheel.h"
|
|
|
|
#include "timer.h"
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
namespace MIDI {
|
|
|
|
class Port;
|
|
|
|
}
|
|
|
|
|
|
|
|
namespace Mackie {
|
|
|
|
class Surface;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
This handles the plugin duties, and the midi encoding and decoding,
|
|
|
|
and the signal callbacks, mostly from ARDOUR::Route.
|
|
|
|
|
|
|
|
The model of the control surface is handled by classes in controls.h
|
|
|
|
|
|
|
|
What happens is that each strip on the control surface has
|
|
|
|
a corresponding route in ControlProtocol::route_table. When
|
|
|
|
an incoming midi message is signaled, the correct route
|
|
|
|
is looked up, and the relevant changes made to it.
|
|
|
|
|
|
|
|
For each route currently in route_table, there's a RouteSignal object
|
|
|
|
which encapsulates the signals that indicate that there are changes
|
|
|
|
to be sent to the surface. The signals are handled by this class.
|
|
|
|
|
|
|
|
Calls to signal handlers pass a Route object which is used to look
|
|
|
|
up the relevant Strip in Surface. Then the state is retrieved from
|
|
|
|
the Route and encoded as the correct midi message.
|
|
|
|
*/
|
2009-12-20 11:49:55 -05:00
|
|
|
|
|
|
|
class MackieControlProtocol
|
|
|
|
: public ARDOUR::ControlProtocol
|
|
|
|
, public Mackie::MackieButtonHandler
|
2008-06-02 17:41:35 -04:00
|
|
|
{
|
|
|
|
public:
|
2009-12-20 11:49:55 -05:00
|
|
|
MackieControlProtocol(ARDOUR::Session &);
|
2008-06-02 17:41:35 -04:00
|
|
|
virtual ~MackieControlProtocol();
|
|
|
|
|
|
|
|
int set_active (bool yn);
|
|
|
|
|
|
|
|
XMLNode& get_state ();
|
2009-10-15 14:56:11 -04:00
|
|
|
int set_state (const XMLNode&, int version);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
static bool probe();
|
|
|
|
|
|
|
|
Mackie::Surface & surface();
|
|
|
|
|
2010-07-29 22:09:39 -04:00
|
|
|
std::list<boost::shared_ptr<ARDOUR::Bundle> > bundles ();
|
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
// control events
|
|
|
|
void handle_control_event(Mackie::SurfacePort & port, Mackie::Control & control, const Mackie::ControlState & state);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
// strip/route related stuff
|
2008-06-02 17:41:35 -04:00
|
|
|
public:
|
|
|
|
/// Signal handler for Route::solo
|
2009-12-20 11:49:55 -05:00
|
|
|
void notify_solo_changed(Mackie::RouteSignal *);
|
2008-06-02 17:41:35 -04:00
|
|
|
/// Signal handler for Route::mute
|
2009-12-20 11:49:55 -05:00
|
|
|
void notify_mute_changed(Mackie::RouteSignal *);
|
2008-06-02 17:41:35 -04:00
|
|
|
/// Signal handler for Route::record_enable_changed
|
2009-12-20 11:49:55 -05:00
|
|
|
void notify_record_enable_changed(Mackie::RouteSignal *);
|
|
|
|
/// Signal handler for Route::gain_changed (from IO)
|
|
|
|
void notify_gain_changed(Mackie::RouteSignal *, bool force_update = true);
|
2008-06-02 17:41:35 -04:00
|
|
|
/// Signal handler for Route::name_change
|
2010-02-19 13:09:08 -05:00
|
|
|
void notify_property_changed(const PBD::PropertyChange&, Mackie::RouteSignal *);
|
2008-06-02 17:41:35 -04:00
|
|
|
/// Signal handler from Panner::Change
|
2009-12-20 11:49:55 -05:00
|
|
|
void notify_panner_changed(Mackie::RouteSignal *, bool force_update = true);
|
2008-06-02 17:41:35 -04:00
|
|
|
/// Signal handler for new routes added
|
2009-12-20 11:49:55 -05:00
|
|
|
void notify_route_added(ARDOUR::RouteList &);
|
2008-12-12 17:55:03 -05:00
|
|
|
/// Signal handler for Route::active_changed
|
2009-12-20 11:49:55 -05:00
|
|
|
void notify_active_changed(Mackie::RouteSignal *);
|
2008-12-12 17:55:03 -05:00
|
|
|
|
2008-06-02 17:41:35 -04:00
|
|
|
void notify_remote_id_changed();
|
|
|
|
|
|
|
|
/// rebuild the current bank. Called on route added/removed and
|
2009-12-20 11:49:55 -05:00
|
|
|
/// remote id changed.
|
2008-06-02 17:41:35 -04:00
|
|
|
void refresh_current_bank();
|
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
// global buttons (ie button not part of strips)
|
|
|
|
|
2008-06-02 17:41:35 -04:00
|
|
|
public:
|
2009-12-20 11:49:55 -05:00
|
|
|
// button-related signals
|
2008-06-02 17:41:35 -04:00
|
|
|
void notify_record_state_changed();
|
2009-12-20 11:49:55 -05:00
|
|
|
void notify_transport_state_changed();
|
|
|
|
// mainly to pick up punch-in and punch-out
|
|
|
|
void notify_parameter_changed(std::string const &);
|
|
|
|
void notify_solo_active_changed(bool);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-10-26 10:38:58 -04:00
|
|
|
/// Turn timecode on and beats off, or vice versa, depending
|
2008-12-12 17:55:03 -05:00
|
|
|
/// on state of _timecode_type
|
2009-10-26 10:38:58 -04:00
|
|
|
void update_timecode_beats_led();
|
2008-12-12 17:55:03 -05:00
|
|
|
|
|
|
|
/// this is called to generate the midi to send in response to a button press.
|
2009-12-20 11:49:55 -05:00
|
|
|
void update_led(Mackie::Button & button, Mackie::LedState);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
void update_global_button(const std::string & name, Mackie::LedState);
|
|
|
|
void update_global_led(const std::string & name, Mackie::LedState);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
// transport button handler methods from MackieButtonHandler
|
|
|
|
virtual Mackie::LedState frm_left_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState frm_left_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState frm_right_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState frm_right_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState stop_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState stop_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState play_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState play_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState record_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState record_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState loop_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState loop_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState punch_in_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState punch_in_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState punch_out_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState punch_out_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState home_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState home_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState end_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState end_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState rewind_press(Mackie::Button & button);
|
|
|
|
virtual Mackie::LedState rewind_release(Mackie::Button & button);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState ffwd_press(Mackie::Button & button);
|
|
|
|
virtual Mackie::LedState ffwd_release(Mackie::Button & button);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
// bank switching button handler methods from MackieButtonHandler
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState left_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState left_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState right_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState right_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState channel_left_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState channel_left_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState channel_right_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState channel_right_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState clicking_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState clicking_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState global_solo_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState global_solo_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
// function buttons
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState marker_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState marker_release(Mackie::Button &);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState drop_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState drop_release(Mackie::Button &);
|
2008-12-12 17:55:03 -05:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState save_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState save_release(Mackie::Button &);
|
2008-12-12 17:55:03 -05:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState timecode_beats_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState timecode_beats_release(Mackie::Button &);
|
2008-12-12 17:55:03 -05:00
|
|
|
|
|
|
|
// jog wheel states
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState zoom_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState zoom_release(Mackie::Button &);
|
2008-12-12 17:55:03 -05:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
virtual Mackie::LedState scrub_press(Mackie::Button &);
|
|
|
|
virtual Mackie::LedState scrub_release(Mackie::Button &);
|
2008-12-12 17:55:03 -05:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
/// This is the main MCU port, ie not an extender port
|
2008-12-12 17:55:03 -05:00
|
|
|
/// Only for use by JogWheel
|
|
|
|
const Mackie::SurfacePort & mcu_port() const;
|
|
|
|
Mackie::SurfacePort & mcu_port();
|
|
|
|
ARDOUR::Session & get_session() { return *session; }
|
|
|
|
|
2008-06-02 17:41:35 -04:00
|
|
|
protected:
|
|
|
|
// create instances of MackiePort, depending on what's found in ardour.rc
|
|
|
|
void create_ports();
|
|
|
|
|
|
|
|
// shut down the surface
|
|
|
|
void close();
|
|
|
|
|
|
|
|
// create the Surface object, with the correct number
|
|
|
|
// of strips for the currently connected ports and
|
|
|
|
// hook up the control event notification
|
|
|
|
void initialize_surface();
|
|
|
|
|
|
|
|
// This sets up the notifications and sets the
|
2009-12-20 11:49:55 -05:00
|
|
|
// controls to the correct values
|
2008-06-02 17:41:35 -04:00
|
|
|
void update_surface();
|
2009-12-20 11:49:55 -05:00
|
|
|
|
|
|
|
// connects global (not strip) signals from the Session to here
|
|
|
|
// so the surface can be notified of changes from the other UIs.
|
|
|
|
void connect_session_signals();
|
|
|
|
|
|
|
|
// set all controls to their zero position
|
2008-06-02 17:41:35 -04:00
|
|
|
void zero_all();
|
|
|
|
|
|
|
|
/**
|
2009-12-20 11:49:55 -05:00
|
|
|
Fetch the set of routes to be considered for control by the
|
|
|
|
surface. Excluding master, hidden and control routes, and inactive routes
|
2008-06-02 17:41:35 -04:00
|
|
|
*/
|
|
|
|
typedef std::vector<boost::shared_ptr<ARDOUR::Route> > Sorted;
|
|
|
|
Sorted get_sorted_routes();
|
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
// bank switching
|
|
|
|
void switch_banks(int initial);
|
|
|
|
void prev_track();
|
|
|
|
void next_track();
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
// delete all RouteSignal objects connecting Routes to Strips
|
|
|
|
void clear_route_signals();
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
typedef std::vector<Mackie::RouteSignal*> RouteSignals;
|
|
|
|
RouteSignals route_signals;
|
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
// return which of the ports a particular route_table
|
|
|
|
// index belongs to
|
|
|
|
Mackie::MackiePort & port_for_id(uint32_t index);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
/**
|
2009-12-20 11:49:55 -05:00
|
|
|
Handle a button press for the control and return whether
|
|
|
|
the corresponding light should be on or off.
|
2008-06-02 17:41:35 -04:00
|
|
|
*/
|
2009-12-20 11:49:55 -05:00
|
|
|
bool handle_strip_button(Mackie::Control &, Mackie::ButtonState, boost::shared_ptr<ARDOUR::Route>);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2010-07-07 21:00:46 -04:00
|
|
|
void add_port (MIDI::Port &, MIDI::Port &, int number);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2008-12-12 09:43:24 -05:00
|
|
|
/**
|
2009-12-20 11:49:55 -05:00
|
|
|
Read session data and send to surface. Includes
|
|
|
|
automation from the currently active routes and
|
|
|
|
timecode displays.
|
2008-12-12 09:43:24 -05:00
|
|
|
*/
|
2008-12-12 17:55:03 -05:00
|
|
|
void poll_session_data();
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
// called from poll_automation to figure out which automations need to be sent
|
2009-12-20 11:49:55 -05:00
|
|
|
void update_automation(Mackie::RouteSignal &);
|
2008-12-12 17:55:03 -05:00
|
|
|
|
2008-12-12 09:43:24 -05:00
|
|
|
// also called from poll_automation to update timecode display
|
|
|
|
void update_timecode_display();
|
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
std::string format_bbt_timecode (ARDOUR::nframes_t now_frame);
|
|
|
|
std::string format_timecode_timecode (ARDOUR::nframes_t now_frame);
|
2008-12-12 17:55:03 -05:00
|
|
|
|
2008-06-02 17:41:35 -04:00
|
|
|
/**
|
2009-12-20 11:49:55 -05:00
|
|
|
notification that the port is about to start it's init sequence.
|
|
|
|
We must make sure that before this exits, the port is being polled
|
|
|
|
for new data.
|
2008-06-02 17:41:35 -04:00
|
|
|
*/
|
2009-12-20 11:49:55 -05:00
|
|
|
void handle_port_init(Mackie::SurfacePort *);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
/// notification from a MackiePort that it's now active
|
2009-12-20 11:49:55 -05:00
|
|
|
void handle_port_active(Mackie::SurfacePort *);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
/// notification from a MackiePort that it's now inactive
|
2009-12-20 11:49:55 -05:00
|
|
|
void handle_port_inactive(Mackie::SurfacePort *);
|
2008-06-02 17:41:35 -04:00
|
|
|
|
|
|
|
boost::shared_ptr<ARDOUR::Route> master_route();
|
|
|
|
Mackie::Strip & master_strip();
|
|
|
|
|
|
|
|
private:
|
|
|
|
boost::shared_ptr<Mackie::RouteSignal> master_route_signal;
|
2009-12-20 11:49:55 -05:00
|
|
|
|
|
|
|
static const char * default_port_name;
|
|
|
|
|
2008-06-02 17:41:35 -04:00
|
|
|
/// The Midi port(s) connected to the units
|
2009-01-14 18:54:46 -05:00
|
|
|
typedef std::vector<Mackie::MackiePort*> MackiePorts;
|
2008-06-02 17:41:35 -04:00
|
|
|
MackiePorts _ports;
|
|
|
|
|
2008-12-12 17:55:03 -05:00
|
|
|
/// Sometimes the real port goes away, and we want to contain the breakage
|
|
|
|
Mackie::DummyPort _dummy_port;
|
|
|
|
|
2008-06-02 17:41:35 -04:00
|
|
|
/// The initial remote_id of the currently switched in bank.
|
2009-12-20 11:49:55 -05:00
|
|
|
uint32_t _current_initial_bank;
|
2008-06-02 17:41:35 -04:00
|
|
|
|
2009-12-20 11:49:55 -05:00
|
|
|
/// protects the port list
|
2008-06-02 17:41:35 -04:00
|
|
|
Glib::Mutex update_mutex;
|
|
|
|
|
2009-12-17 13:24:23 -05:00
|
|
|
PBD::ScopedConnectionList session_connections;
|
|
|
|
PBD::ScopedConnectionList port_connections;
|
|
|
|
PBD::ScopedConnectionList route_connections;
|
|
|
|
|
|
|
|
/// The representation of the physical controls on the surface.
|
2008-06-02 17:41:35 -04:00
|
|
|
Mackie::Surface * _surface;
|
|
|
|
|
|
|
|
bool _transport_previously_rolling;
|
2008-12-12 17:55:03 -05:00
|
|
|
|
|
|
|
// timer for two quick marker left presses
|
|
|
|
Mackie::Timer _frm_left_last;
|
|
|
|
|
|
|
|
Mackie::JogWheel _jog_wheel;
|
|
|
|
|
|
|
|
// last written timecode string
|
|
|
|
std::string _timecode_last;
|
|
|
|
|
2009-10-26 10:38:58 -04:00
|
|
|
// Which timecode are we displaying? BBT or Timecode
|
2008-12-12 17:55:03 -05:00
|
|
|
ARDOUR::AnyTime::Type _timecode_type;
|
2010-07-29 22:09:39 -04:00
|
|
|
|
|
|
|
// Bundle to represent our input ports
|
|
|
|
boost::shared_ptr<ARDOUR::Bundle> _input_bundle;
|
|
|
|
// Bundle to represent our output ports
|
|
|
|
boost::shared_ptr<ARDOUR::Bundle> _output_bundle;
|
2008-06-02 17:41:35 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif // ardour_mackie_control_protocol_h
|