2 Copyright (C) 2006,2007 John Anderson
3 Copyright (C) 2012 Paul Davis
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License
16 along with this program; if not, write to the Free Software
17 Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
22 #include <glibmm/miscutils.h>
24 #include "pbd/xml++.h"
25 #include "pbd/error.h"
26 #include "pbd/file_utils.h"
27 #include "pbd/convert.h"
28 #include "pbd/stl_delete.h"
30 #include "ardour/filesystem_paths.h"
32 #include "device_info.h"
37 using namespace ARDOUR;
38 using namespace ArdourSurface;
39 using namespace Mackie;
44 std::map<std::string,DeviceInfo> DeviceInfo::device_info;
46 DeviceInfo::DeviceInfo()
49 , _master_position (0)
50 , _has_two_character_display (true)
51 , _has_master_fader (true)
52 , _has_timecode_display (true)
53 , _has_global_controls (true)
54 , _has_jog_wheel (true)
55 , _has_touch_sense_faders (true)
56 , _uses_logic_control_buttons (false)
57 , _uses_ipmidi (false)
58 , _no_handshake (false)
61 , _name (X_("Mackie Control Universal Pro"))
63 mackie_control_buttons ();
66 DeviceInfo::~DeviceInfo()
71 DeviceInfo::get_global_button(Button::ID id)
73 GlobalButtonsInfo::iterator it;
75 it = _global_buttons.find (id);
81 DeviceInfo::get_global_button_name(Button::ID id)
83 GlobalButtonsInfo::iterator it;
85 it = _global_buttons.find (id);
86 if (it == _global_buttons.end ()) {
87 _global_button_name = "";
88 return _global_button_name;
90 return it->second.label;
95 DeviceInfo::mackie_control_buttons ()
97 _global_buttons.clear ();
100 _global_buttons[Button::UserA] = GlobalButtonInfo ("Rear Panel User Switch 1", "user", 0x66);
101 _global_buttons[Button::UserB] = GlobalButtonInfo ("Rear Panel User Switch 2", "user", 0x67);
103 //TODO Implement "rear panel external control": a connection for a resistive
104 //TODO element expression pedal . Message: 0xb0 0x2e 0xVV where 0xVV = external
105 //TODO controller position value (0x00 to 0x7f)
107 _strip_buttons[Button::RecEnable] = StripButtonInfo (0x0, "Rec");
111 DeviceInfo::logic_control_buttons ()
113 _global_buttons.clear ();
116 _global_buttons[Button::UserA] = GlobalButtonInfo ("User Switch A", "user", 0x66);
117 _global_buttons[Button::UserB] = GlobalButtonInfo ("User Switch B", "user", 0x67);
119 _strip_buttons[Button::RecEnable] = StripButtonInfo (0x0, "Rec/Rdy");
123 DeviceInfo::shared_buttons ()
125 _global_buttons[Button::Track] = GlobalButtonInfo ("Track", "assignment", 0x28);
126 _global_buttons[Button::Send] = GlobalButtonInfo ("Send", "assignment", 0x29);
127 _global_buttons[Button::Pan] = GlobalButtonInfo ("Pan/Surround", "assignment", 0x2a);
128 _global_buttons[Button::Plugin] = GlobalButtonInfo ("Plugin", "assignment", 0x2b);
129 _global_buttons[Button::Eq] = GlobalButtonInfo ("Eq", "assignment", 0x2c);
130 _global_buttons[Button::Dyn] = GlobalButtonInfo ("Instrument", "assignment", 0x2d);
132 _global_buttons[Button::Left] = GlobalButtonInfo ("Bank Left", "bank", 0x2e);
133 _global_buttons[Button::Right] = GlobalButtonInfo ("Bank Right", "bank", 0x2f);
134 _global_buttons[Button::ChannelLeft] = GlobalButtonInfo ("Channel Left", "bank", 0x30);
135 _global_buttons[Button::ChannelRight] = GlobalButtonInfo ("Channel Right", "bank", 0x31);
136 _global_buttons[Button::Flip] = GlobalButtonInfo ("Flip", "assignment", 0x32);
137 _global_buttons[Button::View] = GlobalButtonInfo ("Global View", "global view", 0x33);
139 _global_buttons[Button::NameValue] = GlobalButtonInfo ("Name/Value", "display", 0x34);
140 _global_buttons[Button::TimecodeBeats] = GlobalButtonInfo ("Timecode/Beats", "display", 0x35);
142 _global_buttons[Button::F1] = GlobalButtonInfo ("F1", "function select", 0x36);
143 _global_buttons[Button::F2] = GlobalButtonInfo ("F2", "function select", 0x37);
144 _global_buttons[Button::F3] = GlobalButtonInfo ("F3", "function select", 0x38);
145 _global_buttons[Button::F4] = GlobalButtonInfo ("F4", "function select", 0x39);
146 _global_buttons[Button::F5] = GlobalButtonInfo ("F5", "function select", 0x3a);
147 _global_buttons[Button::F6] = GlobalButtonInfo ("F6", "function select", 0x3b);
148 _global_buttons[Button::F7] = GlobalButtonInfo ("F7", "function select", 0x3c);
149 _global_buttons[Button::F8] = GlobalButtonInfo ("F8", "function select", 0x3d);
151 _global_buttons[Button::MidiTracks] = GlobalButtonInfo ("MIDI Tracks", "global view", 0x3e);
152 _global_buttons[Button::Inputs] = GlobalButtonInfo ("Inputs", "global view", 0x3f);
153 _global_buttons[Button::AudioTracks] = GlobalButtonInfo ("Audio Tracks", "global view", 0x40);
154 _global_buttons[Button::AudioInstruments] = GlobalButtonInfo ("Audio Instruments", "global view", 0x41);
155 _global_buttons[Button::Aux] = GlobalButtonInfo ("Aux", "global view", 0x42);
156 _global_buttons[Button::Busses] = GlobalButtonInfo ("Busses", "global view", 0x43);
157 _global_buttons[Button::Outputs] = GlobalButtonInfo ("Outputs", "global view", 0x44);
158 _global_buttons[Button::User] = GlobalButtonInfo ("User", "global view", 0x45);
160 _global_buttons[Button::Shift] = GlobalButtonInfo ("Shift", "modifiers", 0x46);
161 _global_buttons[Button::Option] = GlobalButtonInfo ("Option", "modifiers", 0x47);
162 _global_buttons[Button::Ctrl] = GlobalButtonInfo ("Ctrl", "modifiers", 0x48);
163 _global_buttons[Button::CmdAlt] = GlobalButtonInfo ("Cmd/Alt", "modifiers", 0x49);
165 _global_buttons[Button::Read] = GlobalButtonInfo ("Read/Off", "automation", 0x4a);
166 _global_buttons[Button::Write] = GlobalButtonInfo ("Write", "automation", 0x4b);
167 _global_buttons[Button::Trim] = GlobalButtonInfo ("Trim", "automation", 0x4c);
168 _global_buttons[Button::Touch] = GlobalButtonInfo ("Touch", "automation", 0x4d);
169 _global_buttons[Button::Latch] = GlobalButtonInfo ("Latch", "automation", 0x4e);
170 _global_buttons[Button::Grp] = GlobalButtonInfo ("Group", "automation", 0x4f);
172 _global_buttons[Button::Save] = GlobalButtonInfo ("Save", "utilities", 0x50);
173 _global_buttons[Button::Undo] = GlobalButtonInfo ("Undo", "utilities", 0x51);
174 _global_buttons[Button::Cancel] = GlobalButtonInfo ("Cancel", "utilities", 0x52);
175 _global_buttons[Button::Enter] = GlobalButtonInfo ("Enter", "utilities", 0x53);
177 _global_buttons[Button::Marker] = GlobalButtonInfo ("Marker", "transport", 0x54);
178 _global_buttons[Button::Nudge] = GlobalButtonInfo ("Nudge", "transport", 0x55);
179 _global_buttons[Button::Loop] = GlobalButtonInfo ("Cycle", "transport", 0x56);
180 _global_buttons[Button::Drop] = GlobalButtonInfo ("Drop", "transport", 0x57);
181 _global_buttons[Button::Replace] = GlobalButtonInfo ("Replace", "transport", 0x58);
182 _global_buttons[Button::Click] = GlobalButtonInfo ("Click", "transport", 0x59);
183 _global_buttons[Button::ClearSolo] = GlobalButtonInfo ("Solo", "transport", 0x5a);
185 _global_buttons[Button::Rewind] = GlobalButtonInfo ("Rewind", "transport", 0x5b);
186 _global_buttons[Button::Ffwd] = GlobalButtonInfo ("Fast Fwd", "transport", 0x5c);
187 _global_buttons[Button::Stop] = GlobalButtonInfo ("Stop", "transport", 0x5d);
188 _global_buttons[Button::Play] = GlobalButtonInfo ("Play", "transport", 0x5e);
189 _global_buttons[Button::Record] = GlobalButtonInfo ("Record", "transport", 0x5f);
191 _global_buttons[Button::CursorUp] = GlobalButtonInfo ("Cursor Up", "cursor", 0x60);
192 _global_buttons[Button::CursorDown] = GlobalButtonInfo ("Cursor Down", "cursor", 0x61);
193 _global_buttons[Button::CursorLeft] = GlobalButtonInfo ("Cursor Left", "cursor", 0x62);
194 _global_buttons[Button::CursorRight] = GlobalButtonInfo ("Cursor Right", "cursor", 0x63);
195 _global_buttons[Button::Zoom] = GlobalButtonInfo ("Zoom", "cursor", 0x64);
196 _global_buttons[Button::Scrub] = GlobalButtonInfo ("Scrub", "cursor", 0x65);
198 _strip_buttons[Button::Solo] = StripButtonInfo (0x08, "Solo");
199 _strip_buttons[Button::Mute] = StripButtonInfo (0x10, "Mute");
200 _strip_buttons[Button::Select] = StripButtonInfo (0x18, "Select");
201 _strip_buttons[Button::VSelect] = StripButtonInfo (0x20, "V-Select");
203 _strip_buttons[Button::FaderTouch] = StripButtonInfo (0x68, "Fader Touch");
205 _global_buttons[Button::MasterFaderTouch] = GlobalButtonInfo ("Master Fader Touch", "master", 0x70);
209 DeviceInfo::set_state (const XMLNode& node, int /* version */)
211 const XMLProperty* prop;
212 const XMLNode* child;
214 if (node.name() != "MackieProtocolDevice") {
218 /* Device type ought to be mandatory but early versions missed it */
219 if ((child = node.child ("DeviceType")) != 0) {
220 if ((prop = child->property ("value")) != 0) {
221 if (prop->value() == X_("MCU")) {
223 } else if (prop->value() == X_("MCXT")) {
225 } else if (prop->value() == X_("LC")) {
227 } else if (prop->value() == X_("LCXT")) {
229 } else if (prop->value() == X_("HUI")) {
232 error << string_compose (_("Unknown Mackie device type \"%1\" used in device info file, using MCU instead"), prop->value()) << endmsg;
240 /* name is mandatory */
241 if ((child = node.child ("Name")) != 0) {
242 if ((prop = child->property ("value")) != 0) {
243 _name = prop->value();
249 /* strip count is mandatory */
250 if ((child = node.child ("Strips")) != 0) {
251 if ((prop = child->property ("value")) != 0) {
252 if ((_strip_cnt = atoi (prop->value().c_str())) == 0) {
260 if ((child = node.child ("Extenders")) != 0) {
261 if ((prop = child->property ("value")) != 0) {
262 if ((_extenders = atoi (prop->value().c_str())) == 0) {
268 if ((child = node.child ("MasterPosition")) != 0) {
269 if ((prop = child->property ("value")) != 0) {
270 if ((_master_position = atoi (prop->value().c_str())) == 0) {
271 _master_position = 0;
272 } else if (_master_position > 0) {
278 if ((child = node.child ("TwoCharacterDisplay")) != 0) {
279 if ((prop = child->property ("value")) != 0) {
280 _has_two_character_display = string_is_affirmative (prop->value());
284 if ((child = node.child ("MasterFader")) != 0) {
285 if ((prop = child->property ("value")) != 0) {
286 _has_master_fader = string_is_affirmative (prop->value());
290 if ((child = node.child ("TimecodeDisplay")) != 0) {
291 if ((prop = child->property ("value")) != 0) {
292 _has_timecode_display = string_is_affirmative (prop->value());
295 _has_timecode_display = false;
298 if ((child = node.child ("GlobalControls")) != 0) {
299 if ((prop = child->property ("value")) != 0) {
300 _has_global_controls = string_is_affirmative (prop->value());
303 _has_global_controls = false;
306 if ((child = node.child ("JogWheel")) != 0) {
307 if ((prop = child->property ("value")) != 0) {
308 _has_jog_wheel = string_is_affirmative (prop->value());
311 _has_jog_wheel = false;
314 if ((child = node.child ("TouchSenseFaders")) != 0) {
315 if ((prop = child->property ("value")) != 0) {
316 _has_touch_sense_faders = string_is_affirmative (prop->value());
319 _has_touch_sense_faders = false;
322 if ((child = node.child ("UsesIPMIDI")) != 0) {
323 if ((prop = child->property ("value")) != 0) {
324 _uses_ipmidi = string_is_affirmative (prop->value());
327 _uses_ipmidi = false;
330 if ((child = node.child ("NoHandShake")) != 0) {
331 if ((prop = child->property ("value")) != 0) {
332 _no_handshake = string_is_affirmative (prop->value());
335 _no_handshake = false;
338 if ((child = node.child ("HasMeters")) != 0) {
339 if ((prop = child->property ("value")) != 0) {
340 _has_meters = string_is_affirmative (prop->value());
346 if ((child = node.child ("LogicControlButtons")) != 0) {
347 if ((prop = child->property ("value")) != 0) {
348 _uses_logic_control_buttons = string_is_affirmative (prop->value());
350 if (_uses_logic_control_buttons) {
351 logic_control_buttons();
353 mackie_control_buttons ();
358 if ((child = node.child ("Buttons")) != 0) {
359 XMLNodeConstIterator i;
360 const XMLNodeList& nlist (child->children());
362 for (i = nlist.begin(); i != nlist.end(); ++i) {
363 if ((*i)->name() == "GlobalButton") {
364 if ((prop = (*i)->property ("name")) != 0) {
365 int id = Button::name_to_id (prop->value());
367 Button::ID bid = (Button::ID) id;
368 if ((prop = (*i)->property ("id")) != 0) {
369 int val = strtol (prop->value().c_str(), 0, 0);
370 std::map<Button::ID,GlobalButtonInfo>::iterator b = _global_buttons.find (bid);
371 if (b != _global_buttons.end()) {
374 if ((prop = (*i)->property ("label")) != 0) {
375 b->second.label = prop->value();
383 } else if ((*i)->name() == "StripButton") {
384 if ((prop = (*i)->property ("name")) != 0) {
385 int id = Button::name_to_id (prop->value());
387 Button::ID bid = (Button::ID) id;
388 if ((prop = (*i)->property ("baseid")) != 0) {
389 int val = strtol (prop->value().c_str(), 0, 0);
390 std::map<Button::ID,StripButtonInfo>::iterator b = _strip_buttons.find (bid);
391 if (b != _strip_buttons.end()) {
392 b->second.base_id = val;
407 DeviceInfo::name() const
413 DeviceInfo::strip_cnt() const
419 DeviceInfo::extenders() const
425 DeviceInfo::master_position() const
427 return _master_position;
431 DeviceInfo::has_master_fader() const
433 return _has_master_fader;
437 DeviceInfo::has_meters() const
443 DeviceInfo::has_two_character_display() const
445 return _has_two_character_display;
449 DeviceInfo::has_timecode_display () const
451 return _has_timecode_display;
455 DeviceInfo::uses_ipmidi () const
461 DeviceInfo::has_global_controls () const
463 return _has_global_controls;
467 DeviceInfo::has_jog_wheel () const
469 return _has_jog_wheel;
473 DeviceInfo::no_handshake () const
475 return _no_handshake;
479 DeviceInfo::has_touch_sense_faders () const
481 return _has_touch_sense_faders;
484 static const char * const devinfo_env_variable_name = "ARDOUR_MCP_PATH";
485 static const char* const devinfo_dir_name = "mcp";
486 static const char* const devinfo_suffix = ".device";
489 devinfo_search_path ()
491 bool devinfo_path_defined = false;
492 std::string spath_env (Glib::getenv (devinfo_env_variable_name, devinfo_path_defined));
494 if (devinfo_path_defined) {
498 Searchpath spath (ardour_data_search_path());
499 spath.add_subdirectory_to_paths(devinfo_dir_name);
505 devinfo_filter (const string &str, void* /*arg*/)
507 return (str.length() > strlen(devinfo_suffix) &&
508 str.find (devinfo_suffix) == (str.length() - strlen (devinfo_suffix)));
512 DeviceInfo::reload_device_info ()
515 vector<string> devinfos;
516 Searchpath spath (devinfo_search_path());
518 find_files_matching_filter (devinfos, spath, devinfo_filter, 0, false, true);
519 device_info.clear ();
521 if (devinfos.empty()) {
522 error << "No MCP device info files found using " << spath.to_string() << endmsg;
523 std::cerr << "No MCP device info files found using " << spath.to_string() << std::endl;
527 for (vector<string>::iterator i = devinfos.begin(); i != devinfos.end(); ++i) {
528 string fullpath = *i;
529 DeviceInfo di; // has to be initial every loop or info from last added.
533 if (!tree.read (fullpath.c_str())) {
537 XMLNode* root = tree.root ();
542 if (di.set_state (*root, 3000) == 0) { /* version is ignored for now */
543 device_info[di.name()] = di;
548 std::ostream& operator<< (std::ostream& os, const Mackie::DeviceInfo& di)
550 os << di.name() << ' '
551 << di.strip_cnt() << ' '
552 << di.extenders() << ' '
553 << di.master_position() << ' '